2018-12-04 20:37:12 +00:00
|
|
|
extern crate blake2_rfc;
|
|
|
|
|
|
|
|
use self::blake2_rfc::blake2b::blake2b;
|
|
|
|
use super::ethereum_types::{Address, H256};
|
|
|
|
use super::{ssz_encode, TreeHash};
|
2018-12-08 23:25:59 +00:00
|
|
|
use std::cmp::Ord;
|
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::hash::Hash;
|
2018-12-04 20:37:12 +00:00
|
|
|
|
|
|
|
impl TreeHash for u8 {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for u16 {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for u32 {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for u64 {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for Address {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for H256 {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for [u8] {
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
hash(&self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-08 23:25:59 +00:00
|
|
|
impl<T> TreeHash for Vec<T>
|
|
|
|
where
|
|
|
|
T: TreeHash,
|
|
|
|
{
|
|
|
|
/// Appends the tree_hash for each value of 'self' into a byte array
|
|
|
|
/// and returns the hash of said byte array
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
let mut result = Vec::new();
|
|
|
|
for x in self {
|
|
|
|
result.append(&mut x.tree_hash());
|
|
|
|
}
|
|
|
|
|
|
|
|
hash(&result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<K, V> TreeHash for HashMap<K, V>
|
|
|
|
where
|
|
|
|
K: Eq,
|
|
|
|
K: Hash,
|
|
|
|
K: Ord,
|
|
|
|
V: TreeHash,
|
|
|
|
{
|
|
|
|
/// Appends the tree_hash for each value of 'self, sorted by key,
|
|
|
|
/// into a byte array and returns the hash of said byte array
|
|
|
|
fn tree_hash(&self) -> Vec<u8> {
|
|
|
|
let mut items: Vec<_> = self.iter().collect();
|
|
|
|
items.sort_by(|a, b| a.0.cmp(b.0));
|
|
|
|
let mut result = Vec::new();
|
|
|
|
for item in items {
|
|
|
|
result.append(&mut item.1.tree_hash());
|
|
|
|
}
|
|
|
|
|
|
|
|
hash(&result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// From the Spec:
|
|
|
|
/// We define hash(x) as BLAKE2b-512(x)[0:32]
|
2018-12-04 20:37:12 +00:00
|
|
|
fn hash(data: &[u8]) -> Vec<u8> {
|
|
|
|
let result = blake2b(32, &[], &data);
|
|
|
|
result.as_bytes().to_vec()
|
|
|
|
}
|
2018-12-08 23:25:59 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_impl_tree_hash_vec() {
|
|
|
|
let result = vec![1u32, 2, 3, 4, 5, 6, 7].tree_hash();
|
|
|
|
assert_eq!(result.len(), 32);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_impl_tree_hash_hashmap() {
|
|
|
|
let mut map = HashMap::new();
|
|
|
|
map.insert("c", 3);
|
|
|
|
map.insert("b", 2);
|
|
|
|
map.insert("g", 7);
|
|
|
|
map.insert("d", 6);
|
|
|
|
map.insert("e", 4);
|
|
|
|
map.insert("a", 1u32);
|
|
|
|
map.insert("f", 5);
|
|
|
|
let result = map.tree_hash();
|
|
|
|
|
|
|
|
// TODO: resolve inconsistencies between the python sample code and
|
|
|
|
// the spec; and create tests that tie-out to an offical result
|
|
|
|
assert_eq!(
|
|
|
|
result,
|
|
|
|
[
|
|
|
|
59, 110, 242, 24, 177, 184, 73, 109, 190, 19, 172, 39, 74, 94, 224, 198, 0, 170,
|
|
|
|
225, 152, 249, 59, 10, 76, 137, 124, 52, 159, 37, 42, 26, 157
|
|
|
|
]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|