bigdecimal 0.4.10

Arbitrary precision decimal numbers
Documentation
//! Addition operator trait implementation
//!

use super::*;
use stdlib::borrow::ToOwned;


impl Add<BigDecimal> for BigDecimal {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimal) -> BigDecimal {
        arithmetic::addition::add_bigdecimals(self, rhs)
    }
}

impl<'a, T: Into<BigDecimalRef<'a>>> Add<T> for BigDecimal {
    type Output = BigDecimal;

    fn add(mut self, rhs: T) -> BigDecimal {
        self.add_assign(rhs);
        self
    }
}

impl Add<BigInt> for BigDecimal {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigInt) -> BigDecimal {
        self + BigDecimal::from(rhs)
    }
}



impl Add<BigDecimal> for &'_ BigDecimal {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimal) -> BigDecimal {
        rhs + self
    }
}

impl<'a, T: Into<BigDecimalRef<'a>>> Add<T> for &'_ BigDecimal {
    type Output = BigDecimal;
    fn add(self, rhs: T) -> BigDecimal {
        arithmetic::addition::add_bigdecimal_refs(self, rhs, None)
    }
}

impl Add<BigInt> for &'_ BigDecimal {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigInt) -> BigDecimal {
        self.to_ref() + rhs
    }
}


impl Add<BigDecimal> for BigDecimalRef<'_> {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimal) -> BigDecimal {
        rhs + self
    }
}

impl<'a, T: Into<BigDecimalRef<'a>>> Add<T> for BigDecimalRef<'_> {
    type Output = BigDecimal;
    fn add(self, rhs: T) -> BigDecimal {
        arithmetic::addition::add_bigdecimal_refs(self, rhs, None)
    }
}

impl Add<BigInt> for BigDecimalRef<'_> {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigInt) -> BigDecimal {
        self + BigDecimal::from(rhs)
    }
}


impl Add<BigDecimal> for BigInt {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimal) -> BigDecimal {
        BigDecimal::from(self) + rhs
    }
}

impl Add<&BigDecimal> for BigInt {
    type Output = BigDecimal;

    fn add(self, rhs: &BigDecimal) -> BigDecimal {
        BigDecimal::from(self) + rhs
    }
}

impl Add<BigDecimalRef<'_>> for BigInt {
    type Output = BigDecimal;

    fn add(self, rhs: BigDecimalRef<'_>) -> BigDecimal {
        BigDecimal::from(self) + rhs
    }
}


impl Add<BigDecimal> for &BigInt {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimal) -> BigDecimal {
        rhs + self
    }
}

impl Add<&BigDecimal> for &BigInt {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: &BigDecimal) -> BigDecimal {
        rhs + self
    }
}

impl Add<BigDecimalRef<'_>> for &BigInt {
    type Output = BigDecimal;

    #[inline]
    fn add(self, rhs: BigDecimalRef<'_>) -> BigDecimal {
        rhs + self
    }
}


impl AddAssign<BigDecimal> for BigDecimal {
    fn add_assign(&mut self, rhs: BigDecimal) {
        arithmetic::addition::addassign_bigdecimals(self, rhs)
    }
}

impl<'a, N: Into<BigDecimalRef<'a>>> AddAssign<N> for BigDecimal {
    #[inline]
    fn add_assign(&mut self, rhs: N) {
        arithmetic::addition::addassign_bigdecimal_ref(self, rhs)
    }
}

impl AddAssign<BigInt> for BigDecimal {
    #[inline]
    fn add_assign(&mut self, rhs: BigInt) {
        self.add_assign(BigDecimal::from(rhs));
    }
}


#[cfg(test)]
mod test {
    use super::*;
    use paste::paste;

