2018-06-04 10:19:50 +02:00
|
|
|
// Copyright 2015-2018 Parity Technologies (UK) Ltd.
|
2017-01-06 16:18:45 +01:00
|
|
|
// 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/>.
|
|
|
|
|
|
|
|
//! Trie lookup via HashDB.
|
|
|
|
|
|
|
|
use hashdb::HashDB;
|
|
|
|
use nibbleslice::NibbleSlice;
|
2018-01-10 13:35:18 +01:00
|
|
|
use ethereum_types::H256;
|
2017-01-06 16:18:45 +01:00
|
|
|
|
|
|
|
use super::{TrieError, Query};
|
|
|
|
use super::node::Node;
|
|
|
|
|
|
|
|
/// Trie lookup helper object.
|
|
|
|
pub struct Lookup<'a, Q: Query> {
|
|
|
|
/// database to query from.
|
|
|
|
pub db: &'a HashDB,
|
|
|
|
/// Query object to record nodes and transform data.
|
|
|
|
pub query: Q,
|
|
|
|
/// Hash to start at
|
|
|
|
pub hash: H256,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, Q: Query> Lookup<'a, Q> {
|
|
|
|
/// Look up the given key. If the value is found, it will be passed to the given
|
|
|
|
/// function to decode or copy.
|
|
|
|
pub fn look_up(mut self, mut key: NibbleSlice) -> super::Result<Option<Q::Item>> {
|
|
|
|
let mut hash = self.hash;
|
|
|
|
|
|
|
|
// this loop iterates through non-inline nodes.
|
|
|
|
for depth in 0.. {
|
|
|
|
let node_data = match self.db.get(&hash) {
|
|
|
|
Some(value) => value,
|
|
|
|
None => return Err(Box::new(match depth {
|
|
|
|
0 => TrieError::InvalidStateRoot(hash),
|
|
|
|
_ => TrieError::IncompleteDatabase(hash),
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
|
|
|
|
self.query.record(&hash, &node_data, depth);
|
|
|
|
|
|
|
|
// this loop iterates through all inline children (usually max 1)
|
|
|
|
// without incrementing the depth.
|
|
|
|
let mut node_data = &node_data[..];
|
|
|
|
loop {
|
2018-05-04 10:06:54 +02:00
|
|
|
match Node::decoded(node_data)? {
|
2017-01-06 16:18:45 +01:00
|
|
|
Node::Leaf(slice, value) => {
|
|
|
|
return Ok(match slice == key {
|
|
|
|
true => Some(self.query.decode(value)),
|
|
|
|
false => None,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
Node::Extension(slice, item) => {
|
|
|
|
if key.starts_with(&slice) {
|
|
|
|
node_data = item;
|
|
|
|
key = key.mid(slice.len());
|
|
|
|
} else {
|
|
|
|
return Ok(None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Node::Branch(children, value) => match key.is_empty() {
|
|
|
|
true => return Ok(value.map(move |val| self.query.decode(val))),
|
|
|
|
false => {
|
|
|
|
node_data = children[key.at(0) as usize];
|
|
|
|
key = key.mid(1);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => return Ok(None),
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if new node data is inline or hash.
|
2018-03-22 03:08:48 +01:00
|
|
|
if let Some(h) = Node::try_decode_hash(&node_data) {
|
|
|
|
hash = h;
|
2017-01-06 16:18:45 +01:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
}
|