| //! Date and time types. |
| |
| use bitflags::bitflags; |
| use core::fmt::{self, Display, Formatter}; |
| |
| /// Date and time representation. |
| #[derive(Debug, Default, Copy, Clone, Eq)] |
| #[repr(C)] |
| pub struct Time { |
| /// Year. Valid range: `1900..=9999`. |
| pub year: u16, |
| |
| /// Month. Valid range: `1..=12`. |
| pub month: u8, |
| |
| /// Day of the month. Valid range: `1..=31`. |
| pub day: u8, |
| |
| /// Hour. Valid range: `0..=23`. |
| pub hour: u8, |
| |
| /// Minute. Valid range: `0..=59`. |
| pub minute: u8, |
| |
| /// Second. Valid range: `0..=59`. |
| pub second: u8, |
| |
| /// Unused padding. |
| pub pad1: u8, |
| |
| /// Nanosececond. Valid range: `0..=999_999_999`. |
| pub nanosecond: u32, |
| |
| /// Offset in minutes from UTC. Valid range: `-1440..=1440`, or |
| /// [`Time::UNSPECIFIED_TIMEZONE`]. |
| pub time_zone: i16, |
| |
| /// Daylight savings time information. |
| pub daylight: Daylight, |
| |
| /// Unused padding. |
| pub pad2: u8, |
| } |
| |
| impl Time { |
| /// Indicates the time should be interpreted as local time. |
| pub const UNSPECIFIED_TIMEZONE: i16 = 0x07ff; |
| |
| /// Create an invalid `Time` with all fields set to zero. |
| #[must_use] |
| pub const fn invalid() -> Self { |
| Self { |
| year: 0, |
| month: 0, |
| day: 0, |
| hour: 0, |
| minute: 0, |
| second: 0, |
| pad1: 0, |
| nanosecond: 0, |
| time_zone: 0, |
| daylight: Daylight::empty(), |
| pad2: 0, |
| } |
| } |
| |
| /// True if all fields are within valid ranges, false otherwise. |
| #[must_use] |
| pub fn is_valid(&self) -> bool { |
| (1900..=9999).contains(&self.year) |
| && (1..=12).contains(&self.month) |
| && (1..=31).contains(&self.day) |
| && self.hour <= 23 |
| && self.minute <= 59 |
| && self.second <= 59 |
| && self.nanosecond <= 999_999_999 |
| && ((-1440..=1440).contains(&self.time_zone) |
| || self.time_zone == Self::UNSPECIFIED_TIMEZONE) |
| } |
| } |
| |
| impl Display for Time { |
| fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result { |
| write!(f, "{:04}-{:02}-{:02} ", self.year, self.month, self.day)?; |
| write!( |
| f, |
| "{:02}:{:02}:{:02}.{:09}", |
| self.hour, self.minute, self.second, self.nanosecond |
| )?; |
| |
| if self.time_zone == Self::UNSPECIFIED_TIMEZONE { |
| write!(f, " (local)")?; |
| } else { |
| let offset_in_hours = self.time_zone as f32 / 60.0; |
| let integer_part = offset_in_hours as i16; |
| // We can't use "offset_in_hours.fract()" because it is part of `std`. |
| let fraction_part = offset_in_hours - (integer_part as f32); |
| // most time zones |
| if fraction_part == 0.0 { |
| write!(f, "UTC+{offset_in_hours}")?; |
| } |
| // time zones with 30min offset (and perhaps other special time zones) |
| else { |
| write!(f, "UTC+{offset_in_hours:.1}")?; |
| } |
| } |
| |
| Ok(()) |
| } |
| } |
| |
| /// The padding fields of `Time` are ignored for comparison. |
| impl PartialEq for Time { |
| fn eq(&self, other: &Self) -> bool { |
| self.year == other.year |
| && self.month == other.month |
| && self.day == other.day |
| && self.hour == other.hour |
| && self.minute == other.minute |
| && self.second == other.second |
| && self.nanosecond == other.nanosecond |
| && self.time_zone == other.time_zone |
| && self.daylight == other.daylight |
| } |
| } |
| |
| bitflags! { |
| /// A bitmask containing daylight savings time information. |
| #[repr(transparent)] |
| #[derive(Clone, Copy, Debug, Default, PartialEq, Eq, PartialOrd, Ord)] |
| pub struct Daylight: u8 { |
| /// Time is affected by daylight savings time. |
| const ADJUST_DAYLIGHT = 0x01; |
| |
| /// Time has been adjusted for daylight savings time. |
| const IN_DAYLIGHT = 0x02; |
| } |
| } |
| |
| #[cfg(test)] |
| mod tests { |
| extern crate alloc; |
| |
| use super::*; |
| use alloc::string::ToString; |
| |
| #[test] |
| fn test_time_display() { |
| let mut time = Time { |
| year: 2023, |
| month: 5, |
| day: 18, |
| hour: 11, |
| minute: 29, |
| second: 57, |
| nanosecond: 123_456_789, |
| time_zone: Time::UNSPECIFIED_TIMEZONE, |
| daylight: Daylight::empty(), |
| pad1: 0, |
| pad2: 0, |
| }; |
| assert_eq!(time.to_string(), "2023-05-18 11:29:57.123456789 (local)"); |
| |
| time.time_zone = 120; |
| assert_eq!(time.to_string(), "2023-05-18 11:29:57.123456789UTC+2"); |
| |
| time.time_zone = 150; |
| assert_eq!(time.to_string(), "2023-05-18 11:29:57.123456789UTC+2.5"); |
| } |
| } |