mirror of
https://github.com/sigp/lighthouse.git
synced 2026-03-20 05:14:35 +00:00
Minor updates to TreeHash
This commit is contained in:
@@ -1,123 +1,63 @@
|
||||
extern crate hashing;
|
||||
|
||||
use self::hashing::canonical_hash;
|
||||
use super::ethereum_types::{Address, H256};
|
||||
use super::{merkle_hash, ssz_encode, TreeHash};
|
||||
use std::cmp::Ord;
|
||||
use std::collections::HashMap;
|
||||
use std::hash::Hash;
|
||||
|
||||
impl TreeHash for u8 {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for u16 {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for u32 {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for u64 {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for Address {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for H256 {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
ssz_encode(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl TreeHash for [u8] {
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
hash(&self)
|
||||
}
|
||||
}
|
||||
|
||||
impl<T> TreeHash for Vec<T>
|
||||
where
|
||||
T: TreeHash,
|
||||
{
|
||||
/// Returns the merkle_hash of a list of tree_hash values created
|
||||
/// Returns the merkle_hash of a list of hash_tree_root values created
|
||||
/// from the given list.
|
||||
/// Note: A byte vector, Vec<u8>, must be converted to a slice (as_slice())
|
||||
/// to be handled properly (i.e. hashed) as byte array.
|
||||
fn tree_hash(&self) -> Vec<u8> {
|
||||
let mut tree_hashes = self.iter().map(|x| x.tree_hash()).collect();
|
||||
fn hash_tree_root(&self) -> Vec<u8> {
|
||||
let mut tree_hashes = self.iter().map(|x| x.hash_tree_root()).collect();
|
||||
merkle_hash(&mut tree_hashes)
|
||||
}
|
||||
}
|
||||
|
||||
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)
|
||||
}
|
||||
}
|
||||
|
||||
fn hash(data: &[u8]) -> Vec<u8> {
|
||||
canonical_hash(data)
|
||||
}
|
||||
|
||||
#[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();
|
||||
let result = vec![1u32, 2, 3, 4, 5, 6, 7].hash_tree_root();
|
||||
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: create tests that tie-out to an offical result
|
||||
assert_eq!(
|
||||
result,
|
||||
[
|
||||
232, 63, 235, 91, 115, 69, 159, 54, 95, 239, 147, 30, 179, 96, 232, 210, 225, 31,
|
||||
12, 95, 149, 104, 134, 158, 45, 51, 20, 101, 202, 164, 200, 163
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user