Skip to content

FastFilter/fastfilter_java

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Fast Approximate Membership Filters in Java

Java 17 CI

The following filter types are currently implemented:

  • Xor filter: 8 and 16 bit variants; needs less space than cuckoo filters, with faster lookup
  • Xor+ filter: 8 and 16 bit variants; compressed xor filter
  • Xor binary fuse filter: 8. 16 and 32 bit variants; needs less space than xor filters (for large filters), with faster lookup
  • Cuckoo filter: 8 and 16 bit variants; uses cuckoo hashing to store fingerprints
  • Cuckoo+ filter: 8 and 16 bit variants, need a bit less space than regular cuckoo filters
  • Bloom filter: the 'standard' algorithm
  • Blocked Bloom filter: faster than regular Bloom filters, but need a bit more space
  • Counting Bloom filter: allow removing entries, but need 4 times more space
  • Succinct counting Bloom filter: about half the space of regular counting Bloom filters; faster lookup but slower add / remove
  • Succinct counting blocked Bloom filter: same lookup speed as blocked Bloom filter

The following additional types are implemented, but less tested:

  • Golomb Compressed Set (GCS): needs less space than cuckoo filters, but lookup is slow
  • Minimal Perfect Hash filter: needs less space than cuckoo filters, but lookup is slow

Reference

Usage

When using Maven:

<dependency>
  <groupId>io.github.fastfilter</groupId>
  <artifactId>fastfilter</artifactId>
  <version>1.0.4</version>
</dependency>

Other Xor Filter Implementations

Note that the data format in other implementations may not match the data format in Java.

Password Lookup Tool

Included is a tool to build a filter from a list of known password (hashes), and a tool to do lookups. That way, the password list can be queried locally, without requiring a large file. The filter is only 650 MB, instead of the original file which is 11 GB. At the cost of some false positives (unknown passwords reported as known, with about 1% probability).

Generate the Password Filter File

Download the latest SHA-1 password file that is ordered by hash, for example the file pwned-passwords-sha1-ordered-by-hash-v4.7z (~10 GB) from https://haveibeenpwned.com/passwords with about 550 million passwords.

If you have enough disk space, you can extract the hash file (~25 GB), and convert it as follows:

mvn clean install
cat hash.txt | java -cp target/fastfilter*.jar org.fastfilter.tools.BuildFilterFile filter.bin

Converting takes about 2-3 minutes (depending on hardware). To save disk space, you can extract the file on the fly (Mac OS X using Keka):

/Applications/Keka.app/Contents/Resources/keka7z e -so
    pass.7z | java -cp target/fastfilter*.jar org.fastfilter.tools.BuildFilterFile filter.bin

Both will generate a file named filter.bin (~630 MB).

Check Passwords

java -cp target/fastfilter*.jar org.fastfilter.tools.PasswordLookup filter.bin

Enter a password to see if it's in the list. If yes, it will (for sure) either show "Found", or "Found; common", which means it was seen 10 times or more often. Passwords not in the list will show "Not found" with more than 99% probability, and with less than 1% probability "Found" or "Found; common".

Internally, the tool uses a xor+ filter (see above) with 8 bits per fingerprint. Actually, 1024 smaller filters (segments) are made, the segment id being the highest 10 bits of the key. The lowest bit of the key is set to either 0 (regular) or 1 (common), and so two lookups are made per password. Because of that, the false positive rate is twice of what it would be with just one lookup (0.0078 instead of 0.0039). A regular Bloom filter with the same guarantees would be ~760 MB. For each lookup, one filter segment (so, less than 1 MB) are read from the file.

Benchmarks

The project includes JMH (Java Microbenchmark Harness) benchmarks to measure the performance of the filters.

Running Benchmarks

Option 1: Run via Maven (recommended)

To run the benchmarks directly from Maven (with minimal iterations for quick testing):

mvn -pl jmh clean package exec:exec@run-benchmarks

For full benchmarks, modify the pom.xml or run the JAR manually with custom parameters.

This will compile and execute the JMH benchmarks for the XOR filters (XOR_8, XOR_16, XOR_BINARY_FUSE_8, XOR_BINARY_FUSE_16).

Option 2: Run the JAR manually

First, build the project:

mvn clean package

Then run the benchmarks:

java -jar jmh/target/benchmarks.jar org.fastfilter.FilterBenchmark

To run benchmarks for a specific filter type:

java -jar jmh/target/benchmarks.jar org.fastfilter.FilterBenchmark -p filterType=XOR_BINARY_FUSE_8

Available filter types: XOR_8, XOR_16, XOR_BINARY_FUSE_8, XOR_BINARY_FUSE_16.

Benchmark Details

The benchmarks measure:

  • Average time per operation (nanoseconds) for lookups of existing and non-existing keys
  • Throughput (operations per second) for the same operations
  • False positive rate validation

Possible results:


Benchmark                                                     (filterType)   Mode  Cnt          Score   Error  Units
FilterBenchmark.benchmarkContainsExistingThroughput                  XOR_8  thrpt       412364492,755          ops/s
FilterBenchmark.benchmarkContainsExistingThroughput                 XOR_16  thrpt       397627818,837          ops/s
FilterBenchmark.benchmarkContainsExistingThroughput      XOR_BINARY_FUSE_8  thrpt       516262004,459          ops/s
FilterBenchmark.benchmarkContainsExistingThroughput     XOR_BINARY_FUSE_16  thrpt       489256453,340          ops/s
FilterBenchmark.benchmarkContainsNonExistingThroughput               XOR_8  thrpt       429856367,135          ops/s
FilterBenchmark.benchmarkContainsNonExistingThroughput              XOR_16  thrpt       441042890,257          ops/s
FilterBenchmark.benchmarkContainsNonExistingThroughput   XOR_BINARY_FUSE_8  thrpt       533609392,046          ops/s
FilterBenchmark.benchmarkContainsNonExistingThroughput  XOR_BINARY_FUSE_16  thrpt       540058414,150          ops/s
FilterBenchmark.benchmarkContainsExisting                            XOR_8   avgt               2,475          ns/op
FilterBenchmark.benchmarkContainsExisting                           XOR_16   avgt               2,522          ns/op
FilterBenchmark.benchmarkContainsExisting                XOR_BINARY_FUSE_8   avgt               1,965          ns/op
FilterBenchmark.benchmarkContainsExisting               XOR_BINARY_FUSE_16   avgt               2,060          ns/op
FilterBenchmark.benchmarkContainsNonExisting                         XOR_8   avgt               2,347          ns/op
FilterBenchmark.benchmarkContainsNonExisting                        XOR_16   avgt               2,295          ns/op
FilterBenchmark.benchmarkContainsNonExisting             XOR_BINARY_FUSE_8   avgt               1,892          ns/op
FilterBenchmark.benchmarkContainsNonExisting            XOR_BINARY_FUSE_16   avgt               1,903          ns/op

This indicates that we can issue about half a billion queries per second, and sustain a rate of about 2 ns per query.

The benchmarks use 1,000,000 keys by default. You can modify the NUM_KEYS constant in FilterBenchmark.java for smaller/larger test sets.

About

Fast Approximate Membership Filters (Java)

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 7

Languages