// runtime/time.el — Time operations, sleep, and formatting. // // Implements the time surface from el-compiler/runtime/legacy/el_runtime.c // (lines 3334–3440, 3471–3656) in pure El, using seed primitives. // // Seed primitives consumed: // __time_now_ns() -> Int (nanoseconds since Unix epoch) // __sleep_ms(n: Int) // __int_to_str(n: Int) -> String // __str_to_int(s: String) -> Int // __float_to_str(f: Float) -> String // --------------------------------------------------------------------------- // Core — now / sleep // --------------------------------------------------------------------------- // time_now — milliseconds since Unix epoch (UTC). Matches legacy time_now(). fn time_now() -> Int { return __time_now_ns() / 1000000 } // time_now_utc — same as time_now; UTC alias kept for compatibility. fn time_now_utc() -> Int { return __time_now_ns() / 1000000 } // now_ns — nanoseconds since Unix epoch. Matches el_now_instant(). fn now_ns() -> Int { return __time_now_ns() } // unix_timestamp — whole seconds since Unix epoch. Matches unix_timestamp(). fn unix_timestamp() -> Int { return __time_now_ns() / 1000000000 } // sleep_secs — block for n seconds. Clamps negatives to 0. fn sleep_secs(n: Int) { if n < 0 { __sleep_ms(0) } else { __sleep_ms(n * 1000) } } // sleep_ms — block for n milliseconds. Clamps negatives to 0. fn sleep_ms(n: Int) { if n < 0 { __sleep_ms(0) } else { __sleep_ms(n) } } // --------------------------------------------------------------------------- // Gregorian decomposition helpers — pure integer arithmetic. // // Algorithm: civil date from days since Unix epoch (1970-01-01). // Based on Howard Hinnant's public-domain civil_from_days formula // (http://howardhinnant.github.io/date_algorithms.html), which the legacy // gmtime_r call performs under the hood. // // We expose the pieces as private helpers (leading underscore convention). // --------------------------------------------------------------------------- // _is_leap — 1 if year y is a Gregorian leap year, 0 otherwise. fn _is_leap(y: Int) -> Int { if y % 400 == 0 { return 1 } if y % 100 == 0 { return 0 } if y % 4 == 0 { return 1 } return 0 } // _days_in_month — number of days in month m of year y (m: 1..12). fn _days_in_month(y: Int, m: Int) -> Int { if m == 1 { return 31 } if m == 2 { if _is_leap(y) == 1 { return 29 } return 28 } if m == 3 { return 31 } if m == 4 { return 30 } if m == 5 { return 31 } if m == 6 { return 30 } if m == 7 { return 31 } if m == 8 { return 31 } if m == 9 { return 30 } if m == 10 { return 31 } if m == 11 { return 30 } return 31 } // _pad2 — zero-pad an integer to at least 2 digits. fn _pad2(n: Int) -> String { if n < 10 { return "0" + __int_to_str(n) } return __int_to_str(n) } // _pad4 — zero-pad an integer to at least 4 digits. fn _pad4(n: Int) -> String { if n < 10 { return "000" + __int_to_str(n) } if n < 100 { return "00" + __int_to_str(n) } if n < 1000 { return "0" + __int_to_str(n) } return __int_to_str(n) } // _pad3 — zero-pad an integer to at least 3 digits (milliseconds). fn _pad3(n: Int) -> String { if n < 10 { return "00" + __int_to_str(n) } if n < 100 { return "0" + __int_to_str(n) } return __int_to_str(n) } // _civil_year_month_day — decompose days-since-epoch (z, may be negative) // into year/month/day using the civil_from_days algorithm. // Returns a JSON object: {"year":Y,"month":M,"day":D} fn _civil_ymd(z: Int) -> String { // shift epoch to 0000-03-01 (makes leap-day math clean) let zz: Int = z + 719468 // era: 400-year block let era: Int = zz / 146097 if zz < 0 { era = (zz - 146096) / 146097 } let doe: Int = zz - era * 146097 // day-of-era [0, 146096] let yoe: Int = (doe - doe / 1460 + doe / 36524 - doe / 146096) / 365 // year-of-era [0, 399] let y: Int = yoe + era * 400 let doy: Int = doe - (365 * yoe + yoe / 4 - yoe / 100) // day-of-year [0, 365] let mp: Int = (5 * doy + 2) / 153 // month in [0, 11] from March let d: Int = doy - (153 * mp + 2) / 5 + 1 // day [1, 31] let m: Int = mp + 3 if mp >= 10 { m = mp - 9 } if mp >= 10 { y = y + 1 } return "{\"year\":" + __int_to_str(y) + ",\"month\":" + __int_to_str(m) + ",\"day\":" + __int_to_str(d) + "}" } // --------------------------------------------------------------------------- // time_to_parts — decompose a millisecond timestamp into UTC components. // // Returns a JSON string: // {"year":Y,"month":M,"day":D,"hour":H,"minute":M,"second":S,"ms":MS} // // Matches legacy time_to_parts() which returns an ElMap with the same keys. // --------------------------------------------------------------------------- fn time_to_parts(ts: Int) -> String { let ms_raw: Int = ts % 1000 let ms: Int = ms_raw if ms_raw < 0 { ms = ms_raw + 1000 } let s_raw: Int = ts / 1000 let s: Int = s_raw if ms_raw < 0 { s = s_raw - 1 } // seconds within the day and days since epoch let sec_of_day: Int = s % 86400 let day_z: Int = s / 86400 // handle negative: floor division if sec_of_day < 0 { sec_of_day = sec_of_day + 86400 day_z = day_z - 1 } let hour: Int = sec_of_day / 3600 let rem: Int = sec_of_day % 3600 let minute: Int = rem / 60 let second: Int = rem % 60 // date components via civil decomposition let ymd: String = _civil_ymd(day_z) let year: Int = __str_to_int(json_get(ymd, "year")) let month: Int = __str_to_int(json_get(ymd, "month")) let day: Int = __str_to_int(json_get(ymd, "day")) return "{\"year\":" + __int_to_str(year) + ",\"month\":" + __int_to_str(month) + ",\"day\":" + __int_to_str(day) + ",\"hour\":" + __int_to_str(hour) + ",\"minute\":" + __int_to_str(minute) + ",\"second\":" + __int_to_str(second) + ",\"ms\":" + __int_to_str(ms) + "}" } // --------------------------------------------------------------------------- // time_format — format a millisecond timestamp as a string. // // fmt "ISO" (or empty) → "YYYY-MM-DDTHH:MM:SS.mmmZ" (ISO 8601 UTC) // Other fmt values are passed as a strftime-style hint; the El runtime // implements the most common tokens. Unsupported tokens are passed through. // // Matches legacy time_format(). // --------------------------------------------------------------------------- fn time_format(ts: Int, fmt: String) -> String { let parts: String = time_to_parts(ts) let y: Int = __str_to_int(json_get(parts, "year")) let mo: Int = __str_to_int(json_get(parts, "month")) let d: Int = __str_to_int(json_get(parts, "day")) let h: Int = __str_to_int(json_get(parts, "hour")) let mi: Int = __str_to_int(json_get(parts, "minute")) let s: Int = __str_to_int(json_get(parts, "second")) let ms: Int = __str_to_int(json_get(parts, "ms")) // ISO 8601 UTC: YYYY-MM-DDTHH:MM:SS.mmmZ if str_eq(fmt, "ISO") { return _pad4(y) + "-" + _pad2(mo) + "-" + _pad2(d) + "T" + _pad2(h) + ":" + _pad2(mi) + ":" + _pad2(s) + "." + _pad3(ms) + "Z" } if str_eq(fmt, "") { return _pad4(y) + "-" + _pad2(mo) + "-" + _pad2(d) + "T" + _pad2(h) + ":" + _pad2(mi) + ":" + _pad2(s) + "." + _pad3(ms) + "Z" } // strftime-subset: replace common tokens let out: String = fmt let out = str_replace(out, "%Y", _pad4(y)) let out = str_replace(out, "%m", _pad2(mo)) let out = str_replace(out, "%d", _pad2(d)) let out = str_replace(out, "%H", _pad2(h)) let out = str_replace(out, "%M", _pad2(mi)) let out = str_replace(out, "%S", _pad2(s)) let out = str_replace(out, "%3N", _pad3(ms)) return out } // --------------------------------------------------------------------------- // time_from_parts — construct a ms timestamp from seconds + nanosecond offset. // // Matches legacy time_from_parts(secs, ns, tz) — tz is ignored (UTC assumed). // --------------------------------------------------------------------------- fn time_from_parts(secs: Int, ns: Int, tz: String) -> Int { return secs * 1000 + ns / 1000000 } // --------------------------------------------------------------------------- // time_add — add a duration to a millisecond timestamp. // // unit: "ms" | "sec" | "min" | "hour" | "day" // Matches legacy time_add(). // --------------------------------------------------------------------------- fn time_add(ts: Int, n: Int, unit: String) -> Int { if str_eq(unit, "ms") { return ts + n } if str_eq(unit, "sec") { return ts + n * 1000 } if str_eq(unit, "min") { return ts + n * 60000 } if str_eq(unit, "hour") { return ts + n * 3600000 } if str_eq(unit, "day") { return ts + n * 86400000 } // default: treat as ms return ts + n } // --------------------------------------------------------------------------- // time_diff — compute the difference between two millisecond timestamps. // // Returns ts2 - ts1 in the given unit. // unit: "ms" | "sec" | "min" | "hour" | "day" // Matches legacy time_diff(). // --------------------------------------------------------------------------- fn time_diff(ts1: Int, ts2: Int, unit: String) -> Int { let d: Int = ts2 - ts1 if str_eq(unit, "ms") { return d } if str_eq(unit, "sec") { return d / 1000 } if str_eq(unit, "min") { return d / 60000 } if str_eq(unit, "hour") { return d / 3600000 } if str_eq(unit, "day") { return d / 86400000 } return d } // --------------------------------------------------------------------------- // Instant / Duration — nanosecond-precision temporal types. // // These match the el_now_instant, duration_seconds, duration_millis, etc. // family from legacy lines 3471–3656. Both Instant and Duration are Int // (nanoseconds); the type distinction is at the call-site convention level. // --------------------------------------------------------------------------- // now — current Instant in nanoseconds. Alias for __time_now_ns(). fn now() -> Int { return __time_now_ns() } // unix_seconds — Instant from whole seconds since epoch. fn unix_seconds(n: Int) -> Int { return n * 1000000000 } // unix_millis — Instant from milliseconds since epoch. fn unix_millis(n: Int) -> Int { return n * 1000000 } // instant_to_unix_seconds — convert Instant nanoseconds to whole seconds. fn instant_to_unix_seconds(i: Int) -> Int { return i / 1000000000 } // instant_to_unix_millis — convert Instant nanoseconds to milliseconds. fn instant_to_unix_millis(i: Int) -> Int { return i / 1000000 } // instant_to_iso8601 — format an Instant (nanoseconds) as ISO 8601 UTC. fn instant_to_iso8601(i: Int) -> String { let ms: Int = i / 1000000 return time_format(ms, "ISO") } // duration_seconds — Duration from n whole seconds. fn duration_seconds(n: Int) -> Int { return n * 1000000000 } // duration_millis — Duration from n milliseconds. fn duration_millis(n: Int) -> Int { return n * 1000000 } // duration_nanos — Duration from n nanoseconds (identity). fn duration_nanos(n: Int) -> Int { return n } // duration_to_seconds — convert a Duration (nanoseconds) to whole seconds. fn duration_to_seconds(d: Int) -> Int { return d / 1000000000 } // duration_to_millis — convert a Duration (nanoseconds) to milliseconds. fn duration_to_millis(d: Int) -> Int { return d / 1000000 } // duration_to_nanos — return the Duration as nanoseconds (identity). fn duration_to_nanos(d: Int) -> Int { return d } // sleep_duration — sleep for a Duration (nanoseconds). Clamps negatives to 0. fn sleep_duration(dur: Int) { let ms: Int = dur / 1000000 if ms < 0 { __sleep_ms(0) } else { __sleep_ms(ms) } } // --------------------------------------------------------------------------- // TTL cache — time-bounded key/value backed by state. // // Matches legacy ttl_cache_set / ttl_cache_get / ttl_cache_age (lines 3663–3717). // max_age is a Duration (nanoseconds). // --------------------------------------------------------------------------- // ttl_cache_set — store a value and record the current Instant for TTL checks. fn ttl_cache_set(key: String, value: String) { state_set(key, value) let stamp_key: String = "__ttl_at:" + key let now_str: String = __int_to_str(__time_now_ns()) state_set(stamp_key, now_str) } // ttl_cache_get — return value if age < max_age (nanoseconds), else "". fn ttl_cache_get(key: String, max_age: Int) -> String { let stamp_key: String = "__ttl_at:" + key let sv: String = state_get(stamp_key) if str_eq(sv, "") { return "" } let set_at: Int = __str_to_int(sv) let now_ns: Int = __time_now_ns() let age: Int = now_ns - set_at if age < 0 { return "" } if age > max_age { return "" } return state_get(key) } // ttl_cache_age — nanoseconds since a key was last set (INT_MAX sentinel if missing). fn ttl_cache_age(key: String) -> Int { let stamp_key: String = "__ttl_at:" + key let sv: String = state_get(stamp_key) if str_eq(sv, "") { return 9223372036854775807 } let set_at: Int = __str_to_int(sv) return __time_now_ns() - set_at } // --------------------------------------------------------------------------- // uuid_new / uuid_v4 — generate a UUID v4 string. // // Delegates to the __uuid_v4() seed primitive. // Matches legacy uuid_new() / uuid_v4() (lines 4602–4621). // --------------------------------------------------------------------------- fn uuid_new() -> String { return __uuid_v4() } fn uuid_v4() -> String { return __uuid_v4() } // --------------------------------------------------------------------------- // DHARMA-compatible aliases — millisecond-precision timestamps. // // now_millis, unix_timestamp_ms, and time_now_ms all return the same value: // milliseconds since the Unix epoch. They exist because different parts of // the dharma codebase use different names for the same concept. // --------------------------------------------------------------------------- // now_millis — milliseconds since Unix epoch. Alias for time_now(). fn now_millis() -> Int { return __time_now_ns() / 1000000 } // unix_timestamp_ms — same as now_millis. fn unix_timestamp_ms() -> Int { return __time_now_ns() / 1000000 } // time_now_ms — same as now_millis. fn time_now_ms() -> Int { return __time_now_ns() / 1000000 }