diff --git a/common/changes/@subsquid/http-client/portal-api_2024-12-16-21-16.json b/common/changes/@subsquid/http-client/portal-api_2024-12-16-21-16.json new file mode 100644 index 000000000..ff35dda8c --- /dev/null +++ b/common/changes/@subsquid/http-client/portal-api_2024-12-16-21-16.json @@ -0,0 +1,10 @@ +{ + "changes": [ + { + "packageName": "@subsquid/http-client", + "comment": "add `retry-after` header support", + "type": "minor" + } + ], + "packageName": "@subsquid/http-client" +} \ No newline at end of file diff --git a/evm/evm-processor/src/processor.ts b/evm/evm-processor/src/processor.ts index c7c15f5f3..61bfa63ca 100644 --- a/evm/evm-processor/src/processor.ts +++ b/evm/evm-processor/src/processor.ts @@ -118,6 +118,8 @@ export interface PortalSettings { * Request timeout in ms */ requestTimeout?: number + + retryAttempts?: number bufferThreshold?: number @@ -542,7 +544,8 @@ export class EvmBatchProcessor { new PortalClient({ http, url: archive.url, - queryTimeout: archive.requestTimeout, + requestTimeout: archive.requestTimeout, + retryAttempts: archive.retryAttempts, bufferThreshold: archive.bufferThreshold, newBlockTimeout: archive.newBlockTimeout, log, diff --git a/substrate/substrate-processor/src/processor.ts b/substrate/substrate-processor/src/processor.ts index c8a40fd46..59b2152f5 100644 --- a/substrate/substrate-processor/src/processor.ts +++ b/substrate/substrate-processor/src/processor.ts @@ -105,6 +105,8 @@ export interface PortalSettings { * Request timeout in ms */ requestTimeout?: number + + retryAttempts?: number bufferThreshold?: number @@ -535,7 +537,8 @@ export class SubstrateBatchProcessor { client: new PortalClient({ http, url: options.url, - queryTimeout: options.requestTimeout, + requestTimeout: options.requestTimeout, + retryAttempts: options.retryAttempts, bufferThreshold: options.bufferThreshold, newBlockTimeout: options.newBlockTimeout, log, diff --git a/util/http-client/src/client.ts b/util/http-client/src/client.ts index 274d263bf..2fb0f9d47 100644 --- a/util/http-client/src/client.ts +++ b/util/http-client/src/client.ts @@ -104,9 +104,13 @@ export class HttpClient { let res: HttpResponse | Error = await this.performRequestWithTimeout(req).catch(ensureError) if (res instanceof Error || !res.ok) { if (retryAttempts > retries && this.isRetryableError(res, req)) { - let pause = retrySchedule.length + let pause = asRetryAfterPause(res) + if (pause == null) { + pause = retrySchedule.length ? retrySchedule[Math.min(retries, retrySchedule.length - 1)] : 1000 + } + // FIXME: should we count retries if there is retry-after header in response? retries += 1 this.beforeRetryPause(req, res, pause) await wait(pause, req.signal) @@ -334,7 +338,7 @@ export class HttpClient { case 524: return true default: - return false + return error.headers.has('retry-after') } } return false @@ -478,3 +482,20 @@ export function isHttpConnectionError(err: unknown): boolean { && err.type == 'system' && (err.message.startsWith('request to') || err.code == 'ERR_STREAM_PREMATURE_CLOSE') } + + +export function asRetryAfterPause(res: HttpResponse | Error): number | undefined { + if (res instanceof HttpError) { + res = res.response + } + if (res instanceof HttpResponse) { + let retryAfter = res.headers.get('retry-after') + if (retryAfter == null) return undefined + if (/^\d+$/.test(retryAfter)) return parseInt(retryAfter, 10) * 1000 + if (HTTP_DATE_REGEX.test(retryAfter)) return Math.max(0, new Date(retryAfter).getTime() - Date.now()) + } + + return undefined +} + +const HTTP_DATE_REGEX = /^(?:Mon|Tue|Wed|Thu|Fri|Sat|Sun), (\d{2}) (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec) (\d{4}) (\d{2}):(\d{2}):(\d{2}) GMT$/; \ No newline at end of file diff --git a/util/portal-client/src/client.ts b/util/portal-client/src/client.ts index 1abb7a025..8a33d4ec5 100644 --- a/util/portal-client/src/client.ts +++ b/util/portal-client/src/client.ts @@ -26,7 +26,8 @@ export interface PortalClientOptions { url: string http: HttpClient log?: Logger - queryTimeout?: number + requestTimeout?: number + retryAttempts?: number bufferThreshold?: number newBlockTimeout?: number } @@ -34,18 +35,20 @@ export interface PortalClientOptions { export class PortalClient { private url: URL private http: HttpClient - private queryTimeout: number + private requestTimeout: number private bufferThreshold: number private newBlockTimeout: number + private retryAttempts: number private log?: Logger constructor(options: PortalClientOptions) { this.url = new URL(options.url) this.log = options.log this.http = options.http - this.queryTimeout = options.queryTimeout ?? 180_000 + this.requestTimeout = options.requestTimeout ?? 180_000 this.bufferThreshold = options.bufferThreshold ?? 10 * 1024 * 1024 this.newBlockTimeout = options.newBlockTimeout ?? 120_000 + this.retryAttempts = options.retryAttempts ?? Infinity } private getDatasetUrl(path: string): string { @@ -60,8 +63,8 @@ export class PortalClient { async getMetadata(): Promise { let res: {real_time: boolean} = await this.http.get(this.getDatasetUrl('metadata'), { - retryAttempts: 3, - httpTimeout: 10_000, + retryAttempts: this.retryAttempts, + httpTimeout: this.requestTimeout, }) return { isRealTime: !!res.real_time, @@ -70,8 +73,8 @@ export class PortalClient { async getFinalizedHeight(): Promise { let res: string = await this.http.get(this.getDatasetUrl('finalized-stream/height'), { - retryAttempts: 3, - httpTimeout: 10_000, + retryAttempts: this.retryAttempts, + httpTimeout: this.requestTimeout, }) let height = parseInt(res) assert(Number.isSafeInteger(height)) @@ -83,8 +86,8 @@ export class PortalClient { return this.http .request('POST', this.getDatasetUrl(`finalized-stream`), { json: query, - retryAttempts: 3, - httpTimeout: this.queryTimeout, + retryAttempts: this.retryAttempts, + httpTimeout: this.requestTimeout, }) .catch( withErrorContext({ @@ -154,8 +157,8 @@ export class PortalClient { let res = await this.http .request('POST', this.getDatasetUrl(`finalized-stream`), { json: archiveQuery, - retryAttempts: 3, - httpTimeout: this.queryTimeout, + retryAttempts: this.retryAttempts, + httpTimeout: this.requestTimeout, stream: true, }) .catch(