2022-08-17 12:39:41 +00:00
|
|
|
import {TimeoutError} from '../common/Errors.js';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2023-05-26 06:02:17 +00:00
|
|
|
export interface DeferredPromise<T> {
|
2022-08-17 12:39:41 +00:00
|
|
|
finished: () => boolean;
|
|
|
|
resolved: () => boolean;
|
2022-09-15 06:22:20 +00:00
|
|
|
resolve: (value: T) => void;
|
2023-05-26 06:02:17 +00:00
|
|
|
reject: (error: Error) => void;
|
|
|
|
value: () => T | Error | undefined;
|
|
|
|
valueOrThrow: () => Promise<T>;
|
2022-08-17 12:39:41 +00:00
|
|
|
}
|
2022-08-24 10:05:32 +00:00
|
|
|
|
2022-09-05 08:30:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export interface DeferredPromiseOptions {
|
|
|
|
message: string;
|
|
|
|
timeout: number;
|
2022-08-30 14:24:51 +00:00
|
|
|
}
|
|
|
|
|
2022-08-17 12:39:41 +00:00
|
|
|
/**
|
2023-05-26 06:02:17 +00:00
|
|
|
* Creates and returns a deferred object along with the resolve/reject functions.
|
2022-08-17 12:39:41 +00:00
|
|
|
*
|
2023-05-26 06:02:17 +00:00
|
|
|
* If the deferred has not been resolved/rejected within the `timeout` period,
|
|
|
|
* the deferred gets resolves with a timeout error. `timeout` has to be greater than 0 or
|
2022-09-05 08:30:47 +00:00
|
|
|
* it is ignored.
|
2022-08-17 12:39:41 +00:00
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-05 08:30:47 +00:00
|
|
|
export function createDeferredPromise<T>(
|
|
|
|
opts?: DeferredPromiseOptions
|
|
|
|
): DeferredPromise<T> {
|
2022-08-17 12:39:41 +00:00
|
|
|
let isResolved = false;
|
|
|
|
let isRejected = false;
|
2023-05-26 06:02:17 +00:00
|
|
|
let _value: T | Error | undefined;
|
|
|
|
let resolver: (value: void) => void;
|
|
|
|
const taskPromise = new Promise<void>(resolve => {
|
2022-08-17 12:39:41 +00:00
|
|
|
resolver = resolve;
|
|
|
|
});
|
2022-09-05 08:30:47 +00:00
|
|
|
const timeoutId =
|
|
|
|
opts && opts.timeout > 0
|
|
|
|
? setTimeout(() => {
|
2023-05-26 06:02:17 +00:00
|
|
|
reject(new TimeoutError(opts.message));
|
2022-09-05 08:30:47 +00:00
|
|
|
}, opts.timeout)
|
|
|
|
: undefined;
|
2023-05-26 06:02:17 +00:00
|
|
|
|
|
|
|
function finish(value: T | Error) {
|
|
|
|
clearTimeout(timeoutId);
|
|
|
|
_value = value;
|
|
|
|
resolver();
|
|
|
|
}
|
|
|
|
|
|
|
|
function resolve(value: T) {
|
|
|
|
if (isRejected || isResolved) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
isResolved = true;
|
|
|
|
finish(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
function reject(error: Error) {
|
|
|
|
if (isRejected || isResolved) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
isRejected = true;
|
|
|
|
finish(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
2022-08-17 12:39:41 +00:00
|
|
|
resolved: () => {
|
|
|
|
return isResolved;
|
|
|
|
},
|
|
|
|
finished: () => {
|
|
|
|
return isResolved || isRejected;
|
|
|
|
},
|
2023-05-26 06:02:17 +00:00
|
|
|
resolve,
|
|
|
|
reject,
|
2023-05-16 15:18:22 +00:00
|
|
|
value: () => {
|
|
|
|
return _value;
|
|
|
|
},
|
2023-05-26 06:02:17 +00:00
|
|
|
async valueOrThrow() {
|
|
|
|
await taskPromise;
|
|
|
|
if (isRejected) {
|
|
|
|
throw _value;
|
|
|
|
}
|
|
|
|
return _value as T;
|
|
|
|
},
|
|
|
|
};
|
2022-08-17 12:39:41 +00:00
|
|
|
}
|