time-core 0.1.9

This crate is an implementation detail and should not be relied upon directly.
Documentation
//! Conversion between units of time.

use self::sealed::{DefaultOutput, MultipleOf};

/// Given the list of types, stringify them as a list.
macro_rules! stringify_outputs {
    (@inner $first:ty) => {
        concat!("or `", stringify!($first), "`")
    };
    (@inner $first:ty, $($t:ty),+) => {
        concat!(stringify_outputs!($first), ", ", stringify_outputs!(@inner $($t),+))
    };
    ($first:ty) => {
        concat!("`", stringify!($first), "`")
    };
    ($($t:ty),+) => {
        stringify_outputs!(@inner $($t),+)
    };
}

/// Declare all unit types.
macro_rules! declare_types {
    ($($t:ident ($str:literal))+) => {$(
        #[doc = concat!("A unit of time representing exactly `N` ", $str, "s.")]
        #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
        pub struct $t<const N: u128 = 1>;
    )+};
}

/// Implement `Per` for all relevant types. Identity implementations are automatic.
macro_rules! impl_per {
    ($($t:ident per {$(
        $larger:ident : [$default_output:ty]

        $($int_output:ty)|+ = $int_value:expr;
        $($float_output:ty)|+ = $float_value:expr;
    )+})*) => {$(
        impl $t<1> {
            #[doc = concat!("Obtain the number of times `", stringify!($t), "` can fit into `T`.")]
            #[doc = concat!("If `T` is smaller than `", stringify!($t), "`, the code will fail to")]
            /// compile. The return type is the smallest unsigned integer type that can represent
            /// the value.
            ///
            /// Valid calls:
            ///
            $(#[doc = concat!(
                "  - `", stringify!($t), "::per(", stringify!($larger), ")` (returns `",
                stringify!($default_output), "`)"
            )])+
            #[inline]
            pub const fn per<T>(_larger: T) -> <T as DefaultOutput<Self>>::Output
            where
                T: MultipleOf<Self, T::Output> + DefaultOutput<Self> + Copy,
            {
                T::VALUE
            }

            #[doc = concat!("Obtain the number of times `", stringify!($t), "` can fit into `T`.")]
            #[doc = concat!("If `T` is smaller than `", stringify!($t), "`, the code will fail to")]
            /// compile. The return type is any primitive numeric type that can represent the value.
            ///
            /// Valid calls:
            ///
            $(#[doc = concat!(
                "  - `", stringify!($t), "::per(", stringify!($larger), ")` (returns ",
                stringify_outputs!($($int_output),+ , $($float_output),+), ")"
            )])+
            #[inline]
            pub const fn per_t<Output>(larger: impl MultipleOf<Self, Output> + Copy) -> Output {
                multiple_of_value(larger)
            }
        }

        $(
            $(impl MultipleOf<$t, $int_output> for $larger {
                const VALUE: $int_output = $int_value;
            })+

            $(impl MultipleOf<$t, $float_output> for $larger {
                const VALUE: $float_output = $float_value;
            })+

            impl DefaultOutput<$t> for $larger {
                type Output = $default_output;
            }
        )+
    )*};
}

/// Implement `PartialEq` and `PartialOrd` between unit types.
macro_rules! impl_partial_eq_ord {
    ($($self:ident {
        $(> $($bigger:ident)+;)?
        $(< $($smaller:ident)+;)?
    })*) => {$(
        $($(
            impl<const N: u128> PartialEq<$bigger<N>> for $self<N> {
                #[inline]
                fn eq(&self, _: &$bigger<N>) -> bool {
                    false
                }
            }

            impl<const N: u128> PartialOrd<$bigger<N>> for $self<N> {
                #[inline]
                fn partial_cmp(&self, _: &$bigger<N>) -> Option<core::cmp::Ordering> {
                    Some(core::cmp::Ordering::Greater)
                }
            }
        )+)?

        $($(
            impl<const N: u128> PartialEq<$smaller<N>> for $self<N> {
                #[inline]
                fn eq(&self, _: &$smaller<N>) -> bool {
                    false
                }
            }

            impl<const N: u128> PartialOrd<$smaller<N>> for $self<N> {
                #[inline]
                fn partial_cmp(&self, _: &$smaller<N>) -> Option<core::cmp::Ordering> {
                    Some(core::cmp::Ordering::Less)
                }
            }
        )+)?
    )*};
}

/// Symmetrically implement `PartialEq` and `PartialOrd` between `Unit` and all unit types.
macro_rules! impl_partial_eq_ord_for_unit {
    ($($ty:ident)+) => {$(
        impl<const N: u128> PartialEq<$ty<N>> for Unit<N> {
            #[inline]
            fn eq(&self, _: &$ty<N>) -> bool {
                *self == Self::$ty
            }
        }

        impl<const N: u128> PartialEq<Unit<N>> for $ty<N> {
            #[inline]
            fn eq(&self, other: &Unit<N>) -> bool {
                *other == Unit::$ty
            }
        }

        impl<const N: u128> PartialOrd<$ty<N>> for Unit<N> {
            #[inline]
            fn partial_cmp(&self, _: &$ty<N>) -> Option<core::cmp::Ordering> {
                self.partial_cmp(&Unit::$ty)
            }
        }

        impl<const N: u128> PartialOrd<Unit<N>> for $ty<N> {
            #[inline]
            fn partial_cmp(&self, other: &Unit<N>) -> Option<core::cmp::Ordering> {
                Unit::$ty.partial_cmp(other)
            }
        }
    )+};
}

