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

reactor.test.subscriber.DefaultTestSubscriber Maven / Gradle / Ivy

There is a newer version: 3.7.0
Show newest version
/*
 * Copyright (c) 2021 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.test.subscriber;

import java.time.Duration;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.concurrent.atomic.AtomicReference;

import org.reactivestreams.Subscription;

import reactor.core.Fuseable;
import reactor.core.publisher.Operators;
import reactor.core.publisher.Signal;
import reactor.util.annotation.Nullable;
import reactor.util.context.Context;

/**
 * Base version of a {@link TestSubscriber} (aka non-conditional).
 *
 * @author Simon Baslé
 */
class DefaultTestSubscriber implements TestSubscriber {

	final long                                    initialRequest;
	final Context                                 context;
	final DefaultTestSubscriber.FusionRequirement fusionRequirement;
	final int                                     requestedFusionMode;
	final int                                     expectedFusionMode;

	Subscription s;
	@Nullable
	Fuseable.QueueSubscription qs;
	int fusionMode = -1;

	// state tracking
	final AtomicBoolean   cancelled;
	final List         receivedOnNext;
	final List         receivedPostCancellation;
	final List> protocolErrors;

	final CountDownLatch                  doneLatch;
	final AtomicReference subscriptionFailure;

	@Nullable
	volatile Signal terminalSignal;

	volatile     int                                              state;
	@SuppressWarnings("rawtypes")
	static final AtomicIntegerFieldUpdater STATE =
			AtomicIntegerFieldUpdater.newUpdater(DefaultTestSubscriber.class, "state");

	volatile     long                                          requestedTotal;
	@SuppressWarnings("rawtypes")
	static final AtomicLongFieldUpdater REQUESTED_TOTAL =
			AtomicLongFieldUpdater.newUpdater(DefaultTestSubscriber.class, "requestedTotal");

	volatile     long                                          requestedPreSubscription;
	@SuppressWarnings("rawtypes")
	static final AtomicLongFieldUpdater REQUESTED_PRE_SUBSCRIPTION =
			AtomicLongFieldUpdater.newUpdater(DefaultTestSubscriber.class, "requestedPreSubscription");


	DefaultTestSubscriber(TestSubscriberBuilder options) {
		this.initialRequest = options.initialRequest;
		this.context = options.context;
		this.fusionRequirement = options.fusionRequirement;
		this.requestedFusionMode = options.requestedFusionMode;
		this.expectedFusionMode = options.expectedFusionMode;

		this.cancelled = new AtomicBoolean();
		this.receivedOnNext = new CopyOnWriteArrayList<>();
		this.receivedPostCancellation = new CopyOnWriteArrayList<>();
		this.protocolErrors = new CopyOnWriteArrayList<>();
		this.state = 0;

		this.doneLatch = new CountDownLatch(1);
		this.subscriptionFailure = new AtomicReference<>();
		REQUESTED_PRE_SUBSCRIPTION.lazySet(this, initialRequest);
	}

	@Override
	public Context currentContext() {
		return this.context;
	}

	void internalCancel() {
		Subscription s = this.s;
		if (cancelled.compareAndSet(false, true) && s != null) {
			s.cancel();
			safeClearQueue(s);
		}
	}

	void safeClearQueue(@Nullable Subscription s) {
		if (s instanceof Fuseable.QueueSubscription) {
			((Fuseable.QueueSubscription) s).clear();
		}
	}

	void subscriptionFail(String message) {
		if (this.subscriptionFailure.compareAndSet(null, new AssertionError(message))) {
			internalCancel();
			notifyDone();
		}
	}

	final void notifyDone() {
		doneLatch.countDown();
	}

