/*
This file is part of web3.js.
web3.js is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
web3.js is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with web3.js. If not, see .
*/
import { OperationTimeoutError } from '@theqrl/web3-errors';
import { Web3DeferredPromiseInterface } from '@theqrl/web3-types';
/**
* The class is a simple implementation of a deferred promise with optional timeout functionality,
* which can be useful when dealing with asynchronous tasks.
*
*/
export class Web3DeferredPromise implements Promise, Web3DeferredPromiseInterface {
// public tag to treat object as promise by different libs
// eslint-disable-next-line @typescript-eslint/prefer-as-const
public [Symbol.toStringTag]: 'Promise' = 'Promise';
private readonly _promise: Promise;
private _resolve!: (value: T | PromiseLike) => void;
private _reject!: (reason?: unknown) => void;
private _state: 'pending' | 'fulfilled' | 'rejected' = 'pending';
private _timeoutId?: NodeJS.Timeout;
private readonly _timeoutInterval?: number;
private readonly _timeoutMessage: string;
/**
*
* @param timeout - (optional) The timeout in milliseconds.
* @param eagerStart - (optional) If true, the timer starts as soon as the promise is created.
* @param timeoutMessage - (optional) The message to include in the timeout erro that is thrown when the promise times out.
*/
public constructor(
{
timeout,
eagerStart,
timeoutMessage,
}: { timeout: number; eagerStart: boolean; timeoutMessage: string } = {
timeout: 0,
eagerStart: false,
timeoutMessage: 'DeferredPromise timed out',
},
) {
this._promise = new Promise((resolve, reject) => {
this._resolve = resolve;
this._reject = reject;
});
this._timeoutMessage = timeoutMessage;
this._timeoutInterval = timeout;
if (eagerStart) {
this.startTimer();
}
}
/**
* Returns the current state of the promise.
* @returns 'pending' | 'fulfilled' | 'rejected'
*/
public get state(): 'pending' | 'fulfilled' | 'rejected' {
return this._state;
}
/**
*
* @param onfulfilled - (optional) The callback to execute when the promise is fulfilled.
* @param onrejected - (optional) The callback to execute when the promise is rejected.
* @returns
*/
public async then(
onfulfilled?: (value: T) => TResult1 | PromiseLike,
onrejected?: (reason: unknown) => TResult2 | PromiseLike,
): Promise {
return this._promise.then(onfulfilled, onrejected);
}
/**
*
* @param onrejected - (optional) The callback to execute when the promise is rejected.
* @returns
*/
public async catch(
// eslint-disable-next-line @typescript-eslint/no-explicit-any
onrejected?: (reason: any) => TResult | PromiseLike,
): Promise {
return this._promise.catch(onrejected);
}
/**
*
* @param onfinally - (optional) The callback to execute when the promise is settled (fulfilled or rejected).
* @returns
*/
public async finally(onfinally?: (() => void) | undefined): Promise {
return this._promise.finally(onfinally);
}
/**
* Resolves the current promise.
* @param value - The value to resolve the promise with.
*/
public resolve(value: T | PromiseLike): void {
this._resolve(value);
this._state = 'fulfilled';
this._clearTimeout();
}
/**
* Rejects the current promise.
* @param reason - The reason to reject the promise with.
*/
public reject(reason?: unknown): void {
this._reject(reason);
this._state = 'rejected';
this._clearTimeout();
}
/**
* Starts the timeout timer for the promise.
*/
public startTimer() {
if (this._timeoutInterval && this._timeoutInterval > 0) {
this._timeoutId = setTimeout(this._checkTimeout.bind(this), this._timeoutInterval);
}
}
private _checkTimeout() {
if (this._state === 'pending' && this._timeoutId) {
this.reject(new OperationTimeoutError(this._timeoutMessage));
}
}
private _clearTimeout() {
if (this._timeoutId) {
clearTimeout(this._timeoutId);
}
}
}