| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192 |
- import { describe, expect, test } from "bun:test"
- import { getRetryAfterDay, getRetryAfterHour } from "../src/routes/zen/util/rateLimiter"
- describe("getRetryAfterDay", () => {
- test("returns full day at midnight UTC", () => {
- const midnight = Date.UTC(2026, 0, 15, 0, 0, 0, 0)
- expect(getRetryAfterDay(midnight)).toBe(86_400)
- })
- test("returns remaining seconds until next UTC day", () => {
- const noon = Date.UTC(2026, 0, 15, 12, 0, 0, 0)
- expect(getRetryAfterDay(noon)).toBe(43_200)
- })
- test("rounds up to nearest second", () => {
- const almost = Date.UTC(2026, 0, 15, 23, 59, 59, 500)
- expect(getRetryAfterDay(almost)).toBe(1)
- })
- })
- describe("getRetryAfterHour", () => {
- // 14:30:00 UTC — 30 minutes into the current hour
- const now = Date.UTC(2026, 0, 15, 14, 30, 0, 0)
- const intervals = ["2026011514", "2026011513", "2026011512"]
- test("waits 3 hours when all usage is in current hour", () => {
- const rows = [{ interval: "2026011514", count: 10 }]
- // only current hour has usage — it won't leave the window for 3 hours from hour start
- // 3 * 3600 - 1800 = 9000s
- expect(getRetryAfterHour(rows, intervals, 10, now)).toBe(9000)
- })
- test("waits 1 hour when dropping oldest interval is sufficient", () => {
- const rows = [
- { interval: "2026011514", count: 2 },
- { interval: "2026011512", count: 10 },
- ]
- // total=12, drop oldest (-2h, count=10) -> 2 < 10
- // hours = 3 - 2 = 1 -> 1 * 3600 - 1800 = 1800s
- expect(getRetryAfterHour(rows, intervals, 10, now)).toBe(1800)
- })
- test("waits 2 hours when usage spans oldest two intervals", () => {
- const rows = [
- { interval: "2026011513", count: 8 },
- { interval: "2026011512", count: 5 },
- ]
- // total=13, drop -2h (5) -> 8, 8 >= 8, drop -1h (8) -> 0 < 8
- // hours = 3 - 1 = 2 -> 2 * 3600 - 1800 = 5400s
- expect(getRetryAfterHour(rows, intervals, 8, now)).toBe(5400)
- })
- test("waits 1 hour when oldest interval alone pushes over limit", () => {
- const rows = [
- { interval: "2026011514", count: 1 },
- { interval: "2026011513", count: 1 },
- { interval: "2026011512", count: 10 },
- ]
- // total=12, drop -2h (10) -> 2 < 10
- // hours = 3 - 2 = 1 -> 1800s
- expect(getRetryAfterHour(rows, intervals, 10, now)).toBe(1800)
- })
- test("waits 2 hours when middle interval keeps total over limit", () => {
- const rows = [
- { interval: "2026011514", count: 4 },
- { interval: "2026011513", count: 4 },
- { interval: "2026011512", count: 4 },
- ]
- // total=12, drop -2h (4) -> 8, 8 >= 5, drop -1h (4) -> 4 < 5
- // hours = 3 - 1 = 2 -> 5400s
- expect(getRetryAfterHour(rows, intervals, 5, now)).toBe(5400)
- })
- test("rounds up to nearest second", () => {
- const offset = Date.UTC(2026, 0, 15, 14, 30, 0, 500)
- const rows = [
- { interval: "2026011514", count: 2 },
- { interval: "2026011512", count: 10 },
- ]
- // hours=1 -> 3_600_000 - 1_800_500 = 1_799_500ms -> ceil(1799.5) = 1800
- expect(getRetryAfterHour(rows, intervals, 10, offset)).toBe(1800)
- })
- test("fallback returns time until next hour when rows are empty", () => {
- // edge case: rows empty but function called (shouldn't happen in practice)
- // loop drops all zeros, running stays 0 which is < any positive limit on first iteration
- const rows: { interval: string; count: number }[] = []
- // drop -2h (0) -> 0 < 1 -> hours = 3 - 2 = 1 -> 1800s
- expect(getRetryAfterHour(rows, intervals, 1, now)).toBe(1800)
- })
- })
|