#![allow(
clippy::cast_possible_truncation,
clippy::cast_possible_wrap,
clippy::cast_sign_loss,
clippy::cast_precision_loss,
clippy::float_cmp
)]
use fraction::{BigFraction, One, Zero};
use serde_json::Number;
macro_rules! define_num_cmp {
($($trait_fn:ident => $fn_name:ident, $op:tt, $infinity_positive:literal),* $(,)?) => {
$(
pub(crate) fn $fn_name<T>(value: &Number, limit: T) -> bool
where
T: Copy + num_traits::ToPrimitive,
u64: num_cmp::NumCmp<T>,
i64: num_cmp::NumCmp<T>,
f64: num_cmp::NumCmp<T>,
{
if let Some(v) = value.as_u64() {
num_cmp::NumCmp::$trait_fn(v, limit)
} else if let Some(v) = value.as_i64() {
num_cmp::NumCmp::$trait_fn(v, limit)
} else if let Some(v) = value.as_f64() {
num_cmp::NumCmp::$trait_fn(v, limit)
} else {
#[cfg(feature = "arbitrary-precision")]
{
if let Some(big_value) = bignum::try_parse_bigfraction(value) {
if let Some(limit_f64) = num_traits::ToPrimitive::to_f64(&limit) {
let limit_frac = BigFraction::from(limit_f64);
return big_value $op limit_frac;
}
}
let is_negative = value.as_str().starts_with('-');
if $infinity_positive {
!is_negative
} else {
is_negative
}
}
#[cfg(not(feature = "arbitrary-precision"))]
{
unreachable!("Always Some without `arbitrary-precision`")
}
}
}
)*
};
}
define_num_cmp!(
num_ge => ge, >=, true, num_le => le, <=, false, num_gt => gt, >, true,
num_lt => lt, <, false,
);
pub(crate) fn is_multiple_of_float(value: &Number, multiple: f64) -> bool {
if let Some(value_f64) = value.as_f64() {
if value_f64.is_zero() {
return true;
}
if value_f64.abs() < multiple {
return false;
}
(BigFraction::from(value_f64) / BigFraction::from(multiple))
.denom()
.is_none_or(One::is_one)
} else {
false
}
}
#[cfg(feature = "arbitrary-precision")]
const MAX_SAFE_INTEGER: u64 = 1u64 << 53;
pub(crate) fn is_multiple_of_integer(value: &Number, multiple: f64) -> bool {
#[cfg(feature = "arbitrary-precision")]
{
if let Some(v) = value.as_u64() {
if v > MAX_SAFE_INTEGER
&& multiple > 0.0
&& multiple <= MAX_SAFE_INTEGER as f64
&& multiple.fract() == 0.0
{
return (v % (multiple as u64)) == 0;
}
}
if let Some(v) = value.as_i64() {
if v.unsigned_abs() > MAX_SAFE_INTEGER
&& multiple > 0.0
&& multiple <= MAX_SAFE_INTEGER as f64
&& multiple.fract() == 0.0
{
return (v % (multiple as i64)) == 0;
}
}
}
if let Some(value_f64) = value.as_f64() {
value_f64.fract() == 0. && (value_f64 % multiple) == 0.
} else {
#[cfg(feature = "arbitrary-precision")]
{
if let Some(big_value) = bignum::try_parse_bigint(value) {
use num_bigint::BigInt;
let multiple_int = BigInt::from(multiple as i64);
return bignum::is_multiple_of_bigint(&big_value, &multiple_int);
}
false
}
#[cfg(not(feature = "arbitrary-precision"))]
{
unreachable!("Always Some without `arbitrary-precision`")
}
}
}
#[cfg(feature = "arbitrary-precision")]
pub(crate) mod bignum {
use fraction::BigFraction;
use num_bigint::BigInt;
use num_traits::{ToPrimitive, Zero};
use serde_json::Number;
use std::str::FromStr;
const MAX_EXPONENT_ADJUSTMENT: u32 = 1_000_000;
#[derive(Debug, Clone)]
struct DecimalComponents {
negative: bool,
digits: String,
fraction_digits: usize,
exponent: i64,
}
impl DecimalComponents {
fn parse(num_str: &str) -> Option<Self> {
let bytes = num_str.as_bytes();
if bytes.is_empty() {
return None;
}
let mut idx = 0;
let negative = if bytes[idx] == b'-' {
idx += 1;
true
} else {
false
};
if idx >= bytes.len() {
return None;
}
let mut digits = String::with_capacity(bytes.len());
let int_start = idx;
while idx < bytes.len() && bytes[idx].is_ascii_digit() {
idx += 1;
}
if int_start == idx {
return None;
}
digits.push_str(&num_str[int_start..idx]);
let mut fraction_digits = 0usize;
if idx < bytes.len() && bytes[idx] == b'.' {
idx += 1;
let frac_start = idx;
while idx < bytes.len() && bytes[idx].is_ascii_digit() {
idx += 1;
}
if frac_start == idx {
return None;
}
digits.push_str(&num_str[frac_start..idx]);
fraction_digits = idx - frac_start;
}
let mut exponent: i64 = 0;
if idx < bytes.len() && (bytes[idx] == b'e' || bytes[idx] == b'E') {
idx += 1;
if idx >= bytes.len() {
return None;
}
let mut exp_sign: i64 = 1;
if bytes[idx] == b'+' {
idx += 1;
} else if bytes[idx] == b'-' {
exp_sign = -1;
idx += 1;
}
let exp_start = idx;
while idx < bytes.len() && bytes[idx].is_ascii_digit() {
idx += 1;
}
if exp_start == idx {
return None;
}
let exp_value = num_str[exp_start..idx].parse::<i64>().ok()?;
exponent = exp_value.checked_mul(exp_sign)?;
}
if idx != bytes.len() {
return None;
}
Some(Self {
negative,
digits,
fraction_digits,
exponent,
})
}
#[inline]
fn decimal_shift(&self) -> i64 {
self.exponent - self.fraction_digits as i64
}
}
fn digits_are_zero(s: &str) -> bool {
s.bytes().all(|b| b == b'0')
}
fn trailing_zero_count(s: &str) -> usize {
s.as_bytes()
.iter()
.rev()
.take_while(|b| **b == b'0')
.count()
}
fn append_zeros(target: &mut String, count: usize) -> Option<()> {
let new_len = target.len().checked_add(count)?;
target.reserve(count);
target.extend(std::iter::repeat_n('0', count));
debug_assert_eq!(target.len(), new_len);
Some(())
}
fn pow10_bigint(exp: usize) -> Option<BigInt> {
if exp == 0 {
return Some(BigInt::from(1));
}
let exp_u32 = u32::try_from(exp).ok()?;
Some(BigInt::from(10).pow(exp_u32))
}
fn shift_exceeds_limit(shift: i64) -> bool {
if shift <= 0 {
return false;
}
shift as u64 > u64::from(MAX_EXPONENT_ADJUSTMENT)
}
fn exponent_reduction_exceeds_limit(exponent: i64) -> bool {
if exponent >= 0 {
return false;
}
match exponent.checked_abs() {
Some(abs) => abs as u64 > u64::from(MAX_EXPONENT_ADJUSTMENT),
None => true,
}
}
pub(crate) fn try_parse_bigint(num: &Number) -> Option<BigInt> {
use super::MAX_SAFE_INTEGER;
let num_str = num.as_str();
if let Some(v) = num.as_i64() {
if v.unsigned_abs() <= MAX_SAFE_INTEGER {
return None;
}
}
let has_fraction_or_exponent = num_str.bytes().any(|b| b == b'.' || b == b'e' || b == b'E');
if !has_fraction_or_exponent {
return BigInt::from_str(num_str).ok();
}
let mut components = DecimalComponents::parse(num_str)?;
let mut shift = components.decimal_shift();
if shift < 0 {
let needed = (-shift) as usize;
if digits_are_zero(&components.digits) {
components.digits.clear();
components.digits.push('0');
shift = 0;
} else {
if exponent_reduction_exceeds_limit(components.exponent) {
return None;
}
let zeros = trailing_zero_count(&components.digits);
if zeros < needed {
return None;
}
let new_len = components.digits.len() - needed;
components.digits.truncate(new_len);
shift = 0;
}
}
if shift > 0 {
if shift_exceeds_limit(shift) {
return None;
}
append_zeros(&mut components.digits, shift as usize)?;
}
let digits_trimmed = components.digits.trim_start_matches('0');
let digits_ref = if digits_trimmed.is_empty() {
"0"
} else {
digits_trimmed
};
let mut value = BigInt::from_str(digits_ref).ok()?;
if components.negative && !value.is_zero() {
value = -value;
}
Some(value)
}
pub(crate) fn try_parse_bigfraction(num: &Number) -> Option<BigFraction> {
if num.as_i64().is_some() {
return None;
}
let num_str = num.as_str();
let mut has_decimal_point = false;
let mut has_exponent = false;
for b in num_str.bytes() {
if b == b'.' {
has_decimal_point = true;
} else if b == b'e' || b == b'E' {
has_exponent = true;
break;
}
}
if !has_decimal_point && !has_exponent {
return None;
}
if !has_exponent {
return BigFraction::from_str(num_str).ok();
}
let components = DecimalComponents::parse(num_str)?;
let shift = components.decimal_shift();
if shift >= 0 {
return None;
}
if exponent_reduction_exceeds_limit(components.exponent) {
return None;
}
let denom_power = (-shift) as usize;
let denominator = pow10_bigint(denom_power)?;
let mut numerator = BigInt::from_str(&components.digits).ok()?;
if components.negative && !numerator.is_zero() {
numerator = -numerator;
}
Some(BigFraction::from(numerator) / BigFraction::from(denominator))
}
macro_rules! define_bigint_cmp {
($($fn_name:ident, $prim_type:ty, $to_prim:ident, $op:tt, $overflow_sign:expr);* $(;)?) => {
$(
pub(crate) fn $fn_name(bigint: &BigInt, value: $prim_type) -> bool {
if let Some(converted) = bigint.$to_prim() {
converted $op value
} else {
bigint.sign() == $overflow_sign
}
}
)*
};
}
define_bigint_cmp!(
bigint_ge_u64, u64, to_u64, >=, num_bigint::Sign::Plus;
bigint_le_u64, u64, to_u64, <=, num_bigint::Sign::Minus;
bigint_gt_u64, u64, to_u64, >, num_bigint::Sign::Plus;
bigint_lt_u64, u64, to_u64, <, num_bigint::Sign::Minus;
bigint_ge_i64, i64, to_i64, >=, num_bigint::Sign::Plus;
bigint_le_i64, i64, to_i64, <=, num_bigint::Sign::Minus;
bigint_gt_i64, i64, to_i64, >, num_bigint::Sign::Plus;
bigint_lt_i64, i64, to_i64, <, num_bigint::Sign::Minus;
bigint_ge_f64, f64, to_f64, >=, num_bigint::Sign::Plus;
bigint_le_f64, f64, to_f64, <=, num_bigint::Sign::Minus;
bigint_gt_f64, f64, to_f64, >, num_bigint::Sign::Plus;
bigint_lt_f64, f64, to_f64, <, num_bigint::Sign::Minus;
);
macro_rules! define_reverse_cmp {
($($rev_ge:ident, $rev_le:ident, $rev_gt:ident, $rev_lt:ident, $prim_type:ty, $big_type:ty, $fwd_ge:ident, $fwd_le:ident, $fwd_gt:ident, $fwd_lt:ident);* $(;)?) => {
$(
pub(crate) fn $rev_ge(value: $prim_type, big: &$big_type) -> bool {
$fwd_le(big, value)
}
pub(crate) fn $rev_le(value: $prim_type, big: &$big_type) -> bool {
$fwd_ge(big, value)
}
pub(crate) fn $rev_gt(value: $prim_type, big: &$big_type) -> bool {
$fwd_lt(big, value)
}
pub(crate) fn $rev_lt(value: $prim_type, big: &$big_type) -> bool {
$fwd_gt(big, value)
}
)*
};
}
define_reverse_cmp!(
u64_ge_bigint, u64_le_bigint, u64_gt_bigint, u64_lt_bigint, u64, BigInt, bigint_ge_u64, bigint_le_u64, bigint_gt_u64, bigint_lt_u64;
i64_ge_bigint, i64_le_bigint, i64_gt_bigint, i64_lt_bigint, i64, BigInt, bigint_ge_i64, bigint_le_i64, bigint_gt_i64, bigint_lt_i64;
f64_ge_bigint, f64_le_bigint, f64_gt_bigint, f64_lt_bigint, f64, BigInt, bigint_ge_f64, bigint_le_f64, bigint_gt_f64, bigint_lt_f64;
);
pub(crate) fn is_multiple_of_bigint(value: &BigInt, multiple: &BigInt) -> bool {
if value.is_zero() {
return true;
}
(value % multiple).is_zero()
}
macro_rules! define_bigfraction_cmp {
($($fn_name:ident, $prim_type:ty, $op:tt);* $(;)?) => {
$(
pub(crate) fn $fn_name(bigfrac: &BigFraction, value: $prim_type) -> bool {
let value_frac = BigFraction::from(value);
*bigfrac $op value_frac
}
)*
};
}
define_bigfraction_cmp!(
bigfrac_ge_u64, u64, >=;
bigfrac_le_u64, u64, <=;
bigfrac_gt_u64, u64, >;
bigfrac_lt_u64, u64, <;
bigfrac_ge_i64, i64, >=;
bigfrac_le_i64, i64, <=;
bigfrac_gt_i64, i64, >;
bigfrac_lt_i64, i64, <;
bigfrac_ge_f64, f64, >=;
bigfrac_le_f64, f64, <=;
bigfrac_gt_f64, f64, >;
bigfrac_lt_f64, f64, <;
);
define_reverse_cmp!(
u64_ge_bigfrac, u64_le_bigfrac, u64_gt_bigfrac, u64_lt_bigfrac, u64, BigFraction, bigfrac_ge_u64, bigfrac_le_u64, bigfrac_gt_u64, bigfrac_lt_u64;
i64_ge_bigfrac, i64_le_bigfrac, i64_gt_bigfrac, i64_lt_bigfrac, i64, BigFraction, bigfrac_ge_i64, bigfrac_le_i64, bigfrac_gt_i64, bigfrac_lt_i64;
f64_ge_bigfrac, f64_le_bigfrac, f64_gt_bigfrac, f64_lt_bigfrac, f64, BigFraction, bigfrac_ge_f64, bigfrac_le_f64, bigfrac_gt_f64, bigfrac_lt_f64;
);
pub(crate) fn is_multiple_of_bigfrac(value: &BigFraction, multiple: &BigFraction) -> bool {
if value.is_zero() {
return true;
}
if multiple.is_zero() {
return false;
}
(value / multiple).denom().is_none_or(fraction::One::is_one)
}
}
#[cfg(all(test, feature = "arbitrary-precision"))]
mod tests {
use super::bignum;
use fraction::BigFraction;
use num_bigint::BigInt;
use serde_json::{Number, Value};
use test_case::test_case;
fn number_from_str(raw: &str) -> Number {
match serde_json::from_str::<Value>(raw).expect("valid JSON number") {
Value::Number(num) => num,
_ => unreachable!(),
}
}
#[test]
fn bigint_parses_scientific_integer() {
let num = number_from_str("1e19");
let parsed = bignum::try_parse_bigint(&num).expect("parsed bigint");
assert_eq!(
parsed,
BigInt::parse_bytes(b"10000000000000000000", 10).unwrap()
);
}
#[test]
fn bigint_rejects_non_integer_scientific() {
let num = number_from_str("1.25e1");
assert!(bignum::try_parse_bigint(&num).is_none());
}
#[test]
fn bigfraction_parses_scientific_decimal() {
let num = number_from_str("1.5e-5");
let parsed = bignum::try_parse_bigfraction(&num).expect("parsed bigfraction");
let expected =
BigFraction::from(BigInt::from(3)) / BigFraction::from(BigInt::from(200_000));
assert_eq!(parsed, expected);
}
#[test]
fn bigfraction_skips_scientific_integer() {
let num = number_from_str("3e4");
assert!(bignum::try_parse_bigfraction(&num).is_none());
}
const HUGE_POSITIVE: &str = "99999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999.5";
const HUGE_NEGATIVE: &str = "-99999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999999.5";
#[test_case(r#"{"minimum": 0}"#, HUGE_POSITIVE, true ; "huge_positive_vs_minimum_0")]
#[test_case(r#"{"minimum": 0}"#, HUGE_NEGATIVE, false ; "huge_negative_vs_minimum_0")]
#[test_case(r#"{"maximum": 1000}"#, HUGE_POSITIVE, false ; "huge_positive_vs_maximum_1000")]
#[test_case(r#"{"maximum": 0}"#, HUGE_NEGATIVE, true ; "huge_negative_vs_maximum_0")]
#[test_case(r#"{"exclusiveMinimum": 0}"#, HUGE_POSITIVE, true ; "huge_positive_vs_exclusive_minimum_0")]
#[test_case(r#"{"exclusiveMinimum": 0}"#, HUGE_NEGATIVE, false ; "huge_negative_vs_exclusive_minimum_0")]
#[test_case(r#"{"exclusiveMaximum": 1000}"#, HUGE_POSITIVE, false ; "huge_positive_vs_exclusive_maximum_1000")]
#[test_case(r#"{"exclusiveMaximum": 0}"#, HUGE_NEGATIVE, true ; "huge_negative_vs_exclusive_maximum_0")]
#[test_case(r#"{"minimum": -1000, "maximum": 1000}"#, HUGE_POSITIVE, false ; "huge_positive_outside_range")]
#[test_case(r#"{"minimum": -1000, "maximum": 1000}"#, HUGE_NEGATIVE, false ; "huge_negative_outside_range")]
#[test_case(r#"{"multipleOf": 0.5}"#, HUGE_POSITIVE, true ; "huge_positive_multiple_of_0_5")]
#[test_case(r#"{"multipleOf": 0.5}"#, HUGE_NEGATIVE, true ; "huge_negative_multiple_of_0_5")]
#[test_case(r#"{"multipleOf": 2}"#, HUGE_POSITIVE, false ; "huge_positive_not_multiple_of_2")]
fn test_huge_decimal_validation(schema_json: &str, instance_value: &str, expected_valid: bool) {
let schema: serde_json::Value = serde_json::from_str(schema_json).unwrap();
let validator = crate::validator_for(&schema).unwrap();
let instance = serde_json::from_str(instance_value).unwrap();
assert_eq!(validator.is_valid(&instance), expected_valid);
}
}