import { isFunction } from './util/isFunction'; import { UnsubscriptionError } from './util/UnsubscriptionError'; import { SubscriptionLike, TeardownLogic, Unsubscribable } from './types'; import { arrRemove } from './util/arrRemove'; /** * Represents a disposable resource, such as the execution of an Observable. A * Subscription has one important method, `unsubscribe`, that takes no argument * and just disposes the resource held by the subscription. * * Additionally, subscriptions may be grouped together through the `add()` * method, which will attach a child Subscription to the current Subscription. * When a Subscription is unsubscribed, all its children (and its grandchildren) * will be unsubscribed as well. * * @class Subscription */ export class Subscription implements SubscriptionLike { /** @nocollapse */ public static EMPTY = (() => { const empty = new Subscription(); empty.closed = true; return empty; })(); /** * A flag to indicate whether this Subscription has already been unsubscribed. */ public closed = false; private _parentage: Subscription[] | Subscription | null = null; /** * The list of registered finalizers to execute upon unsubscription. Adding and removing from this * list occurs in the {@link #add} and {@link #remove} methods. */ private _finalizers: Exclude[] | null = null; /** * @param initialTeardown A function executed first as part of the finalization * process that is kicked off when {@link #unsubscribe} is called. */ constructor(private initialTeardown?: () => void) {} /** * Disposes the resources held by the subscription. May, for instance, cancel * an ongoing Observable execution or cancel any other type of work that * started when the Subscription was created. * @return {void} */ unsubscribe(): void { let errors: any[] | undefined; if (!this.closed) { this.closed = true; // Remove this from it's parents. const { _parentage } = this; if (_parentage) { this._parentage = null; if (Array.isArray(_parentage)) { for (const parent of _parentage) { parent.remove(this); } } else { _parentage.remove(this); } } const { initialTeardown: initialFinalizer } = this; if (isFunction(initialFinalizer)) { try { initialFinalizer(); } catch (e) { errors = e instanceof UnsubscriptionError ? e.errors : [e]; } } const { _finalizers } = this; if (_finalizers) { this._finalizers = null; for (const finalizer of _finalizers) { try { execFinalizer(finalizer); } catch (err) { errors = errors ?? []; if (err instanceof UnsubscriptionError) { errors = [...errors, ...err.errors]; } else { errors.push(err); } } } } if (errors) { throw new UnsubscriptionError(errors); } } } /** * Adds a finalizer to this subscription, so that finalization will be unsubscribed/called * when this subscription is unsubscribed. If this subscription is already {@link #closed}, * because it has already been unsubscribed, then whatever finalizer is passed to it * will automatically be executed (unless the finalizer itself is also a closed subscription). * * Closed Subscriptions cannot be added as finalizers to any subscription. Adding a closed * subscription to a any subscription will result in no operation. (A noop). * * Adding a subscription to itself, or adding `null` or `undefined` will not perform any * operation at all. (A noop). * * `Subscription` instances that are added to this instance will automatically remove themselves * if they are unsubscribed. Functions and {@link Unsubscribable} objects that you wish to remove * will need to be removed manually with {@link #remove} * * @param teardown The finalization logic to add to this subscription. */ add(teardown: TeardownLogic): void { // Only add the finalizer if it's not undefined // and don't add a subscription to itself. if (teardown && teardown !== this) { if (this.closed) { // If this subscription is already closed, // execute whatever finalizer is handed to it automatically. execFinalizer(teardown); } else { if (teardown instanceof Subscription) { // We don't add closed subscriptions, and we don't add the same subscription // twice. Subscription unsubscribe is idempotent. if (teardown.closed || teardown._hasParent(this)) { return; } teardown._addParent(this); } (this._finalizers = this._finalizers ?? []).push(teardown); } } } /** * Checks to see if a this subscription already has a particular parent. * This will signal that this subscription has already been added to the parent in question. * @param parent the parent to check for */ private _hasParent(parent: Subscription) { const { _parentage } = this; return _parentage === parent || (Array.isArray(_parentage) && _parentage.includes(parent)); } /** * Adds a parent to this subscription so it can be removed from the parent if it * unsubscribes on it's own. * * NOTE: THIS ASSUMES THAT {@link _hasParent} HAS ALREADY BEEN CHECKED. * @param parent The parent subscription to add */ private _addParent(parent: Subscription) { const { _parentage } = this; this._parentage = Array.isArray(_parentage) ? (_parentage.push(parent), _parentage) : _parentage ? [_parentage, parent] : parent; } /** * Called on a child when it is removed via {@link #remove}. * @param parent The parent to remove */ private _removeParent(parent: Subscription) { const { _parentage } = this; if (_parentage === parent) { this._parentage = null; } else if (Array.isArray(_parentage)) { arrRemove(_parentage, parent); } } /** * Removes a finalizer from this subscription that was previously added with the {@link #add} method. * * Note that `Subscription` instances, when unsubscribed, will automatically remove themselves * from every other `Subscription` they have been added to. This means that using the `remove` method * is not a common thing and should be used thoughtfully. * * If you add the same finalizer instance of a function or an unsubscribable object to a `Subscription` instance * more than once, you will need to call `remove` the same number of times to remove all instances. * * All finalizer instances are removed to free up memory upon unsubscription. * * @param teardown The finalizer to remove from this subscription */ remove(teardown: Exclude): void { const { _finalizers } = this; _finalizers && arrRemove(_finalizers, teardown); if (teardown instanceof Subscription) { teardown._removeParent(this); } } } export const EMPTY_SUBSCRIPTION = Subscription.EMPTY; export function isSubscription(value: any): value is Subscription { return ( value instanceof Subscription || (value && 'closed' in value && isFunction(value.remove) && isFunction(value.add) && isFunction(value.unsubscribe)) ); } function execFinalizer(finalizer: Unsubscribable | (() => void)) { if (isFunction(finalizer)) { finalizer(); } else { finalizer.unsubscribe(); } }