|
|
@@ -0,0 +1,92 @@
|
|
|
+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)
|
|
|
+ })
|
|
|
+})
|