use std::fmt::{self, Debug, Formatter, LowerHex, UpperHex};
use std::io;
use std::str::FromStr;
use amplify::Wrapper;
use bitcoin_hashes::hex::{Error, FromHex, ToHex};
use bitcoin_hashes::{sha256, Hash};
use strict_encoding::{StrictDecode, StrictEncode};
#[cfg_attr(
feature = "serde",
derive(Serialize, Deserialize),
serde(crate = "serde_crate")
)]
#[derive(
Wrapper,
Clone,
Copy,
PartialEq,
Eq,
PartialOrd,
Ord,
Hash,
Display,
Default,
From,
)]
#[display(LowerHex)]
pub struct Slice32(
#[cfg_attr(
feature = "serde",
serde(
serialize_with = "serde_helpers::to_hex",
deserialize_with = "serde_helpers::from_hex"
)
)]
[u8; 32],
);
impl Slice32 {
#[cfg(feature = "keygen")]
pub fn random() -> Self {
use bitcoin::secp256k1::rand;
let mut entropy = [0u8; 32];
entropy.copy_from_slice(
&bitcoin::secp256k1::SecretKey::new(&mut rand::thread_rng())[..],
);
Slice32::from_inner(entropy)
}
pub fn from_slice(slice: impl AsRef<[u8]>) -> Option<Slice32> {
if slice.as_ref().len() != 32 {
return None;
}
let mut inner = [0u8; 32];
inner.copy_from_slice(slice.as_ref());
Some(Self(inner))
}
#[allow(clippy::wrong_self_convention)]
pub fn to_vec(&self) -> Vec<u8> {
self.0.to_vec()
}
}
impl StrictEncode for Slice32 {
fn strict_encode<E: io::Write>(
&self,
e: E,
) -> Result<usize, strict_encoding::Error> {
sha256::Hash::from_inner(self.to_inner()).strict_encode(e)
}
}
impl StrictDecode for Slice32 {
fn strict_decode<D: io::Read>(
d: D,
) -> Result<Self, strict_encoding::Error> {
let hash = sha256::Hash::strict_decode(d)?;
Ok(Slice32::from_inner(hash.into_inner()))
}
}
impl Debug for Slice32 {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "Slice32({})", self.to_hex())
}
}
impl FromStr for Slice32 {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::from_hex(s)
}
}
impl FromHex for Slice32 {
fn from_byte_iter<I>(iter: I) -> Result<Self, Error>
where
I: Iterator<Item = Result<u8, Error>>
+ ExactSizeIterator
+ DoubleEndedIterator,
{
let vec = Vec::<u8>::from_byte_iter(iter)?;
if vec.len() != 32 {
return Err(Error::InvalidLength(32, vec.len()));
}
let mut id = [0u8; 32];
id.copy_from_slice(&vec);
Ok(Slice32(id))
}
}
impl LowerHex for Slice32 {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
if f.alternate() {
write!(
f,
"{}..{}",
self.0[..4].to_hex(),
self.0[(self.0.len() - 4)..].to_hex()
)
} else {
f.write_str(&self.0.to_hex())
}
}
}
impl UpperHex for Slice32 {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
if f.alternate() {
write!(
f,
"{}..{}",
self.0[..4].to_hex().to_ascii_uppercase(),
self.0[(self.0.len() - 4)..].to_hex().to_ascii_uppercase()
)
} else {
f.write_str(&self.0.to_hex().to_ascii_uppercase())
}
}
}
#[cfg(feature = "serde")]
pub(crate) mod serde_helpers {
use bitcoin_hashes::hex::{FromHex, ToHex};
use serde::{Deserialize, Deserializer, Serializer};
pub fn to_hex<S>(
buffer: &[u8; 32],
serializer: S,
) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_str(&buffer.as_ref().to_hex())
}
pub fn from_hex<'de, D>(deserializer: D) -> Result<[u8; 32], D::Error>
where
D: Deserializer<'de>,
{
use serde::de::Error;
String::deserialize(deserializer).and_then(|string| {
let vec = Vec::<u8>::from_hex(&string)
.map_err(|_| D::Error::custom("wrong hex data"))?;
if vec.len() != 32 {
return Err(D::Error::custom(
"Wrong 32-byte slice data length",
));
}
let mut slice32 = [0u8; 32];
slice32.copy_from_slice(&vec[0..32]);
Ok(slice32)
})
}
}
#[cfg(test)]
mod test {
use super::{Error, Slice32};
use amplify::Wrapper;
use bitcoin_hashes::hex::FromHex;
use std::str::FromStr;
use strict_encoding::{StrictDecode, StrictEncode};
#[test]
fn test_slice32_str() {
let s =
"a3401bcceb26201b55978ff705fecf7d8a0a03598ebeccf2a947030b91a0ff53";
let slice32 = Slice32::from_hex(s).unwrap();
assert_eq!(Slice32::from_str(s), Ok(slice32));
assert_eq!(Slice32::from_hex(&s.to_uppercase()), Ok(slice32));
assert_eq!(
Slice32::from_str(&s[..30]),
Err(Error::InvalidLength(32, 15))
);
assert_eq!(&slice32.to_string(), s);
assert_eq!(format!("{:x}", slice32), s);
assert_eq!(format!("{:X}", slice32), s.to_uppercase());
assert_eq!(format!("{:?}", slice32), format!("Slice32({})", s));
}
#[test]
fn test_encoding() {
let s =
"a3401bcceb26201b55978ff705fecf7d8a0a03598ebeccf2a947030b91a0ff53";
let slice32 = Slice32::from_hex(s).unwrap();
let ser = slice32.strict_serialize().unwrap();
let data = [
0xa3, 0x40, 0x1b, 0xcc, 0xeb, 0x26, 0x20, 0x1b, 0x55, 0x97, 0x8f,
0xf7, 0x05, 0xfe, 0xcf, 0x7d, 0x8a, 0x0a, 0x03, 0x59, 0x8e, 0xbe,
0xcc, 0xf2, 0xa9, 0x47, 0x03, 0x0b, 0x91, 0xa0, 0xff, 0x53,
];
assert_eq!(ser.len(), 32);
assert_eq!(&ser, &data);
assert_eq!(Slice32::strict_deserialize(&ser), Ok(slice32));
assert_eq!(Slice32::from_slice(&data), Some(slice32));
assert_eq!(Slice32::from_slice(&data[..30]), None);
assert_eq!(&slice32.to_vec(), &data);
assert_eq!(&slice32.as_inner()[..], &data);
assert_eq!(slice32.to_inner(), data);
assert_eq!(slice32.into_inner(), data);
}
}