#![allow(clippy::missing_docs_in_private_items)]
#![allow(unreachable_pub)]
use core::cmp::Ordering;
use core::fmt;
use core::hash::{Hash, Hasher};
use core::mem::size_of;
use core::ops::{Add, AddAssign, Sub, SubAssign};
use core::time::Duration as StdDuration;
#[cfg(feature = "formatting")]
use std::io;
#[cfg(feature = "std")]
use std::time::SystemTime;
use crate::convert::*;
use crate::date::{MAX_YEAR, MIN_YEAR};
#[cfg(feature = "formatting")]
use crate::formatting::Formattable;
#[cfg(feature = "parsing")]
use crate::parsing::{Parsable, Parsed};
use crate::{error, util, Date, Duration, Month, Time, UtcOffset, Weekday};
#[allow(missing_debug_implementations, missing_copy_implementations)]
pub(crate) mod offset_kind {
pub enum None {}
pub enum Fixed {}
}
pub(crate) use sealed::MaybeOffset;
use sealed::*;
mod sealed {
use super::*;
#[allow(unreachable_pub)] pub trait MaybeOffsetType {}
impl MaybeOffsetType for () {}
impl MaybeOffsetType for UtcOffset {}
pub trait MaybeOffset: Sized {
#[cfg(feature = "quickcheck")]
type MemoryOffsetType: Copy + MaybeOffsetType + quickcheck::Arbitrary;
#[cfg(not(feature = "quickcheck"))]
type MemoryOffsetType: Copy + MaybeOffsetType;
type LogicalOffsetType: Copy + MaybeOffsetType;
type Self_;
const HAS_LOGICAL_OFFSET: bool =
size_of::<Self::LogicalOffsetType>() == size_of::<UtcOffset>();
const HAS_MEMORY_OFFSET: bool =
size_of::<Self::MemoryOffsetType>() == size_of::<UtcOffset>();
const STATIC_OFFSET: Option<UtcOffset>;
#[cfg(feature = "parsing")]
fn try_from_parsed(parsed: Parsed) -> Result<Self::MemoryOffsetType, error::TryFromParsed>;
}
pub trait HasLogicalOffset: MaybeOffset<LogicalOffsetType = UtcOffset> {}
impl<T: MaybeOffset<LogicalOffsetType = UtcOffset>> HasLogicalOffset for T {}
pub trait NoLogicalOffset: MaybeOffset<LogicalOffsetType = ()> {}
impl<T: MaybeOffset<LogicalOffsetType = ()>> NoLogicalOffset for T {}
pub trait HasMemoryOffset: MaybeOffset<MemoryOffsetType = UtcOffset> {}
impl<T: MaybeOffset<MemoryOffsetType = UtcOffset>> HasMemoryOffset for T {}
pub trait NoMemoryOffset: MaybeOffset<MemoryOffsetType = ()> {}
impl<T: MaybeOffset<MemoryOffsetType = ()>> NoMemoryOffset for T {}
pub trait IsOffsetKindNone:
MaybeOffset<Self_ = offset_kind::None, MemoryOffsetType = (), LogicalOffsetType = ()>
{
}
impl IsOffsetKindNone for offset_kind::None {}
pub trait IsOffsetKindFixed:
MaybeOffset<
Self_ = offset_kind::Fixed,
MemoryOffsetType = UtcOffset,
LogicalOffsetType = UtcOffset,
>
{
}
impl IsOffsetKindFixed for offset_kind::Fixed {}
}
impl MaybeOffset for offset_kind::None {
type MemoryOffsetType = ();
type LogicalOffsetType = ();
type Self_ = Self;
const STATIC_OFFSET: Option<UtcOffset> = None;
#[cfg(feature = "parsing")]
fn try_from_parsed(_: Parsed) -> Result<(), error::TryFromParsed> {
Ok(())
}
}
impl MaybeOffset for offset_kind::Fixed {
type MemoryOffsetType = UtcOffset;
type LogicalOffsetType = UtcOffset;
type Self_ = Self;
const STATIC_OFFSET: Option<UtcOffset> = None;
#[cfg(feature = "parsing")]
fn try_from_parsed(parsed: Parsed) -> Result<UtcOffset, error::TryFromParsed> {
parsed.try_into()
}
}
const fn maybe_offset_as_offset_opt<O: MaybeOffset>(
offset: O::MemoryOffsetType,
) -> Option<UtcOffset> {
if O::STATIC_OFFSET.is_some() {
O::STATIC_OFFSET
} else if O::HAS_MEMORY_OFFSET {
#[repr(C)] union Convert<O: MaybeOffset> {
input: O::MemoryOffsetType,
output: UtcOffset,
}
Some(unsafe { Convert::<O> { input: offset }.output })
} else {
None
}
}
const fn maybe_offset_as_offset<O: MaybeOffset + HasLogicalOffset>(
offset: O::MemoryOffsetType,
) -> UtcOffset {
match maybe_offset_as_offset_opt::<O>(offset) {
Some(offset) => offset,
None => bug!("`MaybeOffset::as_offset` called on a type without an offset in memory"),
}
}
pub(crate) const fn maybe_offset_from_offset<O: MaybeOffset>(
offset: UtcOffset,
) -> O::MemoryOffsetType {
#[repr(C)] union Convert<O: MaybeOffset> {
input: UtcOffset,
output: O::MemoryOffsetType,
}
unsafe { Convert::<O> { input: offset }.output }
}
const UNIX_EPOCH_JULIAN_DAY: i32 = Date::__from_ordinal_date_unchecked(1970, 1).to_julian_day();
pub struct DateTime<O: MaybeOffset> {
pub(crate) date: Date,
pub(crate) time: Time,
pub(crate) offset: O::MemoryOffsetType,
}
impl<O: MaybeOffset> Clone for DateTime<O> {
fn clone(&self) -> Self {
*self
}
}
impl<O: MaybeOffset> Copy for DateTime<O> {}
impl DateTime<offset_kind::None> {
pub const MIN: Self = Self {
date: Date::MIN,
time: Time::MIN,
offset: (),
};
pub const MAX: Self = Self {
date: Date::MAX,
time: Time::MAX,
offset: (),
};
}
impl DateTime<offset_kind::Fixed> {
pub const UNIX_EPOCH: Self = Self {
date: Date::__from_ordinal_date_unchecked(1970, 1),
time: Time::MIDNIGHT,
offset: UtcOffset::UTC,
};
}
impl<O: MaybeOffset> DateTime<O> {
pub const fn new(date: Date, time: Time) -> Self
where
O: IsOffsetKindNone,
{
Self {
date,
time,
offset: (),
}
}
pub const fn from_unix_timestamp(timestamp: i64) -> Result<Self, error::ComponentRange>
where
O: HasLogicalOffset,
{
#[allow(clippy::missing_docs_in_private_items)]
const MIN_TIMESTAMP: i64 = Date::MIN.midnight().assume_utc().unix_timestamp();
#[allow(clippy::missing_docs_in_private_items)]
const MAX_TIMESTAMP: i64 = Date::MAX.with_time(Time::MAX).assume_utc().unix_timestamp();
ensure_value_in_range!(timestamp in MIN_TIMESTAMP => MAX_TIMESTAMP);
let date = Date::from_julian_day_unchecked(
UNIX_EPOCH_JULIAN_DAY + div_floor!(timestamp, Second.per(Day) as i64) as i32,
);
let seconds_within_day = timestamp.rem_euclid(Second.per(Day) as _);
let time = unsafe {
Time::__from_hms_nanos_unchecked(
(seconds_within_day / Second.per(Hour) as i64) as _,
((seconds_within_day % Second.per(Hour) as i64) / Minute.per(Hour) as i64) as _,
(seconds_within_day % Second.per(Minute) as i64) as _,
0,
)
};
Ok(Self {
date,
time,
offset: maybe_offset_from_offset::<O>(UtcOffset::UTC),
})
}
pub const fn from_unix_timestamp_nanos(timestamp: i128) -> Result<Self, error::ComponentRange>
where
O: HasLogicalOffset,
{
let datetime = const_try!(Self::from_unix_timestamp(div_floor!(
timestamp,
Nanosecond.per(Second) as i128
) as i64));
Ok(Self {
date: datetime.date,
time: unsafe {
Time::__from_hms_nanos_unchecked(
datetime.hour(),
datetime.minute(),
datetime.second(),
timestamp.rem_euclid(Nanosecond.per(Second) as _) as u32,
)
},
offset: maybe_offset_from_offset::<O>(UtcOffset::UTC),
})
}
#[cfg(feature = "std")]
pub fn now_utc() -> DateTime<offset_kind::Fixed>
where
O: IsOffsetKindFixed,
{
#[cfg(all(
target_family = "wasm",
not(any(target_os = "emscripten", target_os = "wasi")),
feature = "wasm-bindgen"
))]
{
js_sys::Date::new_0().into()
}
#[cfg(not(all(
target_family = "wasm",
not(any(target_os = "emscripten", target_os = "wasi")),
feature = "wasm-bindgen"
)))]
SystemTime::now().into()
}
#[cfg(feature = "local-offset")]
pub fn now_local() -> Result<DateTime<offset_kind::Fixed>, error::IndeterminateOffset>
where
O: IsOffsetKindFixed,
{
let t = DateTime::<offset_kind::Fixed>::now_utc();
Ok(t.to_offset(UtcOffset::local_offset_at(crate::OffsetDateTime(t))?))
}
pub const fn date(self) -> Date {
self.date
}
pub const fn time(self) -> Time {
self.time
}
pub const fn offset(self) -> UtcOffset
where
O: HasLogicalOffset,
{
maybe_offset_as_offset::<O>(self.offset)
}
pub const fn year(self) -> i32 {
self.date.year()
}
pub const fn month(self) -> Month {
self.date.month()
}
pub const fn day(self) -> u8 {
self.date.day()
}
pub const fn ordinal(self) -> u16 {
self.date.ordinal()
}
pub const fn iso_week(self) -> u8 {
self.date.iso_week()
}
pub const fn sunday_based_week(self) -> u8 {
self.date.sunday_based_week()
}
pub const fn monday_based_week(self) -> u8 {
self.date.monday_based_week()
}
pub const fn to_calendar_date(self) -> (i32, Month, u8) {
self.date.to_calendar_date()
}
pub const fn to_ordinal_date(self) -> (i32, u16) {
self.date.to_ordinal_date()
}
pub const fn to_iso_week_date(self) -> (i32, u8, Weekday) {
self.date.to_iso_week_date()
}
pub const fn weekday(self) -> Weekday {
self.date.weekday()
}
pub const fn to_julian_day(self) -> i32 {
self.date.to_julian_day()
}
pub const fn as_hms(self) -> (u8, u8, u8) {
self.time.as_hms()
}
pub const fn as_hms_milli(self) -> (u8, u8, u8, u16) {
self.time.as_hms_milli()
}
pub const fn as_hms_micro(self) -> (u8, u8, u8, u32) {
self.time.as_hms_micro()
}
pub const fn as_hms_nano(self) -> (u8, u8, u8, u32) {
self.time.as_hms_nano()
}
pub const fn hour(self) -> u8 {
self.time.hour()
}
pub const fn minute(self) -> u8 {
self.time.minute()
}
pub const fn second(self) -> u8 {
self.time.second()
}
pub const fn millisecond(self) -> u16 {
self.time.millisecond()
}
pub const fn microsecond(self) -> u32 {
self.time.microsecond()
}
pub const fn nanosecond(self) -> u32 {
self.time.nanosecond()
}
pub const fn unix_timestamp(self) -> i64
where
O: HasLogicalOffset,
{
let offset = maybe_offset_as_offset::<O>(self.offset).whole_seconds() as i64;
let days =
(self.to_julian_day() as i64 - UNIX_EPOCH_JULIAN_DAY as i64) * Second.per(Day) as i64;
let hours = self.hour() as i64 * Second.per(Hour) as i64;
let minutes = self.minute() as i64 * Second.per(Minute) as i64;
let seconds = self.second() as i64;
days + hours + minutes + seconds - offset
}
pub const fn unix_timestamp_nanos(self) -> i128
where
O: HasLogicalOffset,
{
self.unix_timestamp() as i128 * Nanosecond.per(Second) as i128 + self.nanosecond() as i128
}
pub const fn assume_offset(self, offset: UtcOffset) -> DateTime<offset_kind::Fixed>
where
O: NoLogicalOffset,
{
DateTime {
date: self.date,
time: self.time,
offset,
}
}
pub const fn assume_utc(self) -> DateTime<offset_kind::Fixed>
where
O: NoLogicalOffset,
{
self.assume_offset(UtcOffset::UTC)
}
pub const fn to_offset(self, offset: UtcOffset) -> DateTime<offset_kind::Fixed>
where
O: HasLogicalOffset,
{
expect_opt!(
self.checked_to_offset(offset),
"local datetime out of valid range"
)
}
pub const fn checked_to_offset(self, offset: UtcOffset) -> Option<DateTime<offset_kind::Fixed>>
where
O: HasLogicalOffset,
{
let self_offset = maybe_offset_as_offset::<O>(self.offset);
if self_offset.whole_hours() == offset.whole_hours()
&& self_offset.minutes_past_hour() == offset.minutes_past_hour()
&& self_offset.seconds_past_minute() == offset.seconds_past_minute()
{
return Some(DateTime {
date: self.date,
time: self.time,
offset,
});
}
let (year, ordinal, time) = self.to_offset_raw(offset);
if year > MAX_YEAR || year < MIN_YEAR {
return None;
}
Some(DateTime {
date: Date::__from_ordinal_date_unchecked(year, ordinal),
time,
offset,
})
}
pub(crate) const fn to_offset_raw(self, offset: UtcOffset) -> (i32, u16, Time) {
let Some(from) = maybe_offset_as_offset_opt::<O>(self.offset) else {
return (self.year(), self.ordinal(), self.time);
};
let to = offset;
if from.whole_hours() == to.whole_hours()
&& from.minutes_past_hour() == to.minutes_past_hour()
&& from.seconds_past_minute() == to.seconds_past_minute()
{
return (self.year(), self.ordinal(), self.time());
}
let mut second = self.second() as i16 - from.seconds_past_minute() as i16
+ to.seconds_past_minute() as i16;
let mut minute =
self.minute() as i16 - from.minutes_past_hour() as i16 + to.minutes_past_hour() as i16;
let mut hour = self.hour() as i8 - from.whole_hours() + to.whole_hours();
let (mut year, ordinal) = self.to_ordinal_date();
let mut ordinal = ordinal as i16;
cascade!(second in 0..Second.per(Minute) as i16 => minute);
cascade!(second in 0..Second.per(Minute) as i16 => minute);
cascade!(minute in 0..Minute.per(Hour) as i16 => hour);
cascade!(minute in 0..Minute.per(Hour) as i16 => hour);
cascade!(hour in 0..Hour.per(Day) as i8 => ordinal);
cascade!(hour in 0..Hour.per(Day) as i8 => ordinal);
cascade!(ordinal => year);
debug_assert!(ordinal > 0);
debug_assert!(ordinal <= crate::util::days_in_year(year) as i16);
(
year,
ordinal as _,
unsafe {
Time::__from_hms_nanos_unchecked(
hour as _,
minute as _,
second as _,
self.nanosecond(),
)
},
)
}
pub const fn checked_add(self, duration: Duration) -> Option<Self> {
let (date_adjustment, time) = self.time.adjusting_add(duration);
let date = const_try_opt!(self.date.checked_add(duration));
Some(Self {
date: match date_adjustment {
util::DateAdjustment::Previous => const_try_opt!(date.previous_day()),
util::DateAdjustment::Next => const_try_opt!(date.next_day()),
util::DateAdjustment::None => date,
},
time,
offset: self.offset,
})
}
pub const fn checked_sub(self, duration: Duration) -> Option<Self> {
let (date_adjustment, time) = self.time.adjusting_sub(duration);
let date = const_try_opt!(self.date.checked_sub(duration));
Some(Self {
date: match date_adjustment {
util::DateAdjustment::Previous => const_try_opt!(date.previous_day()),
util::DateAdjustment::Next => const_try_opt!(date.next_day()),
util::DateAdjustment::None => date,
},
time,
offset: self.offset,
})
}
pub const fn saturating_add(self, duration: Duration) -> Self {
if let Some(datetime) = self.checked_add(duration) {
datetime
} else if duration.is_negative() {
Self {
date: Date::MIN,
time: Time::MIN,
offset: self.offset,
}
} else {
Self {
date: Date::MAX,
time: Time::MAX,
offset: self.offset,
}
}
}
pub const fn saturating_sub(self, duration: Duration) -> Self {
if let Some(datetime) = self.checked_sub(duration) {
datetime
} else if duration.is_negative() {
Self {
date: Date::MAX,
time: Time::MAX,
offset: self.offset,
}
} else {
Self {
date: Date::MIN,
time: Time::MIN,
offset: self.offset,
}
}
}
#[must_use = "this does not modify the original value"]
pub const fn replace_time(self, time: Time) -> Self {
Self {
date: self.date,
time,
offset: self.offset,
}
}
#[must_use = "this does not modify the original value"]
pub const fn replace_date(self, date: Date) -> Self {
Self {
date,
time: self.time,
offset: self.offset,
}
}
#[must_use = "this does not modify the original value"]
pub const fn replace_date_time(self, date_time: DateTime<offset_kind::None>) -> Self
where
O: HasLogicalOffset,
{
Self {
date: date_time.date,
time: date_time.time,
offset: self.offset,
}
}
#[must_use = "this does not modify the original value"]
pub const fn replace_year(self, year: i32) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: const_try!(self.date.replace_year(year)),
time: self.time,
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_month(self, month: Month) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: const_try!(self.date.replace_month(month)),
time: self.time,
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_day(self, day: u8) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: const_try!(self.date.replace_day(day)),
time: self.time,
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_hour(self, hour: u8) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_hour(hour)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_minute(self, minute: u8) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_minute(minute)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_second(self, second: u8) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_second(second)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_millisecond(
self,
millisecond: u16,
) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_millisecond(millisecond)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_microsecond(
self,
microsecond: u32,
) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_microsecond(microsecond)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_nanosecond(self, nanosecond: u32) -> Result<Self, error::ComponentRange> {
Ok(Self {
date: self.date,
time: const_try!(self.time.replace_nanosecond(nanosecond)),
offset: self.offset,
})
}
#[must_use = "this does not modify the original value"]
pub const fn replace_offset(self, offset: UtcOffset) -> DateTime<offset_kind::Fixed>
where
O: IsOffsetKindFixed,
{
DateTime {
date: self.date,
time: self.time,
offset,
}
}
#[cfg(feature = "formatting")]
pub fn format_into(
self,
output: &mut impl io::Write,
format: &(impl Formattable + ?Sized),
) -> Result<usize, error::Format> {
format.format_into(
output,
Some(self.date),
Some(self.time),
maybe_offset_as_offset_opt::<O>(self.offset),
)
}
#[cfg(feature = "formatting")]
pub fn format(self, format: &(impl Formattable + ?Sized)) -> Result<String, error::Format> {
format.format(
Some(self.date),
Some(self.time),
maybe_offset_as_offset_opt::<O>(self.offset),
)
}
#[cfg(feature = "parsing")]
pub fn parse(
input: &str,
description: &(impl Parsable + ?Sized),
) -> Result<Self, error::Parse> {
description.parse_date_time(input.as_bytes())
}
#[cfg(feature = "parsing")]
pub(crate) const fn is_valid_leap_second_stand_in(self) -> bool {
if !O::HAS_LOGICAL_OFFSET {
return false;
}
if self.nanosecond() != 999_999_999 {
return false;
}
let (year, ordinal, time) = self.to_offset_raw(UtcOffset::UTC);
let Ok(date) = Date::from_ordinal_date(year, ordinal) else {
return false;
};
time.hour() == 23
&& time.minute() == 59
&& time.second() == 59
&& date.day() == util::days_in_year_month(year, date.month())
}
#[doc(hidden)]
#[allow(dead_code)] #[deprecated(since = "0.3.18", note = "use `as_hms` instead")]
pub const fn to_hms(self) -> (u8, u8, u8)
where
O: IsOffsetKindFixed,
{
self.time.as_hms()
}
#[doc(hidden)]
#[allow(dead_code)] #[deprecated(since = "0.3.18", note = "use `as_hms_milli` instead")]
pub const fn to_hms_milli(self) -> (u8, u8, u8, u16)
where
O: IsOffsetKindFixed,
{
self.time.as_hms_milli()
}
#[doc(hidden)]
#[allow(dead_code)] #[deprecated(since = "0.3.18", note = "use `as_hms_micro` instead")]
pub const fn to_hms_micro(self) -> (u8, u8, u8, u32)
where
O: IsOffsetKindFixed,
{
self.time.as_hms_micro()
}
#[doc(hidden)]
#[allow(dead_code)] #[deprecated(since = "0.3.18", note = "use `as_hms_nano` instead")]
pub const fn to_hms_nano(self) -> (u8, u8, u8, u32)
where
O: IsOffsetKindFixed,
{
self.time.as_hms_nano()
}
}
impl<O: MaybeOffset> fmt::Debug for DateTime<O> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
<Self as fmt::Display>::fmt(self, f)
}
}
impl<O: MaybeOffset> fmt::Display for DateTime<O> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{} {}", self.date, self.time)?;
if let Some(offset) = maybe_offset_as_offset_opt::<O>(self.offset) {
write!(f, " {offset}")?;
}
Ok(())
}
}
impl<O: MaybeOffset> PartialEq for DateTime<O> {
fn eq(&self, rhs: &Self) -> bool {
if O::HAS_LOGICAL_OFFSET {
self.to_offset_raw(UtcOffset::UTC) == rhs.to_offset_raw(UtcOffset::UTC)
} else {
(self.date, self.time) == (rhs.date, rhs.time)
}
}
}
impl<O: MaybeOffset> Eq for DateTime<O> {}
impl<O: MaybeOffset> PartialOrd for DateTime<O> {
fn partial_cmp(&self, rhs: &Self) -> Option<Ordering> {
Some(self.cmp(rhs))
}
}
impl<O: MaybeOffset> Ord for DateTime<O> {
fn cmp(&self, rhs: &Self) -> Ordering {
if O::HAS_LOGICAL_OFFSET {
self.to_offset_raw(UtcOffset::UTC)
.cmp(&rhs.to_offset_raw(UtcOffset::UTC))
} else {
(self.date, self.time).cmp(&(rhs.date, rhs.time))
}
}
}
impl<O: MaybeOffset> Hash for DateTime<O> {
fn hash<H: Hasher>(&self, hasher: &mut H) {
if O::HAS_LOGICAL_OFFSET {
self.to_offset_raw(UtcOffset::UTC).hash(hasher);
} else {
(self.date, self.time).hash(hasher);
}
}
}
impl<O: MaybeOffset> Add<Duration> for DateTime<O> {
type Output = Self;
fn add(self, duration: Duration) -> Self {
self.checked_add(duration)
.expect("resulting value is out of range")
}
}
impl<O: MaybeOffset> Add<StdDuration> for DateTime<O> {
type Output = Self;
fn add(self, duration: StdDuration) -> Self::Output {
let (is_next_day, time) = self.time.adjusting_add_std(duration);
Self {
date: if is_next_day {
(self.date + duration)
.next_day()
.expect("resulting value is out of range")
} else {
self.date + duration
},
time,
offset: self.offset,
}
}
}
impl<O: MaybeOffset> AddAssign<Duration> for DateTime<O> {
fn add_assign(&mut self, rhs: Duration) {
*self = *self + rhs;
}
}
impl<O: MaybeOffset> AddAssign<StdDuration> for DateTime<O> {
fn add_assign(&mut self, rhs: StdDuration) {
*self = *self + rhs;
}
}
impl<O: MaybeOffset> Sub<Duration> for DateTime<O> {
type Output = Self;
fn sub(self, duration: Duration) -> Self {
self.checked_sub(duration)
.expect("resulting value is out of range")
}
}
impl<O: MaybeOffset> Sub<StdDuration> for DateTime<O> {
type Output = Self;
fn sub(self, duration: StdDuration) -> Self::Output {
let (is_previous_day, time) = self.time.adjusting_sub_std(duration);
Self {
date: if is_previous_day {
(self.date - duration)
.previous_day()
.expect("resulting value is out of range")
} else {
self.date - duration
},
time,
offset: self.offset,
}
}
}
impl<O: MaybeOffset> SubAssign<Duration> for DateTime<O> {
fn sub_assign(&mut self, rhs: Duration) {
*self = *self - rhs;
}
}
impl<O: MaybeOffset> SubAssign<StdDuration> for DateTime<O> {
fn sub_assign(&mut self, rhs: StdDuration) {
*self = *self - rhs;
}
}
impl<O: MaybeOffset> Sub<Self> for DateTime<O> {
type Output = Duration;
fn sub(self, rhs: Self) -> Self::Output {
let base = (self.date - rhs.date) + (self.time - rhs.time);
match (
maybe_offset_as_offset_opt::<O>(self.offset),
maybe_offset_as_offset_opt::<O>(rhs.offset),
) {
(Some(self_offset), Some(rhs_offset)) => {
let adjustment = Duration::seconds(
(self_offset.whole_seconds() - rhs_offset.whole_seconds()) as i64,
);
base - adjustment
}
(left, right) => {
debug_assert!(
left.is_none() && right.is_none(),
"offset type should not be different for the same type"
);
base
}
}
}
}
#[cfg(feature = "std")]
impl Add<Duration> for SystemTime {
type Output = Self;
fn add(self, duration: Duration) -> Self::Output {
if duration.is_zero() {
self
} else if duration.is_positive() {
self + duration.unsigned_abs()
} else {
debug_assert!(duration.is_negative());
self - duration.unsigned_abs()
}
}
}
impl_add_assign!(SystemTime: #[cfg(feature = "std")] Duration);
#[cfg(feature = "std")]
impl Sub<Duration> for SystemTime {
type Output = Self;
fn sub(self, duration: Duration) -> Self::Output {
(DateTime::from(self) - duration).into()
}
}
impl_sub_assign!(SystemTime: #[cfg(feature = "std")] Duration);
#[cfg(feature = "std")]
impl Sub<SystemTime> for DateTime<offset_kind::Fixed> {
type Output = Duration;
fn sub(self, rhs: SystemTime) -> Self::Output {
self - Self::from(rhs)
}
}
#[cfg(feature = "std")]
impl Sub<DateTime<offset_kind::Fixed>> for SystemTime {
type Output = Duration;
fn sub(self, rhs: DateTime<offset_kind::Fixed>) -> Self::Output {
DateTime::<offset_kind::Fixed>::from(self) - rhs
}
}
#[cfg(feature = "std")]
impl PartialEq<SystemTime> for DateTime<offset_kind::Fixed> {
fn eq(&self, rhs: &SystemTime) -> bool {
self == &Self::from(*rhs)
}
}
#[cfg(feature = "std")]
impl PartialEq<DateTime<offset_kind::Fixed>> for SystemTime {
fn eq(&self, rhs: &DateTime<offset_kind::Fixed>) -> bool {
&DateTime::<offset_kind::Fixed>::from(*self) == rhs
}
}
#[cfg(feature = "std")]
impl PartialOrd<SystemTime> for DateTime<offset_kind::Fixed> {
fn partial_cmp(&self, other: &SystemTime) -> Option<Ordering> {
self.partial_cmp(&Self::from(*other))
}
}
#[cfg(feature = "std")]
impl PartialOrd<DateTime<offset_kind::Fixed>> for SystemTime {
fn partial_cmp(&self, other: &DateTime<offset_kind::Fixed>) -> Option<Ordering> {
DateTime::<offset_kind::Fixed>::from(*self).partial_cmp(other)
}
}
#[cfg(feature = "std")]
impl From<SystemTime> for DateTime<offset_kind::Fixed> {
fn from(system_time: SystemTime) -> Self {
match system_time.duration_since(SystemTime::UNIX_EPOCH) {
Ok(duration) => Self::UNIX_EPOCH + duration,
Err(err) => Self::UNIX_EPOCH - err.duration(),
}
}
}
#[allow(clippy::fallible_impl_from)] #[cfg(feature = "std")]
impl From<DateTime<offset_kind::Fixed>> for SystemTime {
fn from(datetime: DateTime<offset_kind::Fixed>) -> Self {
let duration = datetime - DateTime::<offset_kind::Fixed>::UNIX_EPOCH;
if duration.is_zero() {
Self::UNIX_EPOCH
} else if duration.is_positive() {
Self::UNIX_EPOCH + duration.unsigned_abs()
} else {
debug_assert!(duration.is_negative());
Self::UNIX_EPOCH - duration.unsigned_abs()
}
}
}
#[allow(clippy::fallible_impl_from)]
#[cfg(all(
target_family = "wasm",
not(any(target_os = "emscripten", target_os = "wasi")),
feature = "wasm-bindgen"
))]
impl From<js_sys::Date> for DateTime<offset_kind::Fixed> {
fn from(js_date: js_sys::Date) -> Self {
let timestamp_nanos = (js_date.get_time() * Nanosecond.per(Millisecond) as f64) as i128;
Self::from_unix_timestamp_nanos(timestamp_nanos)
.expect("invalid timestamp: Timestamp cannot fit in range")
}
}
#[cfg(all(
target_family = "wasm",
not(any(target_os = "emscripten", target_os = "wasi")),
feature = "wasm-bindgen"
))]
impl From<DateTime<offset_kind::Fixed>> for js_sys::Date {
fn from(datetime: DateTime<offset_kind::Fixed>) -> Self {
let timestamp =
(datetime.unix_timestamp_nanos() / Nanosecond.per(Millisecond) as i128) as f64;
js_sys::Date::new(×tamp.into())
}
}