mirror of
https://github.com/Ed94/Odin.git
synced 2026-06-18 11:52:22 -07:00
72c15d7699
A new package `core:time/datetime` has been added which can represent moments much further in the past and future than `core:time`. It is based on *the* reference work on the subject, Calendrical Calculations Ultimate Edition, Reingold & Dershowitz. More procedures will be added to it in the future, to for example calculate the 3rd Thursday in March to figure out holidays. The package has been tested for more than a year and can handle dates 25 quadrillion years into the past and future with 64-bit day ordinals, or 5 million with 32-bit ones. This also fixes a longstanding bug where converting between YYYY-MM:DD hh:mm:ss and `time.Time` and back could result in a mismatch. RFC 3339 timestamps can now also be parsed using the `core:time` package.
262 lines
7.2 KiB
Odin
262 lines
7.2 KiB
Odin
/*
|
|
Calendrical conversions using a proleptic Gregorian calendar.
|
|
|
|
Implemented using formulas from: Calendrical Calculations Ultimate Edition, Reingold & Dershowitz
|
|
*/
|
|
package datetime
|
|
|
|
import "base:intrinsics"
|
|
|
|
// Procedures that return an Ordinal
|
|
date_to_ordinal :: proc "contextless" (date: Date) -> (ordinal: Ordinal, err: Error) {
|
|
validate(date) or_return
|
|
return unsafe_date_to_ordinal(date), .None
|
|
}
|
|
|
|
components_to_ordinal :: proc "contextless" (year, month, day: int) -> (ordinal: Ordinal, err: Error) {
|
|
return date_to_ordinal(Date{year, month, day})
|
|
}
|
|
|
|
// Procedures that return a Date
|
|
ordinal_to_date :: proc "contextless" (ordinal: Ordinal) -> (date: Date, err: Error) {
|
|
validate(ordinal) or_return
|
|
return unsafe_ordinal_to_date(ordinal), .None
|
|
}
|
|
|
|
components_to_date :: proc "contextless" (year, month, day: int) -> (date: Date, err: Error) {
|
|
date = Date{year, month, day}
|
|
validate(date) or_return
|
|
return date, .None
|
|
}
|
|
|
|
ordinal_to_datetime :: proc "contextless" (ordinal: Ordinal) -> (datetime: DateTime, err: Error) {
|
|
d := ordinal_to_date(ordinal) or_return
|
|
return {Date(d), {}}, .None
|
|
}
|
|
|
|
day_of_week :: proc "contextless" (ordinal: Ordinal) -> (day: Weekday) {
|
|
return Weekday((ordinal - EPOCH) %% 7)
|
|
}
|
|
|
|
subtract_dates :: proc "contextless" (a, b: Date) -> (delta: Delta, err: Error) {
|
|
ord_a := date_to_ordinal(a) or_return
|
|
ord_b := date_to_ordinal(b) or_return
|
|
|
|
delta = Delta{days=ord_a - ord_b}
|
|
return
|
|
}
|
|
|
|
subtract_datetimes :: proc "contextless" (a, b: DateTime) -> (delta: Delta, err: Error) {
|
|
ord_a := date_to_ordinal(a) or_return
|
|
ord_b := date_to_ordinal(b) or_return
|
|
|
|
validate(a.time) or_return
|
|
validate(b.time) or_return
|
|
|
|
seconds_a := a.hour * 3600 + a.minute * 60 + a.second
|
|
seconds_b := b.hour * 3600 + b.minute * 60 + b.second
|
|
|
|
delta = Delta{ord_a - ord_b, seconds_a - seconds_b, a.nano - b.nano}
|
|
return
|
|
}
|
|
|
|
subtract_deltas :: proc "contextless" (a, b: Delta) -> (delta: Delta, err: Error) {
|
|
delta = Delta{a.days - b.days, a.seconds - b.seconds, a.nanos - b.nanos}
|
|
delta = normalize_delta(delta) or_return
|
|
return
|
|
}
|
|
sub :: proc{subtract_datetimes, subtract_dates, subtract_deltas}
|
|
|
|
add_days_to_date :: proc "contextless" (a: Date, days: int) -> (date: Date, err: Error) {
|
|
ord := date_to_ordinal(a) or_return
|
|
ord += days
|
|
return ordinal_to_date(ord)
|
|
}
|
|
|
|
add_delta_to_date :: proc "contextless" (a: Date, delta: Delta) -> (date: Date, err: Error) {
|
|
ord := date_to_ordinal(a) or_return
|
|
// Because the input is a Date, we add only the days from the Delta.
|
|
ord += delta.days
|
|
return ordinal_to_date(ord)
|
|
}
|
|
|
|
add_delta_to_datetime :: proc "contextless" (a: DateTime, delta: Delta) -> (datetime: DateTime, err: Error) {
|
|
days := date_to_ordinal(a) or_return
|
|
|
|
a_seconds := a.hour * 3600 + a.minute * 60 + a.second
|
|
a_delta := Delta{days=days, seconds=a_seconds, nanos=a.nano}
|
|
|
|
sum_delta := Delta{days=a_delta.days + delta.days, seconds=a_delta.seconds + delta.seconds, nanos=a_delta.nanos + delta.nanos}
|
|
sum_delta = normalize_delta(sum_delta) or_return
|
|
|
|
datetime.date = ordinal_to_date(sum_delta.days) or_return
|
|
|
|
r: int
|
|
datetime.hour, r = divmod(sum_delta.seconds, 3600)
|
|
datetime.minute, datetime.second = divmod(r, 60)
|
|
datetime.nano = sum_delta.nanos
|
|
|
|
return
|
|
}
|
|
add :: proc{add_days_to_date, add_delta_to_date, add_delta_to_datetime}
|
|
|
|
day_number :: proc "contextless" (date: Date) -> (day_number: int, err: Error) {
|
|
validate(date) or_return
|
|
|
|
ord := unsafe_date_to_ordinal(date)
|
|
_, day_number = unsafe_ordinal_to_year(ord)
|
|
return
|
|
}
|
|
|
|
days_remaining :: proc "contextless" (date: Date) -> (days_remaining: int, err: Error) {
|
|
// Alternative formulation `day_number` subtracted from 365 or 366 depending on leap year
|
|
validate(date) or_return
|
|
delta := sub(date, Date{date.year, 12, 31}) or_return
|
|
return delta.days, .None
|
|
}
|
|
|
|
last_day_of_month :: proc "contextless" (year, month: int) -> (day: int, err: Error) {
|
|
// Not using formula 2.27 from the book. This is far simpler and gives the same answer.
|
|
|
|
validate(Date{year, month, 1}) or_return
|
|
month_days := MONTH_DAYS
|
|
|
|
day = month_days[month]
|
|
if month == 2 && is_leap_year(year) {
|
|
day += 1
|
|
}
|
|
return
|
|
}
|
|
|
|
new_year :: proc "contextless" (year: int) -> (new_year: Date, err: Error) {
|
|
new_year = {year, 1, 1}
|
|
validate(new_year) or_return
|
|
return
|
|
}
|
|
|
|
year_end :: proc "contextless" (year: int) -> (year_end: Date, err: Error) {
|
|
year_end = {year, 12, 31}
|
|
validate(year_end) or_return
|
|
return
|
|
}
|
|
|
|
year_range :: proc (year: int, allocator := context.allocator) -> (range: []Date) {
|
|
is_leap := is_leap_year(year)
|
|
|
|
days := 366 if is_leap else 365
|
|
range = make([]Date, days, allocator)
|
|
|
|
month_days := MONTH_DAYS
|
|
if is_leap {
|
|
month_days[2] = 29
|
|
}
|
|
|
|
i := 0
|
|
for month in 1..=len(month_days) {
|
|
for day in 1..=month_days[month] {
|
|
range[i] = Date{year, month, day}
|
|
i += 1
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
normalize_delta :: proc "contextless" (delta: Delta) -> (normalized: Delta, err: Error) {
|
|
// Distribute nanos into seconds and remainder
|
|
seconds, nanos := divmod(delta.nanos, 1e9)
|
|
|
|
// Add original seconds to rolled over seconds.
|
|
seconds += delta.seconds
|
|
days: int
|
|
|
|
// Distribute seconds into number of days and remaining seconds.
|
|
days, seconds = divmod(seconds, 24 * 3600)
|
|
|
|
// Add original days
|
|
days += delta.days
|
|
|
|
if days <= MIN_ORD || days >= MAX_ORD {
|
|
return {}, .Invalid_Delta
|
|
}
|
|
return Delta{days, seconds, nanos}, .None
|
|
}
|
|
|
|
// The following procedures don't check whether their inputs are in a valid range.
|
|
// They're still exported for those who know their inputs have been validated.
|
|
|
|
unsafe_date_to_ordinal :: proc "contextless" (date: Date) -> (ordinal: Ordinal) {
|
|
year_minus_one := date.year - 1
|
|
|
|
// Day before epoch
|
|
ordinal = EPOCH - 1
|
|
|
|
// Add non-leap days
|
|
ordinal += 365 * year_minus_one
|
|
|
|
// Add leap days
|
|
ordinal += floor_div(year_minus_one, 4) // Julian-rule leap days
|
|
ordinal -= floor_div(year_minus_one, 100) // Prior century years
|
|
ordinal += floor_div(year_minus_one, 400) // Prior 400-multiple years
|
|
ordinal += floor_div(367 * date.month - 362, 12) // Prior days this year
|
|
|
|
// Apply correction
|
|
if date.month <= 2 {
|
|
ordinal += 0
|
|
} else if is_leap_year(date.year) {
|
|
ordinal -= 1
|
|
} else {
|
|
ordinal -= 2
|
|
}
|
|
|
|
// Add days
|
|
ordinal += date.day
|
|
return
|
|
}
|
|
|
|
unsafe_ordinal_to_year :: proc "contextless" (ordinal: Ordinal) -> (year: int, day_ordinal: int) {
|
|
// Days after epoch
|
|
d0 := ordinal - EPOCH
|
|
|
|
// Number of 400-year cycles and remainder
|
|
n400, d1 := divmod(d0, 146097)
|
|
|
|
// Number of 100-year cycles and remainder
|
|
n100, d2 := divmod(d1, 36524)
|
|
|
|
// Number of 4-year cycles and remainder
|
|
n4, d3 := divmod(d2, 1461)
|
|
|
|
// Number of remaining days
|
|
n1, d4 := divmod(d3, 365)
|
|
|
|
year = 400 * n400 + 100 * n100 + 4 * n4 + n1
|
|
|
|
if n1 != 4 && n100 != 4 {
|
|
day_ordinal = d4 + 1
|
|
} else {
|
|
day_ordinal = 366
|
|
}
|
|
|
|
if n100 == 4 || n1 == 4 {
|
|
return year, day_ordinal
|
|
}
|
|
return year + 1, day_ordinal
|
|
}
|
|
|
|
unsafe_ordinal_to_date :: proc "contextless" (ordinal: Ordinal) -> (date: Date) {
|
|
year, _ := unsafe_ordinal_to_year(ordinal)
|
|
|
|
prior_days := ordinal - unsafe_date_to_ordinal(Date{year, 1, 1})
|
|
correction := Ordinal(2)
|
|
|
|
if ordinal < unsafe_date_to_ordinal(Date{year, 3, 1}) {
|
|
correction = 0
|
|
} else if is_leap_year(year) {
|
|
correction = 1
|
|
}
|
|
|
|
month := floor_div((12 * (prior_days + correction) + 373), 367)
|
|
day := ordinal - unsafe_date_to_ordinal(Date{year, month, 1}) + 1
|
|
|
|
return {year, month, day}
|
|
} |