From 9e88c2e529712f2dfecbb581a8a02cc9161d0a07 Mon Sep 17 00:00:00 2001 From: Ben Lesh Date: Wed, 27 Jan 2016 12:05:18 -0800 Subject: [PATCH] fix(Subscription): fix leaks caused by unsubscribe functions that throw - adds a test for the issue related #1256 --- spec/Subscription-spec.js | 39 +++++++++++++++++++++++++++++++++++++++ src/Rx.DOM.ts | 5 +++-- src/Rx.KitchenSink.ts | 3 ++- src/Rx.ts | 5 +++-- src/Subscription.ts | 37 +++++++++++++++++++++++++++++++++---- 5 files changed, 80 insertions(+), 9 deletions(-) create mode 100644 spec/Subscription-spec.js diff --git a/spec/Subscription-spec.js b/spec/Subscription-spec.js new file mode 100644 index 0000000000..0a5fea3367 --- /dev/null +++ b/spec/Subscription-spec.js @@ -0,0 +1,39 @@ +/* globals describe, it, expect */ +var Rx = require('../dist/cjs/Rx'); +var Subscription = Rx.Subscription; +var Observable = Rx.Observable; + +describe('Subscription', function () { + it('should not leak', function (done) { + var tearDowns = []; + + var source1 = Observable.create(function (observer) { + return function () { + tearDowns.push(1); + }; + }); + + var source2 = Observable.create(function (observer) { + return function () { + tearDowns.push(2); + throw new Error('oops, I am a bad unsubscribe!'); + }; + }); + + var source3 = Observable.create(function (observer) { + return function () { + tearDowns.push(3); + }; + }); + + var subscription = Observable.merge(source1, source2, source3).subscribe(); + + setTimeout(function () { + expect(function () { + subscription.unsubscribe(); + }).toThrow(); + expect(tearDowns).toEqual([1, 2, 3]); + done(); + }); + }); +}); \ No newline at end of file diff --git a/src/Rx.DOM.ts b/src/Rx.DOM.ts index 8bd91e5fff..412d1fae40 100644 --- a/src/Rx.DOM.ts +++ b/src/Rx.DOM.ts @@ -110,7 +110,7 @@ import './add/operator/zip'; import './add/operator/zipAll'; /* tslint:disable:no-unused-variable */ -import {Subscription} from './Subscription'; +import {Subscription, UnsubscriptionError} from './Subscription'; import {Subscriber} from './Subscriber'; import {AsyncSubject} from './subject/AsyncSubject'; import {ReplaySubject} from './subject/ReplaySubject'; @@ -159,5 +159,6 @@ export { Notification, EmptyError, ArgumentOutOfRangeError, - ObjectUnsubscribedError + ObjectUnsubscribedError, + UnsubscriptionError }; diff --git a/src/Rx.KitchenSink.ts b/src/Rx.KitchenSink.ts index 7601b369df..0ae4ad79a1 100644 --- a/src/Rx.KitchenSink.ts +++ b/src/Rx.KitchenSink.ts @@ -139,7 +139,7 @@ import './add/operator/zipAll'; /* tslint:disable:no-unused-variable */ import {Observer} from './Observer'; -import {Subscription} from './Subscription'; +import {Subscription, UnsubscriptionError} from './Subscription'; import {Subscriber} from './Subscriber'; import {AsyncSubject} from './subject/AsyncSubject'; import {ReplaySubject} from './subject/ReplaySubject'; @@ -185,6 +185,7 @@ export { EmptyError, ArgumentOutOfRangeError, ObjectUnsubscribedError, + UnsubscriptionError, TestScheduler, VirtualTimeScheduler, TimeInterval, diff --git a/src/Rx.ts b/src/Rx.ts index ec86e38563..d8c04944f6 100644 --- a/src/Rx.ts +++ b/src/Rx.ts @@ -114,7 +114,7 @@ import './add/operator/zipAll'; /* tslint:disable:no-unused-variable */ import {Operator} from './Operator'; import {Observer} from './Observer'; -import {Subscription} from './Subscription'; +import {Subscription, UnsubscriptionError} from './Subscription'; import {Subscriber} from './Subscriber'; import {AsyncSubject} from './subject/AsyncSubject'; import {ReplaySubject} from './subject/ReplaySubject'; @@ -158,5 +158,6 @@ export { Notification, EmptyError, ArgumentOutOfRangeError, - ObjectUnsubscribedError + ObjectUnsubscribedError, + UnsubscriptionError }; diff --git a/src/Subscription.ts b/src/Subscription.ts index 7f5148aca2..2ceac85434 100644 --- a/src/Subscription.ts +++ b/src/Subscription.ts @@ -1,6 +1,8 @@ import {isArray} from './util/isArray'; import {isObject} from './util/isObject'; import {isFunction} from './util/isFunction'; +import {tryCatch} from './util/tryCatch'; +import {errorObject} from './util/errorObject'; export class Subscription { public static EMPTY: Subscription = (function(empty: any){ @@ -17,6 +19,8 @@ export class Subscription { } unsubscribe(): void { + let hasErrors = false; + let errors: any[]; if (this.isUnsubscribed) { return; @@ -29,7 +33,11 @@ export class Subscription { ( this)._subscriptions = null; if (isFunction(_unsubscribe)) { - _unsubscribe.call(this); + let trial = tryCatch(_unsubscribe).call(this); + if (trial === errorObject) { + hasErrors = true; + (errors = errors || []).push(errorObject.e); + } } if (isArray(_subscriptions)) { @@ -38,12 +46,26 @@ export class Subscription { const len = _subscriptions.length; while (++index < len) { - const subscription = _subscriptions[index]; - if (isObject(subscription)) { - subscription.unsubscribe(); + const sub = _subscriptions[index]; + if (isObject(sub)) { + let trial = tryCatch(sub.unsubscribe).call(sub); + if (trial === errorObject) { + hasErrors = true; + errors = errors || []; + let err = errorObject.e; + if (err instanceof UnsubscriptionError) { + errors = errors.concat(err.errors); + } else { + errors.push(err); + } + } } } } + + if (hasErrors) { + throw new UnsubscriptionError(errors); + } } add(subscription: Subscription | Function | void): void { @@ -98,3 +120,10 @@ export class Subscription { } } } + +export class UnsubscriptionError extends Error { + constructor(public errors: any[]) { + super('unsubscriptoin error(s)'); + this.name = 'UnsubscriptionError'; + } +} \ No newline at end of file