Improve handling of 413 HTTP responses

This commit is contained in:
Evan Hahn 2021-09-27 09:44:09 -05:00 committed by GitHub
parent 8b98035cbf
commit 9791fa43ef
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 188 additions and 78 deletions

View file

@ -0,0 +1,27 @@
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { isRecord } from '../../util/isRecord';
import { parseIntWithFallback } from '../../util/parseIntWithFallback';
/**
* Looks for an HTTP code. First tries the top level error, then looks at its `httpError`
* property.
*/
export function getHttpErrorCode(maybeError: unknown): number {
if (!isRecord(maybeError)) {
return -1;
}
const maybeTopLevelCode = parseIntWithFallback(maybeError.code, -1);
if (maybeTopLevelCode !== -1) {
return maybeTopLevelCode;
}
const { httpError } = maybeError;
if (!isRecord(httpError)) {
return -1;
}
return parseIntWithFallback(httpError.code, -1);
}

View file

@ -2,9 +2,8 @@
// SPDX-License-Identifier: AGPL-3.0-only
import type { LoggerType } from '../../types/Logging';
import { parseIntWithFallback } from '../../util/parseIntWithFallback';
import { HTTPError } from '../../textsecure/Errors';
import { sleepFor413RetryAfterTimeIfApplicable } from './sleepFor413RetryAfterTimeIfApplicable';
import { sleepFor413RetryAfterTime } from './sleepFor413RetryAfterTime';
import { getHttpErrorCode } from './getHttpErrorCode';
export async function handleCommonJobRequestError({
err,
@ -15,17 +14,14 @@ export async function handleCommonJobRequestError({
log: LoggerType;
timeRemaining: number;
}>): Promise<void> {
if (!(err instanceof HTTPError)) {
throw err;
switch (getHttpErrorCode(err)) {
case 413:
await sleepFor413RetryAfterTime({ err, log, timeRemaining });
return;
case 508:
log.info('server responded with 508. Giving up on this job');
return;
default:
throw err;
}
const code = parseIntWithFallback(err.code, -1);
if (code === 508) {
log.info('server responded with 508. Giving up on this job');
return;
}
await sleepFor413RetryAfterTimeIfApplicable({ err, log, timeRemaining });
throw err;
}

View file

@ -7,7 +7,7 @@ import { parseRetryAfter } from '../../util/parseRetryAfter';
import { isRecord } from '../../util/isRecord';
import { HTTPError } from '../../textsecure/Errors';
export async function sleepFor413RetryAfterTimeIfApplicable({
export async function sleepFor413RetryAfterTime({
err,
log,
timeRemaining,
@ -16,17 +16,12 @@ export async function sleepFor413RetryAfterTimeIfApplicable({
log: Pick<LoggerType, 'info'>;
timeRemaining: number;
}>): Promise<void> {
if (
timeRemaining <= 0 ||
!(err instanceof HTTPError) ||
err.code !== 413 ||
!isRecord(err.responseHeaders)
) {
if (timeRemaining <= 0) {
return;
}
const retryAfter = Math.min(
parseRetryAfter(err.responseHeaders['retry-after']),
parseRetryAfter(findRetryAfterTime(err)),
timeRemaining
);
@ -36,3 +31,19 @@ export async function sleepFor413RetryAfterTimeIfApplicable({
await sleep(retryAfter);
}
function findRetryAfterTime(err: unknown): unknown {
if (!isRecord(err)) {
return undefined;
}
if (isRecord(err.responseHeaders)) {
return err.responseHeaders['retry-after'];
}
if (err.httpError instanceof HTTPError) {
return err.httpError.responseHeaders?.['retry-after'];
}
return undefined;
}