From 9703cba92abe88dc0ecec7ae7b6e7ccb5399e11b Mon Sep 17 00:00:00 2001 From: Ignotus Peverell Date: Wed, 19 Jul 2017 14:03:42 +0000 Subject: [PATCH] Formatting --- core/src/core/sumtree.rs | 1436 ++++++++++++++++++++------------------ 1 file changed, 743 insertions(+), 693 deletions(-) diff --git a/core/src/core/sumtree.rs b/core/src/core/sumtree.rs index cb3c5f18b..1f87215b6 100644 --- a/core/src/core/sumtree.rs +++ b/core/src/core/sumtree.rs @@ -16,8 +16,8 @@ //! Generic sum-merkle tree. See `doc/merkle.md` for design and motivation for //! this structure. Most trees in Grin are stored and transmitted as either //! (a) only a root, or (b) the entire unpruned tree. For these it is sufficient -//! to have a root-calculating function, for which the `compute_root` function should -//! be used. +//! to have a root-calculating function, for which the `compute_root` function +//! should be used. //! //! The output set structure has much stronger requirements, as it is updated //! and pruned in place, and needs to be efficiently storable even when it is @@ -29,345 +29,379 @@ use ser::{self, Readable, Reader, Writeable, Writer}; use std::collections::HashMap; use std::{self, mem, ops}; -/// Trait describing an object that has a well-defined sum that the tree can sum over +/// Trait describing an object that has a well-defined sum that the tree can +/// sum over pub trait Summable { - /// The type of an object's sum - type Sum: Clone + ops::Add + Readable + Writeable; + /// The type of an object's sum + type Sum: Clone + ops::Add + Readable + Writeable; - /// Obtain the sum of the object - fn sum(&self) -> Self::Sum; + /// Obtain the sum of the object + fn sum(&self) -> Self::Sum; } /// An empty sum that takes no space #[derive(Copy, Clone)] pub struct NullSum; impl ops::Add for NullSum { - type Output = NullSum; - fn add(self, _: NullSum) -> NullSum { NullSum } + type Output = NullSum; + fn add(self, _: NullSum) -> NullSum { + NullSum + } } impl Readable for NullSum { - fn read(_: &mut Reader) -> Result { - Ok(NullSum) - } + fn read(_: &mut Reader) -> Result { + Ok(NullSum) + } } impl Writeable for NullSum { - fn write(&self, _: &mut W) -> Result<(), ser::Error> { - Ok(()) - } + fn write(&self, _: &mut W) -> Result<(), ser::Error> { + Ok(()) + } } /// Wrapper for a type that allows it to be inserted in a tree without summing pub struct NoSum(T); impl Summable for NoSum { - type Sum = NullSum; - fn sum(&self) -> NullSum { NullSum } + type Sum = NullSum; + fn sum(&self) -> NullSum { + NullSum + } } #[derive(Clone)] enum NodeData { - /// Node with 2^n children which are not stored with the tree - Pruned(T::Sum), - /// Actual data - Leaf(T), - /// Node with 2^n children - Internal { - lchild: Box>, - rchild: Box>, - sum: T::Sum - }, + /// Node with 2^n children which are not stored with the tree + Pruned(T::Sum), + /// Actual data + Leaf(T), + /// Node with 2^n children + Internal { + lchild: Box>, + rchild: Box>, + sum: T::Sum, + }, } impl Summable for NodeData { - type Sum = T::Sum; - fn sum(&self) -> T::Sum { - match *self { - NodeData::Pruned(ref sum) => sum.clone(), - NodeData::Leaf(ref data) => data.sum(), - NodeData::Internal { ref sum, .. } => sum.clone() - } - } + type Sum = T::Sum; + fn sum(&self) -> T::Sum { + match *self { + NodeData::Pruned(ref sum) => sum.clone(), + NodeData::Leaf(ref data) => data.sum(), + NodeData::Internal { ref sum, .. } => sum.clone(), + } + } } #[derive(Clone)] struct Node { - full: bool, - data: NodeData, - hash: Hash, - depth: u8 + full: bool, + data: NodeData, + hash: Hash, + depth: u8, } impl Summable for Node { - type Sum = T::Sum; - fn sum(&self) -> T::Sum { - self.data.sum() - } + type Sum = T::Sum; + fn sum(&self) -> T::Sum { + self.data.sum() + } } impl Node { - /// Get the root hash and sum of the node - fn root_sum(&self) -> (Hash, T::Sum) { - (self.hash, self.sum()) - } - - fn n_children(&self) -> usize { - if self.full { - 1 << self.depth - } else { - if let NodeData::Internal{ ref lchild, ref rchild, .. } = self.data { - lchild.n_children() + rchild.n_children() - } else { - unreachable!() - } - } - } + /// Get the root hash and sum of the node + fn root_sum(&self) -> (Hash, T::Sum) { + (self.hash, self.sum()) + } + fn n_children(&self) -> usize { + if self.full { + 1 << self.depth + } else { + if let NodeData::Internal { + ref lchild, + ref rchild, + .. + } = self.data + { + lchild.n_children() + rchild.n_children() + } else { + unreachable!() + } + } + } } /// An insertion ordered merkle sum tree. #[derive(Clone)] pub struct SumTree { - /// Index mapping data to its index in the tree - index: HashMap, - /// Tree contents - root: Option> + /// Index mapping data to its index in the tree + index: HashMap, + /// Tree contents + root: Option>, } impl SumTree - where T: Summable + Writeable +where + T: Summable + Writeable, { - /// Create a new empty tree - pub fn new() -> SumTree { - SumTree { - index: HashMap::new(), - root: None - } - } + /// Create a new empty tree + pub fn new() -> SumTree { + SumTree { + index: HashMap::new(), + root: None, + } + } - /// Accessor for the tree's root - pub fn root_sum(&self) -> Option<(Hash, T::Sum)> { - self.root.as_ref().map(|node| node.root_sum()) - } + /// Accessor for the tree's root + pub fn root_sum(&self) -> Option<(Hash, T::Sum)> { + self.root.as_ref().map(|node| node.root_sum()) + } - fn insert_right_of(mut old: Node, new: Node) -> Node { - assert!(old.depth >= new.depth); + fn insert_right_of(mut old: Node, new: Node) -> Node { + assert!(old.depth >= new.depth); - // If we are inserting next to a full node, make a parent. If we're - // inserting a tree of equal depth then we get a full node, otherwise - // we get a partial node. Leaves and pruned data both count as full - // nodes. - if old.full { - let parent_depth = old.depth + 1; - let parent_sum = old.sum() + new.sum(); - let parent_hash = (parent_depth, &parent_sum, old.hash, new.hash).hash(); - let parent_full = old.depth == new.depth; - let parent_data = NodeData::Internal { - lchild: Box::new(old), - rchild: Box::new(new), - sum: parent_sum, - }; + // If we are inserting next to a full node, make a parent. If we're + // inserting a tree of equal depth then we get a full node, otherwise + // we get a partial node. Leaves and pruned data both count as full + // nodes. + if old.full { + let parent_depth = old.depth + 1; + let parent_sum = old.sum() + new.sum(); + let parent_hash = (parent_depth, &parent_sum, old.hash, new.hash).hash(); + let parent_full = old.depth == new.depth; + let parent_data = NodeData::Internal { + lchild: Box::new(old), + rchild: Box::new(new), + sum: parent_sum, + }; - Node { - full: parent_full, - data: parent_data, - hash: parent_hash, - depth: parent_depth - } - // If we are inserting next to a partial node, we should actually be - // inserting under the node, so we recurse. The right child of a partial - // node is always another partial node or a leaf. - } else { - if let NodeData::Internal{ ref lchild, ref mut rchild, ref mut sum } = old.data { - // Recurse - let dummy_child = Node { full: true, data: NodeData::Pruned(sum.clone()), hash: old.hash, depth: 0 }; - let moved_rchild = mem::replace(&mut **rchild, dummy_child); - mem::replace(&mut **rchild, SumTree::insert_right_of(moved_rchild, new)); - // Update this node's states to reflect the new right child - if rchild.full && rchild.depth == old.depth - 1 { - old.full = rchild.full; - } - *sum = lchild.sum() + rchild.sum(); - old.hash = (old.depth, &*sum, lchild.hash, rchild.hash).hash(); - } else { - unreachable!() - } - old - } - } + Node { + full: parent_full, + data: parent_data, + hash: parent_hash, + depth: parent_depth, + } + // If we are inserting next to a partial node, we should actually be + // inserting under the node, so we recurse. The right child of a partial + // node is always another partial node or a leaf. + } else { + if let NodeData::Internal { + ref lchild, + ref mut rchild, + ref mut sum, + } = old.data + { + // Recurse + let dummy_child = Node { + full: true, + data: NodeData::Pruned(sum.clone()), + hash: old.hash, + depth: 0, + }; + let moved_rchild = mem::replace(&mut **rchild, dummy_child); + mem::replace(&mut **rchild, SumTree::insert_right_of(moved_rchild, new)); + // Update this node's states to reflect the new right child + if rchild.full && rchild.depth == old.depth - 1 { + old.full = rchild.full; + } + *sum = lchild.sum() + rchild.sum(); + old.hash = (old.depth, &*sum, lchild.hash, rchild.hash).hash(); + } else { + unreachable!() + } + old + } + } - /// Accessor for number of elements in the tree, not including pruned ones - pub fn len(&self) -> usize { - self.index.len() - } + /// Accessor for number of elements in the tree, not including pruned ones + pub fn len(&self) -> usize { + self.index.len() + } - /// Accessor for number of elements in the tree, including pruned ones - pub fn unpruned_len(&self) -> usize { - match self.root { - None => 0, - Some(ref node) => node.n_children() - } - } + /// Accessor for number of elements in the tree, including pruned ones + pub fn unpruned_len(&self) -> usize { + match self.root { + None => 0, + Some(ref node) => node.n_children(), + } + } - /// Add an element to the tree. Returns true if the element was added, - /// false if it already existed in the tree. - pub fn push(&mut self, elem: T) -> bool { - // Compute element hash and depth-0 node hash - let index_hash = Hashed::hash(&elem); - let elem_sum = elem.sum(); - let elem_hash = (0u8, &elem_sum, index_hash).hash(); + /// Add an element to the tree. Returns true if the element was added, + /// false if it already existed in the tree. + pub fn push(&mut self, elem: T) -> bool { + // Compute element hash and depth-0 node hash + let index_hash = Hashed::hash(&elem); + let elem_sum = elem.sum(); + let elem_hash = (0u8, &elem_sum, index_hash).hash(); - if self.index.contains_key(&index_hash) { - return false; - } + if self.index.contains_key(&index_hash) { + return false; + } - // Special-case the first element - if self.root.is_none() { - self.root = Some(Node { - full: true, - data: NodeData::Leaf(elem), - hash: elem_hash, - depth: 0 - }); - self.index.insert(index_hash, 0); - return true; - } + // Special-case the first element + if self.root.is_none() { + self.root = Some(Node { + full: true, + data: NodeData::Leaf(elem), + hash: elem_hash, + depth: 0, + }); + self.index.insert(index_hash, 0); + return true; + } - // Next, move the old root out of the structure so that we are allowed to - // move it. We will move a new root back in at the end of the function - let old_root = mem::replace(&mut self.root, None).unwrap(); + // Next, move the old root out of the structure so that we are allowed to + // move it. We will move a new root back in at the end of the function + let old_root = mem::replace(&mut self.root, None).unwrap(); - // Insert into tree, compute new root - let new_node = Node { - full: true, - data: NodeData::Leaf(elem), - hash: elem_hash, - depth: 0 - }; + // Insert into tree, compute new root + let new_node = Node { + full: true, + data: NodeData::Leaf(elem), + hash: elem_hash, + depth: 0, + }; - // Put new root in place and record insertion - let index = old_root.n_children(); - self.root = Some(SumTree::insert_right_of(old_root, new_node)); - self.index.insert(index_hash, index); - true - } + // Put new root in place and record insertion + let index = old_root.n_children(); + self.root = Some(SumTree::insert_right_of(old_root, new_node)); + self.index.insert(index_hash, index); + true + } - fn replace_recurse(node: &mut Node, index: usize, new_elem: T) { - assert!(index < (1 << node.depth)); + fn replace_recurse(node: &mut Node, index: usize, new_elem: T) { + assert!(index < (1 << node.depth)); - if node.depth == 0 { - assert!(node.full); - node.hash = (0u8, new_elem.sum(), Hashed::hash(&new_elem)).hash(); - node.data = NodeData::Leaf(new_elem); - } else { - match node.data { - NodeData::Internal { ref mut lchild, ref mut rchild, ref mut sum } => { - let bit = index & (1 << (node.depth - 1)); - if bit > 0 { - SumTree::replace_recurse(rchild, index - bit, new_elem); - } else { - SumTree::replace_recurse(lchild, index, new_elem); - } - *sum = lchild.sum() + rchild.sum(); - node.hash = (node.depth, &*sum, lchild.hash, rchild.hash).hash(); - } - // Pruned data would not have been in the index - NodeData::Pruned(_) => unreachable!(), - NodeData::Leaf(_) => unreachable!() - } - } - } + if node.depth == 0 { + assert!(node.full); + node.hash = (0u8, new_elem.sum(), Hashed::hash(&new_elem)).hash(); + node.data = NodeData::Leaf(new_elem); + } else { + match node.data { + NodeData::Internal { + ref mut lchild, + ref mut rchild, + ref mut sum, + } => { + let bit = index & (1 << (node.depth - 1)); + if bit > 0 { + SumTree::replace_recurse(rchild, index - bit, new_elem); + } else { + SumTree::replace_recurse(lchild, index, new_elem); + } + *sum = lchild.sum() + rchild.sum(); + node.hash = (node.depth, &*sum, lchild.hash, rchild.hash).hash(); + } + // Pruned data would not have been in the index + NodeData::Pruned(_) => unreachable!(), + NodeData::Leaf(_) => unreachable!(), + } + } + } - /// Replaces an element in the tree. Returns true if the element existed - /// and was replaced. Returns false if the old element did not exist or - /// if the new element already existed - pub fn replace(&mut self, elem: &T, new_elem: T) -> bool { - let index_hash = Hashed::hash(elem); + /// Replaces an element in the tree. Returns true if the element existed + /// and was replaced. Returns false if the old element did not exist or + /// if the new element already existed + pub fn replace(&mut self, elem: &T, new_elem: T) -> bool { + let index_hash = Hashed::hash(elem); - let root = match self.root { - Some(ref mut node) => node, - None => { return false; } - }; + let root = match self.root { + Some(ref mut node) => node, + None => { + return false; + } + }; - match self.index.remove(&index_hash) { - None => false, - Some(index) => { - let new_index_hash = Hashed::hash(&new_elem); - if self.index.contains_key(&new_index_hash) { - false - } else { - SumTree::replace_recurse(root, index, new_elem); - self.index.insert(new_index_hash, index); - true - } - } - } - } + match self.index.remove(&index_hash) { + None => false, + Some(index) => { + let new_index_hash = Hashed::hash(&new_elem); + if self.index.contains_key(&new_index_hash) { + false + } else { + SumTree::replace_recurse(root, index, new_elem); + self.index.insert(new_index_hash, index); + true + } + } + } + } - /// Determine whether an element exists in the tree. - /// If so, return its index - pub fn contains(&self, elem: &T) -> Option { - let index_hash = Hashed::hash(elem); - self.index.get(&index_hash).map(|x| *x) - } + /// Determine whether an element exists in the tree. + /// If so, return its index + pub fn contains(&self, elem: &T) -> Option { + let index_hash = Hashed::hash(elem); + self.index.get(&index_hash).map(|x| *x) + } - fn prune_recurse(node: &mut Node, index: usize) { - assert!(index < (1 << node.depth)); + fn prune_recurse(node: &mut Node, index: usize) { + assert!(index < (1 << node.depth)); - if node.depth == 0 { - let sum = if let NodeData::Leaf(ref elem) = node.data { - elem.sum() - } else { - unreachable!() - }; - node.data = NodeData::Pruned(sum); - } else { - let mut prune_me = None; - match node.data { - NodeData::Internal { ref mut lchild, ref mut rchild, .. } => { - let bit = index & (1 << (node.depth - 1)); - if bit > 0 { - SumTree::prune_recurse(rchild, index - bit); - } else { - SumTree::prune_recurse(lchild, index); - } - if let (&NodeData::Pruned(ref lsum), &NodeData::Pruned(ref rsum)) = (&lchild.data, &rchild.data) { - if node.full { - prune_me = Some(lsum.clone() + rsum.clone()); - } - } - } - NodeData::Pruned(_) => { - // Already pruned. Ok. - } - NodeData::Leaf(_) => unreachable!() - } - if let Some(sum) = prune_me { - node.data = NodeData::Pruned(sum); - } - } - } + if node.depth == 0 { + let sum = if let NodeData::Leaf(ref elem) = node.data { + elem.sum() + } else { + unreachable!() + }; + node.data = NodeData::Pruned(sum); + } else { + let mut prune_me = None; + match node.data { + NodeData::Internal { + ref mut lchild, + ref mut rchild, + .. + } => { + let bit = index & (1 << (node.depth - 1)); + if bit > 0 { + SumTree::prune_recurse(rchild, index - bit); + } else { + SumTree::prune_recurse(lchild, index); + } + if let (&NodeData::Pruned(ref lsum), &NodeData::Pruned(ref rsum)) = + (&lchild.data, &rchild.data) + { + if node.full { + prune_me = Some(lsum.clone() + rsum.clone()); + } + } + } + NodeData::Pruned(_) => { + // Already pruned. Ok. + } + NodeData::Leaf(_) => unreachable!(), + } + if let Some(sum) = prune_me { + node.data = NodeData::Pruned(sum); + } + } + } - /// Removes an element from storage, not affecting the tree - /// Returns true if the element was actually in the tree - pub fn prune(&mut self, elem: &T) -> bool { - let index_hash = Hashed::hash(elem); + /// Removes an element from storage, not affecting the tree + /// Returns true if the element was actually in the tree + pub fn prune(&mut self, elem: &T) -> bool { + let index_hash = Hashed::hash(elem); - let root = match self.root { - Some(ref mut node) => node, - None => { return false; } - }; + let root = match self.root { + Some(ref mut node) => node, + None => { + return false; + } + }; - match self.index.remove(&index_hash) { - None => false, - Some(index) => { - SumTree::prune_recurse(root, index); - true - } - } - } + match self.index.remove(&index_hash) { + None => false, + Some(index) => { + SumTree::prune_recurse(root, index); + true + } + } + } - // TODO push_many to allow bulk updates + // TODO push_many to allow bulk updates } // A SumTree is encoded as follows: an empty tree is the single byte 0x00. @@ -384,146 +418,154 @@ impl SumTree // For a pruned internal node, it is followed by nothing. // impl Writeable for SumTree - where T: Summable + Writeable +where + T: Summable + Writeable, { - fn write(&self, writer: &mut W) -> Result<(), ser::Error> { - match self.root { - None => writer.write_u8(0), - Some(ref node) => node.write(writer) - } - } + fn write(&self, writer: &mut W) -> Result<(), ser::Error> { + match self.root { + None => writer.write_u8(0), + Some(ref node) => node.write(writer), + } + } } impl Writeable for Node - where T: Summable + Writeable +where + T: Summable + Writeable, { - fn write(&self, writer: &mut W) -> Result<(), ser::Error> { - assert!(self.depth < 64); + fn write(&self, writer: &mut W) -> Result<(), ser::Error> { + assert!(self.depth < 64); - // Compute depth byte: 0x80 means full, 0xc0 means unpruned - let mut depth = 0; - if self.full { - depth |= 0x80; - } - if let NodeData::Pruned(_) = self.data { - } else { - depth |= 0xc0; - } - depth |= self.depth; - // Encode node - try!(writer.write_u8(depth)); - try!(self.hash.write(writer)); - match self.data { - NodeData::Pruned(ref sum) => { - sum.write(writer) - }, - NodeData::Leaf(ref data) => { - data.write(writer) - }, - NodeData::Internal { ref lchild, ref rchild, ref sum } => { - try!(sum.write(writer)); - try!(lchild.write(writer)); - rchild.write(writer) - }, - } - } + // Compute depth byte: 0x80 means full, 0xc0 means unpruned + let mut depth = 0; + if self.full { + depth |= 0x80; + } + if let NodeData::Pruned(_) = self.data { + } else { + depth |= 0xc0; + } + depth |= self.depth; + // Encode node + try!(writer.write_u8(depth)); + try!(self.hash.write(writer)); + match self.data { + NodeData::Pruned(ref sum) => sum.write(writer), + NodeData::Leaf(ref data) => data.write(writer), + NodeData::Internal { + ref lchild, + ref rchild, + ref sum, + } => { + try!(sum.write(writer)); + try!(lchild.write(writer)); + rchild.write(writer) + } + } + } } -fn node_read_recurse(reader: &mut Reader, index: &mut HashMap, tree_index: &mut usize) -> Result, ser::Error> - where T: Summable + Readable + Hashed +fn node_read_recurse( + reader: &mut Reader, + index: &mut HashMap, + tree_index: &mut usize, +) -> Result, ser::Error> +where + T: Summable + Readable + Hashed, { - // Read depth byte - let depth = try!(reader.read_u8()); - let full = depth & 0x80 == 0x80; - let pruned = depth & 0xc0 != 0xc0; - let depth = depth & 0x3f; + // Read depth byte + let depth = try!(reader.read_u8()); + let full = depth & 0x80 == 0x80; + let pruned = depth & 0xc0 != 0xc0; + let depth = depth & 0x3f; - // Sanity-check for zero byte - if pruned && !full { - return Err(ser::Error::CorruptedData); - } + // Sanity-check for zero byte + if pruned && !full { + return Err(ser::Error::CorruptedData); + } - // Read remainder of node - let hash = try!(Readable::read(reader)); - let data = match (depth, pruned) { - (_, true) => { - let sum = try!(Readable::read(reader)); - *tree_index += 1 << depth as usize; - NodeData::Pruned(sum) - } - (0, _) => { - let elem: T = try!(Readable::read(reader)); - index.insert(Hashed::hash(&elem), *tree_index); - *tree_index += 1; - NodeData::Leaf(elem) - } - (_, _) => { - let sum = try!(Readable::read(reader)); - NodeData::Internal { - lchild: Box::new(try!(node_read_recurse(reader, index, tree_index))), - rchild: Box::new(try!(node_read_recurse(reader, index, tree_index))), - sum: sum - } - } - }; + // Read remainder of node + let hash = try!(Readable::read(reader)); + let data = match (depth, pruned) { + (_, true) => { + let sum = try!(Readable::read(reader)); + *tree_index += 1 << depth as usize; + NodeData::Pruned(sum) + } + (0, _) => { + let elem: T = try!(Readable::read(reader)); + index.insert(Hashed::hash(&elem), *tree_index); + *tree_index += 1; + NodeData::Leaf(elem) + } + (_, _) => { + let sum = try!(Readable::read(reader)); + NodeData::Internal { + lchild: Box::new(try!(node_read_recurse(reader, index, tree_index))), + rchild: Box::new(try!(node_read_recurse(reader, index, tree_index))), + sum: sum, + } + } + }; - Ok(Node { - full: full, - data: data, - hash: hash, - depth: depth - }) + Ok(Node { + full: full, + data: data, + hash: hash, + depth: depth, + }) } impl Readable for SumTree - where T: Summable + Writeable + Readable + Hashed +where + T: Summable + Writeable + Readable + Hashed, { - fn read(reader: &mut Reader) -> Result, ser::Error> { - // Read depth byte of root node - let depth = try!(reader.read_u8()); - let full = depth & 0x80 == 0x80; - let pruned = depth & 0xc0 != 0xc0; - let depth = depth & 0x3f; + fn read(reader: &mut Reader) -> Result, ser::Error> { + // Read depth byte of root node + let depth = try!(reader.read_u8()); + let full = depth & 0x80 == 0x80; + let pruned = depth & 0xc0 != 0xc0; + let depth = depth & 0x3f; - // Special-case the zero byte - if pruned && !full { - return Ok(SumTree { - index: HashMap::new(), - root: None - }); - } + // Special-case the zero byte + if pruned && !full { + return Ok(SumTree { + index: HashMap::new(), + root: None, + }); + } - // Otherwise continue reading it - let mut index = HashMap::new(); + // Otherwise continue reading it + let mut index = HashMap::new(); - let hash = try!(Readable::read(reader)); - let data = match (depth, pruned) { - (_, true) => { - let sum = try!(Readable::read(reader)); - NodeData::Pruned(sum) - } - (0, _) => NodeData::Leaf(try!(Readable::read(reader))), - (_, _) => { - let sum = try!(Readable::read(reader)); - let mut tree_index = 0; - NodeData::Internal { - lchild: Box::new(try!(node_read_recurse(reader, &mut index, &mut tree_index))), - rchild: Box::new(try!(node_read_recurse(reader, &mut index, &mut tree_index))), - sum: sum - } - } - }; + let hash = try!(Readable::read(reader)); + let data = match (depth, pruned) { + (_, true) => { + let sum = try!(Readable::read(reader)); + NodeData::Pruned(sum) + } + (0, _) => NodeData::Leaf(try!(Readable::read(reader))), + (_, _) => { + let sum = try!(Readable::read(reader)); + let mut tree_index = 0; + NodeData::Internal { + lchild: Box::new(try!(node_read_recurse(reader, &mut index, &mut tree_index))), + rchild: Box::new(try!(node_read_recurse(reader, &mut index, &mut tree_index))), + sum: sum, + } + } + }; - Ok(SumTree { - index: index, - root: Some(Node { - full: full, - data: data, - hash: hash, - depth: depth - }) - }) - } + Ok(SumTree { + index: index, + root: Some(Node { + full: full, + data: data, + hash: hash, + depth: depth, + }), + }) + } } /// This is used to as a scratch space during root calculation so that we can @@ -535,136 +577,145 @@ const MAX_MMR_HEIGHT: usize = 48; /// https://github.com/opentimestamps/opentimestamps-server/blob/master/python-opentimestamps/opentimestamps/core/timestamp.py#L324 /// fn compute_peaks(iter: I, peaks: &mut [Option<(u8, Hash, S)>]) - where S: Clone + ops::Add + Writeable, - I: Iterator +where + S: Clone + ops::Add + Writeable, + I: Iterator, { - for peak in peaks.iter_mut() { - *peak = None; - } - for (mut new_depth, mut new_hash, mut new_sum) in iter { - let mut index = 0; - while let Some((old_depth, old_hash, old_sum)) = peaks[index].take() { - // Erase current peak (done by `take()` above), then combine - // it with the new addition, to be inserted one higher - index += 1; - new_depth = old_depth + 1; - new_sum = old_sum.clone() + new_sum.clone(); - new_hash = (new_depth, &new_sum, old_hash, new_hash).hash(); - } - peaks[index] = Some((new_depth, new_hash, new_sum)); - } + for peak in peaks.iter_mut() { + *peak = None; + } + for (mut new_depth, mut new_hash, mut new_sum) in iter { + let mut index = 0; + while let Some((old_depth, old_hash, old_sum)) = peaks[index].take() { + // Erase current peak (done by `take()` above), then combine + // it with the new addition, to be inserted one higher + index += 1; + new_depth = old_depth + 1; + new_sum = old_sum.clone() + new_sum.clone(); + new_hash = (new_depth, &new_sum, old_hash, new_hash).hash(); + } + peaks[index] = Some((new_depth, new_hash, new_sum)); + } } /// Directly compute the Merkle root of a sum-tree whose contents are given /// explicitly in the passed iterator. pub fn compute_root<'a, T, I>(iter: I) -> Option<(Hash, T::Sum)> - where T: 'a + Summable + Writeable, - I: Iterator +where + T: 'a + Summable + Writeable, + I: Iterator, { - let mut peaks = vec![None; MAX_MMR_HEIGHT]; - compute_peaks(iter.map(|elem| { - let depth = 0u8; - let sum = elem.sum(); - let hash = (depth, &sum, Hashed::hash(elem)).hash(); - (depth, hash, sum) - }), &mut peaks); + let mut peaks = vec![None; MAX_MMR_HEIGHT]; + compute_peaks( + iter.map(|elem| { + let depth = 0u8; + let sum = elem.sum(); + let hash = (depth, &sum, Hashed::hash(elem)).hash(); + (depth, hash, sum) + }), + &mut peaks, + ); - let mut ret = None; - for peak in peaks { - ret = match (peak, ret) { - (None, x) => x, - (Some((_, hash, sum)), None) => Some((hash, sum)), - (Some((depth, lhash, lsum)), Some((rhash, rsum))) => { - let sum = lsum + rsum; - let hash = (depth + 1, &sum, lhash, rhash).hash(); - Some((hash, sum)) - } - }; - } - ret + let mut ret = None; + for peak in peaks { + ret = match (peak, ret) { + (None, x) => x, + (Some((_, hash, sum)), None) => Some((hash, sum)), + (Some((depth, lhash, lsum)), Some((rhash, rsum))) => { + let sum = lsum + rsum; + let hash = (depth + 1, &sum, lhash, rhash).hash(); + Some((hash, sum)) + } + }; + } + ret } // a couple functions that help debugging #[allow(dead_code)] fn print_node(node: &Node, tab_level: usize) - where T: Summable + Writeable, - T::Sum: std::fmt::Debug +where + T: Summable + Writeable, + T::Sum: std::fmt::Debug, { - for _ in 0..tab_level { - print!(" "); - } - print!("[{:03}] {} {:?}", node.depth, node.hash, node.sum()); - match node.data { - NodeData::Pruned(_) => println!(" X"), - NodeData::Leaf(_) => println!(" L"), - NodeData::Internal { ref lchild, ref rchild, .. } => { - println!(":"); - print_node(lchild, tab_level + 1); - print_node(rchild, tab_level + 1); - } - } + for _ in 0..tab_level { + print!(" "); + } + print!("[{:03}] {} {:?}", node.depth, node.hash, node.sum()); + match node.data { + NodeData::Pruned(_) => println!(" X"), + NodeData::Leaf(_) => println!(" L"), + NodeData::Internal { + ref lchild, + ref rchild, + .. + } => { + println!(":"); + print_node(lchild, tab_level + 1); + print_node(rchild, tab_level + 1); + } + } } #[allow(dead_code)] fn print_tree(tree: &SumTree) - where T: Summable + Writeable, - T::Sum: std::fmt::Debug +where + T: Summable + Writeable, + T::Sum: std::fmt::Debug, { - match tree.root { - None => println!("[empty tree]"), - Some(ref node) => { - print_node(node, 0); - } - } + match tree.root { + None => println!("[empty tree]"), + Some(ref node) => { + print_node(node, 0); + } + } } #[cfg(test)] mod test { - use rand::{thread_rng, Rng}; - use core::hash::Hashed; - use ser; - use super::*; + use rand::{thread_rng, Rng}; + use core::hash::Hashed; + use ser; + use super::*; - #[derive(Copy, Clone, Debug)] - struct TestElem([u32; 4]); - impl Summable for TestElem { - type Sum = u64; - fn sum(&self) -> u64 { - // sums are not allowed to overflow, so we use this simple - // non-injective "sum" function that will still be homomorphic - self.0[0] as u64 * 0x1000 + - self.0[1] as u64 * 0x100 + - self.0[2] as u64 * 0x10 + - self.0[3] as u64 - } - } + #[derive(Copy, Clone, Debug)] + struct TestElem([u32; 4]); + impl Summable for TestElem { + type Sum = u64; + fn sum(&self) -> u64 { + // sums are not allowed to overflow, so we use this simple + // non-injective "sum" function that will still be homomorphic + self.0[0] as u64 * 0x1000 + self.0[1] as u64 * 0x100 + self.0[2] as u64 * 0x10 + + self.0[3] as u64 + } + } - impl Writeable for TestElem { - fn write(&self, writer: &mut W) -> Result<(), ser::Error> { - try!(writer.write_u32(self.0[0])); - try!(writer.write_u32(self.0[1])); - try!(writer.write_u32(self.0[2])); - writer.write_u32(self.0[3]) - } - } + impl Writeable for TestElem { + fn write(&self, writer: &mut W) -> Result<(), ser::Error> { + try!(writer.write_u32(self.0[0])); + try!(writer.write_u32(self.0[1])); + try!(writer.write_u32(self.0[2])); + writer.write_u32(self.0[3]) + } + } - fn sumtree_create_(prune: bool) { - let mut tree = SumTree::new(); + fn sumtree_create_(prune: bool) { + let mut tree = SumTree::new(); - macro_rules! leaf { + macro_rules! leaf { ($data: expr) => ({ (0u8, $data.sum(), $data.hash()) }) }; - macro_rules! node { + macro_rules! node { ($left: expr, $right: expr) => ( ($left.0 + 1, $left.1 + $right.1, $left.hash(), $right.hash()) ) }; - macro_rules! prune { + macro_rules! prune { ($prune: expr, $tree: expr, $elem: expr) => { if $prune { assert_eq!($tree.len(), 1); @@ -679,230 +730,229 @@ mod test { } }; - let mut elems = [TestElem([0, 0, 0, 1]), TestElem([0, 0, 0, 2]), - TestElem([0, 0, 0, 3]), TestElem([0, 0, 0, 4]), - TestElem([0, 0, 0, 5]), TestElem([0, 0, 0, 6]), - TestElem([0, 0, 0, 7]), TestElem([1, 0, 0, 0])]; + let mut elems = [ + TestElem([0, 0, 0, 1]), + TestElem([0, 0, 0, 2]), + TestElem([0, 0, 0, 3]), + TestElem([0, 0, 0, 4]), + TestElem([0, 0, 0, 5]), + TestElem([0, 0, 0, 6]), + TestElem([0, 0, 0, 7]), + TestElem([1, 0, 0, 0]), + ]; - assert_eq!(tree.root_sum(), None); - assert_eq!(tree.root_sum(), compute_root(elems[0..0].iter())); - assert_eq!(tree.len(), 0); - assert_eq!(tree.contains(&elems[0]), None); - assert!(tree.push(elems[0])); - assert_eq!(tree.contains(&elems[0]), Some(0)); + assert_eq!(tree.root_sum(), None); + assert_eq!(tree.root_sum(), compute_root(elems[0..0].iter())); + assert_eq!(tree.len(), 0); + assert_eq!(tree.contains(&elems[0]), None); + assert!(tree.push(elems[0])); + assert_eq!(tree.contains(&elems[0]), Some(0)); - // One element - let expected = leaf!(elems[0]).hash(); - assert_eq!(tree.root_sum(), Some((expected, 1))); - assert_eq!(tree.root_sum(), compute_root(elems[0..1].iter())); - assert_eq!(tree.unpruned_len(), 1); - prune!(prune, tree, elems[0]); + // One element + let expected = leaf!(elems[0]).hash(); + assert_eq!(tree.root_sum(), Some((expected, 1))); + assert_eq!(tree.root_sum(), compute_root(elems[0..1].iter())); + assert_eq!(tree.unpruned_len(), 1); + prune!(prune, tree, elems[0]); - // Two elements - assert_eq!(tree.contains(&elems[1]), None); - assert!(tree.push(elems[1])); - assert_eq!(tree.contains(&elems[1]), Some(1)); - let expected = node!(leaf!(elems[0]), - leaf!(elems[1]) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 3))); - assert_eq!(tree.root_sum(), compute_root(elems[0..2].iter())); - assert_eq!(tree.unpruned_len(), 2); - prune!(prune, tree, elems[1]); + // Two elements + assert_eq!(tree.contains(&elems[1]), None); + assert!(tree.push(elems[1])); + assert_eq!(tree.contains(&elems[1]), Some(1)); + let expected = node!(leaf!(elems[0]), leaf!(elems[1])).hash(); + assert_eq!(tree.root_sum(), Some((expected, 3))); + assert_eq!(tree.root_sum(), compute_root(elems[0..2].iter())); + assert_eq!(tree.unpruned_len(), 2); + prune!(prune, tree, elems[1]); - // Three elements - assert_eq!(tree.contains(&elems[2]), None); - assert!(tree.push(elems[2])); - assert_eq!(tree.contains(&elems[2]), Some(2)); - let expected = node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - leaf!(elems[2]) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 6))); - assert_eq!(tree.root_sum(), compute_root(elems[0..3].iter())); - assert_eq!(tree.unpruned_len(), 3); - prune!(prune, tree, elems[2]); + // Three elements + assert_eq!(tree.contains(&elems[2]), None); + assert!(tree.push(elems[2])); + assert_eq!(tree.contains(&elems[2]), Some(2)); + let expected = node!(node!(leaf!(elems[0]), leaf!(elems[1])), leaf!(elems[2])).hash(); + assert_eq!(tree.root_sum(), Some((expected, 6))); + assert_eq!(tree.root_sum(), compute_root(elems[0..3].iter())); + assert_eq!(tree.unpruned_len(), 3); + prune!(prune, tree, elems[2]); - // Four elements - assert_eq!(tree.contains(&elems[3]), None); - assert!(tree.push(elems[3])); - assert_eq!(tree.contains(&elems[3]), Some(3)); - let expected = node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3])) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 10))); - assert_eq!(tree.root_sum(), compute_root(elems[0..4].iter())); - assert_eq!(tree.unpruned_len(), 4); - prune!(prune, tree, elems[3]); + // Four elements + assert_eq!(tree.contains(&elems[3]), None); + assert!(tree.push(elems[3])); + assert_eq!(tree.contains(&elems[3]), Some(3)); + let expected = node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 10))); + assert_eq!(tree.root_sum(), compute_root(elems[0..4].iter())); + assert_eq!(tree.unpruned_len(), 4); + prune!(prune, tree, elems[3]); - // Five elements - assert_eq!(tree.contains(&elems[4]), None); - assert!(tree.push(elems[4])); - assert_eq!(tree.contains(&elems[4]), Some(4)); - let expected = node!(node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3]))), - leaf!(elems[4]) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 15))); - assert_eq!(tree.root_sum(), compute_root(elems[0..5].iter())); - assert_eq!(tree.unpruned_len(), 5); - prune!(prune, tree, elems[4]); + // Five elements + assert_eq!(tree.contains(&elems[4]), None); + assert!(tree.push(elems[4])); + assert_eq!(tree.contains(&elems[4]), Some(4)); + let expected = node!( + node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ), + leaf!(elems[4]) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 15))); + assert_eq!(tree.root_sum(), compute_root(elems[0..5].iter())); + assert_eq!(tree.unpruned_len(), 5); + prune!(prune, tree, elems[4]); - // Six elements - assert_eq!(tree.contains(&elems[5]), None); - assert!(tree.push(elems[5])); - assert_eq!(tree.contains(&elems[5]), Some(5)); - let expected = node!(node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3]))), - node!(leaf!(elems[4]), - leaf!(elems[5])) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 21))); - assert_eq!(tree.root_sum(), compute_root(elems[0..6].iter())); - assert_eq!(tree.unpruned_len(), 6); - prune!(prune, tree, elems[5]); + // Six elements + assert_eq!(tree.contains(&elems[5]), None); + assert!(tree.push(elems[5])); + assert_eq!(tree.contains(&elems[5]), Some(5)); + let expected = node!( + node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ), + node!(leaf!(elems[4]), leaf!(elems[5])) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 21))); + assert_eq!(tree.root_sum(), compute_root(elems[0..6].iter())); + assert_eq!(tree.unpruned_len(), 6); + prune!(prune, tree, elems[5]); - // Seven elements - assert_eq!(tree.contains(&elems[6]), None); - assert!(tree.push(elems[6])); - assert_eq!(tree.contains(&elems[6]), Some(6)); - let expected = node!(node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3]))), - node!(node!(leaf!(elems[4]), - leaf!(elems[5])), - leaf!(elems[6])) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 28))); - assert_eq!(tree.root_sum(), compute_root(elems[0..7].iter())); - assert_eq!(tree.unpruned_len(), 7); - prune!(prune, tree, elems[6]); + // Seven elements + assert_eq!(tree.contains(&elems[6]), None); + assert!(tree.push(elems[6])); + assert_eq!(tree.contains(&elems[6]), Some(6)); + let expected = node!( + node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ), + node!(node!(leaf!(elems[4]), leaf!(elems[5])), leaf!(elems[6])) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 28))); + assert_eq!(tree.root_sum(), compute_root(elems[0..7].iter())); + assert_eq!(tree.unpruned_len(), 7); + prune!(prune, tree, elems[6]); - // Eight elements - assert_eq!(tree.contains(&elems[7]), None); - assert!(tree.push(elems[7])); - assert_eq!(tree.contains(&elems[7]), Some(7)); - let expected = node!(node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3]))), - node!(node!(leaf!(elems[4]), - leaf!(elems[5])), - node!(leaf!(elems[6]), - leaf!(elems[7]))) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 28 + 0x1000))); - assert_eq!(tree.root_sum(), compute_root(elems[0..8].iter())); - assert_eq!(tree.unpruned_len(), 8); - prune!(prune, tree, elems[7]); + // Eight elements + assert_eq!(tree.contains(&elems[7]), None); + assert!(tree.push(elems[7])); + assert_eq!(tree.contains(&elems[7]), Some(7)); + let expected = node!( + node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ), + node!( + node!(leaf!(elems[4]), leaf!(elems[5])), + node!(leaf!(elems[6]), leaf!(elems[7])) + ) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 28 + 0x1000))); + assert_eq!(tree.root_sum(), compute_root(elems[0..8].iter())); + assert_eq!(tree.unpruned_len(), 8); + prune!(prune, tree, elems[7]); - // If we weren't pruning, try changing some elements - if !prune { - for i in 0..8 { - let old_elem = elems[i]; - elems[i].0[2] += 1 + i as u32; - assert_eq!(tree.contains(&old_elem), Some(i)); - assert_eq!(tree.contains(&elems[i]), None); - assert!(tree.replace(&old_elem, elems[i])); - assert_eq!(tree.contains(&elems[i]), Some(i)); - assert_eq!(tree.contains(&old_elem), None); - } - let expected = node!(node!(node!(leaf!(elems[0]), - leaf!(elems[1])), - node!(leaf!(elems[2]), - leaf!(elems[3]))), - node!(node!(leaf!(elems[4]), - leaf!(elems[5])), - node!(leaf!(elems[6]), - leaf!(elems[7]))) - ).hash(); - assert_eq!(tree.root_sum(), Some((expected, 28 + 36 * 0x10 + 0x1000))); - assert_eq!(tree.root_sum(), compute_root(elems[0..8].iter())); - assert_eq!(tree.unpruned_len(), 8); - } + // If we weren't pruning, try changing some elements + if !prune { + for i in 0..8 { + let old_elem = elems[i]; + elems[i].0[2] += 1 + i as u32; + assert_eq!(tree.contains(&old_elem), Some(i)); + assert_eq!(tree.contains(&elems[i]), None); + assert!(tree.replace(&old_elem, elems[i])); + assert_eq!(tree.contains(&elems[i]), Some(i)); + assert_eq!(tree.contains(&old_elem), None); + } + let expected = node!( + node!( + node!(leaf!(elems[0]), leaf!(elems[1])), + node!(leaf!(elems[2]), leaf!(elems[3])) + ), + node!( + node!(leaf!(elems[4]), leaf!(elems[5])), + node!(leaf!(elems[6]), leaf!(elems[7])) + ) + ).hash(); + assert_eq!(tree.root_sum(), Some((expected, 28 + 36 * 0x10 + 0x1000))); + assert_eq!(tree.root_sum(), compute_root(elems[0..8].iter())); + assert_eq!(tree.unpruned_len(), 8); + } - let mut rng = thread_rng(); - // If we weren't pruning as we went, try pruning everything now - // and make sure nothing breaks. - if !prune { - rng.shuffle(&mut elems); - let mut expected_count = 8; - let expected_root_sum = tree.root_sum(); - for elem in &elems { - assert_eq!(tree.root_sum(), expected_root_sum); - assert_eq!(tree.len(), expected_count); - assert_eq!(tree.unpruned_len(), 8); - tree.prune(elem); - expected_count -= 1; - } - } + let mut rng = thread_rng(); + // If we weren't pruning as we went, try pruning everything now + // and make sure nothing breaks. + if !prune { + rng.shuffle(&mut elems); + let mut expected_count = 8; + let expected_root_sum = tree.root_sum(); + for elem in &elems { + assert_eq!(tree.root_sum(), expected_root_sum); + assert_eq!(tree.len(), expected_count); + assert_eq!(tree.unpruned_len(), 8); + tree.prune(elem); + expected_count -= 1; + } + } - // Build a large random tree and check its root against that computed - // by `compute_root`. - let mut big_elems: Vec = vec![]; - let mut big_tree = SumTree::new(); - for i in 0..1000 { - // To avoid RNG overflow we generate random elements that are small. - // Though to avoid repeat elements they have to be reasonably big. - let new_elem; - let word1 = rng.gen::() as u32; - let word2 = rng.gen::() as u32; - if rng.gen() { - if rng.gen() { - new_elem = TestElem([word1, word2, 0, 0]); - } else { - new_elem = TestElem([word1, 0, word2, 0]); - } - } else { - if rng.gen() { - new_elem = TestElem([0, word1, 0, word2]); - } else { - new_elem = TestElem([0, 0, word1, word2]); - } - } + // Build a large random tree and check its root against that computed + // by `compute_root`. + let mut big_elems: Vec = vec![]; + let mut big_tree = SumTree::new(); + for i in 0..1000 { + // To avoid RNG overflow we generate random elements that are small. + // Though to avoid repeat elements they have to be reasonably big. + let new_elem; + let word1 = rng.gen::() as u32; + let word2 = rng.gen::() as u32; + if rng.gen() { + if rng.gen() { + new_elem = TestElem([word1, word2, 0, 0]); + } else { + new_elem = TestElem([word1, 0, word2, 0]); + } + } else { + if rng.gen() { + new_elem = TestElem([0, word1, 0, word2]); + } else { + new_elem = TestElem([0, 0, word1, word2]); + } + } - big_elems.push(new_elem); - assert!(big_tree.push(new_elem)); - if i % 25 == 0 { - // Verify root - println!("{}", i); - assert_eq!(big_tree.root_sum(), compute_root(big_elems.iter())); - // Do serialization roundtrip - } - } - } + big_elems.push(new_elem); + assert!(big_tree.push(new_elem)); + if i % 25 == 0 { + // Verify root + println!("{}", i); + assert_eq!(big_tree.root_sum(), compute_root(big_elems.iter())); + // Do serialization roundtrip + } + } + } - #[test] - fn sumtree_create() { - sumtree_create_(false); - sumtree_create_(true); - } + #[test] + fn sumtree_create() { + sumtree_create_(false); + sumtree_create_(true); + } - #[test] - fn sumtree_double_add() { - let elem = TestElem([10, 100, 1000, 10000]); + #[test] + fn sumtree_double_add() { + let elem = TestElem([10, 100, 1000, 10000]); - let mut tree = SumTree::new(); - // Cannot prune a nonexistant element - assert!(!tree.prune(&elem)); - // Can add - assert!(tree.push(elem)); - // Cannot double-add - assert!(!tree.push(elem)); - // Can prune but not double-prune - assert!(tree.prune(&elem)); - assert!(!tree.prune(&elem)); - // Can re-add - assert!(tree.push(elem)); - } + let mut tree = SumTree::new(); + // Cannot prune a nonexistant element + assert!(!tree.prune(&elem)); + // Can add + assert!(tree.push(elem)); + // Cannot double-add + assert!(!tree.push(elem)); + // Can prune but not double-prune + assert!(tree.prune(&elem)); + assert!(!tree.prune(&elem)); + // Can re-add + assert!(tree.push(elem)); + } } - - -