-
Notifications
You must be signed in to change notification settings - Fork 54
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce a bitstype just for IANA Variable TimeZones #335
base: master
Are you sure you want to change the base?
Changes from 3 commits
2270ee8
886d2be
091493b
fc2535b
eb841d0
10b6d74
ae37248
96b0ee8
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -163,3 +163,7 @@ function last_valid(local_dt::DateTime, tz::VariableTimeZone) | |
possible = interpret(local_dt, tz, Local) | ||
return isempty(possible) ? first(shift_gap(local_dt, tz)) : last(possible) | ||
end | ||
|
||
|
||
first_valid(dt::DateTime, tz::IANATimeZone, args...) = _do_and_rewrap(first_valid, dt, tz, args...) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Probnably |
||
last_valid(dt::DateTime, tz::IANATimeZone, args...) = _do_and_rewrap(last_valid, dt, tz, args...) |
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,159 @@ | ||||||||||||||||||||||||||||||||||||||||||||||
const IANA_TABLE_SIZE = 1680 | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
perfect_hash(tz::VariableTimeZone) = perfect_hash(tz.name) | ||||||||||||||||||||||||||||||||||||||||||||||
function perfect_hash(str::AbstractString) | ||||||||||||||||||||||||||||||||||||||||||||||
# This was generated via `gperf` and translated by hand. | ||||||||||||||||||||||||||||||||||||||||||||||
# in case of collisions from new keys being added you must *not* regenerate it or | ||||||||||||||||||||||||||||||||||||||||||||||
# change the assoc table, as that will change existing hashes and break any serialized | ||||||||||||||||||||||||||||||||||||||||||||||
# data. Instead add special cases from the exact string via adding special cases. e.g. | ||||||||||||||||||||||||||||||||||||||||||||||
#`name=="New/Timezone" && return 1681` (after adjusting `IANA_TABLE_SIZE`) | ||||||||||||||||||||||||||||||||||||||||||||||
# The code below will (for all timezones in the 2021a release of tzdata) generate a | ||||||||||||||||||||||||||||||||||||||||||||||
# value between 29 and 1680 inclusive. So any new keys that need to be added | ||||||||||||||||||||||||||||||||||||||||||||||
# manually to resolve collisions can freely use anything outside that range. | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
asso_values = ( | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 4, 1681, 1, 3, 3, 241, 48, | ||||||||||||||||||||||||||||||||||||||||||||||
1, 22, 10, 9, 61, 18, 74, 9, 30, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
1681, 1681, 1681, 1681, 1681, 4, 2, 18, 167, 48, | ||||||||||||||||||||||||||||||||||||||||||||||
547, 691, 532, 446, 574, 466, 211, 452, 8, 387, | ||||||||||||||||||||||||||||||||||||||||||||||
110, 475, 601, 262, 83, 124, 580, 536, 60, 1, | ||||||||||||||||||||||||||||||||||||||||||||||
719, 2, 1681, 1681, 1681, 291, 1, 1, 27, 424, | ||||||||||||||||||||||||||||||||||||||||||||||
13, 5, 43, 3, 355, 12, 168, 148, 90, 179, | ||||||||||||||||||||||||||||||||||||||||||||||
4, 1, 315, 3, 3, 3, 2, 37, 5, 65, | ||||||||||||||||||||||||||||||||||||||||||||||
22, 320, 245, 1, 1681, 1681, 1681, | ||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How should i rewrap this? |
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
units = codeunits(str) | ||||||||||||||||||||||||||||||||||||||||||||||
len = length(units) | ||||||||||||||||||||||||||||||||||||||||||||||
hval = len | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
len >= 19 && (hval += asso_values[units[19]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 12 && (hval += asso_values[units[12]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 11 && (hval += asso_values[units[11]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 9 && (hval += asso_values[units[9] + 1]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 8 && (hval += asso_values[units[8]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 6 && (hval += asso_values[units[6] + 1]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 4 && (hval += asso_values[units[4]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 2 && (hval += asso_values[units[2] + 1]) | ||||||||||||||||||||||||||||||||||||||||||||||
len >= 1 && (hval += asso_values[units[1]]) | ||||||||||||||||||||||||||||||||||||||||||||||
len > 0 && (hval += asso_values[units[end]]) # add the last | ||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
This saves some time and simplifies the native code but relies on each code unit being in the range Might be worth it just to pad the array of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
I tried this and didn't notice any improvement.
I tried this and it made it way worse. I think because the tuple gets too lange for julia to be willing to do some optimizations. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. running @btime for x in ("America/Winnipeg", "Etc/UTC", "Canada/Central")
TimeZones.perfect_hash(x)
end Noting that i got rid of the vector in the benchmark code since that was dominating.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
growing assoc to 127 seems to help: fikkiwubg wutgh tgat:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hmm i don't trust my benchmarks I think we should do that though, and i will push that code |
||||||||||||||||||||||||||||||||||||||||||||||
return hval | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
const IANA_TIMEZONES = Vector{VariableTimeZone}(undef, IANA_TABLE_SIZE) | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
# TODO: maybe fill this during build(), probably by generating a julia file. | ||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Or maybe it is fine, since it only queries what files exist, using |
||||||||||||||||||||||||||||||||||||||||||||||
# That way we can avoid actually instantitating every timezone til it is needed. | ||||||||||||||||||||||||||||||||||||||||||||||
oxinabox marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||
const IANA_NAMES = Vector{String}(undef, IANA_TABLE_SIZE) | ||||||||||||||||||||||||||||||||||||||||||||||
function init_IANA_NAMES!() # this is run by __init__ (at least for now) | ||||||||||||||||||||||||||||||||||||||||||||||
for name in timezone_names() | ||||||||||||||||||||||||||||||||||||||||||||||
# TODO: we should workout how to filter out FixedTimeZones here | ||||||||||||||||||||||||||||||||||||||||||||||
oxinabox marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||
id = perfect_hash(name) | ||||||||||||||||||||||||||||||||||||||||||||||
# Important: Make sure our hash is perfect (even module the table size) | ||||||||||||||||||||||||||||||||||||||||||||||
oxinabox marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||
isassigned(IANA_NAMES, id) && error("hash collision for $tz, at $id") | ||||||||||||||||||||||||||||||||||||||||||||||
IANA_NAMES[id] = name | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
return IANA_NAMES | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
function is_standard_iana(str::AbstractString) | ||||||||||||||||||||||||||||||||||||||||||||||
id = perfect_hash(str) | ||||||||||||||||||||||||||||||||||||||||||||||
return isassigned(IANA_NAMES, id) && IANA_NAMES[id] == str | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
function get_iana_timezone!(str::AbstractString) | ||||||||||||||||||||||||||||||||||||||||||||||
id = perfect_hash(str) | ||||||||||||||||||||||||||||||||||||||||||||||
if isassigned(IANA_TIMEZONES, id) | ||||||||||||||||||||||||||||||||||||||||||||||
IANA_TIMEZONES[id] | ||||||||||||||||||||||||||||||||||||||||||||||
else | ||||||||||||||||||||||||||||||||||||||||||||||
tz_path = joinpath(TZData.COMPILED_DIR, split(str, "/")...) | ||||||||||||||||||||||||||||||||||||||||||||||
tz, class = deserialize(tz_path) | ||||||||||||||||||||||||||||||||||||||||||||||
if tz isa VariableTimeZone | ||||||||||||||||||||||||||||||||||||||||||||||
IANA_TIMEZONES[id] = tz | ||||||||||||||||||||||||||||||||||||||||||||||
return IANATimeZone(perfect_hash(str)) | ||||||||||||||||||||||||||||||||||||||||||||||
else | ||||||||||||||||||||||||||||||||||||||||||||||
# it is a FixedTimeZone, we are not going to use a IANATimeZone | ||||||||||||||||||||||||||||||||||||||||||||||
return tz | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
function get_iana_timezone!(id::UInt) | ||||||||||||||||||||||||||||||||||||||||||||||
if isassigned(IANA_NAMES, id) | ||||||||||||||||||||||||||||||||||||||||||||||
name = IANA_NAMES[id] | ||||||||||||||||||||||||||||||||||||||||||||||
return get_iana_timezone!(name) | ||||||||||||||||||||||||||||||||||||||||||||||
else | ||||||||||||||||||||||||||||||||||||||||||||||
error( | ||||||||||||||||||||||||||||||||||||||||||||||
"$id does not correspond to any known IANA timezone. " * | ||||||||||||||||||||||||||||||||||||||||||||||
"Check you are using the right version of the IANA database.", | ||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||
IANATimeZone(::AbstractString) <: AbstractVariableTimeZone | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
A type for representing a standard variable IANA TimeZome from the tzdata. | ||||||||||||||||||||||||||||||||||||||||||||||
Under-the-hood it stores only a unique integer identifier. | ||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||
struct IANATimeZone <: TimeZone | ||||||||||||||||||||||||||||||||||||||||||||||
# id must be a prefect_hash of the corresponding timezone name | ||||||||||||||||||||||||||||||||||||||||||||||
id::UInt | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
function IANATimeZone(name::AbstractString) | ||||||||||||||||||||||||||||||||||||||||||||||
return IANATimeZone(perfect_hash(name)) | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
backing_timezone(itz::IANATimeZone) = get_iana_timezone!(itz.id)::VariableTimeZone | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
Base.:(==)(a::IANATimeZone, b::IANATimeZone) = a.id == b.id | ||||||||||||||||||||||||||||||||||||||||||||||
Base.:(==)(a::IANATimeZone, b::TimeZone) = backing_timezone(a) == b | ||||||||||||||||||||||||||||||||||||||||||||||
Base.:(==)(b::TimeZone, a::IANATimeZone) = backing_timezone(a) == b | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
# TODO: we have the hash, it seems like we should be able to use that to get seeded hash | ||||||||||||||||||||||||||||||||||||||||||||||
oxinabox marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||
Base.hash(a::IANATimeZone, seed::UInt) = hash(backing_timezone(a), seed) | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
name(a::IANATimeZone) = name(backing_timezone(a)) | ||||||||||||||||||||||||||||||||||||||||||||||
transitions(tz::IANATimeZone) = transitions(backing_timezone(tz)) | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
# TODO: should i just make this check the fields of VariableTimeZone and just delegate all? | ||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this should be addressed |
||||||||||||||||||||||||||||||||||||||||||||||
function Base.getproperty(tz::IANATimeZone, s::Symbol) | ||||||||||||||||||||||||||||||||||||||||||||||
if s === :name | ||||||||||||||||||||||||||||||||||||||||||||||
return name(tz) | ||||||||||||||||||||||||||||||||||||||||||||||
elseif s == :transitions | ||||||||||||||||||||||||||||||||||||||||||||||
return transitions(tz) | ||||||||||||||||||||||||||||||||||||||||||||||
else | ||||||||||||||||||||||||||||||||||||||||||||||
return getfield(tz, s) | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
function Base.hasproperty(tz::IANATimeZone, s::Symbol) | ||||||||||||||||||||||||||||||||||||||||||||||
return s === :name || s === :transitions || hasfield(IANATimeZone, s) | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
oxinabox marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
"""" | ||||||||||||||||||||||||||||||||||||||||||||||
_do_and_rewrap(f, arg1, tz::IANATimeZone, args...; kwargs...) | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
Run the function `f(arg1, backing_timezone(tz), args...; kwargs...)` | ||||||||||||||||||||||||||||||||||||||||||||||
which must return a `ZonedDateTime`, with the backing timezone. | ||||||||||||||||||||||||||||||||||||||||||||||
Replace the timezone field with `tz` (which should be equivalent). | ||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||
function _do_and_rewrap(f, arg1, tz::IANATimeZone, args...; kwargs...) | ||||||||||||||||||||||||||||||||||||||||||||||
backed_tz = backing_timezone(tz) | ||||||||||||||||||||||||||||||||||||||||||||||
backed_zdt::ZonedDateTime = f(arg1, backing_timezone(tz), args...; kwargs...) | ||||||||||||||||||||||||||||||||||||||||||||||
# make it store tz rather than the equiv backing timezone, other fields the same | ||||||||||||||||||||||||||||||||||||||||||||||
return ZonedDateTime(backed_zdt.utc_datetime, tz, backed_zdt.zone) | ||||||||||||||||||||||||||||||||||||||||||||||
end | ||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||
Base.show(io::IO, tz::IANATimeZone) = show(io, backing_timezone(tz)) | ||||||||||||||||||||||||||||||||||||||||||||||
Base.print(io::IO, tz::IANATimeZone) = print(io, backing_timezone(tz)) |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -46,7 +46,10 @@ function TimeZone(str::AbstractString, mask::Class=Class(:DEFAULT)) | |
tz, class = get!(TIME_ZONE_CACHE, str) do | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @omus input on the best way to rewrite this function would apprecated. |
||
tz_path = joinpath(TZData.COMPILED_DIR, split(str, "/")...) | ||
|
||
if isfile(tz_path) | ||
if mask == Class(:DEFAULT) && is_standard_iana(str) | ||
# TODO: idk if this as sensible way to handle class and mask | ||
get_iana_timezone!(str), Class(:DEFAULT) | ||
elseif isfile(tz_path) | ||
open(deserialize, tz_path, "r") | ||
elseif occursin(FIXED_TIME_ZONE_REGEX, str) | ||
FixedTimeZone(str), Class(:FIXED) | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It might be premature optimization to pull this out to be done before.
Idea is to avoid relooking up the
backing_timezone
forIANATimeZone
multiple times