2016-02-05 13:40:41 +01:00
|
|
|
// Copyright 2015, 2016 Ethcore (UK) Ltd.
|
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2016-01-15 14:40:51 +01:00
|
|
|
use common::*;
|
2015-12-19 13:35:26 +01:00
|
|
|
use hashdb::*;
|
|
|
|
use nibbleslice::*;
|
|
|
|
use rlp::*;
|
2016-06-07 20:44:09 +02:00
|
|
|
use super::node::Node;
|
2016-08-24 16:53:36 +02:00
|
|
|
use super::recorder::{Recorder, NoOp};
|
2016-08-03 18:35:48 +02:00
|
|
|
use super::{Trie, TrieItem, TrieError};
|
2015-12-19 13:35:26 +01:00
|
|
|
|
|
|
|
/// A `Trie` implementation using a generic `HashDB` backing database.
|
2016-03-07 14:33:00 +01:00
|
|
|
///
|
2015-12-19 13:35:26 +01:00
|
|
|
/// Use it as a `Trie` trait object. You can use `db()` to get the backing database object, `keys`
|
|
|
|
/// to get the keys belonging to the trie in the backing database, and `db_items_remaining()` to get
|
|
|
|
/// which items in the backing database do not belong to this trie. If this is the only trie in the
|
|
|
|
/// backing database, then `db_items_remaining()` should be empty.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
/// ```
|
|
|
|
/// extern crate ethcore_util as util;
|
|
|
|
/// use util::trie::*;
|
|
|
|
/// use util::hashdb::*;
|
|
|
|
/// use util::memorydb::*;
|
|
|
|
/// use util::hash::*;
|
|
|
|
/// use util::rlp::*;
|
|
|
|
///
|
|
|
|
/// fn main() {
|
|
|
|
/// let mut memdb = MemoryDB::new();
|
|
|
|
/// let mut root = H256::new();
|
2016-08-03 18:35:48 +02:00
|
|
|
/// TrieDBMut::new(&mut memdb, &mut root).insert(b"foo", b"bar").unwrap();
|
2016-06-07 20:44:09 +02:00
|
|
|
/// let t = TrieDB::new(&memdb, &root).unwrap();
|
2016-08-03 18:35:48 +02:00
|
|
|
/// assert!(t.contains(b"foo").unwrap());
|
|
|
|
/// assert_eq!(t.get(b"foo").unwrap().unwrap(), b"bar");
|
|
|
|
/// assert!(t.db_items_remaining().unwrap().is_empty());
|
2015-12-19 13:35:26 +01:00
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
pub struct TrieDB<'db> {
|
|
|
|
db: &'db HashDB,
|
|
|
|
root: &'db H256,
|
2016-02-03 14:51:45 +01:00
|
|
|
/// The number of hashes performed so far in operations on this trie.
|
2015-12-19 13:35:26 +01:00
|
|
|
pub hash_count: usize,
|
|
|
|
}
|
|
|
|
|
2016-03-11 11:16:49 +01:00
|
|
|
#[cfg_attr(feature="dev", allow(wrong_self_convention))]
|
2015-12-19 13:35:26 +01:00
|
|
|
impl<'db> TrieDB<'db> {
|
|
|
|
/// Create a new trie with the backing database `db` and `root`
|
2016-06-07 20:44:09 +02:00
|
|
|
/// Returns an error if `root` does not exist
|
2016-08-03 18:35:48 +02:00
|
|
|
pub fn new(db: &'db HashDB, root: &'db H256) -> super::Result<Self> {
|
2016-06-02 21:01:47 +02:00
|
|
|
if !db.contains(root) {
|
2016-08-03 18:35:48 +02:00
|
|
|
Err(Box::new(TrieError::InvalidStateRoot(*root)))
|
2016-06-07 20:44:09 +02:00
|
|
|
} else {
|
|
|
|
Ok(TrieDB {
|
|
|
|
db: db,
|
|
|
|
root: root,
|
|
|
|
hash_count: 0
|
|
|
|
})
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the backing database.
|
2016-03-07 14:33:00 +01:00
|
|
|
pub fn db(&'db self) -> &'db HashDB {
|
|
|
|
self.db
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine all the keys in the backing database that belong to the trie.
|
2016-08-03 18:35:48 +02:00
|
|
|
pub fn keys(&self) -> super::Result<Vec<H256>> {
|
2015-12-19 13:35:26 +01:00
|
|
|
let mut ret: Vec<H256> = Vec::new();
|
|
|
|
ret.push(self.root.clone());
|
2016-08-24 16:53:36 +02:00
|
|
|
try!(self.accumulate_keys(try!(self.root_node(&mut NoOp)), &mut ret));
|
2016-08-03 18:35:48 +02:00
|
|
|
Ok(ret)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
2016-07-19 20:42:23 +02:00
|
|
|
/// Convert a vector of hashes to a hashmap of hash to occurrences.
|
2015-12-19 13:35:26 +01:00
|
|
|
pub fn to_map(hashes: Vec<H256>) -> HashMap<H256, u32> {
|
|
|
|
let mut r: HashMap<H256, u32> = HashMap::new();
|
|
|
|
for h in hashes.into_iter() {
|
2016-07-14 18:06:46 +02:00
|
|
|
*r.entry(h).or_insert(0) += 1;
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
2016-07-19 20:42:23 +02:00
|
|
|
/// Determine occurrences of items in the backing database which are not related to this
|
2015-12-19 13:35:26 +01:00
|
|
|
/// trie.
|
2016-08-03 18:35:48 +02:00
|
|
|
pub fn db_items_remaining(&self) -> super::Result<HashMap<H256, i32>> {
|
2015-12-19 13:35:26 +01:00
|
|
|
let mut ret = self.db.keys();
|
2016-08-03 18:35:48 +02:00
|
|
|
for (k, v) in Self::to_map(try!(self.keys())).into_iter() {
|
2015-12-19 13:35:26 +01:00
|
|
|
let keycount = *ret.get(&k).unwrap_or(&0);
|
2016-01-15 04:02:24 +01:00
|
|
|
match keycount <= v as i32 {
|
2015-12-19 13:35:26 +01:00
|
|
|
true => ret.remove(&k),
|
|
|
|
_ => ret.insert(k, keycount - v as i32),
|
|
|
|
};
|
|
|
|
}
|
2016-08-03 18:35:48 +02:00
|
|
|
Ok(ret)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Recursion helper for `keys`.
|
2016-08-03 18:35:48 +02:00
|
|
|
fn accumulate_keys(&self, node: Node, acc: &mut Vec<H256>) -> super::Result<()> {
|
2015-12-19 13:35:26 +01:00
|
|
|
let mut handle_payload = |payload| {
|
|
|
|
let p = Rlp::new(payload);
|
|
|
|
if p.is_data() && p.size() == 32 {
|
|
|
|
acc.push(p.as_val());
|
|
|
|
}
|
|
|
|
|
2016-08-24 16:53:36 +02:00
|
|
|
self.accumulate_keys(try!(self.get_node(payload, &mut NoOp, 0)), acc)
|
2015-12-19 13:35:26 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
match node {
|
2016-08-03 18:35:48 +02:00
|
|
|
Node::Extension(_, payload) => try!(handle_payload(payload)),
|
|
|
|
Node::Branch(payloads, _) => for payload in &payloads { try!(handle_payload(payload)) },
|
2015-12-19 13:35:26 +01:00
|
|
|
_ => {},
|
|
|
|
}
|
2016-08-03 18:35:48 +02:00
|
|
|
|
|
|
|
Ok(())
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the root node's RLP.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn root_node<'a, R: 'a + Recorder>(&self, r: &'a mut R) -> super::Result<Node> {
|
|
|
|
self.root_data(r).map(Node::decoded)
|
2016-01-17 22:24:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the data of the root node.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn root_data<'a, R: 'a + Recorder>(&self, r: &'a mut R) -> super::Result<&[u8]> {
|
2016-08-03 18:35:48 +02:00
|
|
|
self.db.get(self.root).ok_or_else(|| Box::new(TrieError::InvalidStateRoot(*self.root)))
|
2016-08-24 16:53:36 +02:00
|
|
|
.map(|node| { r.record(self.root, node, 0); node })
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the root node as a `Node`.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn get_node<'a, R: 'a + Recorder>(&'db self, node: &'db [u8], r: &'a mut R, depth: u32) -> super::Result<Node> {
|
|
|
|
self.get_raw_or_lookup(node, r, depth).map(Node::decoded)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Indentation helper for `formal_all`.
|
|
|
|
fn fmt_indent(&self, f: &mut fmt::Formatter, size: usize) -> fmt::Result {
|
2016-03-07 14:33:00 +01:00
|
|
|
for _ in 0..size {
|
2015-12-19 13:35:26 +01:00
|
|
|
try!(write!(f, " "));
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Recursion helper for implementation of formatting trait.
|
|
|
|
fn fmt_all(&self, node: Node, f: &mut fmt::Formatter, deepness: usize) -> fmt::Result {
|
|
|
|
match node {
|
|
|
|
Node::Leaf(slice, value) => try!(writeln!(f, "'{:?}: {:?}.", slice, value.pretty())),
|
2016-08-17 16:06:41 +02:00
|
|
|
Node::Extension(ref slice, item) => {
|
2015-12-19 13:35:26 +01:00
|
|
|
try!(write!(f, "'{:?} ", slice));
|
2016-08-24 16:53:36 +02:00
|
|
|
if let Ok(node) = self.get_node(item, &mut NoOp, 0) {
|
2016-08-03 18:35:48 +02:00
|
|
|
try!(self.fmt_all(node, f, deepness));
|
|
|
|
}
|
2015-12-19 13:35:26 +01:00
|
|
|
},
|
|
|
|
Node::Branch(ref nodes, ref value) => {
|
|
|
|
try!(writeln!(f, ""));
|
2016-01-19 12:14:29 +01:00
|
|
|
if let Some(v) = *value {
|
|
|
|
try!(self.fmt_indent(f, deepness + 1));
|
|
|
|
try!(writeln!(f, "=: {:?}", v.pretty()))
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
for i in 0..16 {
|
2016-08-24 16:53:36 +02:00
|
|
|
match self.get_node(nodes[i], &mut NoOp, 0) {
|
2016-08-03 18:35:48 +02:00
|
|
|
Ok(Node::Empty) => {},
|
|
|
|
Ok(n) => {
|
2015-12-19 13:35:26 +01:00
|
|
|
try!(self.fmt_indent(f, deepness + 1));
|
|
|
|
try!(write!(f, "'{:x} ", i));
|
|
|
|
try!(self.fmt_all(n, f, deepness + 1));
|
|
|
|
}
|
2016-08-03 18:35:48 +02:00
|
|
|
Err(e) => {
|
|
|
|
try!(write!(f, "ERROR: {}", e));
|
|
|
|
}
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// empty
|
|
|
|
Node::Empty => {
|
|
|
|
try!(writeln!(f, "<empty>"));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return optional data for a key given as a `NibbleSlice`. Returns `None` if no data exists.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn do_lookup<'key, R: 'key>(&'db self, key: &NibbleSlice<'key>, r: &'key mut R) -> super::Result<Option<&'db [u8]>>
|
|
|
|
where 'db: 'key, R: Recorder
|
2016-08-03 18:35:48 +02:00
|
|
|
{
|
2016-08-24 16:53:36 +02:00
|
|
|
let root_rlp = try!(self.root_data(r));
|
|
|
|
self.get_from_node(root_rlp, key, r, 1)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Recursible function to retrieve the value given a `node` and a partial `key`. `None` if no
|
|
|
|
/// value exists for the key.
|
|
|
|
///
|
|
|
|
/// Note: Not a public API; use Trie trait functions.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn get_from_node<'key, R: 'key>(
|
|
|
|
&'db self,
|
|
|
|
node: &'db [u8],
|
|
|
|
key: &NibbleSlice<'key>,
|
|
|
|
r: &'key mut R,
|
|
|
|
d: u32
|
|
|
|
) -> super::Result<Option<&'db [u8]>> where 'db: 'key, R: Recorder {
|
2015-12-19 13:35:26 +01:00
|
|
|
match Node::decoded(node) {
|
2016-08-17 16:06:41 +02:00
|
|
|
Node::Leaf(ref slice, value) if key == slice => Ok(Some(value)),
|
|
|
|
Node::Extension(ref slice, item) if key.starts_with(slice) => {
|
2016-08-24 16:53:36 +02:00
|
|
|
let data = try!(self.get_raw_or_lookup(item, r, d));
|
|
|
|
self.get_from_node(data, &key.mid(slice.len()), r, d + 1)
|
2015-12-19 13:35:26 +01:00
|
|
|
},
|
|
|
|
Node::Branch(ref nodes, value) => match key.is_empty() {
|
2016-08-03 18:35:48 +02:00
|
|
|
true => Ok(value),
|
2016-08-24 16:53:36 +02:00
|
|
|
false => {
|
|
|
|
let node = try!(self.get_raw_or_lookup(nodes[key.at(0) as usize], r, d));
|
|
|
|
self.get_from_node(node, &key.mid(1), r, d + 1)
|
|
|
|
}
|
2015-12-19 13:35:26 +01:00
|
|
|
},
|
2016-08-03 18:35:48 +02:00
|
|
|
_ => Ok(None)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Given some node-describing data `node`, return the actual node RLP.
|
|
|
|
/// This could be a simple identity operation in the case that the node is sufficiently small, but
|
|
|
|
/// may require a database lookup.
|
2016-08-24 16:53:36 +02:00
|
|
|
fn get_raw_or_lookup<R: Recorder>(&'db self, node: &'db [u8], rec: &mut R, d: u32) -> super::Result<&'db [u8]> {
|
2015-12-19 13:35:26 +01:00
|
|
|
// check if its sha3 + len
|
|
|
|
let r = Rlp::new(node);
|
|
|
|
match r.is_data() && r.size() == 32 {
|
2016-08-03 18:35:48 +02:00
|
|
|
true => {
|
|
|
|
let key = r.as_val::<H256>();
|
|
|
|
self.db.get(&key).ok_or_else(|| Box::new(TrieError::IncompleteDatabase(key)))
|
2016-08-24 16:53:36 +02:00
|
|
|
.map(|raw| { rec.record(&key, raw, d); raw })
|
2016-08-03 18:35:48 +02:00
|
|
|
}
|
|
|
|
false => Ok(node)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-17 22:18:47 +01:00
|
|
|
#[derive(Clone, Eq, PartialEq)]
|
|
|
|
enum Status {
|
|
|
|
Entering,
|
|
|
|
At,
|
|
|
|
AtChild(usize),
|
|
|
|
Exiting,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Eq, PartialEq)]
|
|
|
|
struct Crumb<'a> {
|
|
|
|
node: Node<'a>,
|
|
|
|
status: Status,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Crumb<'a> {
|
|
|
|
/// Move on to next status in the node's sequence.
|
|
|
|
fn increment(&mut self) {
|
|
|
|
self.status = match (&self.status, &self.node) {
|
|
|
|
(_, &Node::Empty) => Status::Exiting,
|
|
|
|
(&Status::Entering, _) => Status::At,
|
|
|
|
(&Status::At, &Node::Branch(_, _)) => Status::AtChild(0),
|
|
|
|
(&Status::AtChild(x), &Node::Branch(_, _)) if x < 15 => Status::AtChild(x + 1),
|
|
|
|
_ => Status::Exiting,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Iterator for going through all values in the trie.
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct TrieDBIterator<'a> {
|
|
|
|
db: &'a TrieDB<'a>,
|
|
|
|
trail: Vec<Crumb<'a>>,
|
|
|
|
key_nibbles: Bytes,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> TrieDBIterator<'a> {
|
|
|
|
/// Create a new iterator.
|
2016-06-27 09:16:34 +02:00
|
|
|
pub fn new(db: &'a TrieDB) -> TrieDBIterator<'a> {
|
2016-01-17 22:18:47 +01:00
|
|
|
let mut r = TrieDBIterator {
|
|
|
|
db: db,
|
|
|
|
trail: vec![],
|
|
|
|
key_nibbles: Vec::new(),
|
|
|
|
};
|
2016-08-24 16:53:36 +02:00
|
|
|
r.descend(db.root_data(&mut NoOp).unwrap());
|
2016-01-17 22:18:47 +01:00
|
|
|
r
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Descend into a payload.
|
|
|
|
fn descend(&mut self, d: &'a [u8]) {
|
|
|
|
self.trail.push(Crumb {
|
|
|
|
status: Status::Entering,
|
2016-08-24 16:53:36 +02:00
|
|
|
node: self.db.get_node(d, &mut NoOp, 0).unwrap(),
|
2016-01-17 22:18:47 +01:00
|
|
|
});
|
|
|
|
match self.trail.last().unwrap().node {
|
2016-01-18 00:51:55 +01:00
|
|
|
Node::Leaf(n, _) | Node::Extension(n, _) => { self.key_nibbles.extend(n.iter()); },
|
2016-01-17 22:18:47 +01:00
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Descend into a payload and get the next item.
|
|
|
|
fn descend_next(&mut self, d: &'a [u8]) -> Option<(Bytes, &'a [u8])> { self.descend(d); self.next() }
|
|
|
|
|
|
|
|
/// The present key.
|
|
|
|
fn key(&self) -> Bytes {
|
|
|
|
// collapse the key_nibbles down to bytes.
|
|
|
|
self.key_nibbles.iter().step(2).zip(self.key_nibbles.iter().skip(1).step(2)).map(|(h, l)| h * 16 + l).collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Iterator for TrieDBIterator<'a> {
|
|
|
|
type Item = (Bytes, &'a [u8]);
|
|
|
|
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
|
|
let b = match self.trail.last_mut() {
|
2016-02-14 12:54:27 +01:00
|
|
|
Some(mut b) => { b.increment(); b.clone() },
|
2016-01-17 22:18:47 +01:00
|
|
|
None => return None
|
|
|
|
};
|
|
|
|
match (b.status, b.node) {
|
|
|
|
(Status::Exiting, n) => {
|
|
|
|
match n {
|
|
|
|
Node::Leaf(n, _) | Node::Extension(n, _) => {
|
|
|
|
let l = self.key_nibbles.len();
|
|
|
|
self.key_nibbles.truncate(l - n.len());
|
|
|
|
},
|
2016-01-18 00:51:55 +01:00
|
|
|
Node::Branch(_, _) => { self.key_nibbles.pop(); },
|
2016-01-17 22:18:47 +01:00
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
self.trail.pop();
|
|
|
|
self.next()
|
|
|
|
},
|
2016-02-14 12:54:27 +01:00
|
|
|
(Status::At, Node::Leaf(_, v)) | (Status::At, Node::Branch(_, Some(v))) => Some((self.key(), v)),
|
2016-01-17 22:18:47 +01:00
|
|
|
(Status::At, Node::Extension(_, d)) => self.descend_next(d),
|
|
|
|
(Status::At, Node::Branch(_, _)) => self.next(),
|
|
|
|
(Status::AtChild(i), Node::Branch(children, _)) if children[i].len() > 0 => {
|
|
|
|
match i {
|
2016-01-18 00:51:55 +01:00
|
|
|
0 => self.key_nibbles.push(0),
|
2016-01-17 22:18:47 +01:00
|
|
|
i => *self.key_nibbles.last_mut().unwrap() = i as u8,
|
|
|
|
}
|
|
|
|
self.descend_next(children[i])
|
|
|
|
},
|
|
|
|
(Status::AtChild(i), Node::Branch(_, _)) => {
|
2016-01-18 00:51:55 +01:00
|
|
|
if i == 0 { self.key_nibbles.push(0); }
|
2016-01-17 22:18:47 +01:00
|
|
|
self.next()
|
|
|
|
},
|
|
|
|
_ => panic!() // Should never see Entering or AtChild without a Branch here.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-19 13:35:26 +01:00
|
|
|
impl<'db> Trie for TrieDB<'db> {
|
2016-07-04 13:53:55 +02:00
|
|
|
fn iter<'a>(&'a self) -> Box<Iterator<Item = TrieItem> + 'a> {
|
2016-08-24 16:53:36 +02:00
|
|
|
Box::new(TrieDBIterator::new(self))
|
2016-06-27 10:59:59 +02:00
|
|
|
}
|
|
|
|
|
2016-07-26 20:31:25 +02:00
|
|
|
fn root(&self) -> &H256 { self.root }
|
2015-12-19 13:35:26 +01:00
|
|
|
|
2016-08-24 16:53:36 +02:00
|
|
|
fn get_recorded<'a, 'b, R: 'b>(&'a self, key: &'b [u8], rec: &'b mut R) -> super::Result<Option<&'a [u8]>>
|
|
|
|
where 'a: 'b, R: Recorder
|
2016-08-03 18:35:48 +02:00
|
|
|
{
|
2016-08-24 16:53:36 +02:00
|
|
|
self.do_lookup(&NibbleSlice::new(key), rec)
|
2015-12-19 13:35:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'db> fmt::Debug for TrieDB<'db> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
try!(writeln!(f, "c={:?} [", self.hash_count));
|
2016-07-26 20:31:25 +02:00
|
|
|
let root_rlp = self.db.get(self.root).expect("Trie root not found!");
|
2015-12-19 13:35:26 +01:00
|
|
|
try!(self.fmt_all(Node::decoded(root_rlp), f, 0));
|
|
|
|
writeln!(f, "]")
|
|
|
|
}
|
|
|
|
}
|
2016-01-17 22:18:47 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn iterator() {
|
|
|
|
use memorydb::*;
|
2016-08-03 18:35:48 +02:00
|
|
|
use super::TrieMut;
|
2016-01-17 22:18:47 +01:00
|
|
|
use super::triedbmut::*;
|
2016-03-07 14:33:00 +01:00
|
|
|
|
2016-01-17 22:18:47 +01:00
|
|
|
let d = vec![ &b"A"[..], &b"AA"[..], &b"AB"[..], &b"B"[..] ];
|
|
|
|
|
|
|
|
let mut memdb = MemoryDB::new();
|
|
|
|
let mut root = H256::new();
|
|
|
|
{
|
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
|
|
|
for x in &d {
|
2016-08-03 18:35:48 +02:00
|
|
|
t.insert(x, x).unwrap();
|
2016-01-17 22:18:47 +01:00
|
|
|
}
|
|
|
|
}
|
2016-08-24 16:53:36 +02:00
|
|
|
|
|
|
|
let t = TrieDB::new(&memdb, &root).unwrap();
|
|
|
|
assert_eq!(d.iter().map(|i|i.to_vec()).collect::<Vec<_>>(), t.iter().map(|x|x.0).collect::<Vec<_>>());
|
|
|
|
assert_eq!(d, t.iter().map(|x|x.1).collect::<Vec<_>>());
|
2016-01-17 22:18:47 +01:00
|
|
|
}
|