@ -0,0 +1,153 @@ |
|||||
|
use super::{
|
||||
|
BTreeMap, KvMap, MerkleError, MerkleStore, NodeIndex, RpoDigest, StoreNode, Vec, Word,
|
||||
|
};
|
||||
|
use crate::utils::collections::Diff;
|
||||
|
|
||||
|
#[cfg(test)]
|
||||
|
use super::{empty_roots::EMPTY_WORD, Felt, SimpleSmt};
|
||||
|
|
||||
|
// MERKLE STORE DELTA
|
||||
|
// ================================================================================================
|
||||
|
|
||||
|
/// [MerkleStoreDelta] stores a vector of ([RpoDigest], [MerkleTreeDelta]) tuples where the
|
||||
|
/// [RpoDigest] represents the root of the Merkle tree and [MerkleTreeDelta] represents the
|
||||
|
/// differences between the initial and final Merkle tree states.
|
||||
|
#[derive(Default, Debug, Clone, PartialEq, Eq)]
|
||||
|
pub struct MerkleStoreDelta(pub Vec<(RpoDigest, MerkleTreeDelta)>);
|
||||
|
|
||||
|
// MERKLE TREE DELTA
|
||||
|
// ================================================================================================
|
||||
|
|
||||
|
/// [MerkleDelta] stores the differences between the initial and final Merkle tree states.
|
||||
|
///
|
||||
|
/// The differences are represented as follows:
|
||||
|
/// - depth: the depth of the merkle tree.
|
||||
|
/// - cleared_slots: indexes of slots where values were set to [ZERO; 4].
|
||||
|
/// - updated_slots: index-value pairs of slots where values were set to non [ZERO; 4] values.
|
||||
|
#[cfg(not(test))]
|
||||
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||
|
pub struct MerkleTreeDelta {
|
||||
|
depth: u8,
|
||||
|
cleared_slots: Vec<u64>,
|
||||
|
updated_slots: Vec<(u64, Word)>,
|
||||
|
}
|
||||
|
|
||||
|
impl MerkleTreeDelta {
|
||||
|
// CONSTRUCTOR
|
||||
|
// --------------------------------------------------------------------------------------------
|
||||
|
pub fn new(depth: u8) -> Self {
|
||||
|
Self {
|
||||
|
depth,
|
||||
|
cleared_slots: Vec::new(),
|
||||
|
updated_slots: Vec::new(),
|
||||
|
}
|
||||
|
}
|
||||
|
|
||||
|
// ACCESSORS
|
||||
|
// --------------------------------------------------------------------------------------------
|
||||
|
/// Returns the depth of the Merkle tree the [MerkleDelta] is associated with.
|
||||
|
pub fn depth(&self) -> u8 {
|
||||
|
self.depth
|
||||
|
}
|
||||
|
|
||||
|
/// Returns the indexes of slots where values were set to [ZERO; 4].
|
||||
|
pub fn cleared_slots(&self) -> &[u64] {
|
||||
|
&self.cleared_slots
|
||||
|
}
|
||||
|
|
||||
|
/// Returns the index-value pairs of slots where values were set to non [ZERO; 4] values.
|
||||
|
pub fn updated_slots(&self) -> &[(u64, Word)] {
|
||||
|
&self.updated_slots
|
||||
|
}
|
||||
|
|
||||
|
// MODIFIERS
|
||||
|
// --------------------------------------------------------------------------------------------
|
||||
|
/// Adds a slot index to the list of cleared slots.
|
||||
|
pub fn add_cleared_slot(&mut self, index: u64) {
|
||||
|
self.cleared_slots.push(index);
|
||||
|
}
|
||||
|
|
||||
|
/// Adds a slot index and a value to the list of updated slots.
|
||||
|
pub fn add_updated_slot(&mut self, index: u64, value: Word) {
|
||||
|
self.updated_slots.push((index, value));
|
||||
|
}
|
||||
|
}
|
||||
|
|
||||
|
/// Extracts a [MerkleDelta] object by comparing the leaves of two Merkle trees specifies by
|
||||
|
/// their roots and depth.
|
||||
|
pub fn merkle_tree_delta<T: KvMap<RpoDigest, StoreNode>>(
|
||||
|
tree_root_1: RpoDigest,
|
||||
|
tree_root_2: RpoDigest,
|
||||
|
depth: u8,
|
||||
|
merkle_store: &MerkleStore<T>,
|
||||
|
) -> Result<MerkleTreeDelta, MerkleError> {
|
||||
|
if tree_root_1 == tree_root_2 {
|
||||
|
return Ok(MerkleTreeDelta::new(depth));
|
||||
|
}
|
||||
|
|
||||
|
let tree_1_leaves: BTreeMap<NodeIndex, RpoDigest> =
|
||||
|
merkle_store.non_empty_leaves(tree_root_1, depth).collect();
|
||||
|
let tree_2_leaves: BTreeMap<NodeIndex, RpoDigest> =
|
||||
|
merkle_store.non_empty_leaves(tree_root_2, depth).collect();
|
||||
|
let diff = tree_1_leaves.diff(&tree_2_leaves);
|
||||
|
|
||||
|
// TODO: Refactor this diff implementation to prevent allocation of both BTree and Vec.
|
||||
|
Ok(MerkleTreeDelta {
|
||||
|
depth,
|
||||
|
cleared_slots: diff.removed.into_iter().map(|index| index.value()).collect(),
|
||||
|
updated_slots: diff
|
||||
|
.updated
|
||||
|
.into_iter()
|
||||
|
.map(|(index, leaf)| (index.value(), *leaf))
|
||||
|
.collect(),
|
||||
|
})
|
||||
|
}
|
||||
|
|
||||
|
// INTERNALS
|
||||
|
// --------------------------------------------------------------------------------------------
|
||||
|
#[cfg(test)]
|
||||
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||
|
pub struct MerkleTreeDelta {
|
||||
|
pub depth: u8,
|
||||
|
pub cleared_slots: Vec<u64>,
|
||||
|
pub updated_slots: Vec<(u64, Word)>,
|
||||
|
}
|
||||
|
|
||||
|
// MERKLE DELTA
|
||||
|
// ================================================================================================
|
||||
|
#[test]
|
||||
|
fn test_compute_merkle_delta() {
|
||||
|
let entries = vec![
|
||||
|
(10, [Felt::new(0), Felt::new(1), Felt::new(2), Felt::new(3)]),
|
||||
|
(15, [Felt::new(4), Felt::new(5), Felt::new(6), Felt::new(7)]),
|
||||
|
(20, [Felt::new(8), Felt::new(9), Felt::new(10), Felt::new(11)]),
|
||||
|
(31, [Felt::new(12), Felt::new(13), Felt::new(14), Felt::new(15)]),
|
||||
|
];
|
||||
|
let simple_smt = SimpleSmt::with_leaves(30, entries.clone()).unwrap();
|
||||
|
let mut store: MerkleStore = (&simple_smt).into();
|
||||
|
let root = simple_smt.root();
|
||||
|
|
||||
|
// add a new node
|
||||
|
let new_value = [Felt::new(16), Felt::new(17), Felt::new(18), Felt::new(19)];
|
||||
|
let new_index = NodeIndex::new(simple_smt.depth(), 32).unwrap();
|
||||
|
let root = store.set_node(root, new_index, new_value.into()).unwrap().root;
|
||||
|
|
||||
|
// update an existing node
|
||||
|
let update_value = [Felt::new(20), Felt::new(21), Felt::new(22), Felt::new(23)];
|
||||
|
let update_idx = NodeIndex::new(simple_smt.depth(), entries[0].0).unwrap();
|
||||
|
let root = store.set_node(root, update_idx, update_value.into()).unwrap().root;
|
||||
|
|
||||
|
// remove a node
|
||||
|
let remove_idx = NodeIndex::new(simple_smt.depth(), entries[1].0).unwrap();
|
||||
|
let root = store.set_node(root, remove_idx, EMPTY_WORD.into()).unwrap().root;
|
||||
|
|
||||
|
let merkle_delta =
|
||||
|
merkle_tree_delta(simple_smt.root(), root, simple_smt.depth(), &store).unwrap();
|
||||
|
let expected_merkle_delta = MerkleTreeDelta {
|
||||
|
depth: simple_smt.depth(),
|
||||
|
cleared_slots: vec![remove_idx.value()],
|
||||
|
updated_slots: vec![(update_idx.value(), update_value), (new_index.value(), new_value)],
|
||||
|
};
|
||||
|
|
||||
|
assert_eq!(merkle_delta, expected_merkle_delta);
|
||||
|
}
|
@ -1,16 +1,31 @@ |
|||||
/// A trait for computing the difference between two objects.
|
/// A trait for computing the difference between two objects.
|
||||
pub trait Diff<K: Ord + Clone, V: Clone> {
|
pub trait Diff<K: Ord + Clone, V: Clone> {
|
||||
|
/// The type that describes the difference between two objects.
|
||||
type DiffType;
|
type DiffType;
|
||||
|
|
||||
/// Returns a `Self::DiffType` object that represents the difference between this object and
|
|
||||
|
/// Returns a [Self::DiffType] object that represents the difference between this object and
|
||||
/// other.
|
/// other.
|
||||
fn diff(&self, other: &Self) -> Self::DiffType;
|
fn diff(&self, other: &Self) -> Self::DiffType;
|
||||
}
|
}
|
||||
|
|
||||
/// A trait for applying the difference between two objects.
|
/// A trait for applying the difference between two objects.
|
||||
pub trait ApplyDiff<K: Ord + Clone, V: Clone> {
|
pub trait ApplyDiff<K: Ord + Clone, V: Clone> {
|
||||
|
/// The type that describes the difference between two objects.
|
||||
type DiffType;
|
type DiffType;
|
||||
|
|
||||
/// Applies the provided changes described by [DiffType] to the object implementing this trait.
|
|
||||
|
/// Applies the provided changes described by [Self::DiffType] to the object implementing this trait.
|
||||
fn apply(&mut self, diff: Self::DiffType);
|
fn apply(&mut self, diff: Self::DiffType);
|
||||
}
|
}
|
||||
|
|
||||
|
/// A trait for applying the difference between two objects with the possibility of failure.
|
||||
|
pub trait TryApplyDiff<K: Ord + Clone, V: Clone> {
|
||||
|
/// The type that describes the difference between two objects.
|
||||
|
type DiffType;
|
||||
|
|
||||
|
/// An error type that can be returned if the changes cannot be applied.
|
||||
|
type Error;
|
||||
|
|
||||
|
/// Applies the provided changes described by [Self::DiffType] to the object implementing this trait.
|
||||
|
/// Returns an error if the changes cannot be applied.
|
||||
|
fn try_apply(&mut self, diff: Self::DiffType) -> Result<(), Self::Error>;
|
||||
|
}
|