retry.test.ts 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647
  1. import { describe, expect, test } from "bun:test"
  2. import { SessionRetry } from "../../src/session/retry"
  3. import { MessageV2 } from "../../src/session/message-v2"
  4. function apiError(headers?: Record<string, string>): MessageV2.APIError {
  5. return new MessageV2.APIError({
  6. message: "boom",
  7. isRetryable: true,
  8. responseHeaders: headers,
  9. }).toObject() as MessageV2.APIError
  10. }
  11. describe("session.retry.getRetryDelayInMs", () => {
  12. test("doubles delay on each attempt when headers missing", () => {
  13. const error = apiError()
  14. const delays = Array.from({ length: 7 }, (_, index) => SessionRetry.getRetryDelayInMs(error, index + 1))
  15. expect(delays).toStrictEqual([2000, 4000, 8000, 16000, 32000, 64000, 128000])
  16. })
  17. test("prefers retry-after-ms when shorter than exponential", () => {
  18. const error = apiError({ "retry-after-ms": "1500" })
  19. expect(SessionRetry.getRetryDelayInMs(error, 4)).toBe(1500)
  20. })
  21. test("uses retry-after seconds when reasonable", () => {
  22. const error = apiError({ "retry-after": "30" })
  23. expect(SessionRetry.getRetryDelayInMs(error, 3)).toBe(30000)
  24. })
  25. test("falls back to exponential when server delay is long", () => {
  26. const error = apiError({ "retry-after": "120" })
  27. expect(SessionRetry.getRetryDelayInMs(error, 2)).toBe(4000)
  28. })
  29. test("accepts http-date retry-after values", () => {
  30. const date = new Date(Date.now() + 20000).toUTCString()
  31. const error = apiError({ "retry-after": date })
  32. const delay = SessionRetry.getRetryDelayInMs(error, 1)
  33. expect(delay).toBeGreaterThanOrEqual(19000)
  34. expect(delay).toBeLessThanOrEqual(20000)
  35. })
  36. test("ignores invalid retry hints", () => {
  37. const error = apiError({ "retry-after": "not-a-number" })
  38. expect(SessionRetry.getRetryDelayInMs(error, 1)).toBe(2000)
  39. })
  40. })