All Downloads are FREE. Search and download functionalities are using the official Maven repository.

reactor.core.publisher.FluxRetryWhen Maven / Gradle / Ivy

/*
 * Copyright (c) 2011-Present VMware Inc. or its affiliates, All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *        https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package reactor.core.publisher;

import java.util.Objects;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.stream.Stream;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscription;

import reactor.core.CorePublisher;
import reactor.core.CoreSubscriber;
import reactor.core.Scannable;
import reactor.util.annotation.Nullable;
import reactor.util.context.Context;
import reactor.util.context.ContextView;
import reactor.util.retry.Retry;

/**
 * Retries a source when a companion sequence signals
 * an item in response to the main's error signal.
 * 

*

If the companion sequence signals when the main source is active, the repeat * attempt is suppressed and any terminal signal will terminate the main source with the same signal immediately. * * @param the source value type * @see Reactive-Streams-Commons */ final class FluxRetryWhen extends InternalFluxOperator { final Retry whenSourceFactory; FluxRetryWhen(Flux source, Retry whenSourceFactory) { super(source); this.whenSourceFactory = Objects.requireNonNull(whenSourceFactory, "whenSourceFactory"); } static void subscribe(CoreSubscriber s, Retry whenSourceFactory, CorePublisher source) { RetryWhenOtherSubscriber other = new RetryWhenOtherSubscriber(); CoreSubscriber serial = Operators.serialize(s); RetryWhenMainSubscriber main = new RetryWhenMainSubscriber<>(serial, other.completionSignal, source, whenSourceFactory.retryContext()); other.main = main; serial.onSubscribe(main); Publisher p; try { p = Objects.requireNonNull(whenSourceFactory.generateCompanion(other), "The whenSourceFactory returned a null Publisher"); } catch (Throwable e) { s.onError(Operators.onOperatorError(e, s.currentContext())); return; } p.subscribe(other); if (!main.cancelled) { source.subscribe(main); } } @Override public CoreSubscriber subscribeOrReturn(CoreSubscriber actual) { subscribe(actual, whenSourceFactory, source); return null; } @Override public Object scanUnsafe(Attr key) { if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC; return super.scanUnsafe(key); } static final class RetryWhenMainSubscriber extends Operators.MultiSubscriptionSubscriber implements Retry.RetrySignal { final Operators.DeferredSubscription otherArbiter; final Sinks.Many signaller; final CorePublisher source; long totalFailureIndex = 0L; long subsequentFailureIndex = 0L; @Nullable Throwable lastFailure = null; final ContextView retryContext; Context context; volatile int wip; static final AtomicIntegerFieldUpdater WIP = AtomicIntegerFieldUpdater.newUpdater(RetryWhenMainSubscriber.class, "wip"); long produced; RetryWhenMainSubscriber(CoreSubscriber actual, Sinks.Many signaller, CorePublisher source, ContextView retryContext) { super(actual); this.signaller = signaller; this.source = source; this.otherArbiter = new Operators.DeferredSubscription(); this.context = actual.currentContext(); this.retryContext = retryContext; } @Override public long totalRetries() { return this.totalFailureIndex - 1; } @Override public long totalRetriesInARow() { return this.subsequentFailureIndex - 1; } @Override public Throwable failure() { assert this.lastFailure != null; return this.lastFailure; } @Override public ContextView retryContextView() { return retryContext; } @Override public Context currentContext() { return this.context; } @Override public Stream inners() { return Stream.of(Scannable.from(signaller), otherArbiter); } @Override public void cancel() { if (!cancelled) { otherArbiter.cancel(); super.cancel(); } } void swap(Subscription w) { otherArbiter.set(w); } @Override public void onNext(T t) { subsequentFailureIndex = 0; actual.onNext(t); produced++; } @Override public void onError(Throwable t) { totalFailureIndex++; subsequentFailureIndex++; lastFailure = t; long p = produced; if (p != 0L) { produced = 0; produced(p); } signaller.emitNext(this, Sinks.EmitFailureHandler.FAIL_FAST); // request after signalling, otherwise it may race otherArbiter.request(1); } @Override public void onComplete() { lastFailure = null; otherArbiter.cancel(); actual.onComplete(); } void resubscribe(Object trigger) { if (WIP.getAndIncrement(this) == 0) { do { if (cancelled) { return; } //flow that emit a Context as a trigger for the re-subscription are //used to REPLACE the currentContext() if (trigger instanceof ContextView) { this.context = this.context.putAll((ContextView) trigger); } source.subscribe(this); } while (WIP.decrementAndGet(this) != 0); } } void whenError(Throwable e) { super.cancel(); actual.onError(e); } void whenComplete() { super.cancel(); actual.onComplete(); } @Override public Object scanUnsafe(Attr key) { if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC; return super.scanUnsafe(key); } } static final class RetryWhenOtherSubscriber extends Flux implements InnerConsumer, OptimizableOperator { RetryWhenMainSubscriber main; final Sinks.Many completionSignal = Sinks.many().multicast().onBackpressureBuffer(); @Override public Context currentContext() { return main.currentContext(); } @Override @Nullable public Object scanUnsafe(Attr key) { if (key == Attr.PARENT) return main.otherArbiter; if (key == Attr.ACTUAL) return main; if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC; return null; } @Override public void onSubscribe(Subscription s) { main.swap(s); } @Override public void onNext(Object t) { main.resubscribe(t); } @Override public void onError(Throwable t) { main.whenError(t); } @Override public void onComplete() { main.whenComplete(); } @Override public void subscribe(CoreSubscriber actual) { completionSignal.asFlux().subscribe(actual); } @Override public CoreSubscriber subscribeOrReturn(CoreSubscriber actual) { return actual; } @Override public CorePublisher source() { return completionSignal.asFlux(); } @Override public OptimizableOperator nextOptimizableSource() { return null; } } }