Skip to content

Commit

Permalink
refactor: updates to Integer API (#102)
Browse files Browse the repository at this point in the history
See #101
  • Loading branch information
hiltontj authored Aug 6, 2024
1 parent 80ffd2e commit 3b096de
Show file tree
Hide file tree
Showing 4 changed files with 82 additions and 54 deletions.
106 changes: 67 additions & 39 deletions serde_json_path_core/src/spec/integer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,24 +22,23 @@ const MAX: i64 = 9_007_199_254_740_992 - 1;
/// The minimum allowed value (-2^53) + 1
const MIN: i64 = -9_007_199_254_740_992 + 1;

#[inline]
fn check_i64_is_valid(v: i64) -> bool {
(MIN..=MAX).contains(&v)
}

impl Integer {
/// An [`Integer`] with the value 0
pub const ZERO: Self = Self(0);

fn try_new(value: i64) -> Result<Self, IntegerError> {
if !(MIN..=MAX).contains(&value) {
Err(IntegerError::OutOfBounds)
} else {
if check_i64_is_valid(value) {
Ok(Self(value))
} else {
Err(IntegerError::OutOfBounds)
}
}

fn check(&self) -> bool {
(MIN..=MAX).contains(&self.0)
}

/// Produce an [`Integer`] with the value 0
pub fn zero() -> Self {
Self(0)
}

/// Get an [`Integer`] from an `i64`
///
/// This is intended for initializing an integer with small, non-zero numbers.
Expand All @@ -52,32 +51,34 @@ impl Integer {
Self::try_new(value).expect("value is out of the valid range")
}

/// Take the absolute value, producing `None` if the resulting value is outside
/// the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_abs(mut self) -> Option<Self> {
self.0 = self.0.checked_abs()?;
self.check().then_some(self)
/// Take the absolute value, producing a new instance of [`Integer`]
///
/// This is safe and will never panic since no instance of [`Integer`] can be constructed with
/// a value that is outside the valid range and since the absolute of the minimum allowed value
/// is the maximum value.
pub fn abs(self) -> Self {
Self(self.0.abs())
}

/// Add the two values, producing `None` if the resulting value is outside the
/// valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_add(mut self, rhs: Self) -> Option<Self> {
self.0 = self.0.checked_add(rhs.0)?;
self.check().then_some(self)
/// Add the two values, producing a new instance of [`Integer`] or `None` if the
/// resulting value is outside the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1]
pub fn checked_add(self, rhs: Self) -> Option<Self> {
let i = self.0.checked_add(rhs.0)?;
check_i64_is_valid(i).then_some(Self(i))
}

/// Subtract the `rhs` from `self`, producing `None` if the resulting value is
/// outside the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_sub(mut self, rhs: Self) -> Option<Self> {
self.0 = self.0.checked_sub(rhs.0)?;
self.check().then_some(self)
/// Subtract the `rhs` from `self`, producing a new instance of [`Integer`] or `None`
/// if the resulting value is outside the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_sub(self, rhs: Self) -> Option<Self> {
let i = self.0.checked_sub(rhs.0)?;
check_i64_is_valid(i).then_some(Self(i))
}

/// Multiply the two values, producing `None` if the resulting value is outside
/// the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_mul(mut self, rhs: Self) -> Option<Self> {
self.0 = self.0.checked_mul(rhs.0)?;
self.check().then_some(self)
/// Multiply the two values, producing a new instance of [`Integer`] or `None` if the resulting
/// value is outside the valid range [-(2<sup>53</sup>)+1, (2<sup>53</sup>)-1].
pub fn checked_mul(self, rhs: Self) -> Option<Self> {
let i = self.0.checked_mul(rhs.0)?;
check_i64_is_valid(i).then_some(Self(i))
}
}

Expand All @@ -89,16 +90,43 @@ impl TryFrom<i64> for Integer {
}
}

