From 03fb98c4627b5ed1a504193b5e0c5b9dffc07bd4 Mon Sep 17 00:00:00 2001 From: Eden Mikitas Date: Tue, 12 Sep 2023 20:47:50 +0300 Subject: [PATCH 1/2] Make TimeZone::timestamp_millis_opt use NaiveDateTime::from_timestamp_millis to reduce code duplication --- src/offset/mod.rs | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/src/offset/mod.rs b/src/offset/mod.rs index 419550e292..2ce1ff4f7c 100644 --- a/src/offset/mod.rs +++ b/src/offset/mod.rs @@ -403,12 +403,10 @@ pub trait TimeZone: Sized + Clone { /// }; /// ``` fn timestamp_millis_opt(&self, millis: i64) -> LocalResult> { - let (mut secs, mut millis) = (millis / 1000, millis % 1000); - if millis < 0 { - secs -= 1; - millis += 1000; + match NaiveDateTime::from_timestamp_millis(millis) { + Some(dt) => LocalResult::Single(self.from_utc_datetime(&dt)), + None => LocalResult::None, } - self.timestamp_opt(secs, millis as u32 * 1_000_000) } /// Makes a new `DateTime` from the number of non-leap nanoseconds From 81708eb42ac16ad8c44b7b090fc7e41e9d44c28d Mon Sep 17 00:00:00 2001 From: Eden Mikitas Date: Tue, 12 Sep 2023 21:05:20 +0300 Subject: [PATCH 2/2] Add TimeZone::timestamp_micros Use NaiveDateTime::from_timestamp_micros instead of implementing the same logic again. --- src/offset/mod.rs | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) diff --git a/src/offset/mod.rs b/src/offset/mod.rs index 2ce1ff4f7c..b7fb803cd1 100644 --- a/src/offset/mod.rs +++ b/src/offset/mod.rs @@ -431,6 +431,22 @@ pub trait TimeZone: Sized + Clone { self.timestamp_opt(secs, nanos as u32).unwrap() } + /// Makes a new `DateTime` from the number of non-leap microseconds + /// since January 1, 1970 0:00:00 UTC (aka "UNIX timestamp"). + /// + /// #Example + /// ``` + /// use chrono::{Utc, TimeZone}; + /// + /// assert_eq!(Utc.timestamp_micros(1431648000000).unwrap().timestamp(), 1431648); + /// ``` + fn timestamp_micros(&self, micros: i64) -> LocalResult> { + match NaiveDateTime::from_timestamp_micros(micros) { + Some(dt) => LocalResult::Single(self.from_utc_datetime(&dt)), + None => LocalResult::None, + } + } + /// Parses a string with the specified format string and returns a /// `DateTime` with the current offset. /// @@ -556,4 +572,18 @@ mod tests { Utc.timestamp_nanos(i64::default()); Utc.timestamp_nanos(i64::min_value()); } + + #[test] + fn test_negative_micros() { + let dt = Utc.timestamp_micros(-1_000_000).unwrap(); + assert_eq!(dt.to_string(), "1969-12-31 23:59:59 UTC"); + let dt = Utc.timestamp_micros(-999_999).unwrap(); + assert_eq!(dt.to_string(), "1969-12-31 23:59:59.000001 UTC"); + let dt = Utc.timestamp_micros(-1).unwrap(); + assert_eq!(dt.to_string(), "1969-12-31 23:59:59.999999 UTC"); + let dt = Utc.timestamp_micros(-60_000_000).unwrap(); + assert_eq!(dt.to_string(), "1969-12-31 23:59:00 UTC"); + let dt = Utc.timestamp_micros(-3_600_000_000).unwrap(); + assert_eq!(dt.to_string(), "1969-12-31 23:00:00 UTC"); + } }