UNPKG

rxjs

Version:

Reactive Extensions for modern JavaScript

108 lines (102 loc) 3.76 kB
import { Observable } from '../Observable'; import { Subject } from '../Subject'; import { Subscription } from '../Subscription'; import { MonoTypeOperatorFunction } from '../types'; import { operate } from '../util/lift'; import { OperatorSubscriber } from './OperatorSubscriber'; /** * Returns an Observable that mirrors the source Observable with the exception of an `error`. If the source Observable * calls `error`, this method will emit the Throwable that caused the error to the Observable returned from `notifier`. * If that Observable calls `complete` or `error` then this method will call `complete` or `error` on the child * subscription. Otherwise this method will resubscribe to the source Observable. * * ![](retryWhen.png) * * Retry an observable sequence on error based on custom criteria. * * ## Example * ```ts * import { timer, interval } from 'rxjs'; * import { map, tap, retryWhen, delayWhen } from 'rxjs/operators'; * * const source = interval(1000); * const example = source.pipe( * map(val => { * if (val > 5) { * // error will be picked up by retryWhen * throw val; * } * return val; * }), * retryWhen(errors => * errors.pipe( * // log error message * tap(val => console.log(`Value ${val} was too high!`)), * // restart in 5 seconds * delayWhen(val => timer(val * 1000)) * ) * ) * ); * * const subscribe = example.subscribe(val => console.log(val)); * * // results: * // 0 * // 1 * // 2 * // 3 * // 4 * // 5 * // "Value 6 was too high!" * // --Wait 5 seconds then repeat * ``` * * @param {function(errors: Observable): Observable} notifier - Receives an Observable of notifications with which a * user can `complete` or `error`, aborting the retry. * @return A function that returns an Observable that mirrors the source * Observable with the exception of an `error`. */ export function retryWhen<T>(notifier: (errors: Observable<any>) => Observable<any>): MonoTypeOperatorFunction<T> { return operate((source, subscriber) => { let innerSub: Subscription | null; let syncResub = false; let errors$: Subject<any>; const subscribeForRetryWhen = () => { innerSub = source.subscribe( new OperatorSubscriber(subscriber, undefined, undefined, (err) => { if (!errors$) { errors$ = new Subject(); notifier(errors$).subscribe( new OperatorSubscriber(subscriber, () => // If we have an innerSub, this was an asynchronous call, kick off the retry. // Otherwise, if we don't have an innerSub yet, that's because the inner subscription // call hasn't even returned yet. We've arrived here synchronously. // So we flag that we want to resub, such that we can ensure teardown // happens before we resubscribe. innerSub ? subscribeForRetryWhen() : (syncResub = true) ) ); } if (errors$) { // We have set up the notifier without error. errors$.next(err); } }) ); if (syncResub) { // Ensure that the inner subscription is torn down before // moving on to the next subscription in the synchronous case. // If we don't do this here, all inner subscriptions will not be // torn down until the entire observable is done. innerSub.unsubscribe(); innerSub = null; // We may need to do this multiple times, so reset the flag. syncResub = false; // Resubscribe subscribeForRetryWhen(); } }; // Start the subscription subscribeForRetryWhen(); }); }