	@Override
	public void onSubscribe(Subscription s) {
		if (cancelled.get()) {
			s.cancel();
			safeClearQueue(s);
			return;
		}
		if (!Operators.validate(this.s, s)) {
			safeClearQueue(s);
			//s is already cancelled at that point, subscriptionFail will cancel this.s
			subscriptionFail("TestSubscriber must not be reused, but Subscription has already been set.");
			return;
		}
		this.s = s;
		this.fusionMode = -1;
		if (s instanceof Fuseable.QueueSubscription) {
			if (fusionRequirement == FusionRequirement.NOT_FUSEABLE) {
				subscriptionFail("TestSubscriber configured to reject QueueSubscription, got " + s);
				return;
			}

			@SuppressWarnings("unchecked") //intermediate variable to suppress via annotation for compiler's benefit
			Fuseable.QueueSubscription converted = (Fuseable.QueueSubscription) s;
			this.qs = converted;
			int negotiatedMode = qs.requestFusion(this.requestedFusionMode);

			if (expectedFusionMode != negotiatedMode && expectedFusionMode != Fuseable.ANY) {
				subscriptionFail("TestSubscriber negotiated fusion mode inconsistent, expected " +
						Fuseable.fusionModeName(expectedFusionMode) + " got " + Fuseable.fusionModeName(negotiatedMode));
				return;
			}
			this.fusionMode = negotiatedMode;
			if (negotiatedMode == Fuseable.SYNC) {
				for (;;) {
					if (cancelled.get()) {
						safeClearQueue(qs);
						break;
					}
					T v = qs.poll();
					if (v == null) {
						onComplete();
						break;
					}
					onNext(v);
				}
			}
			else {
				long rPre = REQUESTED_PRE_SUBSCRIPTION.getAndSet(this, -1L);
				if (rPre > 0L) {
					upstreamRequest(s, rPre);
				}
			}
		}
		else if (fusionRequirement == FusionRequirement.FUSEABLE) {
			subscriptionFail("TestSubscriber configured to require QueueSubscription, got " + s);
		}
		else if (this.initialRequest > 0L) {
			long rPre = REQUESTED_PRE_SUBSCRIPTION.getAndSet(this, -1L);
			if (rPre > 0L) {
				upstreamRequest(s, rPre);
			}
		}
	}

	@Override
	public void onNext(@Nullable T t) {
		int previousState = markOnNextStart();
		boolean wasTerminated = isMarkedTerminated(previousState);
		boolean wasOnNext = isMarkedOnNext(previousState);
		if (wasTerminated || wasOnNext) {
			//at this point, we know we haven't switched the markedOnNext bit. if it is set, let the other onNext unset it
			if (t != null) {
				this.protocolErrors.add(Signal.next(t));
			}
			else if (wasTerminated) {
				this.protocolErrors.add(Signal.error(
						new AssertionError("onNext(null) received despite SYNC fusion (which has already completed)")
				));
			}
			else {
				//due to the looping nature of SYNC fusion in onSubscribe, this shouldn't happen
				this.protocolErrors.add(Signal.error(
						new AssertionError("onNext(null) received despite SYNC fusion (with concurrent onNext)")
				));
			}
			return;
		}

		if (t == null) {
			if (this.fusionMode == Fuseable.ASYNC) {
				drainAsync(false);
				return;
			}
			else {
				subscriptionFail("onNext(null) received while ASYNC fusion not established");
			}
		}

		this.receivedOnNext.add(t);
		if (cancelled.get()) {
			this.receivedPostCancellation.add(t);
		}

		checkTerminatedAfterOnNext();
	}

	@Override
	public void onComplete() {
		Signal sig = Signal.complete();

		int previousState = markTerminated();

		if (isMarkedTerminated(previousState) || isMarkedTerminating(previousState)) {
			this.protocolErrors.add(sig);
			return;
		}
		if (isMarkedOnNext(previousState)) {
			this.protocolErrors.add(sig);
			this.terminalSignal = sig;
			return; //isTerminating will be detected later, triggering the notifyDone()
		}

		this.terminalSignal = sig;

		if (fusionMode == Fuseable.ASYNC) {
			drainAsync(true);
			return;
		}

		notifyDone();
	}

	@Override
	public void onError(Throwable t) {
		Signal sig = Signal.error(t);

		int previousState = markTerminated();

		if (isMarkedTerminated(previousState) || isMarkedTerminating(previousState)) {
			this.protocolErrors.add(sig);
			return;
		}
		if (isMarkedOnNext(previousState)) {
			this.protocolErrors.add(sig);
			this.terminalSignal = sig;
			return; //isTerminating will be detected later, triggering the notifyDone()
		}

		this.terminalSignal = sig;

		if (fusionMode == Fuseable.ASYNC) {
			drainAsync(true);
			return;
		}

		notifyDone();
	}

