forked from Deuxfleurs/garage
Reorganize table API
This commit is contained in:
parent
a450103ed0
commit
4c1aee42d5
4 changed files with 96 additions and 60 deletions
2
TODO
2
TODO
|
@ -5,7 +5,7 @@ Object table
|
||||||
Rename version table to object table
|
Rename version table to object table
|
||||||
In value handle the different versions
|
In value handle the different versions
|
||||||
|
|
||||||
So that the table becomes (bucket, key) -> CRDT(list of versions)
|
So that the table becomes bucket + Sort key = object key -> CRDT(list of versions)
|
||||||
|
|
||||||
CRDT merge rule:
|
CRDT merge rule:
|
||||||
- keep one complete version (the one with the highest timestamp)
|
- keep one complete version (the one with the highest timestamp)
|
||||||
|
|
|
@ -97,11 +97,9 @@ async fn handle_put(garage: Arc<Garage>,
|
||||||
None => return Err(Error::BadRequest(format!("Empty body"))),
|
None => return Err(Error::BadRequest(format!("Empty body"))),
|
||||||
};
|
};
|
||||||
|
|
||||||
let version_key = VersionMetaKey{
|
let mut version = VersionMeta{
|
||||||
bucket: bucket.to_string(),
|
bucket: bucket.into(),
|
||||||
key: key.to_string(),
|
key: key.into(),
|
||||||
};
|
|
||||||
let mut version_value = VersionMetaValue {
|
|
||||||
timestamp: now_msec(),
|
timestamp: now_msec(),
|
||||||
uuid: version_uuid.clone(),
|
uuid: version_uuid.clone(),
|
||||||
mime_type: mime_type.to_string(),
|
mime_type: mime_type.to_string(),
|
||||||
|
@ -111,15 +109,15 @@ async fn handle_put(garage: Arc<Garage>,
|
||||||
};
|
};
|
||||||
|
|
||||||
if first_block.len() < INLINE_THRESHOLD {
|
if first_block.len() < INLINE_THRESHOLD {
|
||||||
version_value.data = VersionData::Inline(first_block);
|
version.data = VersionData::Inline(first_block);
|
||||||
version_value.is_complete = true;
|
version.is_complete = true;
|
||||||
garage.version_table.insert(&version_key, &version_value).await?;
|
garage.version_table.insert(&version).await?;
|
||||||
return Ok(version_uuid)
|
return Ok(version_uuid)
|
||||||
}
|
}
|
||||||
|
|
||||||
let first_block_hash = hash(&first_block[..]);
|
let first_block_hash = hash(&first_block[..]);
|
||||||
version_value.data = VersionData::FirstBlock(first_block_hash);
|
version.data = VersionData::FirstBlock(first_block_hash);
|
||||||
garage.version_table.insert(&version_key, &version_value).await?;
|
garage.version_table.insert(&version).await?;
|
||||||
|
|
||||||
let block_meta = BlockMeta{
|
let block_meta = BlockMeta{
|
||||||
version_uuid: version_uuid.clone(),
|
version_uuid: version_uuid.clone(),
|
||||||
|
@ -145,8 +143,8 @@ async fn handle_put(garage: Arc<Garage>,
|
||||||
|
|
||||||
// TODO: if at any step we have an error, we should undo everything we did
|
// TODO: if at any step we have an error, we should undo everything we did
|
||||||
|
|
||||||
version_value.is_complete = true;
|
version.is_complete = true;
|
||||||
garage.version_table.insert(&version_key, &version_value).await?;
|
garage.version_table.insert(&version).await?;
|
||||||
Ok(version_uuid)
|
Ok(version_uuid)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
100
src/table.rs
100
src/table.rs
|
@ -52,10 +52,10 @@ impl<F: TableFormat + 'static> TableRpcHandler for TableRpcHandlerAdapter<F> {
|
||||||
pub enum TableRPC<F: TableFormat> {
|
pub enum TableRPC<F: TableFormat> {
|
||||||
Ok,
|
Ok,
|
||||||
|
|
||||||
ReadEntry(F::K, Vec<u8>),
|
ReadEntry(F::P, F::S),
|
||||||
ReadEntryResponse(Option<F::V>),
|
ReadEntryResponse(Option<F::E>),
|
||||||
|
|
||||||
Update(Vec<(F::K, F::V)>),
|
Update(Vec<F::E>),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Partition {
|
pub struct Partition {
|
||||||
|
@ -64,21 +64,59 @@ pub struct Partition {
|
||||||
pub other_nodes: Vec<UUID>,
|
pub other_nodes: Vec<UUID>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait TableKey {
|
pub trait PartitionKey: Clone + PartialEq + Serialize + for<'de> Deserialize<'de> + Send + Sync {
|
||||||
fn hash(&self) -> Hash;
|
fn hash(&self) -> Hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait TableValue {
|
pub trait SortKey: Clone + Serialize + for<'de> Deserialize<'de> + Send + Sync {
|
||||||
fn sort_key(&self) -> Vec<u8>;
|
fn sort_key(&self) -> &[u8];
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait Entry<P: PartitionKey, S: SortKey>: Clone + Serialize + for<'de> Deserialize<'de> + Send + Sync {
|
||||||
|
fn partition_key(&self) -> &P;
|
||||||
|
fn sort_key(&self) -> &S;
|
||||||
|
|
||||||
fn merge(&mut self, other: &Self);
|
fn merge(&mut self, other: &Self);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Serialize, Deserialize)]
|
||||||
|
pub struct EmptySortKey;
|
||||||
|
impl SortKey for EmptySortKey {
|
||||||
|
fn sort_key(&self) -> &[u8] {
|
||||||
|
&[]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
pub struct StringKey(String);
|
||||||
|
impl PartitionKey for StringKey {
|
||||||
|
fn hash(&self) -> Hash {
|
||||||
|
hash(self.0.as_bytes())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl SortKey for StringKey {
|
||||||
|
fn sort_key(&self) -> &[u8] {
|
||||||
|
self.0.as_bytes()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl AsRef<str> for StringKey {
|
||||||
|
fn as_ref(&self) -> &str {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl From<&str> for StringKey {
|
||||||
|
fn from(s: &str) -> StringKey {
|
||||||
|
StringKey(s.to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait TableFormat: Send + Sync {
|
pub trait TableFormat: Send + Sync {
|
||||||
type K: Clone + PartialEq + Serialize + for<'de> Deserialize<'de> + TableKey + Send + Sync;
|
type P: PartitionKey;
|
||||||
type V: Clone + Serialize + for<'de> Deserialize<'de> + TableValue + Send + Sync;
|
type S: SortKey;
|
||||||
|
type E: Entry<Self::P, Self::S>;
|
||||||
|
|
||||||
async fn updated(&self, key: &Self::K, old: Option<&Self::V>, new: &Self::V);
|
async fn updated(&self, old: Option<&Self::E>, new: &Self::E);
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F: TableFormat + 'static> Table<F> {
|
impl<F: TableFormat + 'static> Table<F> {
|
||||||
|
@ -99,12 +137,12 @@ impl<F: TableFormat + 'static> Table<F> {
|
||||||
Box::new(TableRpcHandlerAdapter::<F>{ table: self })
|
Box::new(TableRpcHandlerAdapter::<F>{ table: self })
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn insert(&self, k: &F::K, v: &F::V) -> Result<(), Error> {
|
pub async fn insert(&self, e: &F::E) -> Result<(), Error> {
|
||||||
let hash = k.hash();
|
let hash = e.partition_key().hash();
|
||||||
let who = self.system.members.read().await
|
let who = self.system.members.read().await
|
||||||
.walk_ring(&hash, self.param.replication_factor);
|
.walk_ring(&hash, self.param.replication_factor);
|
||||||
|
|
||||||
let rpc = &TableRPC::<F>::Update(vec![(k.clone(), v.clone())]);
|
let rpc = &TableRPC::<F>::Update(vec![e.clone()]);
|
||||||
|
|
||||||
self.rpc_try_call_many(&who[..],
|
self.rpc_try_call_many(&who[..],
|
||||||
&rpc,
|
&rpc,
|
||||||
|
@ -112,12 +150,12 @@ impl<F: TableFormat + 'static> Table<F> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn get(&self, k: &F::K, sort_key: &[u8]) -> Result<Option<F::V>, Error> {
|
pub async fn get(&self, partition_key: &F::P, sort_key: &F::S) -> Result<Option<F::E>, Error> {
|
||||||
let hash = k.hash();
|
let hash = partition_key.hash();
|
||||||
let who = self.system.members.read().await
|
let who = self.system.members.read().await
|
||||||
.walk_ring(&hash, self.param.replication_factor);
|
.walk_ring(&hash, self.param.replication_factor);
|
||||||
|
|
||||||
let rpc = &TableRPC::<F>::ReadEntry(k.clone(), sort_key.to_vec());
|
let rpc = &TableRPC::<F>::ReadEntry(partition_key.clone(), sort_key.clone());
|
||||||
let resps = self.rpc_try_call_many(&who[..],
|
let resps = self.rpc_try_call_many(&who[..],
|
||||||
&rpc,
|
&rpc,
|
||||||
self.param.read_quorum)
|
self.param.read_quorum)
|
||||||
|
@ -179,36 +217,40 @@ impl<F: TableFormat + 'static> Table<F> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_read_entry(&self, key: &F::K, sort_key: &[u8]) -> Result<Option<F::V>, Error> {
|
fn handle_read_entry(&self, p: &F::P, s: &F::S) -> Result<Option<F::E>, Error> {
|
||||||
let mut tree_key = key.hash().to_vec();
|
let tree_key = self.tree_key(p, s);
|
||||||
tree_key.extend(sort_key);
|
|
||||||
if let Some(bytes) = self.store.get(&tree_key)? {
|
if let Some(bytes) = self.store.get(&tree_key)? {
|
||||||
let (_, v) = rmp_serde::decode::from_read_ref::<_, (F::K, F::V)>(&bytes)?;
|
let e = rmp_serde::decode::from_read_ref::<_, F::E>(&bytes)?;
|
||||||
Ok(Some(v))
|
Ok(Some(e))
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_update(&self, mut pairs: Vec<(F::K, F::V)>) -> Result<(), Error> {
|
async fn handle_update(&self, mut entries: Vec<F::E>) -> Result<(), Error> {
|
||||||
for mut pair in pairs.drain(..) {
|
for mut entry in entries.drain(..) {
|
||||||
let mut tree_key = pair.0.hash().to_vec();
|
let tree_key = self.tree_key(entry.partition_key(), entry.sort_key());
|
||||||
tree_key.extend(pair.1.sort_key());
|
|
||||||
|
|
||||||
let old_val = match self.store.get(&tree_key)? {
|
let old_val = match self.store.get(&tree_key)? {
|
||||||
Some(prev_bytes) => {
|
Some(prev_bytes) => {
|
||||||
let (_, old_val) = rmp_serde::decode::from_read_ref::<_, (F::K, F::V)>(&prev_bytes)?;
|
let old_entry = rmp_serde::decode::from_read_ref::<_, F::E>(&prev_bytes)?;
|
||||||
pair.1.merge(&old_val);
|
entry.merge(&old_entry);
|
||||||
Some(old_val)
|
Some(old_entry)
|
||||||
}
|
}
|
||||||
None => None
|
None => None
|
||||||
};
|
};
|
||||||
|
|
||||||
let new_bytes = rmp_serde::encode::to_vec_named(&pair)?;
|
let new_bytes = rmp_serde::encode::to_vec_named(&entry)?;
|
||||||
self.store.insert(&tree_key, new_bytes)?;
|
self.store.insert(&tree_key, new_bytes)?;
|
||||||
|
|
||||||
self.instance.updated(&pair.0, old_val.as_ref(), &pair.1).await;
|
self.instance.updated(old_val.as_ref(), &entry).await;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn tree_key(&self, p: &F::P, s: &F::S) -> Vec<u8> {
|
||||||
|
let mut ret = p.hash().to_vec();
|
||||||
|
ret.extend(s.sort_key());
|
||||||
|
ret
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,14 +8,11 @@ use crate::table::*;
|
||||||
use crate::server::Garage;
|
use crate::server::Garage;
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
pub struct VersionMetaKey {
|
|
||||||
pub bucket: String,
|
|
||||||
pub key: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
pub struct VersionMetaValue {
|
pub struct VersionMeta {
|
||||||
|
pub bucket: StringKey,
|
||||||
|
pub key: StringKey,
|
||||||
|
|
||||||
pub timestamp: u64,
|
pub timestamp: u64,
|
||||||
pub uuid: UUID,
|
pub uuid: UUID,
|
||||||
|
|
||||||
|
@ -37,16 +34,14 @@ pub struct VersionTable {
|
||||||
pub garage: RwLock<Option<Arc<Garage>>>,
|
pub garage: RwLock<Option<Arc<Garage>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TableKey for VersionMetaKey {
|
impl Entry<StringKey, StringKey> for VersionMeta {
|
||||||
fn hash(&self) -> Hash {
|
fn partition_key(&self) -> &StringKey {
|
||||||
hash(self.bucket.as_bytes())
|
&self.bucket
|
||||||
}
|
}
|
||||||
|
fn sort_key(&self) -> &StringKey {
|
||||||
|
&self.key
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TableValue for VersionMetaValue {
|
|
||||||
fn sort_key(&self) -> Vec<u8> {
|
|
||||||
vec![]
|
|
||||||
}
|
|
||||||
fn merge(&mut self, other: &Self) {
|
fn merge(&mut self, other: &Self) {
|
||||||
unimplemented!()
|
unimplemented!()
|
||||||
}
|
}
|
||||||
|
@ -54,10 +49,11 @@ impl TableValue for VersionMetaValue {
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl TableFormat for VersionTable {
|
impl TableFormat for VersionTable {
|
||||||
type K = VersionMetaKey;
|
type P = StringKey;
|
||||||
type V = VersionMetaValue;
|
type S = StringKey;
|
||||||
|
type E = VersionMeta;
|
||||||
|
|
||||||
async fn updated(&self, key: &Self::K, old: Option<&Self::V>, new: &Self::V) {
|
async fn updated(&self, old: Option<&Self::E>, new: &Self::E) {
|
||||||
unimplemented!()
|
unimplemented!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue