export interface IResolve { (value?: T | PromiseLike): void; } export interface IReject { (reason?: unknown): void; } export type TDeferredStatus = 'pending' | 'fulfilled' | 'rejected'; export class Deferred { public promise: Promise; public resolve!: IResolve; public reject!: IReject; public status!: TDeferredStatus; public claimed = false; public claim() { if (this.claimed) { throw new Error('Deferred already claimed'); } this.claimed = true; } public startedAt!: number; public stoppedAt!: number; public get duration(): number { if (this.stoppedAt) { return this.stoppedAt - this.startedAt; } else { return Date.now() - this.startedAt; } } constructor() { this.promise = new Promise((resolve, reject) => { this.resolve = (valueArg?: T | PromiseLike) => { this.status = 'fulfilled'; this.stoppedAt = Date.now(); resolve(valueArg as T | PromiseLike); }; this.reject = (reason?: unknown) => { this.status = 'rejected'; this.stoppedAt = Date.now(); reject(reason); }; this.startedAt = Date.now(); this.status = 'pending'; }); } } export const defer = () => { return new Deferred(); };