mirror of
https://github.com/hinto-janai/cuprate.git
synced 2025-01-24 11:26:03 +00:00
127 lines
2.9 KiB
Rust
127 lines
2.9 KiB
Rust
|
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<const N: usize>(Bytes);
|
||
|
|
||
|
impl<const N: usize> ByteArray<N> {
|
||
|
pub fn take_bytes(self) -> Bytes {
|
||
|
self.0
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<const N: usize> Deref for ByteArray<N> {
|
||
|
type Target = [u8; N];
|
||
|
|
||
|
fn deref(&self) -> &Self::Target {
|
||
|
self.0.deref().try_into().unwrap()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<const N: usize> TryFrom<Bytes> for ByteArray<N> {
|
||
|
type Error = FixedByteError;
|
||
|
|
||
|
fn try_from(value: Bytes) -> Result<Self, Self::Error> {
|
||
|
if value.len() != N {
|
||
|
return Err(FixedByteError::InvalidLength);
|
||
|
}
|
||
|
Ok(ByteArray(value))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub struct ByteArrayVec<const N: usize>(Bytes);
|
||
|
|
||
|
impl<const N: usize> ByteArrayVec<N> {
|
||
|
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<const N: usize> TryFrom<Bytes> for ByteArrayVec<N> {
|
||
|
type Error = FixedByteError;
|
||
|
|
||
|
fn try_from(value: Bytes) -> Result<Self, Self::Error> {
|
||
|
if value.len() % N != 0 {
|
||
|
return Err(FixedByteError::InvalidLength);
|
||
|
}
|
||
|
|
||
|
Ok(ByteArrayVec(value))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<const N: usize> Index<usize> for ByteArrayVec<N> {
|
||
|
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];
|
||
|
}
|
||
|
}
|