From df5532be5d7e640692211da4c7cf7bbe64f219b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Timoth=C3=A9e=20Delabrouille?= Date: Thu, 26 Dec 2024 00:55:29 +0100 Subject: [PATCH] feat: add checked add/mul/sub --- .../src/felt/num_traits_impl.rs | 50 ++++++++++++++++++- 1 file changed, 49 insertions(+), 1 deletion(-) diff --git a/crates/starknet-types-core/src/felt/num_traits_impl.rs b/crates/starknet-types-core/src/felt/num_traits_impl.rs index 23fd52d2..880437e9 100644 --- a/crates/starknet-types-core/src/felt/num_traits_impl.rs +++ b/crates/starknet-types-core/src/felt/num_traits_impl.rs @@ -1,6 +1,8 @@ use super::Felt; use num_bigint::{ToBigInt, ToBigUint}; -use num_traits::{FromPrimitive, Inv, One, Pow, ToPrimitive, Zero}; +use num_traits::{ + CheckedAdd, CheckedMul, CheckedSub, FromPrimitive, Inv, One, Pow, ToPrimitive, Zero, +}; impl ToBigInt for Felt { /// Converts the value of `self` to a [`BigInt`]. @@ -121,6 +123,39 @@ impl Pow for Felt { } } +impl CheckedAdd for Felt { + fn checked_add(&self, v: &Self) -> Option { + let res = self + v; + if res < *self { + None + } else { + Some(res) + } + } +} + +impl CheckedMul for Felt { + fn checked_mul(&self, v: &Self) -> Option { + let res = self * v; + if res < *self { + None + } else { + Some(res) + } + } +} + +impl CheckedSub for Felt { + fn checked_sub(&self, v: &Self) -> Option { + let res = self - v; + if res > *self { + None + } else { + Some(res) + } + } +} + #[cfg(test)] mod tests { use super::*; @@ -139,4 +174,17 @@ mod tests { fn default_is_zero() { assert!(Felt::default().is_zero()); } + + #[test] + fn checked_ops() { + assert_eq!(Felt::ONE.checked_add(&Felt::TWO), Some(Felt::THREE)); + assert!(Felt::MAX.checked_add(&Felt::ONE).is_none()); + assert_eq!( + Felt::TWO.checked_mul(&Felt::THREE), + Some(Felt::from_hex_unchecked("0x6")) + ); + assert!(Felt::MAX.checked_mul(&Felt::TWO).is_none()); + assert_eq!(Felt::THREE.checked_sub(&Felt::TWO), Some(Felt::ONE)); + assert!(Felt::ONE.checked_sub(&Felt::TWO).is_none()); + } }