use alloc::{string::String, vec::Vec};
use core::fmt;
use core::ops::Deref;
use serde::{de, ser, Deserialize, Deserializer, Serialize};
use crate::{
encoding::{from_hex, to_hex},
Binary, StdError, StdResult,
};
#[derive(
Clone,
Default,
PartialEq,
Eq,
Hash,
PartialOrd,
Ord,
schemars::JsonSchema,
cw_schema::Schemaifier,
)]
#[schemaifier(type = cw_schema::NodeType::HexBinary)]
pub struct HexBinary(#[schemars(with = "String")] Vec<u8>);
impl HexBinary {
pub fn from_hex(input: &str) -> StdResult<Self> {
from_hex(input).map(Self)
}
pub fn to_hex(&self) -> String {
to_hex(&self.0)
}
pub fn as_slice(&self) -> &[u8] {
self.0.as_slice()
}
pub fn to_array<const LENGTH: usize>(&self) -> StdResult<[u8; LENGTH]> {
if self.len() != LENGTH {
return Err(StdError::msg(format_args!(
"invalid length. expected {LENGTH}, got {}",
self.len()
)));
}
let mut out: [u8; LENGTH] = [0; LENGTH];
out.copy_from_slice(&self.0);
Ok(out)
}
}
impl fmt::Display for HexBinary {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.to_hex())
}
}
impl fmt::Debug for HexBinary {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "HexBinary(")?;
for byte in self.0.iter() {
write!(f, "{byte:02x}")?;
}
write!(f, ")")?;
Ok(())
}
}
impl Deref for HexBinary {
type Target = [u8];
fn deref(&self) -> &Self::Target {
self.as_slice()
}
}
impl AsRef<[u8]> for HexBinary {
fn as_ref(&self) -> &[u8] {
self.as_slice()
}
}
impl From<&[u8]> for HexBinary {
fn from(binary: &[u8]) -> Self {
Self(binary.to_vec())
}
}
impl<const LENGTH: usize> From<&[u8; LENGTH]> for HexBinary {
fn from(source: &[u8; LENGTH]) -> Self {
Self(source.to_vec())
}
}
impl<const LENGTH: usize> From<[u8; LENGTH]> for HexBinary {
fn from(source: [u8; LENGTH]) -> Self {
Self(source.into())
}
}
impl From<Vec<u8>> for HexBinary {
fn from(vec: Vec<u8>) -> Self {
Self(vec)
}
}
impl From<HexBinary> for Vec<u8> {
fn from(original: HexBinary) -> Vec<u8> {
original.0
}
}
impl From<Binary> for HexBinary {
fn from(original: Binary) -> Self {
Self(original.into())
}
}
impl From<HexBinary> for Binary {
fn from(original: HexBinary) -> Binary {
Binary::from(original.0)
}
}
impl PartialEq<Vec<u8>> for HexBinary {
fn eq(&self, rhs: &Vec<u8>) -> bool {
self.0 == *rhs
}
}
impl PartialEq<HexBinary> for Vec<u8> {
fn eq(&self, rhs: &HexBinary) -> bool {
*self == rhs.0
}
}
impl PartialEq<&[u8]> for HexBinary {
fn eq(&self, rhs: &&[u8]) -> bool {
self.as_slice() == *rhs
}
}
impl PartialEq<HexBinary> for &[u8] {
fn eq(&self, rhs: &HexBinary) -> bool {
*self == rhs.as_slice()
}
}
impl<const LENGTH: usize> PartialEq<[u8; LENGTH]> for HexBinary {
fn eq(&self, rhs: &[u8; LENGTH]) -> bool {
self.as_slice() == rhs.as_slice()
}
}
impl<const LENGTH: usize> PartialEq<HexBinary> for [u8; LENGTH] {
fn eq(&self, rhs: &HexBinary) -> bool {
self.as_slice() == rhs.as_slice()
}
}
impl<const LENGTH: usize> PartialEq<&[u8; LENGTH]> for HexBinary {
fn eq(&self, rhs: &&[u8; LENGTH]) -> bool {
self.as_slice() == rhs.as_slice()
}
}
impl<const LENGTH: usize> PartialEq<HexBinary> for &[u8; LENGTH] {
fn eq(&self, rhs: &HexBinary) -> bool {
self.as_slice() == rhs.as_slice()
}
}
impl Serialize for HexBinary {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: ser::Serializer,
{
if serializer.is_human_readable() {
serializer.serialize_str(&self.to_hex())
} else {
serializer.serialize_bytes(&self.0)
}
}
}
impl<'de> Deserialize<'de> for HexBinary {
fn deserialize<D>(deserializer: D) -> Result<HexBinary, D::Error>
where
D: Deserializer<'de>,
{
if deserializer.is_human_readable() {
deserializer.deserialize_str(HexVisitor)
} else {
deserializer.deserialize_bytes(BytesVisitor)
}
}
}
struct HexVisitor;
impl de::Visitor<'_> for HexVisitor {
type Value = HexBinary;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("valid hex encoded string")
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
match HexBinary::from_hex(v) {
Ok(data) => Ok(data),
Err(_) => Err(E::custom(format!("invalid hex: {v}"))),
}
}
}
struct BytesVisitor;
impl de::Visitor<'_> for BytesVisitor {
type Value = HexBinary;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("byte array")
}
fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(HexBinary(v.to_vec()))
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{assert_hash_works, errors::ErrorKind};
#[test]
fn from_hex_works() {
let data = HexBinary::from_hex("").unwrap();
assert_eq!(data, b"");
let data = HexBinary::from_hex("61").unwrap();
assert_eq!(data, b"a");
let data = HexBinary::from_hex("00").unwrap();
assert_eq!(data, b"\0");
let data = HexBinary::from_hex("68656c6c6f").unwrap();
assert_eq!(data, b"hello");
let data = HexBinary::from_hex("68656C6C6F").unwrap();
assert_eq!(data, b"hello");
let data = HexBinary::from_hex("72616e646f6d695a").unwrap();
assert_eq!(data.as_slice(), b"randomiZ");
assert!(HexBinary::from_hex("123")
.unwrap_err()
.to_string()
.ends_with("Odd number of digits"));
assert!(HexBinary::from_hex("efgh")
.unwrap_err()
.to_string()
.ends_with("Invalid character 'g' at position 2"));
assert!(HexBinary::from_hex("0xaa")
.unwrap_err()
.to_string()
.ends_with("Invalid character 'x' at position 1"));
assert!(matches!(
HexBinary::from_hex("aa ").unwrap_err().kind(),
ErrorKind::Encoding,
));
assert!(matches!(
HexBinary::from_hex(" aa").unwrap_err().kind(),
ErrorKind::Encoding,
));
assert!(matches!(
HexBinary::from_hex("a a").unwrap_err().kind(),
ErrorKind::Encoding,
));
assert!(matches!(
HexBinary::from_hex(" aa ").unwrap_err().kind(),
ErrorKind::Encoding,
));
}
#[test]
fn to_hex_works() {
let binary: &[u8] = b"";
let encoded = HexBinary::from(binary).to_hex();
assert_eq!(encoded, "");
let binary: &[u8] = b"hello";
let encoded = HexBinary::from(binary).to_hex();
assert_eq!(encoded, "68656c6c6f");
let binary = vec![12u8, 187, 0, 17, 250, 1];
let encoded = HexBinary(binary).to_hex();
assert_eq!(encoded, "0cbb0011fa01");
}
#[test]
fn to_array_works() {
let binary = HexBinary::from(&[1, 2, 3]);
let array: [u8; 3] = binary.to_array().unwrap();
assert_eq!(array, [1, 2, 3]);
let binary = HexBinary::from(&[]);
let array: [u8; 0] = binary.to_array().unwrap();
assert_eq!(array, [] as [u8; 0]);
let binary = HexBinary::from(&[1, 2, 3]);
let error = binary.to_array::<8>().unwrap_err();
assert!(error
.to_string()
.ends_with("invalid length. expected 8, got 3"));
let binary =
HexBinary::from_hex("b75d7d24e428c7859440498efe7caa3997cefb08c99bdd581b6b1f9f866096f0")
.unwrap();
let array: [u8; 32] = binary.to_array().unwrap();
assert_eq!(
array,
[
0xb7, 0x5d, 0x7d, 0x24, 0xe4, 0x28, 0xc7, 0x85, 0x94, 0x40, 0x49, 0x8e, 0xfe, 0x7c,
0xaa, 0x39, 0x97, 0xce, 0xfb, 0x08, 0xc9, 0x9b, 0xdd, 0x58, 0x1b, 0x6b, 0x1f, 0x9f,
0x86, 0x60, 0x96, 0xf0,
]
);
let binary = HexBinary::from_hex(
"b75d7d24e428c7859440498efe7caa3997cefb08c99bdd581b6b1f9f866096f073c8c3b0316abe",
)
.unwrap();
let array: [u8; 39] = binary.to_array().unwrap();
assert_eq!(
array,
[
0xb7, 0x5d, 0x7d, 0x24, 0xe4, 0x28, 0xc7, 0x85, 0x94, 0x40, 0x49, 0x8e, 0xfe, 0x7c,
0xaa, 0x39, 0x97, 0xce, 0xfb, 0x08, 0xc9, 0x9b, 0xdd, 0x58, 0x1b, 0x6b, 0x1f, 0x9f,
0x86, 0x60, 0x96, 0xf0, 0x73, 0xc8, 0xc3, 0xb0, 0x31, 0x6a, 0xbe,
]
);
}
#[test]
fn from_json_works() {
let original: &[u8] = &[0u8, 187, 61, 11, 250, 0];
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8, 187, 61, 11, 250, 0]);
}
#[test]
fn from_fixed_length_array_works() {
let original = &[];
let binary: HexBinary = original.into();
assert_eq!(binary.len(), 0);
let original = &[0u8];
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8]);
let original = &[0u8, 187, 61, 11, 250, 0];
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8, 187, 61, 11, 250, 0]);
let original = &[
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
1, 1, 1,
];
let binary: HexBinary = original.into();
assert_eq!(
binary.as_slice(),
[
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
1, 1, 1, 1,
]
);
}
#[test]
fn from_owned_fixed_length_array_works() {
let original = [];
let binary: HexBinary = original.into();
assert_eq!(binary.len(), 0);
let original = [0u8];
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8]);
let original = [0u8, 187, 61, 11, 250, 0];
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8, 187, 61, 11, 250, 0]);
let original = [
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
1, 1, 1,
];
let binary: HexBinary = original.into();
assert_eq!(
binary.as_slice(),
[
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
1, 1, 1, 1,
]
);
}
#[test]
fn from_literal_works() {
let a: HexBinary = b"".into();
assert_eq!(a.len(), 0);
let a: HexBinary = b".".into();
assert_eq!(a.len(), 1);
let a: HexBinary = b"...".into();
assert_eq!(a.len(), 3);
let a: HexBinary = b"...............................".into();
assert_eq!(a.len(), 31);
let a: HexBinary = b"................................".into();
assert_eq!(a.len(), 32);
let a: HexBinary = (b".................................").into();
assert_eq!(a.len(), 33);
}
#[test]
fn from_vec_works() {
let original = vec![0u8, 187, 61, 11, 250, 0];
let original_ptr = original.as_ptr();
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8, 187, 61, 11, 250, 0]);
assert_eq!(binary.0.as_ptr(), original_ptr, "vector must not be copied");
}
#[test]
fn into_vec_works() {
let original = HexBinary(vec![0u8, 187, 61, 11, 250, 0]);
let original_ptr = original.0.as_ptr();
let vec: Vec<u8> = original.into();
assert_eq!(vec.as_slice(), [0u8, 187, 61, 11, 250, 0]);
assert_eq!(vec.as_ptr(), original_ptr, "vector must not be copied");
let original = HexBinary(vec![7u8, 35, 49, 101, 0, 255]);
let original_ptr = original.0.as_ptr();
let vec = Vec::<u8>::from(original);
assert_eq!(vec.as_slice(), [7u8, 35, 49, 101, 0, 255]);
assert_eq!(vec.as_ptr(), original_ptr, "vector must not be copied");
}
#[test]
fn from_binary_works() {
let original = Binary::from([0u8, 187, 61, 11, 250, 0]);
let original_ptr = original.as_ptr();
let binary: HexBinary = original.into();
assert_eq!(binary.as_slice(), [0u8, 187, 61, 11, 250, 0]);
assert_eq!(binary.0.as_ptr(), original_ptr, "vector must not be copied");
}
#[test]
fn into_binary_works() {
let original = HexBinary(vec![0u8, 187, 61, 11, 250, 0]);
let original_ptr = original.0.as_ptr();
let bin: Binary = original.into();
assert_eq!(bin.as_slice(), [0u8, 187, 61, 11, 250, 0]);
assert_eq!(bin.as_ptr(), original_ptr, "vector must not be copied");
let original = HexBinary(vec![7u8, 35, 49, 101, 0, 255]);
let original_ptr = original.0.as_ptr();
let bin = Binary::from(original);
assert_eq!(bin.as_slice(), [7u8, 35, 49, 101, 0, 255]);
assert_eq!(bin.as_ptr(), original_ptr, "vector must not be copied");
}
#[test]
fn serialization_works() {
let binary = HexBinary(vec![0u8, 187, 61, 11, 250, 0]);
let json = serde_json::to_vec(&binary).unwrap();
let deserialized: HexBinary = serde_json::from_slice(&json).unwrap();
assert_eq!(binary, deserialized);
}
#[test]
fn deserialize_from_valid_string() {
let hex = "00bb3d0bfa00";
let expected = vec![0u8, 187, 61, 11, 250, 0];
let serialized = serde_json::to_vec(&hex).unwrap();
let deserialized: HexBinary = serde_json::from_slice(&serialized).unwrap();
assert_eq!(expected, deserialized.as_slice());
}
#[test]
fn deserialize_from_invalid_string() {
let invalid_str = "**BAD!**";
let serialized = serde_json::to_vec(&invalid_str).unwrap();
let res = serde_json::from_slice::<HexBinary>(&serialized);
assert!(res.is_err());
}
#[test]
fn msgpack_serialization_works() {
let data = HexBinary(vec![0u8, 187, 61, 11, 250, 0]);
let expected = [196, 6, 0, 187, 61, 11, 250, 0];
assert_eq!(rmp_serde::to_vec(&data).unwrap(), expected);
}
#[test]
fn msgpack_deserialize_from_valid_data() {
let serialized = vec![196, 6, 0, 187, 61, 11, 250, 0];
let expected = vec![0u8, 187, 61, 11, 250, 0];
let deserialized: HexBinary = rmp_serde::from_slice(&serialized).unwrap();
assert_eq!(expected, deserialized.as_slice());
}
#[test]
fn msgpack_deserialize_from_invalid_data() {
let invalid_data = vec![0, 1, 2, 3, 4, 5];
let res = rmp_serde::from_slice::<HexBinary>(&invalid_data);
assert!(res.is_err());
}
#[test]
fn hex_binary_implements_debug() {
let data = HexBinary(vec![0x07, 0x35, 0xAA, 0xcb, 0x00, 0xff]);
assert_eq!(format!("{data:?}"), "HexBinary(0735aacb00ff)",);
let data = HexBinary(vec![]);
assert_eq!(format!("{data:?}"), "HexBinary()",);
}
#[test]
fn hex_binary_implements_deref() {
let data = HexBinary(vec![7u8, 35, 49, 101, 0, 255]);
assert_eq!(*data, [7u8, 35, 49, 101, 0, 255]);
let data = HexBinary(vec![7u8, 35, 49, 101, 0, 255]);
assert_eq!(data.len(), 6);
let data_slice: &[u8] = &data;
assert_eq!(data_slice, &[7u8, 35, 49, 101, 0, 255]);
}
#[test]
fn hex_binary_implements_as_ref() {
let want = &[7u8, 35, 49, 101, 0, 255];
let data = HexBinary(want.to_vec());
assert_eq!(want, AsRef::<[u8]>::as_ref(&data));
assert_eq!(want, AsRef::<[u8]>::as_ref(&&data));
}
#[test]
fn hex_binary_implements_hash_eq() {
let a = HexBinary::from([0, 187, 61, 11, 250, 0]);
let b = HexBinary::from([16, 21, 33, 0, 255, 9]);
assert_hash_works!(a, b);
}
#[test]
fn hex_binary_implements_partial_eq_with_vector() {
let a = HexBinary(vec![5u8; 3]);
let b = vec![5u8; 3];
let c = vec![9u8; 3];
assert_eq!(a, b);
assert_eq!(b, a);
assert_ne!(a, c);
assert_ne!(c, a);
}
#[test]
fn hex_binary_implements_partial_eq_with_slice_and_array() {
let a = HexBinary(vec![0xAA, 0xBB]);
assert_eq!(a, b"\xAA\xBB" as &[u8]);
assert_eq!(b"\xAA\xBB" as &[u8], a);
assert_ne!(a, b"\x11\x22" as &[u8]);
assert_ne!(b"\x11\x22" as &[u8], a);
assert_eq!(a, b"\xAA\xBB");
assert_eq!(b"\xAA\xBB", a);
assert_ne!(a, b"\x11\x22");
assert_ne!(b"\x11\x22", a);
assert_eq!(a, [0xAA, 0xBB]);
assert_eq!([0xAA, 0xBB], a);
assert_ne!(a, [0x11, 0x22]);
assert_ne!([0x11, 0x22], a);
}
}