Struct TimeZoneDatabase
struct TimeZoneDatabase { ... }
A handle to a IANA Time Zone Database.
A TimeZoneDatabase provides a way to lookup TimeZones by their
human readable identifiers, such as America/Los_Angeles and
Europe/Warsaw.
It is rare to need to create or use this type directly. Routines
like zoned datetime parsing and time zone conversion provide
convenience routines for using an implicit global time zone database
by default. This global time zone database is available via
jiff::tz::db. But lower level parsing routines
such as
fmt::temporal::DateTimeParser::parse_zoned_with
and
civil::DateTime::to_zoned provide a
means to use a custom copy of a TimeZoneDatabase.
Platform behavior
This behavior is subject to change.
On Unix systems, and when the tzdb-zoneinfo crate feature is enabled
(which it is by default), Jiff will read the /usr/share/zoneinfo
directory for time zone data.
On Windows systems and when the tzdb-bundle-platform crate feature is
enabled (which it is by default), or when the tzdb-bundle-always crate
feature is enabled, then the jiff-tzdb crate will be used to embed the
entire Time Zone Database into the compiled artifact.
On Android systems, and when the tzdb-concatenated crate feature is
enabled (which it is by default), Jiff will attempt to read a concatenated
zoneinfo database using the ANDROID_DATA or ANDROID_ROOT environment
variables.
In general, using /usr/share/zoneinfo (or an equivalent) is heavily
preferred in lieu of embedding the database into your compiled artifact.
The reason is because your system copy of the Time Zone Database may be
updated, perhaps a few times a year, and it is better to get seamless
updates through your system rather than needing to wait on a Rust crate
to update and then rebuild your software. The bundling approach should
only be used when there is no plausible alternative. For example, Windows
has no canonical location for a copy of the Time Zone Database. Indeed,
this is why the Cargo configuration of Jiff specifically does not enabled
bundling by default on Unix systems, but does enable it by default on
Windows systems. Of course, if you really do need a copy of the database
bundled, then you can enable the tzdb-bundle-always crate feature.
Cloning
A TimeZoneDatabase can be cheaply cloned. It will share a thread safe
cache with other copies of the same TimeZoneDatabase.
Caching
Because looking up a time zone on disk, reading the file into memory
and parsing the time zone transitions out of that file requires
a fair amount of work, a TimeZoneDatabase does a fair bit of
caching. This means that the vast majority of calls to, for example,
Timestamp::in_tz don't actually need to hit
disk. It will just find a cached copy of a TimeZone and return that.
Of course, with caching comes problems of cache invalidation. Invariably,
there are parameters that Jiff uses to manage when the cache should be
invalidated. Jiff tries to emit log messages about this when it happens. If
you find the caching behavior of Jiff to be sub-optimal for your use case,
please create an issue. (The plan is likely to expose some options for
configuring the behavior of a TimeZoneDatabase, but I wanted to collect
user feedback first.)
Example: list all available time zones
use jiff::tz;
for tzid in tz::db().available() {
println!("{tzid}");
}
Example: using multiple time zone databases
Jiff supports opening and using multiple time zone databases by default.
All you need to do is point TimeZoneDatabase::from_dir to your own
copy of the Time Zone Database, and it will handle the rest.
This example shows how to utilize multiple databases by parsing a datetime using an older copy of the IANA Time Zone Database. This example leverages the fact that the 2018 copy of the database preceded Brazil's announcement that daylight saving time would be abolished. This meant that datetimes in the future, when parsed with the older copy of the Time Zone Database, would still follow the old daylight saving time rules. But a mere update of the database would otherwise change the meaning of the datetime.
This scenario can come up if one stores datetimes in the future. This is
also why the default offset conflict resolution strategy when parsing zoned
datetimes is OffsetConflict::Reject,
which prevents one from silently re-interpreting datetimes to a different
timestamp.
use jiff::{fmt::temporal::DateTimeParser, tz::{self, TimeZoneDatabase}};
static PARSER: DateTimeParser = DateTimeParser::new();
// Open a version of tzdb from before Brazil announced its abolition
// of daylight saving time.
let tzdb2018 = TimeZoneDatabase::from_dir("path/to/tzdb-2018b")?;
// Open the system tzdb.
let tzdb = tz::db();
// Parse the same datetime string with the same parser, but using two
// different versions of tzdb.
let dt = "2020-01-15T12:00[America/Sao_Paulo]";
let zdt2018 = PARSER.parse_zoned_with(&tzdb2018, dt)?;
let zdt = PARSER.parse_zoned_with(tzdb, dt)?;
// Before DST was abolished, 2020-01-15 was in DST, which corresponded
// to UTC offset -02. Since DST rules applied to datetimes in the
// future, the 2018 version of tzdb would lead one to interpret
// 2020-01-15 as being in DST.
assert_eq!(zdt2018.offset(), tz::offset(-2));
// But DST was abolished in 2019, which means that 2020-01-15 was no
// no longer in DST. So after a tzdb update, the same datetime as above
// now has a different offset.
assert_eq!(zdt.offset(), tz::offset(-3));
// So if you try to parse a datetime serialized from an older copy of
// tzdb, you'll get an error under the default configuration because
// of `OffsetConflict::Reject`. This would succeed if you parsed it
// using tzdb2018!
assert!(PARSER.parse_zoned_with(tzdb, zdt2018.to_string()).is_err());
# Ok::<(), Box<dyn std::error::Error>>(())
Implementations
impl TimeZoneDatabase
const fn none() -> TimeZoneDatabaseReturns a database for which all time zone lookups fail.
Example
use TimeZoneDatabase; let db = none; assert_eq!;fn from_env() -> TimeZoneDatabaseReturns a time zone database initialized from the current environment.
This routine never fails, but it may not be able to find a copy of your Time Zone Database. When this happens, log messages (with some at least at the
WARNlevel) will be emitted. They can be viewed by installing alogcompatible logger such asenv_logger.Typically, one does not need to call this routine directly. Instead, it's done for you as part of
jiff::tz::db. This does require Jiff'sstdfeature to be enabled though. So for example, you might use this constructor when the featuresallocandtzdb-bundle-alwaysare enabled to get access to a bundled copy of the IANA time zone database. (Accessing the system copy at/usr/share/zoneinforequiresstd.)Beware that calling this constructor will create a new distinct handle from the one returned by
jiff::tz::dbwith its own cache.Platform behavior
When the
TZDIRenvironment variable is set, this will attempt to open the Time Zone Database at the directory specified. Otherwise, this will search a list of predefined directories for a system installation of the Time Zone Database. Typically, it's found at/usr/share/zoneinfo.On Windows systems, under the default crate configuration, this will return an embedded copy of the Time Zone Database since Windows does not have a canonical installation of the Time Zone Database.
fn from_dir<P: AsRef<std::path::Path>>(path: P) -> Result<TimeZoneDatabase, Error>Returns a time zone database initialized from the given directory.
Unlike
TimeZoneDatabase::from_env, this always attempts to look for a copy of the Time Zone Database at the directory given. And if it fails to find one at that directory, then an error is returned.Basically, you should use this when you need to use a specific copy of the Time Zone Database, and use
TimeZoneDatabase::from_envwhen you just want Jiff to try and "do the right thing for you."Errors
This returns an error if the given directory does not contain a valid copy of the Time Zone Database. Generally, this means a directory with at least one valid TZif file.
fn from_concatenated_path<P: AsRef<std::path::Path>>(path: P) -> Result<TimeZoneDatabase, Error>Returns a time zone database initialized from a path pointing to a concatenated
tzdatafile. This type of format is only known to be found on Android environments. The specific format for this file isn't defined formally anywhere, but Jiff parses the same format supported by the Android Platform.Unlike
TimeZoneDatabase::from_env, this always attempts to look for a copy of the Time Zone Database at the path given. And if it fails to find one at that path, then an error is returned.Basically, you should use this when you need to use a specific copy of the Time Zone Database in its concatenated format, and use
TimeZoneDatabase::from_envwhen you just want Jiff to try and "do the right thing for you." (TimeZoneDatabase::from_envwill attempt to automatically detect the presence of a system concatenatedtzdatafile on Android.)Errors
This returns an error if the given path does not contain a valid copy of the concatenated Time Zone Database.
fn bundled() -> TimeZoneDatabaseReturns a time zone database initialized from the bundled copy of the IANA Time Zone Database.
While this API is always available, in order to get a non-empty database back, this requires that one of the crate features
tzdb-bundle-alwaysortzdb-bundle-platformis enabled. In the latter case, the bundled database is only available on platforms known to lack a system copy of the IANA Time Zone Database (i.e., non-Unix systems).This routine is infallible, but it may return a database that is definitively empty if the bundled data is not available. To query whether the data is empty or not, use
TimeZoneDatabase::is_definitively_empty.fn get(self: &Self, name: &str) -> Result<TimeZone, Error>Returns a
TimeZonecorresponding to the IANA time zone identifier given.The lookup is performed without regard to ASCII case.
To see a list of all available time zone identifiers for this database, use
TimeZoneDatabase::available.Example
use tz; let tz = db.get?; assert_eq!; # Ok::fn available<'d>(self: &'d Self) -> TimeZoneNameIter<'d>Returns a list of all available time zone identifiers from this database.
Note that time zone identifiers are more of a machine readable abstraction and not an end user level abstraction. Still, users comfortable with configuring their system's default time zone through IANA time zone identifiers are probably comfortable interacting with the identifiers returned here.
Example
use jiff::tz; for tzid in tz::db().available() { println!("{tzid}"); }fn reset(self: &Self)Resets the internal cache of this database.
Subsequent interactions with this database will need to re-read time zone data from disk.
It might be useful to call this if you know the time zone database has changed on disk and want to force Jiff to re-load it immediately without spawning a new process or waiting for Jiff's internal cache invalidation heuristics to kick in.
fn is_definitively_empty(self: &Self) -> boolReturns true if it is known that this time zone database is empty.
When this returns true, it is guaranteed that all
TimeZoneDatabase::getcalls will fail, and thatTimeZoneDatabase::availablewill always return an empty iterator.Note that if this returns false, it is still possible for this database to be empty.
Example
use TimeZoneDatabase; let db = none; assert!;
impl Clone for TimeZoneDatabase
fn clone(self: &Self) -> TimeZoneDatabase
impl Debug for TimeZoneDatabase
fn fmt(self: &Self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result
impl Freeze for TimeZoneDatabase
impl RefUnwindSafe for TimeZoneDatabase
impl Send for TimeZoneDatabase
impl Sync for TimeZoneDatabase
impl Unpin for TimeZoneDatabase
impl UnwindSafe for TimeZoneDatabase
impl<T> Any for TimeZoneDatabase
fn type_id(self: &Self) -> TypeId
impl<T> Borrow for TimeZoneDatabase
fn borrow(self: &Self) -> &T
impl<T> BorrowMut for TimeZoneDatabase
fn borrow_mut(self: &mut Self) -> &mut T
impl<T> CloneToUninit for TimeZoneDatabase
unsafe fn clone_to_uninit(self: &Self, dest: *mut u8)
impl<T> From for TimeZoneDatabase
fn from(t: T) -> TReturns the argument unchanged.
impl<T> ToOwned for TimeZoneDatabase
fn to_owned(self: &Self) -> Tfn clone_into(self: &Self, target: &mut T)
impl<T, U> Into for TimeZoneDatabase
fn into(self: Self) -> UCalls
U::from(self).That is, this conversion is whatever the implementation of
[From]<T> for Uchooses to do.
impl<T, U> TryFrom for TimeZoneDatabase
fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>
impl<T, U> TryInto for TimeZoneDatabase
fn try_into(self: Self) -> Result<U, <U as TryFrom<T>>::Error>