2018-03-30 10:43:38 -07:00
|
|
|
//! The `hash` module provides functions for creating SHA-256 hashes.
|
2018-03-29 12:20:54 -06:00
|
|
|
|
2018-08-01 12:23:52 -06:00
|
|
|
use bs58;
|
2018-05-11 13:06:05 -06:00
|
|
|
use generic_array::typenum::U32;
|
2018-05-29 21:20:28 -06:00
|
|
|
use generic_array::GenericArray;
|
2018-03-06 17:20:37 -07:00
|
|
|
use sha2::{Digest, Sha256};
|
2018-08-01 12:23:52 -06:00
|
|
|
use std::fmt;
|
2018-03-06 17:20:37 -07:00
|
|
|
|
2018-08-01 12:23:52 -06:00
|
|
|
#[derive(Serialize, Deserialize, Clone, Copy, Default, Eq, PartialEq, Ord, PartialOrd, Hash)]
|
|
|
|
pub struct Hash(GenericArray<u8, U32>);
|
2018-03-06 17:20:37 -07:00
|
|
|
|
2018-09-21 21:01:13 -07:00
|
|
|
#[derive(Clone, Default)]
|
|
|
|
pub struct Hasher {
|
|
|
|
hasher: Sha256,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hasher {
|
2018-12-07 20:01:28 -07:00
|
|
|
pub fn hash(&mut self, val: &[u8]) {
|
2018-09-21 21:01:13 -07:00
|
|
|
self.hasher.input(val);
|
|
|
|
}
|
2018-12-07 20:01:28 -07:00
|
|
|
pub fn hashv(&mut self, vals: &[&[u8]]) {
|
2018-09-21 21:01:13 -07:00
|
|
|
for val in vals {
|
|
|
|
self.hash(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn result(self) -> Hash {
|
|
|
|
// At the time of this writing, the sha2 library is stuck on an old version
|
|
|
|
// of generic_array (0.9.0). Decouple ourselves with a clone to our version.
|
|
|
|
Hash(GenericArray::clone_from_slice(
|
|
|
|
self.hasher.result().as_slice(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-01 12:23:52 -06:00
|
|
|
impl AsRef<[u8]> for Hash {
|
|
|
|
fn as_ref(&self) -> &[u8] {
|
|
|
|
&self.0[..]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for Hash {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "{}", bs58::encode(self.0).into_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Hash {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "{}", bs58::encode(self.0).into_string())
|
|
|
|
}
|
|
|
|
}
|
2018-09-20 15:02:24 -07:00
|
|
|
|
2018-09-20 23:27:06 -06:00
|
|
|
impl Hash {
|
|
|
|
pub fn new(hash_slice: &[u8]) -> Self {
|
|
|
|
Hash(GenericArray::clone_from_slice(&hash_slice))
|
|
|
|
}
|
|
|
|
}
|
2018-03-06 17:20:37 -07:00
|
|
|
/// Return a Sha256 hash for the given data.
|
2018-09-20 15:02:24 -07:00
|
|
|
pub fn hashv(vals: &[&[u8]]) -> Hash {
|
2018-09-21 21:01:13 -07:00
|
|
|
let mut hasher = Hasher::default();
|
|
|
|
hasher.hashv(vals);
|
|
|
|
hasher.result()
|
2018-03-06 17:20:37 -07:00
|
|
|
}
|
|
|
|
|
2018-09-20 15:02:24 -07:00
|
|
|
/// Return a Sha256 hash for the given data.
|
|
|
|
pub fn hash(val: &[u8]) -> Hash {
|
|
|
|
hashv(&[val])
|
|
|
|
}
|
|
|
|
|
2018-03-06 17:20:37 -07:00
|
|
|
/// Return the hash of the given hash extended with the given value.
|
2018-03-06 17:36:45 -07:00
|
|
|
pub fn extend_and_hash(id: &Hash, val: &[u8]) -> Hash {
|
2018-08-01 12:23:52 -06:00
|
|
|
let mut hash_data = id.as_ref().to_vec();
|
2018-03-06 17:20:37 -07:00
|
|
|
hash_data.extend_from_slice(val);
|
|
|
|
hash(&hash_data)
|
|
|
|
}
|