use num_bigint::{BigInt, BigUint};
use num_traits::Zero;
use crate::result::{illegal_operation, IonError};
use crate::types::UInt;
use std::convert::TryFrom;
use std::fmt::{Display, Formatter};
use std::ops::{MulAssign, Neg};
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Debug)]
pub enum Sign {
Negative,
Positive,
}
#[derive(Debug, Clone, Ord, PartialOrd, Eq, PartialEq)]
pub struct Coefficient {
pub(crate) sign: Sign,
pub(crate) magnitude: UInt,
}
impl Coefficient {
pub(crate) fn new<I: Into<UInt>>(sign: Sign, magnitude: I) -> Self {
let magnitude = magnitude.into();
Coefficient { sign, magnitude }
}
pub(crate) fn sign(&self) -> Sign {
self.sign
}
pub(crate) fn magnitude(&self) -> &UInt {
&self.magnitude
}
pub(crate) fn number_of_decimal_digits(&self) -> u64 {
self.magnitude.number_of_decimal_digits()
}
pub(crate) fn negative_zero() -> Self {
Coefficient {
sign: Sign::Negative,
magnitude: UInt::U64(0),
}
}
pub(crate) fn is_negative_zero(&self) -> bool {
match (self.sign, &self.magnitude) {
(Sign::Negative, UInt::U64(0)) => true,
(Sign::Negative, UInt::BigUInt(b)) if b.is_zero() => true,
_ => false,
}
}
pub(crate) fn is_positive_zero(&self) -> bool {
match (self.sign, &self.magnitude) {
(Sign::Positive, UInt::U64(0)) => true,
(Sign::Positive, UInt::BigUInt(b)) if b.is_zero() => true,
_ => false,
}
}
pub(crate) fn is_zero(&self) -> bool {
match (self.sign, &self.magnitude) {
(_, UInt::U64(0)) => true,
(_, UInt::BigUInt(b)) if b.is_zero() => true,
_ => false,
}
}
pub(crate) fn as_i64(&self) -> Option<i64> {
match self.magnitude {
UInt::U64(unsigned) => match i64::try_from(unsigned) {
Ok(signed) => match self.sign {
Sign::Negative => Some(signed.neg()), Sign::Positive => Some(signed),
},
Err(_) => None,
},
UInt::BigUInt(_) => None,
}
}
}
macro_rules! impl_coefficient_from_unsigned_int_types {
($($t:ty),*) => ($(
impl From<$t> for Coefficient {
fn from(value: $t) -> Coefficient {
Coefficient::new(Sign::Positive, value)
}
}
)*)
}
impl_coefficient_from_unsigned_int_types!(u8, u16, u32, u64, u128, usize, BigUint);
macro_rules! impl_coefficient_from_signed_int_types {
($($t:ty),*) => ($(
impl From<$t> for Coefficient {
fn from(value: $t) -> Coefficient {
let sign = if value < <$t>::zero() { Sign::Negative } else { Sign::Positive };
Coefficient::new(sign, value)
}
}
)*)
}
impl_coefficient_from_signed_int_types!(i8, i16, i32, i64, i128, isize);
impl TryFrom<Coefficient> for BigInt {
type Error = IonError;
fn try_from(value: Coefficient) -> Result<Self, Self::Error> {
if value.is_negative_zero() {
illegal_operation("Cannot convert negative zero Decimal to BigInt")?;
}
let mut big_int: BigInt = match value.magnitude {
UInt::U64(m) => m.into(),
UInt::BigUInt(m) => m.into(),
};
if value.sign == Sign::Negative {
big_int.mul_assign(-1);
}
Ok(big_int)
}
}
impl TryFrom<BigInt> for Coefficient {
type Error = IonError;
fn try_from(value: BigInt) -> Result<Self, Self::Error> {
let (sign, magnitude) = value.into_parts();
let sign = match sign {
num_bigint::Sign::Minus => Sign::Negative,
num_bigint::Sign::Plus => Sign::Positive,
num_bigint::Sign::NoSign => {
if magnitude.is_zero() {
Sign::Positive
} else {
return illegal_operation(
"Cannot convert sign-less non-zero BigInt to Decimal.",
);
}
}
};
Ok(Coefficient::new(sign, magnitude))
}
}
impl Display for Coefficient {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self.sign {
Sign::Positive => {}
Sign::Negative => write!(f, "-")?,
};
match &self.magnitude {
UInt::U64(m) => write!(f, "{}", *m),
UInt::BigUInt(m) => write!(f, "{m}"),
}
}
}
#[cfg(test)]
mod coefficient_tests {
use crate::ion_data::IonEq;
use num_bigint::BigUint;
use crate::types::{Coefficient, Decimal};
fn eq_test<I1, I2>(c1: I1, c2: I2)
where
I1: Into<Coefficient>,
I2: Into<Coefficient>,
{
let c1 = c1.into();
let c2 = c2.into();
assert_eq!(c1, c2);
}
#[test]
fn test_coefficient_eq() {
eq_test(0u64, 0u64);
eq_test(0u64, BigUint::from(0u64));
eq_test(BigUint::from(0u64), 0u64);
eq_test(BigUint::from(0u64), BigUint::from(0u64));
eq_test(u64::MAX, u64::MAX);
eq_test(u64::MAX, BigUint::from(u64::MAX));
eq_test(BigUint::from(u64::MAX), u64::MAX);
eq_test(BigUint::from(u64::MAX), BigUint::from(u64::MAX));
eq_test(BigUint::from(u128::MAX), BigUint::from(u128::MAX));
}
#[test]
fn test_negative_zero_eq() {
let neg_zero = Decimal::new(Coefficient::negative_zero(), 0);
let pos_zero = Decimal::new(0, 0);
assert_eq!(neg_zero, neg_zero);
assert!(neg_zero.ion_eq(&neg_zero));
assert_eq!(neg_zero, pos_zero);
assert!(!neg_zero.ion_eq(&pos_zero));
assert_eq!(pos_zero, pos_zero);
assert!(pos_zero.ion_eq(&pos_zero));
}
}