const MILLIS_PER_SECOND: i64 = 1_000;
const MILLIS_PER_MINUTE: i64 = 60 * MILLIS_PER_SECOND;
const MILLIS_PER_HOUR: i64 = 60 * MILLIS_PER_MINUTE;
const MILLIS_PER_DAY: i64 = 24 * MILLIS_PER_HOUR;
const DAYS_IN_MONTH: [i64; 12] = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];
pub fn datetime_now() -> i64 {
std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.unwrap_or_default()
.as_millis() as i64
}
pub fn datetime_from_epoch(millis: i64) -> i64 {
millis
}
pub fn datetime_from_parts(year: i64, month: i64, day: i64, hour: i64, min: i64, sec: i64) -> i64 {
let days = days_from_civil(year, month, day);
days * MILLIS_PER_DAY + hour * MILLIS_PER_HOUR + min * MILLIS_PER_MINUTE + sec * MILLIS_PER_SECOND
}
pub fn datetime_year(millis: i64) -> i64 {
let (y, _, _) = civil_from_days(millis.div_euclid(MILLIS_PER_DAY));
y
}
pub fn datetime_month(millis: i64) -> i64 {
let (_, m, _) = civil_from_days(millis.div_euclid(MILLIS_PER_DAY));
m
}
pub fn datetime_day(millis: i64) -> i64 {
let (_, _, d) = civil_from_days(millis.div_euclid(MILLIS_PER_DAY));
d
}
pub fn datetime_hour(millis: i64) -> i64 {
let day_millis = millis.rem_euclid(MILLIS_PER_DAY);
day_millis / MILLIS_PER_HOUR
}
pub fn datetime_minute(millis: i64) -> i64 {
let day_millis = millis.rem_euclid(MILLIS_PER_DAY);
(day_millis % MILLIS_PER_HOUR) / MILLIS_PER_MINUTE
}
pub fn datetime_second(millis: i64) -> i64 {
let day_millis = millis.rem_euclid(MILLIS_PER_DAY);
(day_millis % MILLIS_PER_MINUTE) / MILLIS_PER_SECOND
}
pub fn datetime_diff(a: i64, b: i64) -> i64 {
a - b
}
pub fn datetime_add_millis(dt: i64, millis: i64) -> i64 {
dt + millis
}
pub fn datetime_format(millis: i64) -> String {
let days = millis.div_euclid(MILLIS_PER_DAY);
let (year, month, day) = civil_from_days(days);
let day_millis = millis.rem_euclid(MILLIS_PER_DAY);
let hour = day_millis / MILLIS_PER_HOUR;
let minute = (day_millis % MILLIS_PER_HOUR) / MILLIS_PER_MINUTE;
let second = (day_millis % MILLIS_PER_MINUTE) / MILLIS_PER_SECOND;
format!(
"{:04}-{:02}-{:02}T{:02}:{:02}:{:02}Z",
year, month, day, hour, minute, second
)
}
fn is_leap_year(year: i64) -> bool {
(year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)
}
fn days_from_civil(year: i64, month: i64, day: i64) -> i64 {
let (y, m) = if month <= 2 {
(year - 1, month + 9)
} else {
(year, month - 3)
};
let era = y.div_euclid(400);
let yoe = y.rem_euclid(400); let doy = (153 * m + 2) / 5 + day - 1; let doe = yoe * 365 + yoe / 4 - yoe / 100 + doy; era * 146097 + doe - 719468 }
fn civil_from_days(days: i64) -> (i64, i64, i64) {
let z = days + 719468;
let era = z.div_euclid(146097);
let doe = z.rem_euclid(146097); let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146096) / 365; let y = yoe + era * 400;
let doy = doe - (365 * yoe + yoe / 4 - yoe / 100); let mp = (5 * doy + 2) / 153; let d = doy - (153 * mp + 2) / 5 + 1; let m = if mp < 10 { mp + 3 } else { mp - 9 };
let y = if m <= 2 { y + 1 } else { y };
(y, m, d)
}
pub fn days_in_month(year: i64, month: i64) -> i64 {
if month == 2 && is_leap_year(year) {
29
} else if month >= 1 && month <= 12 {
DAYS_IN_MONTH[(month - 1) as usize]
} else {
0
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_epoch_origin() {
let dt = datetime_from_parts(1970, 1, 1, 0, 0, 0);
assert_eq!(dt, 0);
}
#[test]
fn test_known_date() {
let dt = datetime_from_parts(2000, 1, 1, 0, 0, 0);
assert_eq!(datetime_year(dt), 2000);
assert_eq!(datetime_month(dt), 1);
assert_eq!(datetime_day(dt), 1);
assert_eq!(datetime_hour(dt), 0);
}
#[test]
fn test_extraction_roundtrip() {
let dt = datetime_from_parts(2024, 6, 15, 14, 30, 45);
assert_eq!(datetime_year(dt), 2024);
assert_eq!(datetime_month(dt), 6);
assert_eq!(datetime_day(dt), 15);
assert_eq!(datetime_hour(dt), 14);
assert_eq!(datetime_minute(dt), 30);
assert_eq!(datetime_second(dt), 45);
}
#[test]
fn test_leap_year() {
assert!(is_leap_year(2000));
assert!(is_leap_year(2024));
assert!(!is_leap_year(1900));
assert!(!is_leap_year(2023));
}
#[test]
fn test_days_in_feb_leap() {
assert_eq!(days_in_month(2024, 2), 29);
assert_eq!(days_in_month(2023, 2), 28);
}
#[test]
fn test_format_iso8601() {
let dt = datetime_from_parts(2024, 3, 14, 9, 26, 53);
let s = datetime_format(dt);
assert_eq!(s, "2024-03-14T09:26:53Z");
}
#[test]
fn test_diff() {
let a = datetime_from_parts(2024, 1, 2, 0, 0, 0);
let b = datetime_from_parts(2024, 1, 1, 0, 0, 0);
assert_eq!(datetime_diff(a, b), MILLIS_PER_DAY);
}
#[test]
fn test_add_millis() {
let dt = datetime_from_parts(2024, 1, 1, 0, 0, 0);
let dt2 = datetime_add_millis(dt, MILLIS_PER_HOUR);
assert_eq!(datetime_hour(dt2), 1);
}
#[test]
fn test_format_epoch() {
assert_eq!(datetime_format(0), "1970-01-01T00:00:00Z");
}
#[test]
fn test_determinism() {
let a = datetime_from_parts(2024, 12, 31, 23, 59, 59);
let b = datetime_from_parts(2024, 12, 31, 23, 59, 59);
assert_eq!(a, b);
assert_eq!(datetime_format(a), datetime_format(b));
}
}