mirror of
https://github.com/chronotope/chrono.git
synced 2025-09-29 22:11:59 +00:00

There's no pretty way of gating doctests contingent on cfg presence/absense so this is kind of ugly. First, a doc comment `///` is converted into its equivalent `#[doc(...)]`, which is then converted into a conditional rust attribute via `#[cfg_attr(...)]`. We need two of these, one to ignore the tests if the cfg predicate isn't met and another to indicate the start of a code block and enable testing + syntax highlighting.
291 lines
9.0 KiB
Rust
291 lines
9.0 KiB
Rust
// This is a part of Chrono.
|
|
// See README.md and LICENSE.txt for details.
|
|
|
|
//! The time zone which has a fixed offset from UTC.
|
|
|
|
use core::fmt;
|
|
use core::ops::{Add, Sub};
|
|
|
|
use num_integer::div_mod_floor;
|
|
#[cfg(feature = "rkyv")]
|
|
use rkyv::{Archive, Deserialize, Serialize};
|
|
|
|
use super::{LocalResult, Offset, TimeZone};
|
|
use crate::naive::{NaiveDate, NaiveDateTime, NaiveTime};
|
|
use crate::oldtime::Duration as OldDuration;
|
|
use crate::DateTime;
|
|
use crate::Timelike;
|
|
|
|
/// The time zone with fixed offset, from UTC-23:59:59 to UTC+23:59:59.
|
|
///
|
|
/// Using the [`TimeZone`](./trait.TimeZone.html) methods
|
|
/// on a `FixedOffset` struct is the preferred way to construct
|
|
/// `DateTime<FixedOffset>` instances. See the [`east_opt`](#method.east_opt) and
|
|
/// [`west_opt`](#method.west_opt) methods for examples.
|
|
#[derive(PartialEq, Eq, Hash, Copy, Clone)]
|
|
#[cfg_attr(feature = "rkyv", derive(Archive, Deserialize, Serialize))]
|
|
pub struct FixedOffset {
|
|
local_minus_utc: i32,
|
|
}
|
|
|
|
impl FixedOffset {
|
|
/// Makes a new `FixedOffset` for the Eastern Hemisphere with given timezone difference.
|
|
/// The negative `secs` means the Western Hemisphere.
|
|
///
|
|
/// Panics on the out-of-bound `secs`.
|
|
#[deprecated(since = "0.4.23", note = "use `east_opt()` instead")]
|
|
#[must_use]
|
|
pub fn east(secs: i32) -> FixedOffset {
|
|
FixedOffset::east_opt(secs).expect("FixedOffset::east out of bounds")
|
|
}
|
|
|
|
/// Makes a new `FixedOffset` for the Eastern Hemisphere with given timezone difference.
|
|
/// The negative `secs` means the Western Hemisphere.
|
|
///
|
|
/// Returns `None` on the out-of-bound `secs`.
|
|
///
|
|
/// # Example
|
|
///
|
|
#[cfg_attr(not(feature = "std"), doc = "```ignore")]
|
|
#[cfg_attr(feature = "std", doc = "```")]
|
|
/// use chrono::{FixedOffset, TimeZone};
|
|
/// let hour = 3600;
|
|
/// let datetime = FixedOffset::east_opt(5 * hour).unwrap().ymd_opt(2016, 11, 08).unwrap()
|
|
/// .and_hms_opt(0, 0, 0).unwrap();
|
|
/// assert_eq!(&datetime.to_rfc3339(), "2016-11-08T00:00:00+05:00")
|
|
/// ```
|
|
#[must_use]
|
|
pub fn east_opt(secs: i32) -> Option<FixedOffset> {
|
|
if -86_400 < secs && secs < 86_400 {
|
|
Some(FixedOffset { local_minus_utc: secs })
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Makes a new `FixedOffset` for the Western Hemisphere with given timezone difference.
|
|
/// The negative `secs` means the Eastern Hemisphere.
|
|
///
|
|
/// Panics on the out-of-bound `secs`.
|
|
#[deprecated(since = "0.4.23", note = "use `west_opt()` instead")]
|
|
#[must_use]
|
|
pub fn west(secs: i32) -> FixedOffset {
|
|
FixedOffset::west_opt(secs).expect("FixedOffset::west out of bounds")
|
|
}
|
|
|
|
/// Makes a new `FixedOffset` for the Western Hemisphere with given timezone difference.
|
|
/// The negative `secs` means the Eastern Hemisphere.
|
|
///
|
|
/// Returns `None` on the out-of-bound `secs`.
|
|
///
|
|
/// # Example
|
|
///
|
|
#[cfg_attr(not(feature = "std"), doc = "```ignore")]
|
|
#[cfg_attr(feature = "std", doc = "```")]
|
|
/// use chrono::{FixedOffset, TimeZone};
|
|
/// let hour = 3600;
|
|
/// let datetime = FixedOffset::west_opt(5 * hour).unwrap().ymd_opt(2016, 11, 08).unwrap()
|
|
/// .and_hms_opt(0, 0, 0).unwrap();
|
|
/// assert_eq!(&datetime.to_rfc3339(), "2016-11-08T00:00:00-05:00")
|
|
/// ```
|
|
#[must_use]
|
|
pub fn west_opt(secs: i32) -> Option<FixedOffset> {
|
|
if -86_400 < secs && secs < 86_400 {
|
|
Some(FixedOffset { local_minus_utc: -secs })
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Returns the number of seconds to add to convert from UTC to the local time.
|
|
#[inline]
|
|
pub const fn local_minus_utc(&self) -> i32 {
|
|
self.local_minus_utc
|
|
}
|
|
|
|
/// Returns the number of seconds to add to convert from the local time to UTC.
|
|
#[inline]
|
|
pub const fn utc_minus_local(&self) -> i32 {
|
|
-self.local_minus_utc
|
|
}
|
|
}
|
|
|
|
impl TimeZone for FixedOffset {
|
|
type Offset = FixedOffset;
|
|
|
|
fn from_offset(offset: &FixedOffset) -> FixedOffset {
|
|
*offset
|
|
}
|
|
|
|
fn offset_from_local_date(&self, _local: &NaiveDate) -> LocalResult<FixedOffset> {
|
|
LocalResult::Single(*self)
|
|
}
|
|
fn offset_from_local_datetime(&self, _local: &NaiveDateTime) -> LocalResult<FixedOffset> {
|
|
LocalResult::Single(*self)
|
|
}
|
|
|
|
fn offset_from_utc_date(&self, _utc: &NaiveDate) -> FixedOffset {
|
|
*self
|
|
}
|
|
fn offset_from_utc_datetime(&self, _utc: &NaiveDateTime) -> FixedOffset {
|
|
*self
|
|
}
|
|
}
|
|
|
|
impl Offset for FixedOffset {
|
|
fn fix(&self) -> FixedOffset {
|
|
*self
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for FixedOffset {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
let offset = self.local_minus_utc;
|
|
let (sign, offset) = if offset < 0 { ('-', -offset) } else { ('+', offset) };
|
|
let (mins, sec) = div_mod_floor(offset, 60);
|
|
let (hour, min) = div_mod_floor(mins, 60);
|
|
if sec == 0 {
|
|
write!(f, "{}{:02}:{:02}", sign, hour, min)
|
|
} else {
|
|
write!(f, "{}{:02}:{:02}:{:02}", sign, hour, min, sec)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for FixedOffset {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
fmt::Debug::fmt(self, f)
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "arbitrary")]
|
|
impl arbitrary::Arbitrary<'_> for FixedOffset {
|
|
fn arbitrary(u: &mut arbitrary::Unstructured) -> arbitrary::Result<FixedOffset> {
|
|
let secs = u.int_in_range(-86_399..=86_399)?;
|
|
let fixed_offset = FixedOffset::east_opt(secs)
|
|
.expect("Could not generate a valid chrono::FixedOffset. It looks like implementation of Arbitrary for FixedOffset is erroneous.");
|
|
Ok(fixed_offset)
|
|
}
|
|
}
|
|
|
|
// addition or subtraction of FixedOffset to/from Timelike values is the same as
|
|
// adding or subtracting the offset's local_minus_utc value
|
|
// but keep keeps the leap second information.
|
|
// this should be implemented more efficiently, but for the time being, this is generic right now.
|
|
|
|
fn add_with_leapsecond<T>(lhs: &T, rhs: i32) -> T
|
|
where
|
|
T: Timelike + Add<OldDuration, Output = T>,
|
|
{
|
|
// extract and temporarily remove the fractional part and later recover it
|
|
let nanos = lhs.nanosecond();
|
|
let lhs = lhs.with_nanosecond(0).unwrap();
|
|
(lhs + OldDuration::seconds(i64::from(rhs))).with_nanosecond(nanos).unwrap()
|
|
}
|
|
|
|
impl Add<FixedOffset> for NaiveTime {
|
|
type Output = NaiveTime;
|
|
|
|
#[inline]
|
|
fn add(self, rhs: FixedOffset) -> NaiveTime {
|
|
add_with_leapsecond(&self, rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
impl Sub<FixedOffset> for NaiveTime {
|
|
type Output = NaiveTime;
|
|
|
|
#[inline]
|
|
fn sub(self, rhs: FixedOffset) -> NaiveTime {
|
|
add_with_leapsecond(&self, -rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
impl Add<FixedOffset> for NaiveDateTime {
|
|
type Output = NaiveDateTime;
|
|
|
|
#[inline]
|
|
fn add(self, rhs: FixedOffset) -> NaiveDateTime {
|
|
add_with_leapsecond(&self, rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
impl Sub<FixedOffset> for NaiveDateTime {
|
|
type Output = NaiveDateTime;
|
|
|
|
#[inline]
|
|
fn sub(self, rhs: FixedOffset) -> NaiveDateTime {
|
|
add_with_leapsecond(&self, -rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
impl<Tz: TimeZone> Add<FixedOffset> for DateTime<Tz> {
|
|
type Output = DateTime<Tz>;
|
|
|
|
#[inline]
|
|
fn add(self, rhs: FixedOffset) -> DateTime<Tz> {
|
|
add_with_leapsecond(&self, rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
impl<Tz: TimeZone> Sub<FixedOffset> for DateTime<Tz> {
|
|
type Output = DateTime<Tz>;
|
|
|
|
#[inline]
|
|
fn sub(self, rhs: FixedOffset) -> DateTime<Tz> {
|
|
add_with_leapsecond(&self, -rhs.local_minus_utc)
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::FixedOffset;
|
|
use crate::offset::TimeZone;
|
|
|
|
#[test]
|
|
fn test_date_extreme_offset() {
|
|
// starting from 0.3 we don't have an offset exceeding one day.
|
|
// this makes everything easier!
|
|
assert_eq!(
|
|
format!(
|
|
"{:?}",
|
|
FixedOffset::east_opt(86399)
|
|
.unwrap()
|
|
.with_ymd_and_hms(2012, 2, 29, 5, 6, 7)
|
|
.unwrap()
|
|
),
|
|
"2012-02-29T05:06:07+23:59:59".to_string()
|
|
);
|
|
assert_eq!(
|
|
format!(
|
|
"{:?}",
|
|
FixedOffset::east_opt(86399)
|
|
.unwrap()
|
|
.with_ymd_and_hms(2012, 2, 29, 5, 6, 7)
|
|
.unwrap()
|
|
),
|
|
"2012-02-29T05:06:07+23:59:59".to_string()
|
|
);
|
|
assert_eq!(
|
|
format!(
|
|
"{:?}",
|
|
FixedOffset::west_opt(86399)
|
|
.unwrap()
|
|
.with_ymd_and_hms(2012, 3, 4, 5, 6, 7)
|
|
.unwrap()
|
|
),
|
|
"2012-03-04T05:06:07-23:59:59".to_string()
|
|
);
|
|
assert_eq!(
|
|
format!(
|
|
"{:?}",
|
|
FixedOffset::west_opt(86399)
|
|
.unwrap()
|
|
.with_ymd_and_hms(2012, 3, 4, 5, 6, 7)
|
|
.unwrap()
|
|
),
|
|
"2012-03-04T05:06:07-23:59:59".to_string()
|
|
);
|
|
}
|
|
}
|