1use core::fmt;
7use core::str::FromStr;
8
9#[cfg(any(feature = "rkyv", feature = "rkyv-16", feature = "rkyv-32", feature = "rkyv-64"))]
10use rkyv::{Archive, Deserialize, Serialize};
11
12use super::{LocalResult, Offset, TimeZone};
13use crate::format::{scan, ParseError, OUT_OF_RANGE};
14use crate::naive::{NaiveDate, NaiveDateTime};
15
16#[derive(PartialEq, Eq, Hash, Copy, Clone)]
23#[cfg_attr(
24 any(feature = "rkyv", feature = "rkyv-16", feature = "rkyv-32", feature = "rkyv-64"),
25 derive(Archive, Deserialize, Serialize),
26 archive(compare(PartialEq)),
27 archive_attr(derive(Clone, Copy, PartialEq, Eq, Hash, Debug))
28)]
29#[cfg_attr(feature = "rkyv-validation", archive(check_bytes))]
30pub struct FixedOffset {
31 local_minus_utc: i32,
32}
33
34impl FixedOffset {
35 #[deprecated(since = "0.4.23", note = "use `east_opt()` instead")]
40 #[must_use]
41 pub fn east(secs: i32) -> FixedOffset {
42 FixedOffset::east_opt(secs).expect("FixedOffset::east out of bounds")
43 }
44
45 #[cfg_attr(not(feature = "std"), doc = "```ignore")]
53 #[cfg_attr(feature = "std", doc = "```")]
54 #[must_use]
63 pub const fn east_opt(secs: i32) -> Option<FixedOffset> {
64 if -86_400 < secs && secs < 86_400 {
65 Some(FixedOffset { local_minus_utc: secs })
66 } else {
67 None
68 }
69 }
70
71 #[deprecated(since = "0.4.23", note = "use `west_opt()` instead")]
76 #[must_use]
77 pub fn west(secs: i32) -> FixedOffset {
78 FixedOffset::west_opt(secs).expect("FixedOffset::west out of bounds")
79 }
80
81 #[cfg_attr(not(feature = "std"), doc = "```ignore")]
89 #[cfg_attr(feature = "std", doc = "```")]
90 #[must_use]
99 pub const fn west_opt(secs: i32) -> Option<FixedOffset> {
100 if -86_400 < secs && secs < 86_400 {
101 Some(FixedOffset { local_minus_utc: -secs })
102 } else {
103 None
104 }
105 }
106
107 #[inline]
109 pub const fn local_minus_utc(&self) -> i32 {
110 self.local_minus_utc
111 }
112
113 #[inline]
115 pub const fn utc_minus_local(&self) -> i32 {
116 -self.local_minus_utc
117 }
118}
119
120impl FromStr for FixedOffset {
122 type Err = ParseError;
123 fn from_str(s: &str) -> Result<Self, Self::Err> {
124 let (_, offset) = scan::timezone_offset(s, scan::colon_or_space, false, false, true)?;
125 Self::east_opt(offset).ok_or(OUT_OF_RANGE)
126 }
127}
128
129impl TimeZone for FixedOffset {
130 type Offset = FixedOffset;
131
132 fn from_offset(offset: &FixedOffset) -> FixedOffset {
133 *offset
134 }
135
136 fn offset_from_local_date(&self, _local: &NaiveDate) -> LocalResult<FixedOffset> {
137 LocalResult::Single(*self)
138 }
139 fn offset_from_local_datetime(&self, _local: &NaiveDateTime) -> LocalResult<FixedOffset> {
140 LocalResult::Single(*self)
141 }
142
143 fn offset_from_utc_date(&self, _utc: &NaiveDate) -> FixedOffset {
144 *self
145 }
146 fn offset_from_utc_datetime(&self, _utc: &NaiveDateTime) -> FixedOffset {
147 *self
148 }
149}
150
151impl Offset for FixedOffset {
152 fn fix(&self) -> FixedOffset {
153 *self
154 }
155}
156
157impl fmt::Debug for FixedOffset {
158 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
159 let offset = self.local_minus_utc;
160 let (sign, offset) = if offset < 0 { ('-', -offset) } else { ('+', offset) };
161 let sec = offset.rem_euclid(60);
162 let mins = offset.div_euclid(60);
163 let min = mins.rem_euclid(60);
164 let hour = mins.div_euclid(60);
165 if sec == 0 {
166 write!(f, "{}{:02}:{:02}", sign, hour, min)
167 } else {
168 write!(f, "{}{:02}:{:02}:{:02}", sign, hour, min, sec)
169 }
170 }
171}
172
173impl fmt::Display for FixedOffset {
174 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
175 fmt::Debug::fmt(self, f)
176 }
177}
178
179#[cfg(all(feature = "arbitrary", feature = "std"))]
180impl arbitrary::Arbitrary<'_> for FixedOffset {
181 fn arbitrary(u: &mut arbitrary::Unstructured) -> arbitrary::Result<FixedOffset> {
182 let secs = u.int_in_range(-86_399..=86_399)?;
183 let fixed_offset = FixedOffset::east_opt(secs)
184 .expect("Could not generate a valid chrono::FixedOffset. It looks like implementation of Arbitrary for FixedOffset is erroneous.");
185 Ok(fixed_offset)
186 }
187}
188
189#[cfg(test)]
190mod tests {
191 use super::FixedOffset;
192 use crate::offset::TimeZone;
193 use std::str::FromStr;
194
195 #[test]
196 fn test_date_extreme_offset() {
197 let offset = FixedOffset::east_opt(86399).unwrap();
200 assert_eq!(
201 format!("{:?}", offset.with_ymd_and_hms(2012, 2, 29, 5, 6, 7).unwrap()),
202 "2012-02-29T05:06:07+23:59:59"
203 );
204 let offset = FixedOffset::east_opt(-86399).unwrap();
205 assert_eq!(
206 format!("{:?}", offset.with_ymd_and_hms(2012, 2, 29, 5, 6, 7).unwrap()),
207 "2012-02-29T05:06:07-23:59:59"
208 );
209 let offset = FixedOffset::west_opt(86399).unwrap();
210 assert_eq!(
211 format!("{:?}", offset.with_ymd_and_hms(2012, 3, 4, 5, 6, 7).unwrap()),
212 "2012-03-04T05:06:07-23:59:59"
213 );
214 let offset = FixedOffset::west_opt(-86399).unwrap();
215 assert_eq!(
216 format!("{:?}", offset.with_ymd_and_hms(2012, 3, 4, 5, 6, 7).unwrap()),
217 "2012-03-04T05:06:07+23:59:59"
218 );
219 }
220
221 #[test]
222 fn test_parse_offset() {
223 let offset = FixedOffset::from_str("-0500").unwrap();
224 assert_eq!(offset.local_minus_utc, -5 * 3600);
225 let offset = FixedOffset::from_str("-08:00").unwrap();
226 assert_eq!(offset.local_minus_utc, -8 * 3600);
227 let offset = FixedOffset::from_str("+06:30").unwrap();
228 assert_eq!(offset.local_minus_utc, (6 * 3600) + 1800);
229 }
230
231 #[test]
232 #[cfg(feature = "rkyv-validation")]
233 fn test_rkyv_validation() {
234 let offset = FixedOffset::from_str("-0500").unwrap();
235 let bytes = rkyv::to_bytes::<_, 4>(&offset).unwrap();
236 assert_eq!(rkyv::from_bytes::<FixedOffset>(&bytes).unwrap(), offset);
237 }
238}