fix(zone.js): zone.js patches rxjs should check null for unsubscribe (#35990)
Close #31687, #31684 Zone.js patches rxjs internal `_subscribe` and `_unsubscribe` methods, but zone.js doesn't do null check, so in some operator such as `retryWhen`, the `_unsubscribe` will be set to null, and will cause zone patched version throw error. In this PR, if `_subscribe` and `_unsubscribe` is null, will not do the patch. PR Close #35990
This commit is contained in:
parent
54634628ac
commit
3fa895298d
|
@ -50,22 +50,28 @@ type ZoneSubscriberContext = {
|
||||||
},
|
},
|
||||||
set: function(this: Observable<any>, subscribe: any) {
|
set: function(this: Observable<any>, subscribe: any) {
|
||||||
(this as any)._zone = Zone.current;
|
(this as any)._zone = Zone.current;
|
||||||
(this as any)._zoneSubscribe = function(this: ZoneSubscriberContext) {
|
if (!subscribe) {
|
||||||
if (this._zone && this._zone !== Zone.current) {
|
(this as any)._zoneSubscribe = subscribe;
|
||||||
const tearDown = this._zone.run(subscribe, this, arguments as any);
|
} else {
|
||||||
if (tearDown && typeof tearDown === 'function') {
|
(this as any)._zoneSubscribe = function(this: ZoneSubscriberContext) {
|
||||||
const zone = this._zone;
|
if (this._zone && this._zone !== Zone.current) {
|
||||||
return function(this: ZoneSubscriberContext) {
|
const tearDown = this._zone.run(subscribe, this, arguments as any);
|
||||||
if (zone !== Zone.current) {
|
if (typeof tearDown === 'function') {
|
||||||
return zone.run(tearDown, this, arguments as any);
|
const zone = this._zone;
|
||||||
}
|
return function(this: ZoneSubscriberContext) {
|
||||||
return tearDown.apply(this, arguments);
|
if (zone !== Zone.current) {
|
||||||
};
|
return zone.run(tearDown, this, arguments as any);
|
||||||
|
}
|
||||||
|
return tearDown.apply(this, arguments);
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return tearDown;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return subscribe.apply(this, arguments);
|
||||||
}
|
}
|
||||||
return tearDown;
|
};
|
||||||
}
|
}
|
||||||
return subscribe.apply(this, arguments);
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
subjectFactory: {
|
subjectFactory: {
|
||||||
|
@ -113,12 +119,17 @@ type ZoneSubscriberContext = {
|
||||||
},
|
},
|
||||||
set: function(this: Subscription, unsubscribe: any) {
|
set: function(this: Subscription, unsubscribe: any) {
|
||||||
(this as any)._zone = Zone.current;
|
(this as any)._zone = Zone.current;
|
||||||
(this as any)._zoneUnsubscribe = function() {
|
if (!unsubscribe) {
|
||||||
if (this._zone && this._zone !== Zone.current) {
|
(this as any)._zoneUnsubscribe = unsubscribe;
|
||||||
return this._zone.run(unsubscribe, this, arguments);
|
} else {
|
||||||
}
|
(this as any)._zoneUnsubscribe = function() {
|
||||||
return unsubscribe.apply(this, arguments);
|
if (this._zone && this._zone !== Zone.current) {
|
||||||
};
|
return this._zone.run(unsubscribe, this, arguments);
|
||||||
|
} else {
|
||||||
|
return unsubscribe.apply(this, arguments);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -0,0 +1,54 @@
|
||||||
|
/**
|
||||||
|
* @license
|
||||||
|
* Copyright Google Inc. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style license that can be
|
||||||
|
* found in the LICENSE file at https://angular.io/license
|
||||||
|
*/
|
||||||
|
import {Observable, of , timer} from 'rxjs';
|
||||||
|
import {delayWhen, map, retryWhen} from 'rxjs/operators';
|
||||||
|
|
||||||
|
describe('Observable.retryWhen', () => {
|
||||||
|
let log: any[];
|
||||||
|
let observable1: Observable<any>;
|
||||||
|
let defaultTimeout = jasmine.DEFAULT_TIMEOUT_INTERVAL;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
log = [];
|
||||||
|
jasmine.DEFAULT_TIMEOUT_INTERVAL = 10000;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => { jasmine.DEFAULT_TIMEOUT_INTERVAL = defaultTimeout; });
|
||||||
|
|
||||||
|
it('retryWhen func callback should run in the correct zone', (done: DoneFn) => {
|
||||||
|
const constructorZone1: Zone = Zone.current.fork({name: 'Constructor Zone'});
|
||||||
|
const subscriptionZone: Zone = Zone.current.fork({name: 'Subscription Zone'});
|
||||||
|
let isErrorHandled = false;
|
||||||
|
observable1 = constructorZone1.run(() => {
|
||||||
|
return of (1, 2, 3).pipe(
|
||||||
|
map(v => {
|
||||||
|
if (v > 2 && !isErrorHandled) {
|
||||||
|
isErrorHandled = true;
|
||||||
|
throw v;
|
||||||
|
}
|
||||||
|
return v;
|
||||||
|
}),
|
||||||
|
retryWhen(err => err.pipe(delayWhen(v => timer(v)))));
|
||||||
|
});
|
||||||
|
|
||||||
|
subscriptionZone.run(() => {
|
||||||
|
observable1.subscribe(
|
||||||
|
(result: any) => {
|
||||||
|
log.push(result);
|
||||||
|
expect(Zone.current.name).toEqual(subscriptionZone.name);
|
||||||
|
},
|
||||||
|
(err: any) => { fail('should not call error'); },
|
||||||
|
() => {
|
||||||
|
log.push('completed');
|
||||||
|
expect(Zone.current.name).toEqual(subscriptionZone.name);
|
||||||
|
expect(log).toEqual([1, 2, 1, 2, 3, 'completed']);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -50,5 +50,6 @@ import './rxjs.Observable.map.spec';
|
||||||
import './rxjs.Observable.race.spec';
|
import './rxjs.Observable.race.spec';
|
||||||
import './rxjs.Observable.sample.spec';
|
import './rxjs.Observable.sample.spec';
|
||||||
import './rxjs.Observable.take.spec';
|
import './rxjs.Observable.take.spec';
|
||||||
|
import './rxjs.Observable.retry.spec';
|
||||||
import './rxjs.Observable.timeout.spec';
|
import './rxjs.Observable.timeout.spec';
|
||||||
import './rxjs.Observable.window.spec';
|
import './rxjs.Observable.window.spec';
|
||||||
|
|
Loading…
Reference in New Issue