2025-04-02 06:50:39 -04:00

344 lines
9.4 KiB
JavaScript

'use strict';
const assert = require('node:assert');
const { kRetryHandlerDefaultRetry } = require('../core/symbols');
const { RequestRetryError } = require('../core/errors');
const WrapHandler = require('./wrap-handler');
const {
isDisturbed,
parseRangeHeader,
wrapRequestBody,
} = require('../core/util');
function calculateRetryAfterHeader(retryAfter) {
const retryTime = new Date(retryAfter).getTime();
return isNaN(retryTime) ? 0 : retryTime - Date.now();
}
class RetryHandler {
constructor(opts, { dispatch, handler }) {
const { retryOptions, ...dispatchOpts } = opts;
const {
// Retry scoped
retry: retryFn,
maxRetries,
maxTimeout,
minTimeout,
timeoutFactor,
// Response scoped
methods,
errorCodes,
retryAfter,
statusCodes,
} = retryOptions ?? {};
this.dispatch = dispatch;
this.handler = WrapHandler.wrap(handler);
this.opts = { ...dispatchOpts, body: wrapRequestBody(opts.body) };
this.retryOpts = {
retry: retryFn ?? RetryHandler[kRetryHandlerDefaultRetry],
retryAfter: retryAfter ?? true,
maxTimeout: maxTimeout ?? 30 * 1000, // 30s,
minTimeout: minTimeout ?? 500, // .5s
timeoutFactor: timeoutFactor ?? 2,
maxRetries: maxRetries ?? 5,
// What errors we should retry
methods: methods ?? ['GET', 'HEAD', 'OPTIONS', 'PUT', 'DELETE', 'TRACE'],
// Indicates which errors to retry
statusCodes: statusCodes ?? [500, 502, 503, 504, 429],
// List of errors to retry
errorCodes: errorCodes ?? [
'ECONNRESET',
'ECONNREFUSED',
'ENOTFOUND',
'ENETDOWN',
'ENETUNREACH',
'EHOSTDOWN',
'EHOSTUNREACH',
'EPIPE',
'UND_ERR_SOCKET',
],
};
this.retryCount = 0;
this.retryCountCheckpoint = 0;
this.headersSent = false;
this.start = 0;
this.end = null;
this.etag = null;
}
onRequestStart(controller, context) {
if (!this.headersSent) {
this.handler.onRequestStart?.(controller, context);
}
}
onRequestUpgrade(controller, statusCode, headers, socket) {
this.handler.onRequestUpgrade?.(controller, statusCode, headers, socket);
}
static [kRetryHandlerDefaultRetry](err, { state, opts }, cb) {
const { statusCode, code, headers } = err;
const { method, retryOptions } = opts;
const {
maxRetries,
minTimeout,
maxTimeout,
timeoutFactor,
statusCodes,
errorCodes,
methods,
} = retryOptions;
const { counter } = state;
// Any code that is not a Undici's originated and allowed to retry
if (code && code !== 'UND_ERR_REQ_RETRY' && !errorCodes.includes(code)) {
cb(err);
return;
}
// If a set of method are provided and the current method is not in the list
if (Array.isArray(methods) && !methods.includes(method)) {
cb(err);
return;
}
// If a set of status code are provided and the current status code is not in the list
if (
statusCode != null &&
Array.isArray(statusCodes) &&
!statusCodes.includes(statusCode)
) {
cb(err);
return;
}
// If we reached the max number of retries
if (counter > maxRetries) {
cb(err);
return;
}
let retryAfterHeader = headers?.['retry-after'];
if (retryAfterHeader) {
retryAfterHeader = Number(retryAfterHeader);
retryAfterHeader =
Number.isNaN(retryAfterHeader) ?
calculateRetryAfterHeader(headers['retry-after'])
: retryAfterHeader * 1e3; // Retry-After is in seconds
}
const retryTimeout =
retryAfterHeader > 0 ?
Math.min(retryAfterHeader, maxTimeout)
: Math.min(minTimeout * timeoutFactor ** (counter - 1), maxTimeout);
setTimeout(() => cb(null), retryTimeout);
}
onResponseStart(controller, statusCode, headers, statusMessage) {
this.retryCount += 1;
if (statusCode >= 300) {
if (this.retryOpts.statusCodes.includes(statusCode) === false) {
this.headersSent = true;
this.handler.onResponseStart?.(
controller,
statusCode,
headers,
statusMessage
);
return;
} else {
throw new RequestRetryError('Request failed', statusCode, {
headers,
data: {
count: this.retryCount,
},
});
}
}
// Checkpoint for resume from where we left it
if (this.headersSent) {
// Only Partial Content 206 supposed to provide Content-Range,
// any other status code that partially consumed the payload
// should not be retried because it would result in downstream
// wrongly concatenate multiple responses.
if (statusCode !== 206 && (this.start > 0 || statusCode !== 200)) {
throw new RequestRetryError(
'server does not support the range header and the payload was partially consumed',
statusCode,
{
headers,
data: { count: this.retryCount },
}
);
}
const contentRange = parseRangeHeader(headers['content-range']);
// If no content range
if (!contentRange) {
throw new RequestRetryError('Content-Range mismatch', statusCode, {
headers,
data: { count: this.retryCount },
});
}
// Let's start with a weak etag check
if (this.etag != null && this.etag !== headers.etag) {
throw new RequestRetryError('ETag mismatch', statusCode, {
headers,
data: { count: this.retryCount },
});
}
const { start, size, end = size ? size - 1 : null } = contentRange;
assert(this.start === start, 'content-range mismatch');
assert(this.end == null || this.end === end, 'content-range mismatch');
return;
}
if (this.end == null) {
if (statusCode === 206) {
// First time we receive 206
const range = parseRangeHeader(headers['content-range']);
if (range == null) {
this.headersSent = true;
this.handler.onResponseStart?.(
controller,
statusCode,
headers,
statusMessage
);
return;
}
const { start, size, end = size ? size - 1 : null } = range;
assert(
start != null && Number.isFinite(start),
'content-range mismatch'
);
assert(end != null && Number.isFinite(end), 'invalid content-length');
this.start = start;
this.end = end;
}
// We make our best to checkpoint the body for further range headers
if (this.end == null) {
const contentLength = headers['content-length'];
this.end = contentLength != null ? Number(contentLength) - 1 : null;
}
assert(Number.isFinite(this.start));
assert(
this.end == null || Number.isFinite(this.end),
'invalid content-length'
);
this.resume = true;
this.etag = headers.etag != null ? headers.etag : null;
// Weak etags are not useful for comparison nor cache
// for instance not safe to assume if the response is byte-per-byte
// equal
if (this.etag != null && this.etag[0] === 'W' && this.etag[1] === '/') {
this.etag = null;
}
this.headersSent = true;
this.handler.onResponseStart?.(
controller,
statusCode,
headers,
statusMessage
);
} else {
throw new RequestRetryError('Request failed', statusCode, {
headers,
data: { count: this.retryCount },
});
}
}
onResponseData(controller, chunk) {
this.start += chunk.length;
this.handler.onResponseData?.(controller, chunk);
}
onResponseEnd(controller, trailers) {
this.retryCount = 0;
return this.handler.onResponseEnd?.(controller, trailers);
}
onResponseError(controller, err) {
if (controller?.aborted || isDisturbed(this.opts.body)) {
this.handler.onResponseError?.(controller, err);
return;
}
// We reconcile in case of a mix between network errors
// and server error response
if (this.retryCount - this.retryCountCheckpoint > 0) {
// We count the difference between the last checkpoint and the current retry count
this.retryCount =
this.retryCountCheckpoint +
(this.retryCount - this.retryCountCheckpoint);
} else {
this.retryCount += 1;
}
this.retryOpts.retry(
err,
{
state: { counter: this.retryCount },
opts: { retryOptions: this.retryOpts, ...this.opts },
},
onRetry.bind(this)
);
/**
* @this {RetryHandler}
* @param {Error} [err]
* @returns
*/
function onRetry(err) {
if (err != null || controller?.aborted || isDisturbed(this.opts.body)) {
return this.handler.onResponseError?.(controller, err);
}
if (this.start !== 0) {
const headers = { range: `bytes=${this.start}-${this.end ?? ''}` };
// Weak etag check - weak etags will make comparison algorithms never match
if (this.etag != null) {
headers['if-match'] = this.etag;
}
this.opts = {
...this.opts,
headers: {
...this.opts.headers,
...headers,
},
};
}
try {
this.retryCountCheckpoint = this.retryCount;
this.dispatch(this.opts, this);
} catch (err) {
this.handler.onResponseError?.(controller, err);
}
}
}
}
module.exports = RetryHandler;