use core::ops::Deref; use std::fmt::{Debug, Formatter}; use std::ops::Index; use bytes::Bytes; #[cfg_attr(feature = "std", derive(thiserror::Error))] pub enum FixedByteError { #[cfg_attr( feature = "std", error("Cannot create fix byte array, input has invalid length.") )] InvalidLength, } impl FixedByteError { fn field_name(&self) -> &'static str { match self { FixedByteError::InvalidLength => "input", } } fn field_data(&self) -> &'static str { match self { FixedByteError::InvalidLength => { "Cannot create fix byte array, input has invalid length." } } } } impl Debug for FixedByteError { fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { f.debug_struct("FixedByteError") .field(self.field_name(), &self.field_data()) .finish() } } /// A fixed size byte slice. /// /// Internally this is just a wrapper around [`Bytes`], with the constructors checking that the length is equal to [`N`]. /// This implements [`Deref`] with the target being `[u8; N]`. pub struct ByteArray(Bytes); impl ByteArray { pub fn take_bytes(self) -> Bytes { self.0 } } impl Deref for ByteArray { type Target = [u8; N]; fn deref(&self) -> &Self::Target { self.0.deref().try_into().unwrap() } } impl TryFrom for ByteArray { type Error = FixedByteError; fn try_from(value: Bytes) -> Result { if value.len() != N { return Err(FixedByteError::InvalidLength); } Ok(ByteArray(value)) } } pub struct ByteArrayVec(Bytes); impl ByteArrayVec { pub fn len(&self) -> usize { self.0.len() / N } pub fn is_empty(&self) -> bool { self.len() == 0 } pub fn take_bytes(self) -> Bytes { self.0 } } impl TryFrom for ByteArrayVec { type Error = FixedByteError; fn try_from(value: Bytes) -> Result { if value.len() % N != 0 { return Err(FixedByteError::InvalidLength); } Ok(ByteArrayVec(value)) } } impl Index for ByteArrayVec { type Output = [u8; 32]; fn index(&self, index: usize) -> &Self::Output { if (index + 1) * N > self.0.len() { panic!("Index out of range, idx: {}, length: {}", index, self.len()); } self.0[index * N..(index + 1) * N] .as_ref() .try_into() .unwrap() } } #[cfg(test)] mod tests { use super::*; #[test] fn byte_array_vec_len() { let bytes = vec![0; 32 * 100]; let bytes = ByteArrayVec::<32>::try_from(Bytes::from(bytes)).unwrap(); assert_eq!(bytes.len(), 100); let _ = bytes[99]; } }