1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/packages/utility-features/utilities/src/backoff-caller.ts
Sebastian Malton 13673eaac4 Move all utility functions to separate package
Signed-off-by: Sebastian Malton <sebastian@malton.name>
2023-03-02 13:30:24 -05:00

71 lines
1.6 KiB
TypeScript

/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import type { AsyncResult } from "./async-result";
import { delay } from "./delay";
import { noop } from "@k8slens/utilities/src/noop";
/**
* @param error The error that resulted in the failure
* @param attempt The 1-index attempt count
*/
export type OnIntermediateError<E> = (error: E, attempt: number) => void;
export interface BackoffCallerOptions<E> {
/**
* Called when an attempt fails
*/
onIntermediateError?: OnIntermediateError<E>;
/**
* @default 5
*/
maxAttempts?: number;
/**
* In miliseconds
* @default 1000
*/
initialTimeout?: number;
/**
* @default 2
*/
scaleFactor?: number;
}
/**
* Calls `fn` once and then again (with exponential delay between each attempt) up to `options.maxAttempts` times.
* @param fn The function to repeatedly attempt
* @returns The first success or the last failure
*/
export const backoffCaller = async <T, E, R extends AsyncResult<T, E>>(fn: () => Promise<R>, options?: BackoffCallerOptions<E>): Promise<R> => {
const {
initialTimeout = 1000,
maxAttempts = 5,
onIntermediateError = noop as OnIntermediateError<E>,
scaleFactor = 2,
} = options ?? {};
let timeout = initialTimeout;
let attempt = 0;
let result: R;
do {
result = await fn();
if (result.callWasSuccessful) {
return result;
}
onIntermediateError(result.error, attempt + 1);
await delay(timeout);
timeout *= scaleFactor;
} while (attempt += 1, attempt < maxAttempts);
return result;
};