use crate::utils::CastErrorToAnyhow;
use crate::{FromCell, ToCell};
use num_bigint::BigUint;
use std::fmt::{Display, Formatter, LowerHex, UpperHex};
use tonlib_core::cell::{CellBuilder, CellParser};
#[derive(Debug, PartialEq, Default)]
pub struct Uint<const SIZE: usize>(BigUint);
impl<const SIZE: usize> From<BigUint> for Uint<SIZE> {
fn from(value: BigUint) -> Self {
Self(value)
}
}
impl<const SIZE: usize> From<Uint<SIZE>> for BigUint {
fn from(value: Uint<SIZE>) -> Self {
value.0
}
}
macro_rules! try_into {
($structname: ty) => {
impl<const SIZE: usize> TryFrom<Uint<SIZE>> for $structname {
type Error = anyhow::Error;
fn try_from(value: Uint<SIZE>) -> Result<Self, Self::Error> {
value
.0
.try_into()
.map_err(|err| anyhow::Error::msg(format!("Cannot cast Uint: {:?}", err)))
}
}
};
}
try_into!(u32);
try_into!(u64);
try_into!(u128);
try_into!(usize);
impl<const SIZE: usize> Display for Uint<SIZE> {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.0)
}
}
impl<const SIZE: usize> LowerHex for Uint<SIZE> {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(f, "0x{:x}", self.0)
}
}
impl<const SIZE: usize> UpperHex for Uint<SIZE> {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(f, "0x{:X}", self.0)
}
}
impl<const SIZE: usize> ToCell for Uint<SIZE> {
fn store<'a>(&self, builder: &'a mut CellBuilder) -> anyhow::Result<&'a mut CellBuilder> {
builder.store_uint(SIZE, &self.0).map_err_to_anyhow()
}
}
impl<const SIZE: usize> FromCell for Uint<SIZE> {
fn load(parser: &mut CellParser) -> anyhow::Result<Self> {
parser.load_uint(SIZE).map(Self).map_err_to_anyhow()
}
}
#[cfg(test)]
mod tests {
use super::*;
const INT_VALUE: usize = 100500;
const BITS: usize = 69;
#[test]
fn test_from() {
assert_eq!(
Uint::<BITS>::from(BigUint::from(INT_VALUE)),
Uint(BigUint::from(INT_VALUE))
);
}
#[test]
fn test_into() {
assert_eq!(
<Uint::<BITS> as Into<BigUint>>::into(Uint(BigUint::from(INT_VALUE))),
BigUint::from(INT_VALUE)
);
}
#[test]
fn test_try_into() {
assert_eq!(
<Uint<BITS> as TryInto<u32>>::try_into(Uint(BigUint::from(INT_VALUE))).unwrap(),
INT_VALUE as u32
);
assert_eq!(
<Uint<BITS> as TryInto<u64>>::try_into(Uint(BigUint::from(INT_VALUE))).unwrap(),
INT_VALUE as u64
);
assert_eq!(
<Uint<BITS> as TryInto<u128>>::try_into(Uint(BigUint::from(INT_VALUE))).unwrap(),
INT_VALUE as u128
);
assert_eq!(
<Uint<BITS> as TryInto<usize>>::try_into(Uint(BigUint::from(INT_VALUE))).unwrap(),
INT_VALUE
);
}
#[test]
fn test_default() {
assert_eq!(Uint::<BITS>::default(), Uint(BigUint::ZERO));
}
#[test]
fn test_from_cell() {
let cell = CellBuilder::new()
.store_uint(BITS, &BigUint::from(INT_VALUE))
.unwrap()
.build()
.unwrap();
assert_eq!(
Uint::<BITS>::from_cell(cell).unwrap(),
Uint(BigUint::from(INT_VALUE))
);
}
#[test]
fn test_to_cell() {
assert_eq!(
Uint::<BITS>(BigUint::from(INT_VALUE)).to_cell().unwrap(),
CellBuilder::new()
.store_uint(BITS, &BigUint::from(INT_VALUE))
.unwrap()
.build()
.unwrap(),
);
}
#[test]
fn test_from_to_cell() {
let first_iter = Uint::<BITS>::from(BigUint::from(INT_VALUE));
let cell = first_iter.to_cell().unwrap();
let second_iter = Uint::<BITS>::from_cell(cell).unwrap();
assert_eq!(first_iter, second_iter);
}
#[test]
fn test_fmt() {
let value = Uint::<32>::from(BigUint::from(0x1234abcdu32));
assert_eq!("Uint(305441741)", format!("{:?}", value));
assert_eq!("305441741", format!("{}", value));
assert_eq!("0x1234abcd", format!("{:x}", value));
assert_eq!("0x1234ABCD", format!("{:X}", value));
}
}