use std::ops::Add;
use std::ops::AddAssign;
use std::ops::Div;
use std::ops::Mul;
use std::ops::Sub;
use std::ops::SubAssign;
use crate::DateTime;
#[cfg(feature = "macros")]
#[doc(hidden)]
pub mod __private_api {
pub use datetime_rs_macros::nanoseconds;
}
#[macro_export]
#[cfg(feature = "macros")]
#[cfg_attr(docsrs, doc(cfg(feature = "macros")))]
macro_rules! time_interval {
($($interval:tt)+) => { const {
$crate::interval::TimeInterval::from_nanoseconds(
$crate::interval::__private_api::nanoseconds!($($interval)*)
)
}}
}
#[derive(Copy, Clone, Debug, Default, Eq, PartialEq, PartialOrd, Ord)]
pub struct TimeInterval {
seconds: i64,
nanos: u32,
}
impl TimeInterval {
pub const fn new(seconds: i64, nanos: u32) -> Self {
Self { seconds, nanos }
}
pub const fn from_milliseconds(millis: i64) -> Self {
Self::new(millis.div_euclid(1_000), millis.rem_euclid(1_000) as u32 * 1_000_000)
}
pub const fn from_microseconds(micros: i64) -> Self {
Self::new(micros.div_euclid(1_000_000), micros.rem_euclid(1_000_000) as u32 * 1_000)
}
pub const fn from_nanoseconds(nanos: i128) -> Self {
Self::new(nanos.div_euclid(1_000_000_000) as i64, nanos.rem_euclid(1_000_000_000) as u32)
}
pub const fn seconds(&self) -> i64 {
self.seconds
}
pub const fn nanoseconds(&self) -> u32 {
self.nanos
}
pub const fn as_milliseconds(&self) -> i64 {
self.seconds * 1_000 + (self.nanos / 1_000_000) as i64
}
pub const fn as_microseconds(&self) -> i64 {
self.seconds * 1_000_000 + (self.nanos / 1_000) as i64
}
pub const fn as_nanoseconds(&self) -> i128 {
self.seconds as i128 * 1_000_000_000 + self.nanos as i128
}
}
impl Add<TimeInterval> for DateTime {
type Output = DateTime;
fn add(self, rhs: TimeInterval) -> Self::Output {
let seconds = self.seconds + rhs.seconds + ((self.nanos + rhs.nanos) / 1_000_000_000) as i64;
let nanos = (self.nanos + rhs.nanos) % 1_000_000_000;
Self {
seconds,
nanos,
#[cfg(feature = "tz")]
tz: self.tz,
}
}
}
impl AddAssign<TimeInterval> for DateTime {
fn add_assign(&mut self, rhs: TimeInterval) {
self.seconds += rhs.seconds;
self.nanos += rhs.nanos;
while self.nanos >= 1_000_000_000 {
self.seconds += 1;
self.nanos -= 1_000_000_000;
}
}
}
impl Sub<TimeInterval> for DateTime {
type Output = DateTime;
#[allow(clippy::suspicious_arithmetic_impl)]
fn sub(self, rhs: TimeInterval) -> Self::Output {
let mut seconds = self.seconds - rhs.seconds;
let nanos = self.nanos.checked_sub(rhs.nanos).unwrap_or_else(|| {
seconds -= 1;
self.nanos + 1_000_000_000 - rhs.nanos
});
Self {
seconds,
nanos,
#[cfg(feature = "tz")]
tz: self.tz,
}
}
}
impl SubAssign<TimeInterval> for DateTime {
fn sub_assign(&mut self, rhs: TimeInterval) {
self.seconds -= rhs.seconds;
match self.nanos >= rhs.nanos {
true => self.nanos -= rhs.nanos,
false => {
self.nanos += 1_000_000_000;
self.seconds -= 1;
self.nanos -= rhs.nanos;
},
}
}
}
impl Sub for DateTime {
type Output = TimeInterval;
#[allow(clippy::suspicious_arithmetic_impl)]
fn sub(self, rhs: Self) -> Self::Output {
let mut seconds = self.seconds - rhs.seconds;
let nanos = self.nanos.checked_sub(rhs.nanos).unwrap_or_else(|| {
seconds -= 1;
self.nanos + 1_000_000_000 - rhs.nanos
});
TimeInterval { seconds, nanos }
}
}
impl<I: Into<i128>> Mul<I> for TimeInterval {
type Output = Self;
fn mul(self, rhs: I) -> Self::Output {
Self::from_nanoseconds(self.as_nanoseconds() * rhs.into())
}
}
impl<I: Into<i128>> Div<I> for TimeInterval {
type Output = Self;
fn div(self, rhs: I) -> Self::Output {
Self::from_nanoseconds(self.as_nanoseconds() / rhs.into())
}
}
impl Div for TimeInterval {
type Output = f64;
fn div(self, rhs: Self) -> Self::Output {
self.as_nanoseconds() as f64 / rhs.as_nanoseconds() as f64
}
}
#[cfg(feature = "syn")]
mod syn {
use datetime_rs_codegen::Delta;
use syn::Result;
use syn::parse::Parse;
use syn::parse::ParseStream;
use super::TimeInterval;
#[cfg_attr(docsrs, doc(cfg(feature = "syn")))]
impl Parse for TimeInterval {
fn parse(input: ParseStream) -> Result<Self> {
let delta = Delta::parse(input)?;
Ok(Self::new(delta.seconds(), delta.nanos()))
}
}
#[cfg(test)]
mod tests {
use assert2::check;
use quote::quote;
use super::*;
#[test]
fn test_parse() -> Result<()> {
check!(syn::parse2::<TimeInterval>(quote! { 15m 30s })? == time_interval!(15m 30s));
check!(syn::parse2::<TimeInterval>(quote! { -1h 15m })? == time_interval!(-1h 15m));
Ok(())
}
}
}
#[cfg(test)]
mod tests {
use assert2::check;
use super::*;
use crate::DateTime;
#[test]
fn test_interval_macro() {
check!(time_interval!(1d) == TimeInterval::new(86_400, 0));
check!(time_interval!(2h) == TimeInterval::new(7_200, 0));
check!(time_interval!(1m) == TimeInterval::new(60, 0));
check!(time_interval!(1h30m) == TimeInterval::new(5_400, 0));
check!(time_interval!(1h 30m) == TimeInterval::new(5_400, 0));
check!(time_interval!(-1h 30m) == TimeInterval::new(-5_400, 0));
check!(time_interval!(1m 30s) == TimeInterval::new(90, 0));
check!(time_interval!(-1m 20s) == TimeInterval::new(-80, 0));
check!(time_interval!(-20s) == TimeInterval::new(-20, 0));
check!(time_interval!(-1.25s) == TimeInterval::new(-2, 750_000_000));
check!(time_interval!(1m 1.5s) == TimeInterval::new(61, 500_000_000));
check!(time_interval!(-0.5s) == TimeInterval::new(-1, 500_000_000));
check!(time_interval!(0.5s) == TimeInterval::new(0, 500_000_000));
}
#[test]
fn test_from_fractionals() {
for (millis, secs, nanos) in [(2_400, 2, 400_000_000), (-2_400, -3, 600_000_000)] {
let interval = TimeInterval::from_milliseconds(millis);
check!(interval.seconds() == secs);
check!(interval.nanoseconds() == nanos);
}
for (micros, secs, nanos) in [(2_400_000, 2, 400_000_000), (-2_400_000, -3, 600_000_000)] {
let interval = TimeInterval::from_microseconds(micros);
check!(interval.seconds() == secs);
check!(interval.nanoseconds() == nanos);
}
for (ns, s, n) in [(2_400_000_000, 2, 400_000_000), (-2_400_000_000, -3, 600_000_000)] {
let interval = TimeInterval::from_nanoseconds(ns);
check!(interval.seconds() == s);
check!(interval.nanoseconds() == n);
}
}
#[test]
fn test_add() {
check!(
datetime! { 2012-04-21 11:00:00 } + TimeInterval::new(3600, 0)
== datetime! { 2012-04-21 12:00:00 }
);
check!(
datetime! { 2012-04-21 11:00:00 } + TimeInterval::new(1800, 0)
== datetime! { 2012-04-21 11:30:00 }
);
check!(
datetime! { 2012-04-21 11:00:00 } + TimeInterval::new(0, 500_000_000)
== DateTime::ymd(2012, 4, 21).hms(11, 0, 0).nanos(500_000_000).build()
);
let incr = datetime! { 2012-04-21 11:00:00 }
+ TimeInterval::new(0, 500_000_000)
+ TimeInterval::new(0, 500_000_000);
check!(incr.seconds % 10 == 1);
check!(incr.nanos == 0);
}
#[test]
fn test_add_assign() {
let mut dt = datetime! { 2012-04-21 11:00:00 };
dt += TimeInterval::new(3600, 0);
check!(dt == datetime! { 2012-04-21 12:00:00 });
dt += TimeInterval::new(0, 750_000_000);
dt += TimeInterval::new(0, 250_000_000);
check!(dt == datetime! { 2012-04-21 12:00:01 });
}
#[test]
fn test_sub() {
check!(
datetime! { 2012-04-21 11:00:00 } - TimeInterval::new(3600, 0)
== datetime! { 2012-04-21 10:00:00 }
);
check!(
datetime! { 2012-04-21 11:00:00 } - TimeInterval::new(0, 500_000_000)
== DateTime::ymd(2012, 4, 21).hms(10, 59, 59).nanos(500_000_000).build()
);
}
#[test]
fn test_sub_assign() {
let mut dt = datetime! { 2012-04-21 11:00:00 };
dt -= TimeInterval::new(3600, 0);
check!(dt == datetime! { 2012-04-21 10:00:00 });
dt -= TimeInterval::new(0, 750_000_000);
dt -= TimeInterval::new(0, 350_000_000);
dt -= TimeInterval::new(0, 900_000_000);
check!(dt == datetime! { 2012-04-21 09:59:58 });
}
#[test]
fn test_sub_dt() {
check!(
datetime! { 2012-04-21 11:00:00 } - datetime! { 2012-04-21 10:00:00 }
== TimeInterval::new(3600, 0)
);
check!(
datetime! { 2012-04-21 11:00:00 } - datetime! { 2012-04-21 12:00:00 }
== TimeInterval::new(-3600, 0)
);
}
#[test]
fn test_mul_int() {
let interval = TimeInterval::new(3, 500_000_000) * 3;
check!(interval.seconds() == 10);
check!(interval.nanoseconds() == 500_000_000);
}
#[test]
fn test_div_int() {
let interval = TimeInterval::new(4, 500_000_000) / 3;
check!(interval.seconds() == 1);
check!(interval.nanoseconds() == 500_000_000);
}
#[test]
fn test_div() {
check!(TimeInterval::new(3600, 0) / TimeInterval::new(1800, 0) == 2.0);
check!(TimeInterval::new(-1800, 0) / TimeInterval::new(-3600, 0) == 0.5);
check!(TimeInterval::new(-1800, 0) / TimeInterval::new(3600, 0) == -0.5);
check!(TimeInterval::new(0, 3600) / TimeInterval::new(0, 1800) == 2.0);
}
#[test]
fn test_as() {
let dur = TimeInterval::new(5, 0);
check!(dur.as_milliseconds() == 5_000);
check!(dur.as_microseconds() == 5_000_000);
check!(dur.as_nanoseconds() == 5_000_000_000);
}
}