	/**
	 * Drain the subscriber in asynchronous fusion mode (assumes there is a this.qs).
	 *
	 * @param isTerminal is the draining happening from onComplete/onError?
	 */
	void drainAsync(boolean isTerminal) {
		assert this.qs != null;

		//onComplete and onError move to terminated/terminating and call drainAsync ONLY if no work in progress
		int previousState = this.state;
		if (isTerminal && isMarkedOnNext(previousState)) {
			return;
		}

		if (isMarkedTerminated(previousState)) {
			safeClearQueue(qs);
			notifyDone();
			return;
		}

		T t;

		for (; ; ) {
			if (cancelled.get()) {
				safeClearQueue(qs);
				notifyDone();
				return;
			}

			long r = REQUESTED_TOTAL.get(this);
			if (r != Long.MAX_VALUE && r - this.receivedOnNext.size() < 1) {
				//no more request for data, until next request (or termination)
				if (checkTerminatedAfterOnNext()) {
					safeClearQueue(qs);
				}
				return;
			}

			t = qs.poll();
			if (t == null) {
				//no more available data, until next request (or termination)
				if (checkTerminatedAfterOnNext()) {
					safeClearQueue(qs);
				}
				return;
			}
			this.receivedOnNext.add(t);
		}
	}

	@Nullable
	@Override
	public Object scanUnsafe(Attr key) {
		if (key == Attr.TERMINATED) return terminalSignal != null || subscriptionFailure.get() != null;
		if (key == Attr.CANCELLED) return cancelled.get();
		if (key == Attr.ERROR) {
			Throwable subFailure = subscriptionFailure.get();
			Signal sig = terminalSignal;
			if (sig != null && sig.getThrowable() != null) {
				return sig.getThrowable();
			}
			else return subFailure; //simplified: ok to return null if subscriptionFailure holds null
		}
		if (key == Attr.PARENT) return s;
		if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC;
		if (key == Attr.REQUESTED_FROM_DOWNSTREAM) return REQUESTED_TOTAL.get(this);

		return null;
	}

	void upstreamRequest(Subscription s, long n) {
		long prev = Operators.addCap(REQUESTED_TOTAL, this, n);
		if (prev != Long.MAX_VALUE) {
			s.request(n);
		}
	}

	static final int MASK_TERMINATED       = 0b1000;
	static final int MASK_TERMINATING      = 0b0100;
	static final int MASK_ON_NEXT          = 0b0001;

	boolean checkTerminatedAfterOnNext() {
		int donePreviousState = markOnNextDone();
		if (isMarkedTerminating(donePreviousState)) {
			notifyDone();
			return true;
		}
		return false;
	}

	static boolean isMarkedTerminated(int state) {
		return (state & MASK_TERMINATED) == MASK_TERMINATED;
	}

	static boolean isMarkedOnNext(int state) {
		return (state & MASK_ON_NEXT) == MASK_ON_NEXT;
	}

	static boolean isMarkedTerminating(int state) {
		return (state & MASK_TERMINATING) == MASK_TERMINATING
				&& (state & MASK_TERMINATED) != MASK_TERMINATED;
	}

	/**
	 * Attempt to mark the TestSubscriber as terminated. Does nothing if already terminated.
	 * Mark as {@link #isMarkedTerminating(int)} if a concurrent onNext is detected.
	 *
	 * @return the previous state
	 */
	int markTerminated() {
		for(;;) {
			int state = this.state;
			if (isMarkedTerminated(state) || isMarkedTerminating(state)) {
				return state;
			}

			int newState;
			if (isMarkedOnNext(state)) {
				newState = state | MASK_TERMINATING;
			}
			else {
				newState = MASK_TERMINATED;
			}

			if (STATE.compareAndSet(this, state, newState)) {
				return state;
			}
		}
	}

	/**
	 * Mark that onNext processing has started (work in progress) and return the previous state.
	 * @return the previous state
	 */
	int markOnNextStart() {
		for(;;) {
			int state = this.state;
			if (state != 0) {
				return state;
			}

			if (STATE.compareAndSet(this, state, MASK_ON_NEXT)) {
				return state;
			}
		}
	}

	/**
	 * Mark that onNext processing has terminated and return the previous state.
	 * @return the previous state
	 */
	int markOnNextDone() {
		for(;;) {
			int state = this.state;
			int nextState = state & ~MASK_ON_NEXT;
			if (STATE.compareAndSet(this, state, nextState)) {
				return state;
			}
		}
	}

	// == public subscription-like methods

	@Override
	public void cancel() {
		if (cancelled.compareAndSet(false, true)) {
			if (this.s != null) {
				this.s.cancel();
			}

			if (requestedFusionMode == Fuseable.ASYNC) {
				int st = this.state;
				Fuseable.QueueSubscription q = this.qs;
				if (!isMarkedOnNext(st) && q != null) {
					q.clear();
				}
			}

			notifyDone();
		}
	}

