import { Operator } from '../Operator'; import { Subscriber } from '../Subscriber'; import { Observable } from '../Observable'; import { Subject } from '../Subject'; import { Subscription, TeardownLogic } from '../Subscription'; import { tryCatch } from '../util/tryCatch'; import { errorObject } from '../util/errorObject'; import { OuterSubscriber } from '../OuterSubscriber'; import { InnerSubscriber } from '../InnerSubscriber'; import { subscribeToResult } from '../util/subscribeToResult'; /** * Returns an Observable that emits the same values as the source observable with the exception of a `complete`. * A `complete` will cause the emission of the Throwable that cause the complete to the Observable returned from * notificationHandler. If that Observable calls onComplete or `complete` then retry will call `complete` or `error` * on the child subscription. Otherwise, this Observable will resubscribe to the source observable, on a particular * Scheduler. * * * * @param {notificationHandler} receives an Observable of notifications with which a user can `complete` or `error`, * aborting the retry. * @param {scheduler} the Scheduler on which to subscribe to the source Observable. * @return {Observable} the source Observable modified with retry logic. * @method repeatWhen * @owner Observable */ export function repeatWhen(this: Observable, notifier: (notifications: Observable) => Observable): Observable { return this.lift(new RepeatWhenOperator(notifier, this)); } class RepeatWhenOperator implements Operator { constructor(protected notifier: (notifications: Observable) => Observable, protected source: Observable) { } call(subscriber: Subscriber, source: any): TeardownLogic { return source.subscribe(new RepeatWhenSubscriber(subscriber, this.notifier, this.source)); } } /** * We need this JSDoc comment for affecting ESDoc. * @ignore * @extends {Ignored} */ class RepeatWhenSubscriber extends OuterSubscriber { private notifications: Subject; private retries: Observable; private retriesSubscription: Subscription; constructor(destination: Subscriber, private notifier: (notifications: Observable) => Observable, private source: Observable) { super(destination); } complete() { if (!this.isStopped) { let notifications = this.notifications; let retries: any = this.retries; let retriesSubscription = this.retriesSubscription; if (!retries) { notifications = new Subject(); retries = tryCatch(this.notifier)(notifications); if (retries === errorObject) { return super.complete(); } retriesSubscription = subscribeToResult(this, retries); } else { this.notifications = null; this.retriesSubscription = null; } this.unsubscribe(); this.closed = false; this.notifications = notifications; this.retries = retries; this.retriesSubscription = retriesSubscription; notifications.next(); } } protected _unsubscribe() { const { notifications, retriesSubscription } = this; if (notifications) { notifications.unsubscribe(); this.notifications = null; } if (retriesSubscription) { retriesSubscription.unsubscribe(); this.retriesSubscription = null; } this.retries = null; } notifyNext(outerValue: T, innerValue: R, outerIndex: number, innerIndex: number, innerSub: InnerSubscriber): void { const { notifications, retries, retriesSubscription } = this; this.notifications = null; this.retries = null; this.retriesSubscription = null; this.unsubscribe(); this.isStopped = false; this.closed = false; this.notifications = notifications; this.retries = retries; this.retriesSubscription = retriesSubscription; this.source.subscribe(this); } }