2023-04-14 15:41:01 +00:00
|
|
|
use core::fmt::Debug;
|
|
|
|
use std::{
|
|
|
|
sync::{Arc, RwLock},
|
|
|
|
collections::{HashSet, HashMap},
|
|
|
|
};
|
|
|
|
|
|
|
|
/// An object implementing get.
|
|
|
|
pub trait Get: Send + Sync + Debug {
|
|
|
|
fn get(&self, key: impl AsRef<[u8]>) -> Option<Vec<u8>>;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An atomic database operation.
|
2023-04-14 19:04:26 +00:00
|
|
|
#[must_use]
|
2023-04-14 15:41:01 +00:00
|
|
|
pub trait DbTxn: Send + Sync + Debug + Get {
|
|
|
|
fn put(&mut self, key: impl AsRef<[u8]>, value: impl AsRef<[u8]>);
|
|
|
|
fn del(&mut self, key: impl AsRef<[u8]>);
|
|
|
|
fn commit(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A database supporting atomic operations.
|
|
|
|
pub trait Db: 'static + Send + Sync + Clone + Debug + Get {
|
|
|
|
type Transaction<'a>: DbTxn;
|
|
|
|
fn key(db_dst: &'static [u8], item_dst: &'static [u8], key: impl AsRef<[u8]>) -> Vec<u8> {
|
|
|
|
let db_len = u8::try_from(db_dst.len()).unwrap();
|
|
|
|
let dst_len = u8::try_from(item_dst.len()).unwrap();
|
2023-07-08 15:29:05 +00:00
|
|
|
[[db_len].as_ref(), db_dst, [dst_len].as_ref(), item_dst, key.as_ref()].concat()
|
2023-04-14 15:41:01 +00:00
|
|
|
}
|
|
|
|
fn txn(&mut self) -> Self::Transaction<'_>;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An atomic operation for the in-memory databae.
|
2023-04-14 19:04:26 +00:00
|
|
|
#[must_use]
|
2023-04-14 19:51:43 +00:00
|
|
|
#[derive(PartialEq, Eq, Debug)]
|
2023-04-14 15:41:01 +00:00
|
|
|
pub struct MemDbTxn<'a>(&'a MemDb, HashMap<Vec<u8>, Vec<u8>>, HashSet<Vec<u8>>);
|
|
|
|
|
|
|
|
impl<'a> Get for MemDbTxn<'a> {
|
|
|
|
fn get(&self, key: impl AsRef<[u8]>) -> Option<Vec<u8>> {
|
|
|
|
if self.2.contains(key.as_ref()) {
|
|
|
|
return None;
|
|
|
|
}
|
2023-07-08 15:29:05 +00:00
|
|
|
self
|
|
|
|
.1
|
|
|
|
.get(key.as_ref())
|
|
|
|
.cloned()
|
|
|
|
.or_else(|| self.0 .0.read().unwrap().get(key.as_ref()).cloned())
|
2023-04-14 15:41:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
impl<'a> DbTxn for MemDbTxn<'a> {
|
|
|
|
fn put(&mut self, key: impl AsRef<[u8]>, value: impl AsRef<[u8]>) {
|
|
|
|
self.2.remove(key.as_ref());
|
|
|
|
self.1.insert(key.as_ref().to_vec(), value.as_ref().to_vec());
|
|
|
|
}
|
|
|
|
fn del(&mut self, key: impl AsRef<[u8]>) {
|
|
|
|
self.1.remove(key.as_ref());
|
|
|
|
self.2.insert(key.as_ref().to_vec());
|
|
|
|
}
|
|
|
|
fn commit(mut self) {
|
|
|
|
let mut db = self.0 .0.write().unwrap();
|
|
|
|
for (key, value) in self.1.drain() {
|
|
|
|
db.insert(key, value);
|
|
|
|
}
|
|
|
|
for key in self.2 {
|
|
|
|
db.remove(&key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An in-memory database.
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct MemDb(Arc<RwLock<HashMap<Vec<u8>, Vec<u8>>>>);
|
|
|
|
|
2023-04-14 19:51:43 +00:00
|
|
|
impl PartialEq for MemDb {
|
|
|
|
fn eq(&self, other: &MemDb) -> bool {
|
|
|
|
*self.0.read().unwrap() == *other.0.read().unwrap()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Eq for MemDb {}
|
|
|
|
|
2023-04-14 15:41:01 +00:00
|
|
|
impl Default for MemDb {
|
|
|
|
fn default() -> MemDb {
|
|
|
|
MemDb(Arc::new(RwLock::new(HashMap::new())))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MemDb {
|
|
|
|
/// Create a new in-memory database.
|
|
|
|
pub fn new() -> MemDb {
|
|
|
|
MemDb::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Get for MemDb {
|
|
|
|
fn get(&self, key: impl AsRef<[u8]>) -> Option<Vec<u8>> {
|
|
|
|
self.0.read().unwrap().get(key.as_ref()).cloned()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Db for MemDb {
|
|
|
|
type Transaction<'a> = MemDbTxn<'a>;
|
|
|
|
fn txn(&mut self) -> MemDbTxn<'_> {
|
|
|
|
MemDbTxn(self, HashMap::new(), HashSet::new())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Also bind RocksDB
|