mod sealed {
    /// A trait for defining the ratio of two units of time.
    ///
    /// This trait is used to implement the `per` method on the various structs.
    #[diagnostic::on_unimplemented(message = "`{Self}` is not an integer multiple of `{T}`")]
    pub trait MultipleOf<T, Output> {
        /// The number of one unit of time in the other.
        const VALUE: Output;
    }

    /// A trait for defining the default output type for the `per` method.
    pub trait DefaultOutput<T> {
        /// The default output type for the `per` method.
        type Output;
    }
}

// Split this out to a separate function to permit naming `T` while also using `impl Trait` as a
// parameter in the public API.`
const fn multiple_of_value<T, U, Output>(_: T) -> Output
where
    T: MultipleOf<U, Output> + Copy,
{
    T::VALUE
}

declare_types! {
    Nanosecond ("nanosecond")
    Microsecond ("microsecond")
    Millisecond ("millisecond")
    Second ("second")
    Minute ("minute")
    Hour ("hour")
    Day ("day")
    Week ("week")
}

impl_per! {
    Nanosecond per {
        Nanosecond: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Microsecond: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 1_000; f32|f64 = 1_000.;
        Millisecond: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 1_000_000; f32|f64 = 1_000_000.;
        Second:
            [u32] u32|u64|u128|usize|i32|i64|i128|isize = 1_000_000_000; f32|f64 = 1_000_000_000.;
        Minute: [u64] u64|u128|i64|i128 = 60_000_000_000; f32|f64 = 60_000_000_000.;
        Hour: [u64] u64|u128|i64|i128 = 3_600_000_000_000; f32|f64 = 3_600_000_000_000.;
        Day: [u64] u64|u128|i64|i128 = 86_400_000_000_000; f32|f64 = 86_400_000_000_000.;
        Week: [u64] u64|u128|i64|i128 = 604_800_000_000_000; f32|f64 = 604_800_000_000_000.;
    }
    Microsecond per {
        Microsecond: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Millisecond: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 1_000; f32|f64 = 1_000.;
        Second: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 1_000_000; f32|f64 = 1_000_000.;
        Minute: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 60_000_000; f32|f64 = 60_000_000.;
        Hour: [u32] u32|u64|u128|i64|i128 = 3_600_000_000; f32|f64 = 3_600_000_000.;
        Day: [u64] u64|u128|i64|i128 = 86_400_000_000; f32|f64 = 86_400_000_000.;
        Week: [u64] u64|u128|i64|i128 = 604_800_000_000; f32|f64 = 604_800_000_000.;
    }
    Millisecond per {
        Millisecond: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Second: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 1_000; f32|f64 = 1_000.;
        Minute: [u16] u16|u32|u64|u128|usize|i32|i64|i128|isize = 60_000; f32|f64 = 60_000.;
        Hour: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 3_600_000; f32|f64 = 3_600_000.;
        Day: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 86_400_000; f32|f64 = 86_400_000.;
        Week: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 604_800_000; f32|f64 = 604_800_000.;
    }
    Second per {
        Second: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Minute: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 60; f32|f64 = 60.;
        Hour: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 3_600; f32|f64 = 3_600.;
        Day: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 86_400; f32|f64 = 86_400.;
        Week: [u32] u32|u64|u128|usize|i32|i64|i128|isize = 604_800; f32|f64 = 604_800.;
    }
    Minute per {
        Minute: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Hour: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 60; f32|f64 = 60.;
        Day: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 1_440; f32|f64 = 1_440.;
        Week: [u16] u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 10_080; f32|f64 = 10_080.;
    }
    Hour per {
        Hour: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Day: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 24; f32|f64 = 24.;
        Week: [u8] u8|u16|u32|u64|u128|usize|i16|i32|i64|i128|isize = 168; f32|f64 = 168.;
    }
    Day per {
        Day: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
        Week: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 7; f32|f64 = 7.;
    }
    Week per {
        Week: [u8] u8|u16|u32|u64|u128|usize|i8|i16|i32|i64|i128|isize = 1; f32|f64 = 1.;
    }
}

impl_partial_eq_ord! {
    Nanosecond {
        < Microsecond Millisecond Second Minute Hour Day Week;
    }
    Microsecond {
        > Nanosecond;
        < Millisecond Second Minute Hour Day Week;
    }
    Millisecond {
        > Nanosecond Microsecond;
        < Second Minute Hour Day Week;
    }
    Second {
        > Nanosecond Microsecond Millisecond;
        < Minute Hour Day Week;
    }
    Minute {
        > Nanosecond Microsecond Millisecond Second;
        < Hour Day Week;
    }
    Hour {
        > Nanosecond Microsecond Millisecond Second Minute;
        < Day Week;
    }
    Day {
        > Nanosecond Microsecond Millisecond Second Minute Hour;
        < Week;
    }
    Week {
        > Nanosecond Microsecond Millisecond Second Minute Hour Day;
    }
}

/// A statically-known multiple of a given unit of time.
#[non_exhaustive]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum Unit<const N: u128 = 1> {
    #[expect(missing_docs)]
    Nanosecond,
    #[expect(missing_docs)]
    Microsecond,
    #[expect(missing_docs)]
    Millisecond,
    #[expect(missing_docs)]
    Second,
    #[expect(missing_docs)]
    Minute,
    #[expect(missing_docs)]
    Hour,
    #[expect(missing_docs)]
    Day,
    #[expect(missing_docs)]
    Week,
}

impl_partial_eq_ord_for_unit![Nanosecond Microsecond Millisecond Second Minute Hour Day Week];