<div align="center">
<h1>SkipDB</h1>
</div>
<div align="center">
Blazing fast ACID and MVCC in memory database based on lock-free skiplist.
`async-skipdb` uses the same SSI (Serializable Snapshot Isolation) transaction model used in [`badger`](https://github.com/dgraph-io/badger).
[<img alt="github" src="https://img.shields.io/badge/github-al8n/async--skipdb-8da0cb?style=for-the-badge&logo=Github" height="22">][Github-url]
[<img alt="Build" src="https://img.shields.io/github/actions/workflow/status/al8n/skipdb/ci.yml?logo=Github-Actions&style=for-the-badge" height="22">][CI-url]
[<img alt="codecov" src="https://img.shields.io/codecov/c/gh/al8n/skipdb?style=for-the-badge&token=6R3QFWRWHL&logo=codecov" height="22">][codecov-url]
<img alt="license" src="https://img.shields.io/badge/License-Apache%202.0/MIT-blue.svg?style=for-the-badge&fontColor=white&logoColor=f5c076&logo=" height="22">
[<img alt="LoC" src="https://img.shields.io/endpoint?url=https%3A%2F%2Fgist.githubusercontent.com%2Fal8n%2F327b2a8aef9003246e45c6e47fe63937%2Fraw%2Fskipdb" height="22">](https://gist.github.com/al8n/327b2a8aef9003246e45c6e47fe63937#file-skipdb)
[<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-async--skipdb-66c2a5?style=for-the-badge&labelColor=555555&logo=" height="20">][doc-url]
[<img alt="crates.io" src="https://img.shields.io/crates/v/async-skipdb?style=for-the-badge&logo=" height="22">][crates-url]
[<img alt="crates.io" src="https://img.shields.io/crates/d/skipdb?color=critical&logo=&style=for-the-badge" height="22">][crates-url]
English | [简体中文][zh-cn-url]
</div>
## Introduction
Blazing fast ACID and MVCC in memory database based on lock-free skiplist.
`async-skipdb` uses the same SSI (Serializable Snapshot Isolation) transaction model used in [`badger`](https://github.com/dgraph-io/badger).
For sync version, please see [`skipdb`](https://crates.io/crates/skipdb).
## Features
- ACID, MVCC, serializable snapshot isolation, concurrent safe and almost lock-free.
- No extra allocation and copy, there is no `Arc` wrapper for both key and value stored in the database, which means that users provide `K` and `V`, and database store `K` and `V` directly.
- Zero-copy and in-place compaction, which means there is no copy, no extra allocation when compacting.
- Concurrent execution of transactions, providing serializable snapshot isolation, avoiding write skews.
- Both read transaction and write transaction are `Send + Sync + 'static`, which means you do not need to handle annoying lifetime problem anymore.
- Lock-free and concurrent safe read transaction: the read transaction is totally concurrent safe and can be shared in multiple threads, there is no lock in read transaction.
- `BTreeMap` like user friendly API and all iterators implement `Iterator` trait, which means users use Rust powerful conbinators when iterating over the database.
- Runtime agnostic, `tokio`, `async-std`, `smol`, `wasm-bindgen-futures` and any other async runtime.
- 100% safe, skipdb sets `[forbid(unsafe_code)]`.
## Installation
- `tokio`
```toml
[dependencies]
async-skipdb = { version = "0.1", features = ["tokio"] }
```
- `async-std`
```toml
[dependencies]
async-skipdb = { version = "0.1", features = ["async-std"] }
```
- `smol`
```toml
[dependencies]
async-skipdb = { version = "0.1", features = ["smol"] }
```
- `wasm-bindgen-futures`
```toml
[dependencies]
async-skipdb = { version = "0.1", features = ["wasm"] }
```
## Example
- If your `K` implement `Hash`.
```rust
use async_skipdb::equivalent::TokioEquivalentDb;
#[derive(Debug)]
struct Person {
hobby: String,
age: u8,
}
#[tokio::main]
async fn main() {
let db: TokioEquivalentDb<String, Person> = TokioEquivalentDb::new().await;
{
let alice = Person { hobby: "swim".to_string(), age: 20 };
let bob = Person { hobby: "run".to_string(), age: 30 };
let mut txn = db.write().await;
txn.insert("Alice".to_string(), alice).unwrap();
txn.insert("Bob".to_string(), bob).unwrap();
{
let alice = txn.get("Alice").unwrap().unwrap();
assert_eq!(alice.value().age, 20);
assert_eq!(alice.value().hobby, "swim");
}
txn.commit().await.unwrap();
}
{
let txn = db.read().await;
let alice = txn.get("Alice").unwrap();
assert_eq!(alice.value().age, 20);
assert_eq!(alice.value().hobby, "swim");
let bob = txn.get("Bob").unwrap();
assert_eq!(bob.value().age, 30);
assert_eq!(bob.value().hobby, "run");
}
}
```
- If your key cannot implement `Hash` for some reasons.
Then you can use `ComparableDb`, but this will require `K: CheapClone + Ord`, you can see [`cheap_clone`](https://crates.io/crates/cheap-clone) trait for more details.
```rust
use async_skipdb::comparable::TokioComparableDb;
#[derive(Debug)]
struct Person {
name: String,
hobby: String,
age: u8,
}
#[tokio::main]
async fn main() {
let db: TokioComparableDb<u64, Person> = TokioComparableDb::new().await;
{
let alice = Person { name: "Alice".to_string(), hobby: "swim".to_string(), age: 20 };
let bob = Person { name: "Bob".to_string(), hobby: "run".to_string(), age: 30 };
let mut txn = db.write().await;
txn.insert(1, alice).unwrap();
txn.insert(2, bob).unwrap();
{
let alice = txn.get(&1).unwrap().unwrap();
assert_eq!(alice.value().name, "Alice");
assert_eq!(alice.value().age, 20);
assert_eq!(alice.value().hobby, "swim");
}
txn.commit().await.unwrap();
}
{
let txn = db.read().await;
let alice = txn.get(&1).unwrap();
assert_eq!(alice.value().name, "Alice");
assert_eq!(alice.value().age, 20);
assert_eq!(alice.value().hobby, "swim");
let bob = txn.get(&2).unwrap();
assert_eq!(bob.value().name, "Bob");
assert_eq!(bob.value().age, 30);
assert_eq!(bob.value().hobby, "run");
}
}
```
#### License
`async-skipdb` is under the terms of both the MIT license and the
Apache License (Version 2.0).
See [LICENSE-APACHE](LICENSE-APACHE), [LICENSE-MIT](LICENSE-MIT) for details.
Copyright (c) 2024 Al Liu.
[Github-url]: https://github.com/al8n/skipdb/
[CI-url]: https://github.com/al8n/skipdb/actions/workflows/ci.yml
[doc-url]: https://docs.rs/async-skipdb
[crates-url]: https://crates.io/crates/async-skipdb
[codecov-url]: https://app.codecov.io/gh/al8n/skipdb/
[zh-cn-url]: https://github.com/al8n/skipdb/tree/main/README-zh_CN.md