    macro_rules! impl_case {
        ( $name:ident: $a:literal + $b:literal => $c:literal ) => {
            #[test]
            fn $name() {
                let a: BigDecimal = $a.parse().unwrap();
                let b: BigDecimal = $b.parse().unwrap();
                let c: BigDecimal = $c.parse().unwrap();

                assert_eq!(c, a.clone() + b.clone());
                assert_eq!(c, a.clone() + b.to_ref());
                assert_eq!(c, a.clone() + &b);

                assert_eq!(c, &a + b.clone());
                assert_eq!(c, &a + b.to_ref());
                assert_eq!(c, &a + &b);

                assert_eq!(c, a.to_ref() + b.clone());
                assert_eq!(c, a.to_ref() + b.to_ref());
                assert_eq!(c, a.to_ref() + &b);

                // Reversed

                assert_eq!(c, b.clone() + a.clone());
                assert_eq!(c, b.clone() + a.to_ref());
                assert_eq!(c, b.clone() + &a);

                assert_eq!(c, &b + a.clone());
                assert_eq!(c, &b + a.to_ref());
                assert_eq!(c, &b + &a);

                assert_eq!(c, b.to_ref() + a.clone());
                assert_eq!(c, b.to_ref() + a.to_ref());
                assert_eq!(c, b.to_ref() + &a);

                let mut n = a.clone();
                n += b.clone();
                assert_eq!(c, n);

                let mut n = a.clone();
                n += &b;
                assert_eq!(c, n);

                let mut n = a.clone();
                n += b.to_ref();
                assert_eq!(c, n);

                let mut n = b.clone();
                n += a.clone();
                assert_eq!(c, n);

                let mut n = b.clone();
                n += &a;
                assert_eq!(c, n);

                let mut n = b.clone();
                n += a.to_ref();
                assert_eq!(c, n);
            }
        };
        ( $name:ident: $a:literal + (int) $b:literal => $c:literal ) => {
            #[test]
            fn $name() {
                let a: BigDecimal = $a.parse().unwrap();
                let b: BigInt = $b.parse().unwrap();
                let c: BigDecimal = $c.parse().unwrap();

                assert_eq!(c, a.clone() + b.clone());
                assert_eq!(c, a.clone() + &b);
                assert_eq!(c, &a + &b);
                assert_eq!(c, &a + b.clone());
                assert_eq!(c, a.to_ref() + &b);

                assert_eq!(c, b.clone() + a.clone());
                assert_eq!(c, b.clone() + a.to_ref());
                assert_eq!(c, b.clone() + &a);

                assert_eq!(c, &b + a.clone());
                assert_eq!(c, &b + a.to_ref());
                assert_eq!(c, &b + &a);

                let mut n = a.clone();
                n += b.clone();
                assert_eq!(c, n);

                let mut n = a.clone();
                n += &b;
                assert_eq!(c, n);
            }
        };
    }

    impl_case!(case_1234en2_1234en3: "12.34" + "1.234" => "13.574");
    impl_case!(case_1234en2_n1234en3: "12.34" + "-1.234" => "11.106");
    impl_case!(case_1234en2_n1234en2: "12.34" + "-12.34" => "0");
    impl_case!(case_1234e6_1234en6: "1234e6" + "1234e-6" => "1234000000.001234");
    impl_case!(case_1234en6_1234e6: "1234e6" + "1234e-6" => "1234000000.001234");
    impl_case!(case_18446744073709551616_1: "18446744073709551616.0" + "1" => "18446744073709551617");
    impl_case!(case_184467440737e3380_0: "184467440737e3380" + "0" => "184467440737e3380");
    impl_case!(case_0_776en1: "0" + "77.6" => "77.6");

    impl_case!(case_80802295e5_int0: "80802295e5" + (int)"0" => "80802295e5");
    impl_case!(case_239200en4_intneg101: "23.9200" + (int)"-101" => "-77.0800");
    impl_case!(case_46636423395767125en15_int0: "46.636423395767125" + (int)"123" => "169.636423395767125");


    #[cfg(property_tests)]
    mod prop {
        use super::*;
        use proptest::*;
        use num_traits::FromPrimitive;

        proptest! {
            #[test]
            fn add_refs_and_owners(f: f32, g: f32) {
                // ignore non-normal numbers
                prop_assume!(f.is_normal());
                prop_assume!(g.is_normal());

                let a = BigDecimal::from_f32(f).unwrap();
                let b = BigDecimal::from_f32(g).unwrap();
                let own_plus_ref = a.clone() + &b;
                let ref_plus_own = &a + b.clone();

                let mut c = a.clone();
                c += &b;

                let mut d = a.clone();
                d += b;

                prop_assert_eq!(&own_plus_ref, &ref_plus_own);
                prop_assert_eq!(&c, &ref_plus_own);
                prop_assert_eq!(&d, &ref_plus_own);
            }

            #[test]
            fn addition_is_communative(f: f32, g: f32) {
                // ignore non-normal numbers
                prop_assume!(f.is_normal());
                prop_assume!(g.is_normal());

                let a = BigDecimal::from_f32(f).unwrap();
                let b = BigDecimal::from_f32(g).unwrap();
                let a_plus_b = &a + &b;
                let b_plus_a = &b + &a;

                prop_assert_eq!(a_plus_b, b_plus_a)
            }

            #[test]
            fn addition_is_associative(f: f32, g: f32, h: f32) {
                // ignore non-normal numbers
                prop_assume!(f.is_normal());
                prop_assume!(g.is_normal());
                prop_assume!(h.is_normal());

                let a = BigDecimal::from_f32(f).unwrap();
                let b = BigDecimal::from_f32(g).unwrap();
                let c = BigDecimal::from_f32(h).unwrap();

                let ab = &a + &b;
                let ab_c = ab + &c;

                let bc = &b + &c;
                let a_bc = a + bc;

                prop_assert_eq!(ab_c, a_bc)
            }
        }
    }
}