46 releases (29 stable)

new 2.5.0 Jan 8, 2025
2.5.0-pre.1 Dec 20, 2024
2.4.0 Nov 15, 2024
1.5.0 Aug 5, 2024
0.1.0 Dec 21, 2023

#3 in Database implementations

Download history 515/week @ 2024-09-23 241/week @ 2024-09-30 447/week @ 2024-10-07 543/week @ 2024-10-14 222/week @ 2024-10-21 597/week @ 2024-10-28 259/week @ 2024-11-04 432/week @ 2024-11-11 152/week @ 2024-11-18 295/week @ 2024-11-25 454/week @ 2024-12-02 349/week @ 2024-12-09 453/week @ 2024-12-16 333/week @ 2024-12-23 228/week @ 2024-12-30 616/week @ 2025-01-06

1,655 downloads per month
Used in 5 crates

MIT/Apache

350KB
6K SLoC

CI docs.rs Crates.io MSRV dependency status

Discord Bluesky

Fjall is an LSM-based embeddable key-value storage engine written in Rust. It features:

  • Thread-safe BTreeMap-like API
  • 100% safe & stable Rust
  • Range & prefix searching with forward and reverse iteration
  • Partitions (a.k.a. column families) with cross-partition atomic semantics
  • Built-in compression (default = LZ4)
  • Serializable transactions (optional)
  • Key-value separation for large blob use cases (optional)
  • Automatic background maintenance

It is not:

  • a standalone server
  • a relational database
  • a wide-column database: it has no built-in notion of columns

Keys are limited to 65536 bytes, values are limited to 2^32 bytes. As is normal with any kind of storage engine, larger keys and values have a bigger performance impact.

Like any typical key-value store, keys are stored in lexicographic order. If you are storing integer keys (e.g. timeseries data), you should use the big endian form to adhere to locality.

Basic usage

cargo add fjall
use fjall::{Config, PersistMode, Keyspace, PartitionCreateOptions};

// A keyspace is a database, which may contain multiple collections ("partitions")
// You should probably only use a single keyspace for your application
//
let keyspace = Config::new(folder).open()?; // or open_transactional for transactional semantics

// Each partition is its own physical LSM-tree
let items = keyspace.open_partition("my_items", PartitionCreateOptions::default())?;

// Write some data
items.insert("a", "hello")?;

// And retrieve it
let bytes = items.get("a")?;

// Or remove it again
items.remove("a")?;

// Search by prefix
for kv in items.prefix("prefix") {
  // ...
}

// Search by range
for kv in items.range("a"..="z") {
  // ...
}

// Iterators implement DoubleEndedIterator, so you can search backwards, too!
for kv in items.prefix("prefix").rev() {
  // ...
}

// Sync the journal to disk to make sure data is definitely durable
// When the keyspace is dropped, it will try to persist with `PersistMode::SyncAll` as well
keyspace.persist(PersistMode::SyncAll)?;

Durability

To support different kinds of workloads, Fjall is agnostic about the type of durability your application needs. After writing data (insert, remove or committing a write batch/transaction), you can choose to call Keyspace::persist which takes a PersistMode parameter. By default, any operation will flush to OS buffers, but not to disk. This is in line with RocksDB's default durability. Also, when dropped, the keyspace will try to persist the journal to disk synchronously.

Multithreading, Async and Multiprocess

!!! A single keyspace may not be loaded in parallel from separate processes.

However, Fjall is internally synchronized for multi-threaded access, so you can clone around the Keyspace and Partitions as needed, without needing to lock yourself.

For an async example, see the tokio example.

Feature flags

lz4

Allows using LZ4 compression, powered by lz4_flex.

Enabled by default.

miniz

Allows using DEFLATE/zlib compression, powered by miniz_oxide.

Disabled by default.

single_writer_tx

Allows opening a transactional Keyspace for single-writer (serialized) transactions, allowing RYOW (read-your-own-write), fetch-and-update and other atomic operations.

Enabled by default.

ssi_tx

Allows opening a transactional Keyspace for multi-writer, serializable transactions, allowing RYOW (read-your-own-write), fetch-and-update and other atomic operations. Conflict checking is done using optimistic concurrency control.

Disabled by default.

bytes

Uses bytes as the underlying Slice type.

Disabled by default.

bloom [deprecated]

Uses bloom filters to reduce disk I/O when serving point reads, but increases memory usage.

Enabled by default.

Will be removed in the future. If you are absolutely, 100% sure you do not need bloom filters: they will be togglable on a per-partition basis.

Stable disk format

The disk format is stable as of 1.0.0.

2.0.0 uses a new disk format and needs a manual format migration.

Future breaking changes will result in a major version bump and a migration path.

For the underlying LSM-tree implementation, see: https://crates.io/crates/lsm-tree.

Examples

See here for practical examples.

And checkout Smoltable, a standalone Bigtable-inspired toy wide-column database using fjall as its storage engine.

Contributing

How can you help?

License

All source code is licensed under MIT OR Apache-2.0.

All contributions are to be licensed as MIT OR Apache-2.0.

Dependencies

~5–14MB
~191K SLoC