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(); | |
} | |
} | |