use core::{
fmt::{self, Debug, Display},
str::FromStr,
};
use bytes::Bytes;
use serde::{de::Error as _, Deserialize, Deserializer, Serialize, Serializer};
use subtle_encoding::{base64, Encoding, Hex};
use tendermint_proto::Protobuf;
use crate::{error::Error, prelude::*};
pub const SHA256_HASH_SIZE: usize = 32;
#[derive(Copy, Clone, Debug, Eq, Hash, PartialEq)]
pub enum Algorithm {
Sha256,
}
#[derive(Copy, Clone, Hash, Eq, PartialEq, PartialOrd, Ord, Default)]
pub enum Hash {
Sha256([u8; SHA256_HASH_SIZE]),
#[default]
None,
}
impl Protobuf<Vec<u8>> for Hash {}
impl TryFrom<Vec<u8>> for Hash {
type Error = Error;
fn try_from(value: Vec<u8>) -> Result<Self, Self::Error> {
if value.is_empty() {
return Ok(Hash::None);
}
Hash::from_bytes(Algorithm::Sha256, &value)
}
}
impl From<Hash> for Vec<u8> {
fn from(value: Hash) -> Self {
match value {
Hash::Sha256(s) => s.to_vec(),
Hash::None => vec![],
}
}
}
impl AsRef<[u8]> for Hash {
fn as_ref(&self) -> &[u8] {
match self {
Hash::Sha256(ref h) => h.as_ref(),
Hash::None => &[],
}
}
}
impl From<Hash> for Bytes {
fn from(h: Hash) -> Self {
Self::copy_from_slice(h.as_ref())
}
}
impl TryFrom<Bytes> for Hash {
type Error = Error;
fn try_from(value: Bytes) -> Result<Self, Self::Error> {
Self::from_bytes(Algorithm::Sha256, value.as_ref())
}
}
impl Hash {
pub fn from_bytes(alg: Algorithm, bytes: &[u8]) -> Result<Hash, Error> {
if bytes.is_empty() {
return Ok(Hash::None);
}
match alg {
Algorithm::Sha256 => {
if bytes.len() == SHA256_HASH_SIZE {
let mut h = [0u8; SHA256_HASH_SIZE];
h.copy_from_slice(bytes);
Ok(Hash::Sha256(h))
} else {
Err(Error::invalid_hash_size())
}
},
}
}
pub fn from_hex_upper(alg: Algorithm, s: &str) -> Result<Hash, Error> {
if s.is_empty() {
return Ok(Hash::None);
}
match alg {
Algorithm::Sha256 => {
let mut h = [0u8; SHA256_HASH_SIZE];
Hex::upper_case()
.decode_to_slice(s.as_bytes(), &mut h)
.map_err(Error::subtle_encoding)?;
Ok(Hash::Sha256(h))
},
}
}
pub fn algorithm(self) -> Algorithm {
match self {
Hash::Sha256(_) => Algorithm::Sha256,
Hash::None => Algorithm::Sha256,
}
}
pub fn as_bytes(&self) -> &[u8] {
match self {
Hash::Sha256(ref h) => h.as_ref(),
Hash::None => &[],
}
}
pub fn is_empty(&self) -> bool {
self == &Hash::None
}
}
impl Debug for Hash {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Hash::Sha256(_) => write!(f, "Hash::Sha256({self})"),
Hash::None => write!(f, "Hash::None"),
}
}
}
impl Display for Hash {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let hex = match self {
Hash::Sha256(ref h) => Hex::upper_case().encode_to_string(h).unwrap(),
Hash::None => String::new(),
};
write!(f, "{hex}")
}
}
impl FromStr for Hash {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
Self::from_hex_upper(Algorithm::Sha256, s)
}
}
impl<'de> Deserialize<'de> for Hash {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
let hex = <&str>::deserialize(deserializer)?;
if hex.is_empty() {
Err(D::Error::custom("empty hash"))
} else {
Ok(Self::from_str(hex).map_err(|e| D::Error::custom(format!("{e}")))?)
}
}
}
impl Serialize for Hash {
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
self.to_string().serialize(serializer)
}
}
pub mod allow_empty {
use super::*;
pub fn serialize<S>(value: &Hash, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
value.to_string().serialize(serializer)
}
pub fn deserialize<'de, D>(deserializer: D) -> Result<Hash, D::Error>
where
D: Deserializer<'de>,
{
let hex = <&str>::deserialize(deserializer)?;
Hash::from_str(hex).map_err(serde::de::Error::custom)
}
}
#[derive(Clone, PartialEq, Eq, Default)]
pub struct AppHash(Vec<u8>);
impl Protobuf<Vec<u8>> for AppHash {}
impl TryFrom<Vec<u8>> for AppHash {
type Error = Error;
fn try_from(value: Vec<u8>) -> Result<Self, Self::Error> {
Ok(AppHash(value))
}
}
impl From<AppHash> for Vec<u8> {
fn from(value: AppHash) -> Self {
value.0
}
}
impl TryFrom<Bytes> for AppHash {
type Error = Error;
fn try_from(value: Bytes) -> Result<Self, Self::Error> {
Ok(AppHash(value.to_vec()))
}
}
impl From<AppHash> for Bytes {
fn from(value: AppHash) -> Self {
value.0.into()
}
}
impl AppHash {
pub fn as_bytes(&self) -> &[u8] {
&self.0
}
pub fn from_hex_upper(s: &str) -> Result<Self, Error> {
if s.len() % 2 != 0 {
return Err(Error::invalid_app_hash_length());
}
let mut h = vec![0; s.len() / 2];
Hex::upper_case()
.decode_to_slice(s.as_bytes(), &mut h)
.map_err(Error::subtle_encoding)?;
Ok(AppHash(h))
}
pub fn from_base64(s: &str) -> Result<Self, Error> {
let h = base64::decode(s).map_err(Error::subtle_encoding)?;
Ok(AppHash(h))
}
}
impl AsRef<[u8]> for AppHash {
fn as_ref(&self) -> &[u8] {
self.0.as_ref()
}
}
impl Debug for AppHash {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"AppHash({})",
Hex::upper_case().encode_to_string(&self.0).unwrap()
)
}
}
impl Display for AppHash {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"{}",
Hex::upper_case().encode_to_string(&self.0).unwrap()
)
}
}
impl FromStr for AppHash {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
Self::from_hex_upper(s).or_else(|_| Self::from_base64(s))
}
}
#[cfg(test)]
mod tests {
use super::*;
#[derive(Debug, serde::Deserialize)]
struct Test {
#[serde(default)]
#[serde(with = "crate::serializers::apphash")]
pub app_hash: AppHash,
}
#[test]
fn apphash_decode_base64() {
let test = serde_json::from_str::<Test>(
r#"{"app_hash":"MfX9f+bYoI8IioRb4YT/8/VhPvtNjgWFgTi4mmMSkBc="}"#,
)
.unwrap();
assert_eq!(
test.app_hash.as_ref(),
&[
0x31, 0xF5, 0xFD, 0x7F, 0xE6, 0xD8, 0xA0, 0x8F, 0x08, 0x8A, 0x84, 0x5B, 0xE1, 0x84,
0xFF, 0xF3, 0xF5, 0x61, 0x3E, 0xFB, 0x4D, 0x8E, 0x05, 0x85, 0x81, 0x38, 0xB8, 0x9A,
0x63, 0x12, 0x90, 0x17
]
);
}
#[test]
fn apphash_decode_hex() {
let test = serde_json::from_str::<Test>(
r#"{"app_hash":"31F5FD7FE6D8A08F088A845BE184FFF3F5613EFB4D8E05858138B89A63129017"}"#,
)
.unwrap();
assert_eq!(
test.app_hash.as_ref(),
&[
0x31, 0xF5, 0xFD, 0x7F, 0xE6, 0xD8, 0xA0, 0x8F, 0x08, 0x8A, 0x84, 0x5B, 0xE1, 0x84,
0xFF, 0xF3, 0xF5, 0x61, 0x3E, 0xFB, 0x4D, 0x8E, 0x05, 0x85, 0x81, 0x38, 0xB8, 0x9A,
0x63, 0x12, 0x90, 0x17
]
);
}
}