diff --git a/src/main/Executor.ts b/src/main/Executor.ts index 0c6d407..8e4071c 100644 --- a/src/main/Executor.ts +++ b/src/main/Executor.ts @@ -1,7 +1,15 @@ import { AbortablePromise } from './AbortablePromise'; import { PubSub } from './PubSub'; import { AbortableCallback, Awaitable } from './types'; -import { isEqual, isPromiseLike } from './utils'; +import { isEqual, isPromiseLike, withSignal } from './utils'; + +type ExecutorEvent = + | { type: 'pending'; target: Executor } + | { type: 'cleared'; target: Executor } + | { type: 'aborted'; target: Executor } + | { type: 'invalidated'; target: Executor } + | { type: 'fulfilled'; target: Executor } + | { type: 'rejected'; target: Executor }; /** * Manages async callback execution process and provides ways to access execution results, abort or replace an @@ -54,12 +62,12 @@ export class Executor { return this.promise !== null; } - private _pubSub = new PubSub(); - /** * The last callback passed to {@link execute}. */ - private _executee: AbortableCallback | null = null; + private _lastCallback: AbortableCallback | null = null; + + private _pubSub = new PubSub(); /** * Instantly aborts pending execution (if any), marks executor as pending and invokes the callback. @@ -71,27 +79,28 @@ export class Executor { * @returns The promise that is resolved with the result of the callback execution. */ execute(cb: AbortableCallback): AbortablePromise { - this._executee = cb; + this._lastCallback = cb; - const nextPromise = new AbortablePromise((resolve, reject, signal) => { + const promise = new AbortablePromise((resolve, reject, signal) => { signal.addEventListener('abort', () => { - if (this.promise === nextPromise) { - this.abort(); + if (this.promise === promise) { + this.promise = null; + this._pubSub.publish({ type: 'aborted', target: this }); } }); new Promise(resolve => { - resolve(cb(signal)); + resolve(withSignal(cb(signal), signal)); }).then( value => { - if (this.promise === nextPromise) { + if (this.promise === promise) { this.promise = null; this.resolve(value); } resolve(value); }, reason => { - if (this.promise === nextPromise) { + if (this.promise === promise) { this.promise = null; this.reject(reason); } @@ -102,15 +111,25 @@ export class Executor { const prevPromise = this.promise; - this.promise = nextPromise; + this.promise = promise; if (prevPromise !== null) { prevPromise.abort(); } - this._pubSub.publish(); + this._pubSub.publish({ type: 'pending', target: this }); + + return promise; + } - return nextPromise; + /** + * If + */ + retry(): this { + if (this._lastCallback !== null) { + this.execute(this._lastCallback); + } + return this; } /** @@ -129,7 +148,7 @@ export class Executor { if (this.isSettled) { this.isFulfilled = this.isRejected = this.isInvalidated = false; this.value = this.reason = undefined; - this._pubSub.publish(); + this._pubSub.publish({ type: 'cleared', target: this }); } return this; } @@ -141,13 +160,7 @@ export class Executor { * @param reason The abort reason passed to the pending promise. */ abort(reason?: unknown): this { - const promise = this.promise; - - if (promise !== null) { - this.promise = null; - promise.abort(reason); - this._pubSub.publish(); - } + this.promise?.abort(reason); return this; } @@ -156,14 +169,8 @@ export class Executor { * to {@link execute} is executed again and pending execution is aborted. */ invalidate(): this { - if (!this.isSettled || this.isInvalidated) { - return this; - } - - this.isInvalidated = true; - - if (this._executee !== null) { - this.execute(this._executee); + if (this.isInvalidated !== (this.isInvalidated = this.isSettled)) { + this._pubSub.publish({ type: 'invalidated', target: this }); } return this; } @@ -175,12 +182,9 @@ export class Executor { const promise = this.promise; if (isPromiseLike(value)) { - const prevExecutee = this._executee; this.execute(() => value); - this._executee = prevExecutee; return this; } - if ( (promise !== null && (promise.abort(), !(this.promise = null))) || this.isInvalidated || @@ -191,7 +195,7 @@ export class Executor { this.isRejected = this.isInvalidated = false; this.value = value; this.reason = undefined; - this._pubSub.publish(); + this._pubSub.publish({ type: 'fulfilled', target: this }); } return this; } @@ -212,7 +216,7 @@ export class Executor { this.isRejected = true; this.value = undefined; this.reason = reason; - this._pubSub.publish(); + this._pubSub.publish({ type: 'rejected', target: this }); } return this; } @@ -223,7 +227,7 @@ export class Executor { * @param listener The listener that would be notified. * @returns The callback to unsubscribe the listener. */ - subscribe(listener: () => void): () => void { + subscribe(listener: (event: ExecutorEvent) => void): () => void { return this._pubSub.subscribe(listener); } }