344 lines
9.4 KiB
JavaScript
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;
|