feat: document crud
This commit is contained in:
parent
0fa6536c7a
commit
2bc939479c
|
|
@ -2250,6 +2250,8 @@ dependencies = [
|
|||
name = "revdb"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"bincode",
|
||||
"serde",
|
||||
"sled",
|
||||
"thiserror",
|
||||
]
|
||||
|
|
@ -2400,9 +2402,9 @@ checksum = "58bc9567378fc7690d6b2addae4e60ac2eeea07becb2c64b9f218b53865cba2a"
|
|||
|
||||
[[package]]
|
||||
name = "serde"
|
||||
version = "1.0.154"
|
||||
version = "1.0.156"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8cdd151213925e7f1ab45a9bbfb129316bd00799784b174b7cc7bcd16961c49e"
|
||||
checksum = "314b5b092c0ade17c00142951e50ced110ec27cea304b1037c6969246c2469a4"
|
||||
dependencies = [
|
||||
"serde_derive",
|
||||
]
|
||||
|
|
@ -2420,9 +2422,9 @@ dependencies = [
|
|||
|
||||
[[package]]
|
||||
name = "serde_derive"
|
||||
version = "1.0.154"
|
||||
version = "1.0.156"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4fc80d722935453bcafdc2c9a73cd6fac4dc1938f0346035d84bf99fa9e33217"
|
||||
checksum = "d7e29c4601e36bcec74a223228dce795f4cd3616341a4af93520ca1a837c087d"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
|
|
|
|||
|
|
@ -2,7 +2,6 @@
|
|||
set -x
|
||||
set -eo pipefail
|
||||
|
||||
# if a redis container is running, print instructions to kill it and exit
|
||||
RUNNING_CONTAINER=$(docker ps --filter 'name=redis' --format '{{.ID}}')
|
||||
if [[ -n $RUNNING_CONTAINER ]]; then
|
||||
echo >&2 "there is a redis container already running, kill it with"
|
||||
|
|
|
|||
|
|
@ -8,3 +8,5 @@ edition = "2021"
|
|||
[dependencies]
|
||||
sled = "0.34.7"
|
||||
thiserror = "1.0.30"
|
||||
serde = { version = "1.0", features = ["derive"] }
|
||||
bincode = "1.3.3"
|
||||
|
|
@ -1,5 +1,6 @@
|
|||
use crate::document::Document;
|
||||
use crate::error::RevDBError;
|
||||
use sled::Db;
|
||||
use sled::{Batch, Db, IVec};
|
||||
use std::path::Path;
|
||||
|
||||
pub struct RevDB {
|
||||
|
|
@ -12,24 +13,44 @@ impl RevDB {
|
|||
Ok(Self { db })
|
||||
}
|
||||
|
||||
pub fn insert(&self, uid: i64, rev_id: i64, data: &[u8]) -> Result<(), RevDBError> {
|
||||
let key = make_seq_key(uid, rev_id);
|
||||
let _ = self.db.insert(key, data)?;
|
||||
pub fn document(&self) -> Document {
|
||||
Document { db: self }
|
||||
}
|
||||
|
||||
pub fn get<K: AsRef<[u8]>>(&self, key: K) -> Result<Option<IVec>, RevDBError> {
|
||||
let value = self.db.get(key)?;
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
pub fn batch_get<K: AsRef<[u8]>>(
|
||||
&self,
|
||||
from_key: K,
|
||||
to_key: K,
|
||||
) -> Result<Vec<IVec>, RevDBError> {
|
||||
let iter = self.db.range(from_key..to_key);
|
||||
let mut items = vec![];
|
||||
for item in iter {
|
||||
let (_, value) = item?;
|
||||
items.push(value)
|
||||
}
|
||||
Ok(items)
|
||||
}
|
||||
|
||||
pub fn insert<K: AsRef<[u8]>>(&self, key: K, value: &[u8]) -> Result<(), RevDBError> {
|
||||
let _ = self.db.insert(key, value)?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn get(&self, uid: i64, rev_id: i64) -> Result<Option<Vec<u8>>, RevDBError> {
|
||||
let key = make_seq_key(uid, rev_id);
|
||||
let value = self.db.get(key)?;
|
||||
Ok(value.map(|value| value.to_vec()))
|
||||
pub fn batch_insert<'a, K: AsRef<[u8]>>(
|
||||
&self,
|
||||
items: impl IntoIterator<Item = (K, &'a [u8])>,
|
||||
) -> Result<(), RevDBError> {
|
||||
let mut batch = Batch::default();
|
||||
let items = items.into_iter();
|
||||
items.for_each(|(key, value)| {
|
||||
batch.insert(key.as_ref(), value);
|
||||
});
|
||||
let _ = self.db.apply_batch(batch)?;
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
// Optimize your data layout: Sled's B-Tree implementation works best when the keys are sequential,
|
||||
// so try to organize the data in a way that maximizes sequential access.
|
||||
fn make_seq_key(uid: i64, rev_id: i64) -> [u8; 16] {
|
||||
let mut key = [0; 16];
|
||||
key[0..8].copy_from_slice(&uid.to_be_bytes());
|
||||
key[8..16].copy_from_slice(&rev_id.to_be_bytes());
|
||||
key
|
||||
}
|
||||
|
|
|
|||
|
|
@ -0,0 +1,112 @@
|
|||
use crate::db::RevDB;
|
||||
use crate::error::RevDBError;
|
||||
use crate::range::RevRange;
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
pub struct Document<'a> {
|
||||
pub(crate) db: &'a RevDB,
|
||||
}
|
||||
|
||||
impl<'a> Document<'a> {
|
||||
pub fn insert(
|
||||
&self,
|
||||
uid: i64,
|
||||
document_id: i64,
|
||||
rev_id: i64,
|
||||
data: &[u8],
|
||||
) -> Result<(), RevDBError> {
|
||||
let key = make_document_key(uid, document_id, rev_id);
|
||||
let _ = self.db.insert(key, data)?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn get(
|
||||
&self,
|
||||
uid: i64,
|
||||
document_id: i64,
|
||||
rev_id: i64,
|
||||
) -> Result<Option<Vec<u8>>, RevDBError> {
|
||||
let key = make_document_key(uid, document_id, rev_id);
|
||||
let value = self.db.get(key)?;
|
||||
Ok(value.map(|value| value.to_vec()))
|
||||
}
|
||||
|
||||
pub fn get_with_range(
|
||||
&self,
|
||||
uid: i64,
|
||||
document_id: i64,
|
||||
range: RevRange,
|
||||
) -> Result<Vec<DocumentRevData>, RevDBError> {
|
||||
let from = make_document_key(uid, document_id, range.start);
|
||||
let to = make_document_key(uid, document_id, range.end);
|
||||
self.batch_get(from, to)
|
||||
}
|
||||
|
||||
pub fn get_after(
|
||||
&self,
|
||||
uid: i64,
|
||||
document_id: i64,
|
||||
rev_id: i64,
|
||||
) -> Result<Vec<DocumentRevData>, RevDBError> {
|
||||
let from = make_document_key(uid, document_id, rev_id);
|
||||
let to = make_document_key(uid, document_id, i64::MAX);
|
||||
self.batch_get(from, to)
|
||||
}
|
||||
|
||||
pub fn get_before(
|
||||
&self,
|
||||
uid: i64,
|
||||
document_id: i64,
|
||||
rev_id: i64,
|
||||
) -> Result<Vec<DocumentRevData>, RevDBError> {
|
||||
let from = make_document_key(uid, document_id, 0);
|
||||
let to = make_document_key(uid, document_id, rev_id);
|
||||
self.batch_get(from, to)
|
||||
}
|
||||
|
||||
pub fn batch_get<K: AsRef<[u8]>>(
|
||||
&self,
|
||||
from: K,
|
||||
to: K,
|
||||
) -> Result<Vec<DocumentRevData>, RevDBError> {
|
||||
let items = self.db.batch_get(from, to)?;
|
||||
let mut document_revs = vec![];
|
||||
for item in items {
|
||||
let rev_data = DocumentRevData::from_vec(item.as_ref())?;
|
||||
document_revs.push(rev_data);
|
||||
}
|
||||
Ok(document_revs)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize)]
|
||||
pub struct DocumentRevData {
|
||||
#[serde(rename = "rid")]
|
||||
pub rev_id: i64,
|
||||
|
||||
#[serde(rename = "bid")]
|
||||
pub base_rev_id: i64,
|
||||
|
||||
#[serde(rename = "data")]
|
||||
pub content: String,
|
||||
}
|
||||
|
||||
impl DocumentRevData {
|
||||
pub fn from_vec(data: &[u8]) -> Result<Self, RevDBError> {
|
||||
bincode::deserialize::<Self>(data).map_err(|_e| RevDBError::SerdeError)
|
||||
}
|
||||
|
||||
pub fn to_vec(&self) -> Result<Vec<u8>, RevDBError> {
|
||||
bincode::serialize(self).map_err(|_e| RevDBError::SerdeError)
|
||||
}
|
||||
}
|
||||
|
||||
// Optimize your data layout: Sled's B-Tree implementation works best when the keys are sequential,
|
||||
// so try to organize the data in a way that maximizes sequential access.
|
||||
fn make_document_key(uid: i64, document_id: i64, rev_id: i64) -> [u8; 24] {
|
||||
let mut key = [0; 24];
|
||||
key[0..8].copy_from_slice(&uid.to_be_bytes());
|
||||
key[8..16].copy_from_slice(&document_id.to_be_bytes());
|
||||
key[16..24].copy_from_slice(&rev_id.to_be_bytes());
|
||||
key
|
||||
}
|
||||
|
|
@ -3,6 +3,9 @@ pub enum RevDBError {
|
|||
#[error(transparent)]
|
||||
Db(#[from] sled::Error),
|
||||
|
||||
#[error("Serde error")]
|
||||
SerdeError,
|
||||
|
||||
#[error("invalid data")]
|
||||
InvalidData,
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,2 +1,4 @@
|
|||
mod db;
|
||||
pub mod document;
|
||||
pub mod error;
|
||||
pub mod range;
|
||||
|
|
|
|||
|
|
@ -0,0 +1,48 @@
|
|||
use std::ops::{Range, RangeInclusive, RangeToInclusive};
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct RevRange {
|
||||
pub(crate) start: i64,
|
||||
pub(crate) end: i64,
|
||||
}
|
||||
|
||||
impl RevRange {
|
||||
/// Construct a new `RevRange` representing the range [start..end).
|
||||
/// It is an invariant that `start <= end`.
|
||||
pub fn new(start: i64, end: i64) -> RevRange {
|
||||
debug_assert!(start <= end);
|
||||
RevRange { start, end }
|
||||
}
|
||||
}
|
||||
|
||||
impl From<RangeInclusive<i64>> for RevRange {
|
||||
fn from(src: RangeInclusive<i64>) -> RevRange {
|
||||
RevRange::new(*src.start(), src.end().saturating_add(1))
|
||||
}
|
||||
}
|
||||
|
||||
impl From<RangeToInclusive<i64>> for RevRange {
|
||||
fn from(src: RangeToInclusive<i64>) -> RevRange {
|
||||
RevRange::new(0, src.end.saturating_add(1))
|
||||
}
|
||||
}
|
||||
|
||||
impl From<Range<i64>> for RevRange {
|
||||
fn from(src: Range<i64>) -> RevRange {
|
||||
let Range { start, end } = src;
|
||||
RevRange { start, end }
|
||||
}
|
||||
}
|
||||
|
||||
impl Iterator for RevRange {
|
||||
type Item = i64;
|
||||
|
||||
fn next(&mut self) -> Option<i64> {
|
||||
if self.start > self.end {
|
||||
return None;
|
||||
}
|
||||
let val = self.start;
|
||||
self.start += 1;
|
||||
Some(val)
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,2 @@
|
|||
#[test]
|
||||
fn read_document() {}
|
||||
|
|
@ -0,0 +1 @@
|
|||
mod document;
|
||||
Loading…
Reference in New Issue