uom 0.38.0

Units of measurement
Documentation
//! Tests for `uom` macros.

#[allow(unused_imports)]
use self::fmt::{Arguments, QuantityArguments};
use self::length::{kilometer, meter};
use self::mass::kilogram;
use self::thermodynamic_temperature::{degree_fahrenheit, kelvin};
use crate::fmt::DisplayStyle;
use crate::lib::fmt::Debug;
use crate::lib::marker::PhantomData;
#[allow(unused_imports)]
use crate::num::{Float, FromPrimitive, One, Saturating, Signed, Zero};
use crate::str::ParseQuantityError;
use crate::{ConstantOp, Conversion, ConversionFactor};
use quickcheck::TestResult;
#[allow(unused_imports)]
use typenum::{N1, P1, P2, P3, Z0};

#[macro_use]
mod length {
    quantity! {
        quantity: Length; "length";
        dimension: Q<P1, Z0, Z0>;
        units {
            @kilometer: 1.0_E3; "km", "kilometer", "kilometers";
            @meter: 1.0_E0; "m", "meter", "meters";
        }
    }
}

#[macro_use]
mod mass {
    quantity! {
        quantity: Mass; "mass";
        dimension: Q<Z0, P1, Z0>;
        units {
            @kilogram: 1.0_E0; "kg", "kilogram", "kilograms";
        }
    }
}

#[macro_use]
mod thermodynamic_temperature {
    quantity! {
        quantity: ThermodynamicTemperature; "thermodynamic temperature";
        dimension: Q<Z0, Z0, P1>;
        units {
            @kelvin: 1.0_E0; "K", "kelvin", "kelvins";
            @degree_fahrenheit: 5.0_E0 / 9.0_E0, 459.67_E0; "°F", "degree Fahrenheit",
                "degrees Fahrenheit";
        }
    }
}

system! {
    quantities: Q {
        length: meter, L;
        mass: kilogram, M;
        thermodynamic_temperature: kelvin, Th;
    }
    units: U {
        mod length::Length,
        mod mass::Mass,
        mod thermodynamic_temperature::ThermodynamicTemperature,
    }
}

/// Test trait to allow tests to perform storage-type sensitive comparisons.
#[rustfmt::skip]
#[allow(dead_code)] // Not all functions are called based on feature selection.
pub trait Test:
    Debug
    + Sized
    + PartialEq
{
    /// Assert that `lhs` and `rhs` are exactly equal.
    fn assert_eq(lhs: &Self, rhs: &Self) {
        assert_eq!(lhs, rhs);
    }

    /// Assert that `lhs` and `rhs` are approximately equal for floating point types or exactly
    /// equal for other types.
    fn assert_approx_eq(lhs: &Self, rhs: &Self) {
        Test::assert_eq(lhs, rhs);
    }

    /// Exactly compare `lhs` and `rhs` and return the result.
    fn eq(lhs: &Self, rhs: &Self) -> bool {
        lhs == rhs
    }

    /// Approximately compare `lhs` and `rhs` for floating point types or exactly compare for other
    /// types and return the result.
    fn approx_eq(lhs: &Self, rhs: &Self) -> bool {
        Test::eq(lhs, rhs)
    }
}

mod test_trait {
    storage_types! {
        types: Float;

        use crate::num::Float;

        // const EPSILON: V = 64.0 * V::epsilon(); //error[E0015]; calls in constants are limited...
        const EPS_FACTOR: V = 0.5;
        const ULPS: u32 = 3;

        impl super::super::Test for V {
            /// Assert that `lhs` and `rhs` are exactly equal.
            fn assert_eq(lhs: &Self, rhs: &Self) {
                match (lhs.is_nan(), rhs.is_nan()) {
                    (true, true) => {}
                    _ => { assert_eq!(lhs, rhs); }
                }
            }

            /// Assert that `lhs` and `rhs` are approximately equal for floating point types or
            /// exactly equal for other types.
            fn assert_approx_eq(lhs: &Self, rhs: &Self) {
                match (lhs.is_nan(), rhs.is_nan()) {
                    (true, true) => {}
                    _ => {
                        assert_ulps_eq!(lhs, rhs, epsilon = EPS_FACTOR * V::epsilon(),
                            max_ulps = ULPS);
                    }
                }
            }

            /// Exactly compare `lhs` and `rhs` and return the result.
            fn eq(lhs: &Self, rhs: &Self) -> bool {
                (lhs.is_nan() && rhs.is_nan())
                    || lhs == rhs
            }

            /// Approximately compare `lhs` and `rhs` for floating point types or exactly compare
            /// for other types and return the result.
            fn approx_eq(lhs: &Self, rhs: &Self) -> bool {
                (lhs.is_nan() && rhs.is_nan())
                    || ulps_eq!(lhs, rhs, epsilon = EPS_FACTOR * V::epsilon(), max_ulps = ULPS)
            }
        }
    }

