retry.ts 2.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. import type { NamedError } from "@opencode-ai/util/error"
  2. import { MessageV2 } from "./message-v2"
  3. export namespace SessionRetry {
  4. export const RETRY_INITIAL_DELAY = 2000
  5. export const RETRY_BACKOFF_FACTOR = 2
  6. export const RETRY_MAX_DELAY_NO_HEADERS = 30_000 // 30 seconds
  7. export async function sleep(ms: number, signal: AbortSignal): Promise<void> {
  8. return new Promise((resolve, reject) => {
  9. const timeout = setTimeout(resolve, ms)
  10. signal.addEventListener(
  11. "abort",
  12. () => {
  13. clearTimeout(timeout)
  14. reject(new DOMException("Aborted", "AbortError"))
  15. },
  16. { once: true },
  17. )
  18. })
  19. }
  20. export function delay(attempt: number, error?: MessageV2.APIError) {
  21. if (error) {
  22. const headers = error.data.responseHeaders
  23. if (headers) {
  24. const retryAfterMs = headers["retry-after-ms"]
  25. if (retryAfterMs) {
  26. const parsedMs = Number.parseFloat(retryAfterMs)
  27. if (!Number.isNaN(parsedMs)) {
  28. return parsedMs
  29. }
  30. }
  31. const retryAfter = headers["retry-after"]
  32. if (retryAfter) {
  33. const parsedSeconds = Number.parseFloat(retryAfter)
  34. if (!Number.isNaN(parsedSeconds)) {
  35. // convert seconds to milliseconds
  36. return Math.ceil(parsedSeconds * 1000)
  37. }
  38. // Try parsing as HTTP date format
  39. const parsed = Date.parse(retryAfter) - Date.now()
  40. if (!Number.isNaN(parsed) && parsed > 0) {
  41. return Math.ceil(parsed)
  42. }
  43. }
  44. return RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1)
  45. }
  46. }
  47. return Math.min(RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1), RETRY_MAX_DELAY_NO_HEADERS)
  48. }
  49. export function retryable(error: ReturnType<NamedError["toObject"]>) {
  50. if (MessageV2.APIError.isInstance(error)) {
  51. if (!error.data.isRetryable) return undefined
  52. return error.data.message.includes("Overloaded") ? "Provider is overloaded" : error.data.message
  53. }
  54. if (typeof error.data?.message === "string") {
  55. try {
  56. const json = JSON.parse(error.data.message)
  57. if (json.type === "error" && json.error?.type === "too_many_requests") {
  58. return "Too Many Requests"
  59. }
  60. } catch {}
  61. }
  62. return undefined
  63. }
  64. }