signal-desktop/ts/textsecure/TaskWithTimeout.ts
2021-09-17 14:27:53 -04:00

80 lines
2 KiB
TypeScript

// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import * as durations from '../util/durations';
import * as log from '../logging/log';
export default function createTaskWithTimeout<T, Args extends Array<unknown>>(
task: (...args: Args) => Promise<T>,
id: string,
options: { timeout?: number } = {}
): (...args: Args) => Promise<T> {
const timeout = options.timeout || 2 * durations.MINUTE;
const errorForStack = new Error('for stack');
return async (...args: Args) =>
new Promise((resolve, reject) => {
let complete = false;
let timer: NodeJS.Timeout | null = setTimeout(() => {
if (!complete) {
const message = `${
id || ''
} task did not complete in time. Calling stack: ${
errorForStack.stack
}`;
log.error(message);
reject(new Error(message));
return undefined;
}
return null;
}, timeout);
const clearTimer = () => {
try {
const localTimer = timer;
if (localTimer) {
timer = null;
clearTimeout(localTimer);
}
} catch (error) {
log.error(
id || '',
'task ran into problem canceling timer. Calling stack:',
errorForStack.stack
);
}
};
const success = (result: T) => {
clearTimer();
complete = true;
resolve(result);
};
const failure = (error: Error) => {
clearTimer();
complete = true;
reject(error);
};
let promise;
try {
promise = task(...args);
} catch (error) {
clearTimer();
throw error;
}
if (!promise || !promise.then) {
clearTimer();
complete = true;
resolve(promise);
return undefined;
}
// eslint-disable-next-line more/no-then
return promise.then(success, failure);
});
}