use alloc::vec::Vec;
use core::borrow::Borrow;
use super::{
EmptySubtreeRoots, InnerNodeInfo, MerkleError, MerklePath, MerkleProof, MerkleTree, NodeIndex,
PartialMerkleTree, Poseidon2, RootPath, Word,
mmr::Mmr,
smt::{SimpleSmt, Smt},
};
use crate::{
Map,
utils::{ByteReader, ByteWriter, Deserializable, DeserializationError, Serializable},
};
#[cfg(test)]
mod tests;
#[derive(Debug, Default, Copy, Clone, Eq, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
pub struct StoreNode {
left: Word,
right: Word,
}
#[derive(Debug, Clone, Eq, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
pub struct MerkleStore {
nodes: Map<Word, StoreNode>,
}
impl Default for MerkleStore {
fn default() -> Self {
Self::new()
}
}
impl MerkleStore {
pub fn new() -> MerkleStore {
let nodes = empty_hashes().collect();
MerkleStore { nodes }
}
pub fn num_internal_nodes(&self) -> usize {
self.nodes.len()
}
pub fn get_node(&self, root: Word, index: NodeIndex) -> Result<Word, MerkleError> {
let mut hash = root;
self.nodes.get(&hash).ok_or(MerkleError::RootNotInStore(hash))?;
for i in (0..index.depth()).rev() {
let node = self
.nodes
.get(&hash)
.ok_or(MerkleError::NodeIndexNotFoundInStore(hash, index))?;
let is_right = index.is_nth_bit_odd(i);
hash = if is_right { node.right } else { node.left };
}
Ok(hash)
}
pub fn get_path(&self, root: Word, index: NodeIndex) -> Result<MerkleProof, MerkleError> {
let mut hash = root;
let mut path = Vec::with_capacity(index.depth().into());
self.nodes.get(&hash).ok_or(MerkleError::RootNotInStore(hash))?;
for i in (0..index.depth()).rev() {
let node = self
.nodes
.get(&hash)
.ok_or(MerkleError::NodeIndexNotFoundInStore(hash, index))?;
let is_right = index.is_nth_bit_odd(i);
hash = if is_right {
path.push(node.left);
node.right
} else {
path.push(node.right);
node.left
}
}
path.reverse();
Ok(MerkleProof::new(hash, MerklePath::new(path)))
}
pub fn has_path(&self, root: Word, index: NodeIndex) -> bool {
if !self.nodes.contains_key(&root) {
return false;
}
let mut hash = root;
for i in (0..index.depth()).rev() {
let node = match self.nodes.get(&hash) {
Some(node) => node,
None => return false,
};
let is_right = index.is_nth_bit_odd(i);
hash = if is_right { node.right } else { node.left };
}
true
}
pub fn get_leaf_depth(
&self,
root: Word,
tree_depth: u8,
index: u64,
) -> Result<u8, MerkleError> {
if tree_depth > 64 {
return Err(MerkleError::DepthTooBig(tree_depth as u64));
}
NodeIndex::new(tree_depth, index)?;
let empty = EmptySubtreeRoots::empty_hashes(tree_depth);
let mut hash = root;
if !self.nodes.contains_key(&hash) {
return Err(MerkleError::RootNotInStore(hash));
}
let mut path = (index << (64 - tree_depth)).reverse_bits();
for depth in 0..=tree_depth {
if hash == empty[depth as usize] {
return Ok(depth);
}
let children = match self.nodes.get(&hash) {
Some(node) => node,
None => return Ok(depth),
};
hash = if path & 1 == 0 { children.left } else { children.right };
path >>= 1;
}
Err(MerkleError::DepthTooBig(tree_depth as u64 + 1))
}
pub fn find_lone_leaf(
&self,
root: Word,
root_index: NodeIndex,
tree_depth: u8,
) -> Result<Option<(NodeIndex, Word)>, MerkleError> {
const MAX_DEPTH: u8 = u64::BITS as u8;
if tree_depth > MAX_DEPTH {
return Err(MerkleError::DepthTooBig(tree_depth as u64));
}
let empty = EmptySubtreeRoots::empty_hashes(MAX_DEPTH);
let mut node = root;
if !self.nodes.contains_key(&node) {
return Err(MerkleError::RootNotInStore(node));
}
let mut index = root_index;
if index.depth() > tree_depth {
return Err(MerkleError::DepthTooBig(index.depth() as u64));
}
for depth in index.depth()..tree_depth {
let children = match self.nodes.get(&node) {
Some(node) => node,
None => return Ok(Some((index, node))),
};
let empty_node = empty[depth as usize + 1];
node = if children.left != empty_node && children.right == empty_node {
index = index.left_child();
children.left
} else if children.left == empty_node && children.right != empty_node {
index = index.right_child();
children.right
} else {
return Ok(None);
};
}
if self.nodes.contains_key(&node) {
Err(MerkleError::DepthTooBig(tree_depth as u64 + 1))
} else {
Ok(Some((index, node)))
}
}
pub fn subset<I, R>(&self, roots: I) -> MerkleStore
where
I: Iterator<Item = R>,
R: Borrow<Word>,
{
let mut store = MerkleStore::new();
for root in roots {
let root = *root.borrow();
store.clone_tree_from(root, self);
}
store
}
pub fn inner_nodes(&self) -> impl Iterator<Item = InnerNodeInfo> + '_ {
self.nodes
.iter()
.map(|(r, n)| InnerNodeInfo { value: *r, left: n.left, right: n.right })
}
pub fn non_empty_leaves(
&self,
root: Word,
max_depth: u8,
) -> impl Iterator<Item = (NodeIndex, Word)> + '_ {
let empty_roots = EmptySubtreeRoots::empty_hashes(max_depth);
let mut stack = Vec::new();
stack.push((NodeIndex::new_unchecked(0, 0), root));
core::iter::from_fn(move || {
while let Some((index, node_hash)) = stack.pop() {
if index.depth() == max_depth {
return Some((index, node_hash));
}
if let Some(node) = self.nodes.get(&node_hash) {
if !empty_roots.contains(&node.left) {
stack.push((index.left_child(), node.left));
}
if !empty_roots.contains(&node.right) {
stack.push((index.right_child(), node.right));
}
} else {
return Some((index, node_hash));
}
}
None
})
}
pub fn add_merkle_path(
&mut self,
index: u64,
node: Word,
path: MerklePath,
) -> Result<Word, MerkleError> {
let root = path.authenticated_nodes(index, node)?.fold(Word::default(), |_, node| {
let value: Word = node.value;
let left: Word = node.left;
let right: Word = node.right;
debug_assert_eq!(Poseidon2::merge(&[left, right]), value);
self.nodes.insert(value, StoreNode { left, right });
node.value
});
Ok(root)
}
pub fn add_merkle_paths<I>(&mut self, paths: I) -> Result<(), MerkleError>
where
I: IntoIterator<Item = (u64, Word, MerklePath)>,
{
for (index_value, node, path) in paths.into_iter() {
self.add_merkle_path(index_value, node, path)?;
}
Ok(())
}
pub fn set_node(
&mut self,
mut root: Word,
index: NodeIndex,
value: Word,
) -> Result<RootPath, MerkleError> {
let node = value;
let MerkleProof { value, path } = self.get_path(root, index)?;
if node != value {
root = self.add_merkle_path(index.position(), node, path.clone())?;
}
Ok(RootPath { root, path })
}
pub fn merge_roots(&mut self, left_root: Word, right_root: Word) -> Result<Word, MerkleError> {
let parent = Poseidon2::merge(&[left_root, right_root]);
self.nodes.insert(parent, StoreNode { left: left_root, right: right_root });
Ok(parent)
}
pub fn into_inner(self) -> Map<Word, StoreNode> {
self.nodes
}
fn clone_tree_from(&mut self, root: Word, source: &Self) {
if let Some(node) = source.nodes.get(&root) {
if self.nodes.insert(root, *node).is_none() {
self.clone_tree_from(node.left, source);
self.clone_tree_from(node.right, source);
}
}
}
}
impl From<&MerkleTree> for MerkleStore {
fn from(value: &MerkleTree) -> Self {
let nodes = combine_nodes_with_empty_hashes(value.inner_nodes()).collect();
Self { nodes }
}
}
impl<const DEPTH: u8> From<&SimpleSmt<DEPTH>> for MerkleStore {
fn from(value: &SimpleSmt<DEPTH>) -> Self {
let nodes = combine_nodes_with_empty_hashes(value.inner_nodes()).collect();
Self { nodes }
}
}
impl From<&Smt> for MerkleStore {
fn from(value: &Smt) -> Self {
let nodes = combine_nodes_with_empty_hashes(value.inner_nodes()).collect();
Self { nodes }
}
}
impl From<&Mmr> for MerkleStore {
fn from(value: &Mmr) -> Self {
let nodes = combine_nodes_with_empty_hashes(value.inner_nodes()).collect();
Self { nodes }
}
}
impl From<&PartialMerkleTree> for MerkleStore {
fn from(value: &PartialMerkleTree) -> Self {
let nodes = combine_nodes_with_empty_hashes(value.inner_nodes()).collect();
Self { nodes }
}
}
impl FromIterator<InnerNodeInfo> for MerkleStore {
fn from_iter<I: IntoIterator<Item = InnerNodeInfo>>(iter: I) -> Self {
let nodes = combine_nodes_with_empty_hashes(iter).collect();
Self { nodes }
}
}
impl FromIterator<(Word, StoreNode)> for MerkleStore {
fn from_iter<I: IntoIterator<Item = (Word, StoreNode)>>(iter: I) -> Self {
let nodes = iter.into_iter().chain(empty_hashes()).collect();
Self { nodes }
}
}
impl Extend<InnerNodeInfo> for MerkleStore {
fn extend<I: IntoIterator<Item = InnerNodeInfo>>(&mut self, iter: I) {
self.nodes.extend(
iter.into_iter()
.map(|info| (info.value, StoreNode { left: info.left, right: info.right })),
);
}
}
impl Serializable for StoreNode {
fn write_into<W: ByteWriter>(&self, target: &mut W) {
self.left.write_into(target);
self.right.write_into(target);
}
}
impl Deserializable for StoreNode {
fn read_from<R: ByteReader>(source: &mut R) -> Result<Self, DeserializationError> {
let left = Word::read_from(source)?;
let right = Word::read_from(source)?;
Ok(StoreNode { left, right })
}
}
impl Serializable for MerkleStore {
fn write_into<W: ByteWriter>(&self, target: &mut W) {
target.write_u64(self.nodes.len() as u64);
for (k, v) in self.nodes.iter() {
k.write_into(target);
v.write_into(target);
}
}
}
impl Deserializable for MerkleStore {
fn read_from<R: ByteReader>(source: &mut R) -> Result<Self, DeserializationError> {
let len_u64 = source.read_u64()?;
let len = usize::try_from(len_u64).map_err(|_| {
DeserializationError::InvalidValue("MerkleStore node count too large".into())
})?;
let element_size = <(Word, StoreNode) as Deserializable>::min_serialized_size();
let required_bytes = len.checked_mul(element_size).ok_or_else(|| {
DeserializationError::InvalidValue("MerkleStore node count too large".into())
})?;
source.check_eor(required_bytes)?;
let nodes: Vec<(Word, StoreNode)> =
source.read_many_iter(len)?.collect::<Result<_, _>>()?;
Ok(nodes.into_iter().collect())
}
fn min_serialized_size() -> usize {
8
}
}
fn empty_hashes() -> impl Iterator<Item = (Word, StoreNode)> {
let subtrees = EmptySubtreeRoots::empty_hashes(255);
subtrees
.iter()
.rev()
.copied()
.zip(subtrees.iter().rev().skip(1).copied())
.map(|(child, parent)| (parent, StoreNode { left: child, right: child }))
}
fn combine_nodes_with_empty_hashes(
nodes: impl IntoIterator<Item = InnerNodeInfo>,
) -> impl Iterator<Item = (Word, StoreNode)> {
nodes
.into_iter()
.map(|info| (info.value, StoreNode { left: info.left, right: info.right }))
.chain(empty_hashes())
}