    storage_types! {
        types: PrimInt, BigInt, BigUint, Ratio;

        impl super::super::Test for V {}
    }

    storage_types! {
        types: Complex;

        use crate::num::Float;

        // const EPSILON: VV = 64.0 * VV::epsilon(); //error[E0015]; calls in constants are limited...
        const EPS_FACTOR: VV = 0.5;
        const ULPS: u32 = 3;

        impl super::super::Test for V {
            /// Assert that `lhs` and `rhs` are exactly equal.
            fn assert_eq(lhs: &Self, rhs: &Self) {
                match (lhs.is_nan(), rhs.is_nan()) {
                    (true, true) => {}
                    _ => { assert_eq!(lhs, rhs); }
                }
            }

            /// Assert that `lhs` and `rhs` are approximately equal for floating point types or
            /// exactly equal for other types.
            fn assert_approx_eq(lhs: &Self, rhs: &Self) {
                match (lhs.is_nan(), rhs.is_nan()) {
                    (true, true) => {}
                    _ => {
                        assert_ulps_eq!(lhs.re, rhs.re, epsilon = EPS_FACTOR * VV::epsilon(),
                            max_ulps = ULPS);
                        assert_ulps_eq!(lhs.im, rhs.im, epsilon = EPS_FACTOR * VV::epsilon(),
                            max_ulps = ULPS);
                    }
                }
            }

            /// Exactly compare `lhs` and `rhs` and return the result.
            fn eq(lhs: &Self, rhs: &Self) -> bool {
                (lhs.is_nan() && rhs.is_nan())
                    || lhs == rhs
            }

            /// Approximately compare `lhs` and `rhs` for floating point types or exactly compare
            /// for other types and return the result.
            fn approx_eq(lhs: &Self, rhs: &Self) -> bool {
                (lhs.is_nan() && rhs.is_nan())
                    || ulps_eq!(lhs.re, rhs.re,
                        epsilon = EPS_FACTOR * VV::epsilon(), max_ulps = ULPS)
                    || ulps_eq!(lhs.im, rhs.im,
                        epsilon = EPS_FACTOR * VV::epsilon(), max_ulps = ULPS)
            }
        }
    }
}

#[derive(Clone, Debug)]
pub struct A<V> {
    v: V,
}

impl<V> crate::lib::ops::Deref for A<V> {
    type Target = V;

    fn deref(&self) -> &Self::Target {
        &self.v
    }
}

mod a_struct {
    storage_types! {
        types: Float, PrimInt;

        use super::super::A;

        impl quickcheck::Arbitrary for A<V> {
            fn arbitrary(g: &mut quickcheck::Gen) -> Self {
                A { v: V::arbitrary(g), }
            }
        }
    }

    storage_types! {
        types: BigInt, BigUint, Ratio;

        use crate::num::FromPrimitive;
        use super::super::A;

        impl quickcheck::Arbitrary for A<V> {
            fn arbitrary(g: &mut quickcheck::Gen) -> Self {
                A {
                    v: loop {
                        let v = V::from_f64(<f64 as quickcheck::Arbitrary>::arbitrary(g));

                        if let Some(a) = v {
                            break a;
                        }
                    },
                }
            }
        }
    }

    storage_types! {
        types: Complex;

        use super::super::A;

        impl quickcheck::Arbitrary for A<V> {
            fn arbitrary(g: &mut quickcheck::Gen) -> Self {
                A {
                    v: V::new(
                        <VV as quickcheck::Arbitrary>::arbitrary(g),
                        <VV as quickcheck::Arbitrary>::arbitrary(g)),
                }
            }
        }
    }
}

mod asserts;
mod quantities;
mod quantity;
mod system;