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<TeardownLogic, void>[] | 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<TeardownLogic, void>): 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();
|
}
|
}
|