	@Override
	public void request(long n) {
		if (this.s == null) {
			for (;;) {
				long prevReq = REQUESTED_PRE_SUBSCRIPTION.get(this);
				if (prevReq == -1L) {
					request(n); //will propagate upstream
					return;
				}
				long newReq = Operators.addCap(prevReq, n);
				if (REQUESTED_PRE_SUBSCRIPTION.compareAndSet(this, prevReq, newReq)) {
					return;
				}
			}
		}

		if (Operators.validate(n)) {
			if (this.fusionMode == Fuseable.SYNC) {
				internalCancel();
				throw new IllegalStateException("Request is short circuited in SYNC fusion mode, and should not be explicitly used");
			}
			upstreamRequest(this.s, n);
		}
	}

	void checkSubscriptionFailure() {
		AssertionError subscriptionFailure = this.subscriptionFailure.get();
		if (subscriptionFailure != null) {
			throw subscriptionFailure;
		}
	}

	// == public accessors

	@Override
	public boolean isTerminatedOrCancelled() {
		checkSubscriptionFailure();
		return doneLatch.getCount() == 0;
	}

	@Override
	public boolean isTerminated() {
		checkSubscriptionFailure();
		return terminalSignal != null;
	}

	@Override
	public boolean isTerminatedComplete() {
		checkSubscriptionFailure();
		Signal ts = this.terminalSignal;
		return ts != null && ts.isOnComplete();
	}

	@Override
	public boolean isTerminatedError() {
		checkSubscriptionFailure();
		Signal ts = this.terminalSignal;
		return ts != null && ts.isOnError();
	}

	@Override
	public boolean isCancelled() {
		checkSubscriptionFailure();
		return cancelled.get();
	}

	@Override
	@Nullable
	public Signal getTerminalSignal() {
		checkSubscriptionFailure();
		return this.terminalSignal;
	}

	@Override
	public Signal expectTerminalSignal() {
		checkSubscriptionFailure();
		Signal sig = this.terminalSignal;
		if (sig == null || (!sig.isOnError() && !sig.isOnComplete())) {
			cancel();
			throw new AssertionError("Expected subscriber to be terminated, but it has not been terminated yet: cancelling subscription.");
		}
		return sig;
	}

	@Override
	public Throwable expectTerminalError() {
		checkSubscriptionFailure();
		Signal sig = this.terminalSignal;
		if (sig == null) {
			cancel();
			throw new AssertionError("Expected subscriber to have errored, but it has not been terminated yet.");
		}
		if (sig.isOnComplete()) {
			throw new AssertionError("Expected subscriber to have errored, but it has completed instead.");
		}
		Throwable terminal = sig.getThrowable();
		if (terminal == null) {
			cancel();
			throw new AssertionError("Expected subscriber to have errored, got unexpected terminal signal <" + sig + ">.");
		}
		return terminal;
	}

	@Override
	public List getReceivedOnNext() {
		checkSubscriptionFailure();
		return new ArrayList<>(this.receivedOnNext);
	}

	@Override
	public List getReceivedOnNextAfterCancellation() {
		checkSubscriptionFailure();
		return new ArrayList<>(this.receivedPostCancellation);
	}

	@Override
	public List> getProtocolErrors() {
		checkSubscriptionFailure();
		return new ArrayList<>(this.protocolErrors);
	}

	@Override
	public int getFusionMode() {
		checkSubscriptionFailure();
		return this.fusionMode;
	}


	// == blocking and awaiting termination

	@Override
	public void block() {
		try {
			this.doneLatch.await();
			checkSubscriptionFailure();
		}
		catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new AssertionError("Block() interrupted", e);
		}
	}

	@Override
	public void block(Duration timeout) {
		long timeoutMs = timeout.toMillis();
		try {
			boolean done = this.doneLatch.await(timeoutMs, TimeUnit.MILLISECONDS);
			checkSubscriptionFailure();
			if (!done) {
				throw new AssertionError("TestSubscriber timed out, not terminated after " + timeout + " (" + timeoutMs + "ms)");
			}
		}
		catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new AssertionError("Block(" + timeout +") interrupted", e);
		}
	}
	//TODO should we add a method to await the latch without throwing ?
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy