/// Date represented as days since epoch (compact, sortable). /// Use helper functions for formatting and parsing. pub const Date = struct { /// Days since 1970-01-01 days: i32, pub const epoch = Date{ .days = 0 }; pub fn year(self: Date) i16 { return epochDaysToYmd(self.days).year; } pub fn month(self: Date) u8 { return epochDaysToYmd(self.days).month; } pub fn day(self: Date) u8 { return epochDaysToYmd(self.days).day; } pub fn fromYmd(y: i16, m: u8, d: u8) Date { return .{ .days = ymdToEpochDays(y, m, d) }; } /// Parse "YYYY-MM-DD" format pub fn parse(str: []const u8) !Date { if (str.len != 10 or str[4] != '-' or str[7] != '-') return error.InvalidDateFormat; const y = std.fmt.parseInt(i16, str[0..4], 10) catch return error.InvalidDateFormat; const m = std.fmt.parseInt(u8, str[5..7], 10) catch return error.InvalidDateFormat; const d = std.fmt.parseInt(u8, str[8..10], 10) catch return error.InvalidDateFormat; return fromYmd(y, m, d); } /// Format as "YYYY-MM-DD" pub fn format(self: Date, buf: *[10]u8) []const u8 { const ymd = epochDaysToYmd(self.days); const y: u16 = @intCast(ymd.year); buf[0] = '0' + @as(u8, @intCast(y / 1000)); buf[1] = '0' + @as(u8, @intCast((y / 100) % 10)); buf[2] = '0' + @as(u8, @intCast((y / 10) % 10)); buf[3] = '0' + @as(u8, @intCast(y % 10)); buf[4] = '-'; buf[5] = '0' + @as(u8, @intCast(ymd.month / 10)); buf[6] = '0' + @as(u8, @intCast(ymd.month % 10)); buf[7] = '-'; buf[8] = '0' + @as(u8, @intCast(ymd.day / 10)); buf[9] = '0' + @as(u8, @intCast(ymd.day % 10)); return buf[0..10]; } /// Day of week: 0=Monday, 1=Tuesday, ..., 4=Friday, 5=Saturday, 6=Sunday. pub fn dayOfWeek(self: Date) u8 { // 1970-01-01 was a Thursday (day 3 in 0=Mon scheme) const d = @mod(self.days + 3, @as(i32, 7)); return @intCast(if (d < 0) d + 7 else d); } pub fn eql(a: Date, b: Date) bool { return a.days == b.days; } pub fn lessThan(a: Date, b: Date) bool { return a.days < b.days; } pub fn addDays(self: Date, n: i32) Date { return .{ .days = self.days + n }; } /// Subtract N calendar years. Clamps Feb 29 -> Feb 28 if target is not a leap year. pub fn subtractYears(self: Date, n: u16) Date { const ymd = epochDaysToYmd(self.days); const new_year: i16 = ymd.year - @as(i16, @intCast(n)); const new_day: u8 = if (ymd.month == 2 and ymd.day == 29 and !isLeapYear(new_year)) 28 else ymd.day; return .{ .days = ymdToEpochDays(new_year, ymd.month, new_day) }; } /// Subtract N calendar months. Clamps day to end of month if needed (e.g. Mar 31 - 1M = Feb 28). pub fn subtractMonths(self: Date, n: u16) Date { const ymd = epochDaysToYmd(self.days); var m: i32 = @as(i32, ymd.month) - @as(i32, n); var y: i16 = ymd.year; while (m < 1) { m += 12; y -= 1; } const new_month: u8 = @intCast(m); const max_day = daysInMonth(y, new_month); const new_day: u8 = if (ymd.day > max_day) max_day else ymd.day; return .{ .days = ymdToEpochDays(y, new_month, new_day) }; } /// Return the last day of the previous month. /// E.g., if self is 2026-02-24, returns 2026-01-31. pub fn lastDayOfPriorMonth(self: Date) Date { const ymd = epochDaysToYmd(self.days); if (ymd.month == 1) { return fromYmd(ymd.year - 1, 12, 31); } else { return fromYmd(ymd.year, ymd.month - 1, daysInMonth(ymd.year, ymd.month - 1)); } } fn daysInMonth(y: i16, m: u8) u8 { const table = [_]u8{ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; if (m == 2 and isLeapYear(y)) return 29; return table[m - 1]; } /// Returns approximate number of years between two dates pub fn yearsBetween(from: Date, to: Date) f64 { return @as(f64, @floatFromInt(to.days - from.days)) / 365.25; } fn isLeapYear(y: i16) bool { const yu: u16 = @bitCast(y); return (yu % 4 == 0 and yu % 100 != 0) or (yu % 400 == 0); } }; const Ymd = struct { year: i16, month: u8, day: u8 }; fn epochDaysToYmd(days: i32) Ymd { // Algorithm from http://howardhinnant.github.io/date_algorithms.html // Using i64 throughout to avoid overflow on unsigned intermediate values. const z: i64 = @as(i64, days) + 719468; const era: i64 = @divFloor(if (z >= 0) z else z - 146096, 146097); const doe_i: i64 = z - era * 146097; // [0, 146096] const doe: u64 = @intCast(doe_i); const yoe_val: u64 = (doe - doe / 1460 + doe / 36524 - doe / 146096) / 365; // [0, 399] const y: i64 = @as(i64, @intCast(yoe_val)) + era * 400; const doy: u64 = doe - (365 * yoe_val + yoe_val / 4 - yoe_val / 100); const mp: u64 = (5 * doy + 2) / 153; const d: u8 = @intCast(doy - (153 * mp + 2) / 5 + 1); const m_raw: u64 = if (mp < 10) mp + 3 else mp - 9; const m: u8 = @intCast(m_raw); const y_adj: i16 = @intCast(if (m <= 2) y + 1 else y); return .{ .year = y_adj, .month = m, .day = d }; } fn ymdToEpochDays(y: i16, m: u8, d: u8) i32 { const y_adj: i64 = @as(i64, y) - @as(i64, if (m <= 2) @as(i64, 1) else @as(i64, 0)); const era: i64 = @divFloor(if (y_adj >= 0) y_adj else y_adj - 399, 400); const yoe: u64 = @intCast(y_adj - era * 400); const m_adj: u64 = if (m > 2) @as(u64, m) - 3 else @as(u64, m) + 9; const doy: u64 = (153 * m_adj + 2) / 5 + @as(u64, d) - 1; const doe: u64 = yoe * 365 + yoe / 4 -| yoe / 100 + doy; return @intCast(era * 146097 + @as(i64, @intCast(doe)) - 719468); } const std = @import("std"); test "date roundtrip" { const d = Date.fromYmd(2024, 6, 15); try std.testing.expectEqual(@as(i16, 2024), d.year()); try std.testing.expectEqual(@as(u8, 6), d.month()); try std.testing.expectEqual(@as(u8, 15), d.day()); } test "date parse" { const d = try Date.parse("2024-06-15"); try std.testing.expectEqual(@as(i16, 2024), d.year()); try std.testing.expectEqual(@as(u8, 6), d.month()); try std.testing.expectEqual(@as(u8, 15), d.day()); } test "date format" { const d = Date.fromYmd(2024, 1, 5); var buf: [10]u8 = undefined; const s = d.format(&buf); try std.testing.expectEqualStrings("2024-01-05", s); } test "subtractYears" { const d = Date.fromYmd(2026, 2, 24); const d1 = d.subtractYears(1); try std.testing.expectEqual(@as(i16, 2025), d1.year()); try std.testing.expectEqual(@as(u8, 2), d1.month()); try std.testing.expectEqual(@as(u8, 24), d1.day()); const d3 = d.subtractYears(3); try std.testing.expectEqual(@as(i16, 2023), d3.year()); // Leap year edge case: Feb 29 2024 - 1 year = Feb 28 2023 const leap = Date.fromYmd(2024, 2, 29); const non_leap = leap.subtractYears(1); try std.testing.expectEqual(@as(i16, 2023), non_leap.year()); try std.testing.expectEqual(@as(u8, 2), non_leap.month()); try std.testing.expectEqual(@as(u8, 28), non_leap.day()); } test "lastDayOfPriorMonth" { // Feb 24 -> Jan 31 const d1 = Date.fromYmd(2026, 2, 24).lastDayOfPriorMonth(); try std.testing.expectEqual(@as(i16, 2026), d1.year()); try std.testing.expectEqual(@as(u8, 1), d1.month()); try std.testing.expectEqual(@as(u8, 31), d1.day()); // Jan 15 -> Dec 31 of prior year const d2 = Date.fromYmd(2026, 1, 15).lastDayOfPriorMonth(); try std.testing.expectEqual(@as(i16, 2025), d2.year()); try std.testing.expectEqual(@as(u8, 12), d2.month()); try std.testing.expectEqual(@as(u8, 31), d2.day()); // Mar 1 leap year -> Feb 29 const d3 = Date.fromYmd(2024, 3, 1).lastDayOfPriorMonth(); try std.testing.expectEqual(@as(u8, 2), d3.month()); try std.testing.expectEqual(@as(u8, 29), d3.day()); // Mar 1 non-leap -> Feb 28 const d4 = Date.fromYmd(2025, 3, 1).lastDayOfPriorMonth(); try std.testing.expectEqual(@as(u8, 2), d4.month()); try std.testing.expectEqual(@as(u8, 28), d4.day()); } test "dayOfWeek" { // 1970-01-01 was a Thursday (3 in 0=Mon scheme) try std.testing.expectEqual(@as(u8, 3), Date.fromYmd(1970, 1, 1).dayOfWeek()); // 2024-01-01 (Monday) try std.testing.expectEqual(@as(u8, 0), Date.fromYmd(2024, 1, 1).dayOfWeek()); // 2024-01-19 (Friday) try std.testing.expectEqual(@as(u8, 4), Date.fromYmd(2024, 1, 19).dayOfWeek()); // 2024-01-20 (Saturday) try std.testing.expectEqual(@as(u8, 5), Date.fromYmd(2024, 1, 20).dayOfWeek()); // 2024-01-21 (Sunday) try std.testing.expectEqual(@as(u8, 6), Date.fromYmd(2024, 1, 21).dayOfWeek()); } test "eql and lessThan" { const a = Date.fromYmd(2024, 6, 15); const b = Date.fromYmd(2024, 6, 15); const c = Date.fromYmd(2024, 6, 16); try std.testing.expect(a.eql(b)); try std.testing.expect(!a.eql(c)); try std.testing.expect(a.lessThan(c)); try std.testing.expect(!c.lessThan(a)); try std.testing.expect(!a.lessThan(b)); } test "addDays" { const d = Date.fromYmd(2024, 1, 30); const next = d.addDays(2); try std.testing.expectEqual(@as(u8, 1), next.day()); try std.testing.expectEqual(@as(u8, 2), next.month()); // Feb 1 // Negative days const prev = d.addDays(-30); try std.testing.expectEqual(@as(u8, 31), prev.day()); try std.testing.expectEqual(@as(u8, 12), prev.month()); // Dec 31 try std.testing.expectEqual(@as(i16, 2023), prev.year()); } test "subtractMonths" { // Normal case const d1 = Date.fromYmd(2024, 6, 15).subtractMonths(3); try std.testing.expectEqual(@as(u8, 3), d1.month()); try std.testing.expectEqual(@as(u8, 15), d1.day()); // Cross year boundary: Jan - 1 = Dec prior year const d2 = Date.fromYmd(2024, 1, 15).subtractMonths(1); try std.testing.expectEqual(@as(u8, 12), d2.month()); try std.testing.expectEqual(@as(i16, 2023), d2.year()); // Day clamping: Mar 31 - 1M = Feb 29 (leap year 2024) const d3 = Date.fromYmd(2024, 3, 31).subtractMonths(1); try std.testing.expectEqual(@as(u8, 2), d3.month()); try std.testing.expectEqual(@as(u8, 29), d3.day()); // Day clamping: Mar 31 - 1M = Feb 28 (non-leap year 2025) const d4 = Date.fromYmd(2025, 3, 31).subtractMonths(1); try std.testing.expectEqual(@as(u8, 2), d4.month()); try std.testing.expectEqual(@as(u8, 28), d4.day()); } test "yearsBetween" { const a = Date.fromYmd(2024, 1, 1); const b = Date.fromYmd(2025, 1, 1); try std.testing.expectApproxEqAbs(@as(f64, 1.0), Date.yearsBetween(a, b), 0.01); // Half year const c = Date.fromYmd(2024, 7, 1); try std.testing.expectApproxEqAbs(@as(f64, 0.5), Date.yearsBetween(a, c), 0.02); // Same date try std.testing.expectApproxEqAbs(@as(f64, 0.0), Date.yearsBetween(a, a), 0.001); } test "parse error cases" { try std.testing.expectError(error.InvalidDateFormat, Date.parse("not-a-date")); try std.testing.expectError(error.InvalidDateFormat, Date.parse("20240115")); // no dashes try std.testing.expectError(error.InvalidDateFormat, Date.parse("2024/01/15")); // wrong separator }