impl TryFrom<usize> for Integer {
type Error = IntegerError;
macro_rules! impl_try_from {
($type:ty) => {
impl TryFrom<$type> for Integer {
type Error = IntegerError;

fn try_from(value: usize) -> Result<Self, Self::Error> {
i64::try_from(value)
.map_err(|_| IntegerError::OutOfBounds)
.and_then(Self::try_new)
}
fn try_from(value: $type) -> Result<Self, Self::Error> {
i64::try_from(value)
.map_err(|_| IntegerError::OutOfBounds)
.and_then(Self::try_from)
}
}
};
}

impl_try_from!(i128);
impl_try_from!(u64);
impl_try_from!(u128);
impl_try_from!(usize);
impl_try_from!(isize);

macro_rules! impl_from {
($type:ty) => {
impl From<$type> for Integer {
fn from(value: $type) -> Self {
Self(value.into())
}
}
};
}

impl_from!(i8);
impl_from!(i16);
impl_from!(i32);
impl_from!(u8);
impl_from!(u16);
impl_from!(u32);

impl FromStr for Integer {
type Err = IntegerError;

Expand Down
12 changes: 6 additions & 6 deletions serde_json_path_core/src/spec/selector/index.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,9 +24,9 @@ impl Queryable for Index {
fn query<'b>(&self, current: &'b Value, _root: &'b Value) -> Vec<&'b Value> {
if let Some(list) = current.as_array() {
if self.0 < 0 {
self.0
.checked_abs()
.and_then(|i| usize::try_from(i).ok())
let abs = self.0.abs();
usize::try_from(abs)
.ok()
.and_then(|i| list.len().checked_sub(i))
.and_then(|i| list.get(i))
.into_iter()
Expand All @@ -51,9 +51,9 @@ impl Queryable for Index {
) -> Vec<LocatedNode<'b>> {
if let Some((index, node)) = current.as_array().and_then(|list| {
if self.0 < 0 {
self.0
.checked_abs()
.and_then(|i| usize::try_from(i).ok())
let abs = self.0.abs();
usize::try_from(abs)
.ok()
.and_then(|i| list.len().checked_sub(i))
.and_then(|i| list.get(i).map(|v| (i, v)))
} else {
Expand Down
16 changes: 8 additions & 8 deletions serde_json_path_core/src/spec/selector/slice.rs
Original file line number Diff line number Diff line change
Expand Up @@ -81,14 +81,14 @@ impl Slice {

#[inline]
fn bounds_on_forward_slice(&self, len: Integer) -> (Integer, Integer) {
let start_default = self.start.unwrap_or(Integer::zero());
let start_default = self.start.unwrap_or(Integer::ZERO);
let end_default = self.end.unwrap_or(len);
let start = normalize_slice_index(start_default, len)
.unwrap_or(Integer::zero())
.max(Integer::zero());
.unwrap_or(Integer::ZERO)
.max(Integer::ZERO);
let end = normalize_slice_index(end_default, len)
.unwrap_or(Integer::zero())
.max(Integer::zero());
.unwrap_or(Integer::ZERO)
.max(Integer::ZERO);
let lower = start.min(len);
let upper = end.min(len);
(lower, upper)
Expand All @@ -104,10 +104,10 @@ impl Slice {
l.checked_sub(Integer::from_i64_unchecked(1))
})?;
let start = normalize_slice_index(start_default, len)
.unwrap_or(Integer::zero())
.unwrap_or(Integer::ZERO)
.max(Integer::from_i64_unchecked(-1));
let end = normalize_slice_index(end_default, len)
.unwrap_or(Integer::zero())
.unwrap_or(Integer::ZERO)
.max(Integer::from_i64_unchecked(-1));
let lower = end.min(
len.checked_sub(Integer::from_i64_unchecked(1))
Expand Down Expand Up @@ -235,6 +235,6 @@ fn normalize_slice_index(index: Integer, len: Integer) -> Option<Integer> {
if index >= 0 {
Some(index)
} else {
index.checked_abs().and_then(|i| len.checked_sub(i))
len.checked_sub(index.abs())
}
}

0 comments on commit 3b096de

Please sign in to comment.