2021-07-23 22:02:36 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2024-05-29 23:46:43 +00:00
|
|
|
import { strictAssert } from './assert';
|
2021-08-26 14:10:58 +00:00
|
|
|
import * as durations from './durations';
|
2021-07-23 22:02:36 +00:00
|
|
|
|
|
|
|
const BACKOFF_FACTOR = 1.9;
|
2021-08-26 14:10:58 +00:00
|
|
|
const MAX_BACKOFF = 15 * durations.MINUTE;
|
2024-05-29 23:46:43 +00:00
|
|
|
const FIRST_BACKOFFS = [0, 190];
|
2021-07-23 22:02:36 +00:00
|
|
|
/**
|
|
|
|
* For a given attempt, how long should we sleep (in milliseconds)?
|
|
|
|
*
|
|
|
|
* The attempt should be a positive integer, and it is 1-indexed. The first attempt is 1,
|
|
|
|
* the second is 2, and so on.
|
|
|
|
*
|
|
|
|
* This is modified from [iOS's codebase][0].
|
|
|
|
*
|
|
|
|
* [0]: https://github.com/signalapp/Signal-iOS/blob/6069741602421744edfb59923d2fb3a66b1b23c1/SignalServiceKit/src/Util/OWSOperation.swift
|
|
|
|
*/
|
2024-04-16 00:11:48 +00:00
|
|
|
|
|
|
|
export type ExponentialBackoffOptionsType = {
|
|
|
|
maxBackoffTime: number;
|
|
|
|
multiplier: number;
|
2024-05-29 23:46:43 +00:00
|
|
|
firstBackoffs: Array<number>;
|
2024-04-16 00:11:48 +00:00
|
|
|
};
|
|
|
|
export function exponentialBackoffSleepTime(
|
|
|
|
attempt: number,
|
|
|
|
options: ExponentialBackoffOptionsType = {
|
|
|
|
maxBackoffTime: MAX_BACKOFF,
|
|
|
|
multiplier: BACKOFF_FACTOR,
|
2024-05-29 23:46:43 +00:00
|
|
|
firstBackoffs: FIRST_BACKOFFS,
|
2024-04-16 00:11:48 +00:00
|
|
|
}
|
|
|
|
): number {
|
2024-05-29 23:46:43 +00:00
|
|
|
const numHardcodedBackoffs = options.firstBackoffs.length;
|
|
|
|
strictAssert(
|
|
|
|
numHardcodedBackoffs > 0,
|
|
|
|
'must include explicit first backoffs'
|
|
|
|
);
|
|
|
|
|
|
|
|
if (attempt - 1 < numHardcodedBackoffs) {
|
|
|
|
return options.firstBackoffs[attempt - 1];
|
2021-07-23 22:02:36 +00:00
|
|
|
}
|
2024-04-16 00:11:48 +00:00
|
|
|
|
2024-05-29 23:46:43 +00:00
|
|
|
const lastHardcodedBackoff = options.firstBackoffs.at(-1);
|
|
|
|
strictAssert(
|
|
|
|
lastHardcodedBackoff != null && lastHardcodedBackoff > 0,
|
|
|
|
'lastHardcodedBackoff must be a positive number'
|
|
|
|
);
|
2024-04-16 00:11:48 +00:00
|
|
|
return Math.min(
|
|
|
|
options.maxBackoffTime,
|
2024-05-29 23:46:43 +00:00
|
|
|
(lastHardcodedBackoff / options.multiplier) *
|
|
|
|
options.multiplier ** (attempt - numHardcodedBackoffs + 1)
|
2024-04-16 00:11:48 +00:00
|
|
|
);
|
2021-07-23 22:02:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If I want to retry for X milliseconds, how many attempts is that, roughly? For example,
|
|
|
|
* 24 hours (86,400,000 milliseconds) is 111 attempts.
|
|
|
|
*
|
|
|
|
* `desiredDurationMs` should be at least 1.
|
|
|
|
*/
|
|
|
|
export function exponentialBackoffMaxAttempts(
|
2024-04-16 00:11:48 +00:00
|
|
|
desiredDurationMs: number,
|
|
|
|
options?: ExponentialBackoffOptionsType
|
2021-07-23 22:02:36 +00:00
|
|
|
): number {
|
|
|
|
let attempts = 0;
|
|
|
|
let total = 0;
|
|
|
|
// There's probably some algebra we could do here instead of this loop, but this is
|
|
|
|
// fast even for giant numbers, and is typically called just once at startup.
|
|
|
|
do {
|
|
|
|
attempts += 1;
|
2024-04-16 00:11:48 +00:00
|
|
|
total += exponentialBackoffSleepTime(attempts, options);
|
2021-07-23 22:02:36 +00:00
|
|
|
} while (total < desiredDurationMs);
|
|
|
|
return attempts;
|
|
|
|
}
|