use std::fmt::{Display, Formatter};
use crate::{bytes_data, constants, from_hex, Hash, HashFormatType, SafeBytesAccess};
use crate::structs::{ErrorInfo, HashType};
use sha3::{Digest, Sha3_256};
impl Into<Hash> for Vec<u8> {
fn into(self) -> Hash {
Hash::new(self)
}
}
impl Hash {
pub fn vec(&self) -> Vec<u8> {
self.safe_bytes().expect("a")
}
pub fn hex(&self) -> String {
hex::encode(self.vec())
}
pub fn new(vec: Vec<u8>) -> Self {
Self {
bytes: bytes_data(vec),
hash_format_type: HashFormatType::Sha3256 as i32,
hash_type: HashType::Transaction as i32,
}
}
pub fn validate_size(&self) -> Result<&Self, ErrorInfo> {
if self.safe_bytes()?.len() == 32 {
Ok(self)
} else {
Err(ErrorInfo::error_info("Invalid hash size"))
}
}
pub fn from_hex<S: Into<String>>(s: S) -> Result<Self, ErrorInfo> {
let hash = Self::new(from_hex(s.into())?);
hash.validate_size()?;
Ok(hash)
}
pub fn from_string_calculate(s: &str) -> Self {
Self::digest(s.as_bytes().to_vec())
}
pub fn digest(s: Vec<u8>) -> Self {
Self::new(Sha3_256::digest(&s).to_vec())
}
pub fn merkle_combine(&self, right: Hash) -> Self {
let mut vec = self.vec();
vec.extend(right.vec());
Self::digest(vec)
}
pub fn checksum(&self) -> Result<Vec<u8>, ErrorInfo> {
Ok(self.safe_bytes()?[0..4].to_vec())
}
pub fn checksum_hex(&self) -> Result<String, ErrorInfo> {
Ok(hex::encode(self.checksum()?))
}
}
#[test]
fn hash_rendering() {
let h = Hash::from_string_calculate("test");
println!("hash: {}", h.hex());
}