tz/lib.rs
1#![forbid(unsafe_code)]
2#![deny(missing_docs)]
3#![cfg_attr(not(feature = "std"), no_std)]
4#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))]
5
6//! This crate provides the [`TimeZone`] and [`DateTime`] types, which can be used to determine local time on a given time zone.
7//!
8//! This allows to convert between an [Unix timestamp](https://en.wikipedia.org/wiki/Unix_time) and a calendar time expressed in the [proleptic gregorian calendar](https://en.wikipedia.org/wiki/Proleptic_Gregorian_calendar) with a provided time zone.
9//!
10//! Time zones are provided to the library with a [POSIX `TZ` string](https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap08.html) which can be read from the environment.
11//!
12//! Two formats are currently accepted for the `TZ` string:
13//! * `std offset[dst[offset][,start[/time],end[/time]]]` providing a time zone description,
14//! * `file` or `:file` providing the path to a [TZif file](https://datatracker.ietf.org/doc/html/rfc8536), which is absolute or relative to the system timezone directory.
15//!
16//! See also the [Linux manual page of tzset(3)](https://man7.org/linux/man-pages/man3/tzset.3.html) and the [glibc documentation of the `TZ` environment variable](https://www.gnu.org/software/libc/manual/html_node/TZ-Variable.html).
17//!
18//! # Usage
19//!
20//! ## Time zone
21//!
22//! ```rust
23//! # fn main() -> Result<(), tz::Error> {
24//! # #[cfg(feature = "std")] {
25//! use tz::TimeZone;
26//!
27//! // 2000-01-01T00:00:00Z
28//! let unix_time = 946684800;
29//!
30//! // Get UTC time zone
31//! let time_zone_utc = TimeZone::utc();
32//! assert_eq!(time_zone_utc.find_local_time_type(unix_time)?.ut_offset(), 0);
33//!
34//! // Get fixed time zone at GMT-1
35//! let time_zone_fixed = TimeZone::fixed(-3600)?;
36//! assert_eq!(time_zone_fixed.find_local_time_type(unix_time)?.ut_offset(), -3600);
37//!
38//! // Get local time zone (UNIX only)
39//! let time_zone_local = TimeZone::local()?;
40//! // Get the current local time type
41//! let _current_local_time_type = time_zone_local.find_current_local_time_type()?;
42//!
43//! // Get time zone from a TZ string:
44//! // From an absolute file
45//! let _ = TimeZone::from_posix_tz("/usr/share/zoneinfo/Pacific/Auckland");
46//! // From a file relative to the system timezone directory
47//! let _ = TimeZone::from_posix_tz("Pacific/Auckland");
48//! // From a time zone description
49//! TimeZone::from_posix_tz("HST10")?;
50//! TimeZone::from_posix_tz("<-03>3")?;
51//! TimeZone::from_posix_tz("NZST-12:00:00NZDT-13:00:00,M10.1.0,M3.3.0")?;
52//! // Use a leading colon to force searching for a corresponding file
53//! let _ = TimeZone::from_posix_tz(":UTC");
54//! # }
55//! # Ok(())
56//! # }
57//! ```
58//!
59//! ## Date time
60//!
61//! ```rust
62//! # fn main() -> Result<(), tz::Error> {
63//! # #[cfg(feature = "std")] {
64//! use tz::{DateTime, LocalTimeType, TimeZone, UtcDateTime};
65//!
66//! // Get the current UTC date time
67//! let _current_utc_date_time = UtcDateTime::now()?;
68//!
69//! // Create a new UTC date time (2000-01-01T00:00:00.123456789Z)
70//! let utc_date_time = UtcDateTime::new(2000, 1, 1, 0, 0, 0, 123_456_789)?;
71//! assert_eq!(utc_date_time.year(), 2000);
72//! assert_eq!(utc_date_time.month(), 1);
73//! assert_eq!(utc_date_time.month_day(), 1);
74//! assert_eq!(utc_date_time.hour(), 0);
75//! assert_eq!(utc_date_time.minute(), 0);
76//! assert_eq!(utc_date_time.second(), 0);
77//! assert_eq!(utc_date_time.week_day(), 6);
78//! assert_eq!(utc_date_time.year_day(), 0);
79//! assert_eq!(utc_date_time.unix_time(), 946684800);
80//! assert_eq!(utc_date_time.nanoseconds(), 123_456_789);
81//! assert_eq!(utc_date_time.to_string(), "2000-01-01T00:00:00.123456789Z");
82//!
83//! // Create a new UTC date time from a Unix time with nanoseconds (2000-01-01T00:00:00.123456789Z)
84//! let other_utc_date_time = UtcDateTime::from_timespec(946684800, 123_456_789)?;
85//! assert_eq!(other_utc_date_time, utc_date_time);
86//!
87//! // Project the UTC date time to a time zone
88//! let date_time = utc_date_time.project(TimeZone::fixed(-3600)?.as_ref())?;
89//! assert_eq!(date_time.year(), 1999);
90//! assert_eq!(date_time.month(), 12);
91//! assert_eq!(date_time.month_day(), 31);
92//! assert_eq!(date_time.hour(), 23);
93//! assert_eq!(date_time.minute(), 0);
94//! assert_eq!(date_time.second(), 0);
95//! assert_eq!(date_time.week_day(), 5);
96//! assert_eq!(date_time.year_day(), 364);
97//! assert_eq!(date_time.local_time_type().ut_offset(), -3600);
98//! assert_eq!(date_time.unix_time(), 946684800);
99//! assert_eq!(date_time.nanoseconds(), 123_456_789);
100//! assert_eq!(date_time.to_string(), "1999-12-31T23:00:00.123456789-01:00");
101//!
102//! // Project the date time to another time zone
103//! let other_date_time = date_time.project(TimeZone::fixed(3600)?.as_ref())?;
104//! assert_eq!(other_date_time.year(), 2000);
105//! assert_eq!(other_date_time.month(), 1);
106//! assert_eq!(other_date_time.month_day(), 1);
107//! assert_eq!(other_date_time.hour(), 1);
108//! assert_eq!(other_date_time.minute(), 0);
109//! assert_eq!(other_date_time.second(), 0);
110//! assert_eq!(other_date_time.week_day(), 6);
111//! assert_eq!(other_date_time.year_day(), 0);
112//! assert_eq!(other_date_time.local_time_type().ut_offset(), 3600);
113//! assert_eq!(other_date_time.unix_time(), 946684800);
114//! assert_eq!(other_date_time.nanoseconds(), 123_456_789);
115//! assert_eq!(other_date_time.to_string(), "2000-01-01T01:00:00.123456789+01:00");
116//!
117//! // Create a new date time from a Unix time with nanoseconds and a time zone (2000-01-01T00:00:00.123456789Z)
118//! let another_date_time = DateTime::from_timespec(946684800, 123_456_789, TimeZone::fixed(86400)?.as_ref())?;
119//!
120//! // DateTime objects are compared by their Unix time and nanoseconds
121//! assert_eq!(another_date_time, other_date_time);
122//!
123//! // Get the current date time at the local time zone (UNIX only)
124//! let time_zone_local = TimeZone::local()?;
125//! let _date_time = DateTime::now(time_zone_local.as_ref())?;
126//!
127//! // Create a new date time with an UTC offset (2000-01-01T01:00:00.123456789+01:00)
128//! let date_time = DateTime::new(2000, 1, 1, 1, 0, 0, 123_456_789, LocalTimeType::with_ut_offset(3600)?)?;
129//! assert_eq!(date_time.year(), 2000);
130//! assert_eq!(date_time.month(), 1);
131//! assert_eq!(date_time.month_day(), 1);
132//! assert_eq!(date_time.hour(), 1);
133//! assert_eq!(date_time.minute(), 0);
134//! assert_eq!(date_time.second(), 0);
135//! assert_eq!(date_time.week_day(), 6);
136//! assert_eq!(date_time.year_day(), 0);
137//! assert_eq!(date_time.unix_time(), 946684800);
138//! assert_eq!(date_time.nanoseconds(), 123_456_789);
139//! assert_eq!(date_time.to_string(), "2000-01-01T01:00:00.123456789+01:00");
140//!
141//! //
142//! // Find the possible date times corresponding to a date, a time and a time zone
143//! //
144//! let time_zone = TimeZone::from_posix_tz("CET-1CEST,M3.5.0,M10.5.0/3")?;
145//!
146//! // Found date time is unique
147//! let found_date_times = DateTime::find(2000, 1, 1, 0, 0, 0, 0, time_zone.as_ref())?;
148//! let unique = found_date_times.unique().unwrap();
149//! assert_eq!(unique, found_date_times.earliest().unwrap());
150//! assert_eq!(unique, found_date_times.latest().unwrap());
151//! assert_eq!(unique.local_time_type().ut_offset(), 3600);
152//! assert_eq!(unique.local_time_type().time_zone_designation(), "CET");
153//!
154//! // Found date time was skipped by a forward transition
155//! let found_date_times = DateTime::find(2000, 3, 26, 2, 30, 0, 0, time_zone.as_ref())?;
156//!
157//! assert_eq!(found_date_times.unique(), None);
158//!
159//! let earliest = found_date_times.earliest().unwrap();
160//! assert_eq!(earliest.hour(), 2);
161//! assert_eq!(earliest.minute(), 0);
162//! assert_eq!(earliest.local_time_type().ut_offset(), 3600);
163//! assert_eq!(earliest.local_time_type().time_zone_designation(), "CET");
164//!
165//! let latest = found_date_times.latest().unwrap();
166//! assert_eq!(latest.hour(), 3);
167//! assert_eq!(latest.minute(), 0);
168//! assert_eq!(latest.local_time_type().ut_offset(), 7200);
169//! assert_eq!(latest.local_time_type().time_zone_designation(), "CEST");
170//!
171//! // Found date time is ambiguous because of a backward transition
172//! let found_date_times = DateTime::find(2000, 10, 29, 2, 30, 0, 0, time_zone.as_ref())?;
173//!
174//! assert_eq!(found_date_times.unique(), None);
175//!
176//! let earliest = found_date_times.earliest().unwrap();
177//! assert_eq!(earliest.hour(), 2);
178//! assert_eq!(earliest.minute(), 30);
179//! assert_eq!(earliest.local_time_type().ut_offset(), 7200);
180//! assert_eq!(earliest.local_time_type().time_zone_designation(), "CEST");
181//!
182//! let latest = found_date_times.latest().unwrap();
183//! assert_eq!(latest.hour(), 2);
184//! assert_eq!(latest.minute(), 30);
185//! assert_eq!(latest.local_time_type().ut_offset(), 3600);
186//! assert_eq!(latest.local_time_type().time_zone_designation(), "CET");
187//! # }
188//! # Ok(())
189//! # }
190//! ```
191//!
192//! # No std
193//!
194//! This crate can be used in `no_std` context.
195//!
196//! The `settings.rs` example shows how to construct a [`TimeZone`] by specifying a custom `read_file` function via the [`TimeZoneSettings`] struct.
197//!
198
199#[cfg(feature = "alloc")]
200extern crate alloc;
201
202mod constants;
203mod utils;
204
205#[cfg(feature = "alloc")]
206mod parse;
207
208pub mod datetime;
209pub mod error;
210pub mod timezone;
211
212#[doc(inline)]
213pub use datetime::{DateTime, UtcDateTime};
214
215#[doc(inline)]
216pub use error::{Error, TzError};
217
218#[doc(inline)]
219pub use timezone::{LocalTimeType, TimeZoneRef};
220
221#[doc(inline)]
222#[cfg(feature = "alloc")]
223pub use timezone::{TimeZone, TimeZoneSettings};