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

reactor.test.publisher.ColdTestPublisher Maven / Gradle / Ivy

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

import java.util.ArrayList;
import java.util.Collections;
import java.util.EnumSet;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import java.util.stream.Stream;

import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import reactor.core.Exceptions;
import reactor.core.Fuseable;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.Operators;
import reactor.util.annotation.Nullable;

import static reactor.test.publisher.TestPublisher.Violation.ALLOW_NULL;
import static reactor.test.publisher.TestPublisher.Violation.DEFER_CANCELLATION;
import static reactor.test.publisher.TestPublisher.Violation.REQUEST_OVERFLOW;

/**
 * A cold implementation of a {@link TestPublisher}.
 *
 * @author Simon Basle
 * @author Stephane Maldini
 */
final class ColdTestPublisher extends TestPublisher {

	@SuppressWarnings("rawtypes")
	private static final ColdTestPublisherSubscription[] EMPTY = new ColdTestPublisherSubscription[0];
	@SuppressWarnings("rawtypes")
	private static final ColdTestPublisherSubscription[] TERMINATED = new ColdTestPublisherSubscription[0];

	final List values;

	boolean done;
	/** Non-null if either {@link #error(Throwable) or {@link #complete()}} have been called. */
	Throwable error;

	/** If true, emit an overflow error when there is more values than request. If false, buffer until data is requested. */
	final boolean errorOnOverflow;

	/** If misbehaving, the set of violations this publisher will exhibit. */
	final EnumSet violations;

	volatile boolean hasOverflown;
	volatile boolean wasRequested;

	@SuppressWarnings("unchecked")
	volatile ColdTestPublisherSubscription[] subscribers = EMPTY;
	static final AtomicReferenceFieldUpdater SUBSCRIBERS =
			AtomicReferenceFieldUpdater.newUpdater(ColdTestPublisher.class, ColdTestPublisherSubscription[].class, "subscribers");

	volatile int cancelCount;
	static final AtomicIntegerFieldUpdater CANCEL_COUNT =
			AtomicIntegerFieldUpdater.newUpdater(ColdTestPublisher.class, "cancelCount");

	volatile long subscribeCount;
	static final AtomicLongFieldUpdater SUBSCRIBED_COUNT =
			AtomicLongFieldUpdater.newUpdater(ColdTestPublisher.class, "subscribeCount");

	ColdTestPublisher(boolean errorOnOverflow, EnumSet violations) {
		this.errorOnOverflow = errorOnOverflow;
		this.values = Collections.synchronizedList(new ArrayList<>());
		this.violations = violations;
	}

	@Override
	public void subscribe(Subscriber s) {
		Objects.requireNonNull(s, "s");

		ColdTestPublisherSubscription p = new ColdTestPublisherSubscription<>(s, this);
		ColdTestPublisher.SUBSCRIBED_COUNT.incrementAndGet(this);

		if (!add(p)) {
			s.onSubscribe(p); // will trigger drain() via request()
			if (p.cancelled) {
				return;
			}
			p.drain(); // ensures that empty source terminal signal is propagated without waiting for a request from the subscriber
			return;
		}

		s.onSubscribe(p); // will trigger drain() via request()
		p.drain(); // ensures that empty source terminal signal is propagated without waiting for a request from the subscriber
	}

	boolean add(ColdTestPublisherSubscription s) {
		for (;;) {
			ColdTestPublisherSubscription[] a = subscribers;

			if (a == TERMINATED) {
				return false;
			}

			int len = a.length;

			@SuppressWarnings("unchecked")
			ColdTestPublisherSubscription[] b = new ColdTestPublisherSubscription[len + 1];
			System.arraycopy(a, 0, b, 0, len);
			b[len] = s;

			if (SUBSCRIBERS.compareAndSet(this, a, b)) {
				return true;
			}
		}
	}

	@SuppressWarnings("unchecked")
	void remove(ColdTestPublisherSubscription s) {
		ColdTestPublisherSubscription[] a = subscribers;

		if (a == EMPTY || a == TERMINATED) {
			return;
		}

		for (;;) {
			a = subscribers;
			if (a == EMPTY || a == TERMINATED) {
				return;
			}
			int len = a.length;

			int j = -1;

			for (int i = 0; i < len; i++) {
				if (a[i] == s) {
					j = i;
					break;
				}
			}
			if (j < 0) {
				return;
			}

			ColdTestPublisherSubscription[] b;
			if (len == 1) {
				b = EMPTY;
			}
			else {
				b = new ColdTestPublisherSubscription[len - 1];
				System.arraycopy(a, 0, b, 0, j);
				System.arraycopy(a, j + 1, b, j, len - j - 1);
			}

			if (SUBSCRIBERS.compareAndSet(this, a, b)) {
				return;
			}
		}
	}

	static final class ColdTestPublisherSubscription implements Subscription {

		final Subscriber                     actual;
		final Fuseable.ConditionalSubscriber actualConditional;

		final ColdTestPublisher parent;

		volatile boolean cancelled;

		volatile long requested;

		@SuppressWarnings("rawtypes")
		static final AtomicLongFieldUpdater REQUESTED =
				AtomicLongFieldUpdater.newUpdater(ColdTestPublisherSubscription.class, "requested");

		volatile long wip;

		@SuppressWarnings("rawtypes")
		static final AtomicLongFieldUpdater WIP =
				AtomicLongFieldUpdater.newUpdater(ColdTestPublisherSubscription.class, "wip");

		/** Where in the {@link ColdTestPublisher#values} buffer this subscription is at. */
		int index;


		@SuppressWarnings("unchecked")
		ColdTestPublisherSubscription(Subscriber actual, ColdTestPublisher parent) {
			this.actual = actual;
			if(actual instanceof Fuseable.ConditionalSubscriber){
				this.actualConditional = (Fuseable.ConditionalSubscriber) actual;
			}
			else {
				this.actualConditional = null;
			}
			this.parent = parent;
		}

		@Override
		public void request(long n) {
			if (Operators.validate(n)) {
				if (Operators.addCap(REQUESTED, this, n) == 0) {
					parent.wasRequested = true;
				}
				drain();
			}
		}

		@Override
		public void cancel() {
			if (!cancelled) {
				ColdTestPublisher.CANCEL_COUNT.incrementAndGet(parent);
				if (parent.violations.contains(DEFER_CANCELLATION) || parent.violations.contains(REQUEST_OVERFLOW)) {
					return;
				}
				cancelled = true;
				parent.remove(this);
			}
		}

		private void drain() {
			if (WIP.getAndIncrement(this) > 0) {
				return;
			}
			for (; ; ) {
				int i = index;
				// Re-read the volatile 'requested' which could have grown via another thread
				long r = requested;
				int emitted = 0;
				if (cancelled) {
					return;
				}

				// This list can only grow while we're in drain(), so no risk in get(i) being out of bounds
				int s = parent.values.size();
				while (i < s) {
					if (emitted == r && !parent.violations.contains(REQUEST_OVERFLOW)) {
						break;
					}
					T t = parent.values.get(i);
					if (t == null && !parent.violations.contains(ALLOW_NULL)) {
						parent.remove(this);
						actual.onError(new NullPointerException("The " + i + "th element was null"));
						return;
					}

					//emit and increase count, potentially using conditional subscriber
					if (actualConditional != null) {
						//noinspection ConstantConditions
						if (actualConditional.tryOnNext(t)) {
							emitted++;
						}
					} else {
						//noinspection ConstantConditions
						actual.onNext(t);
						emitted++;
					}
					i++;
					if (cancelled) {
						return;
					}
				}

				index = i;
				boolean hasMoreData = i < s;
				boolean hasMoreRequest;
				if (emitted > r) { //we did clearly overflow
					assert parent.violations.contains(REQUEST_OVERFLOW);
					parent.hasOverflown = true;
				}
				//let's update the REQUESTED unless we're in fastpath
				if (r != Long.MAX_VALUE) {
					hasMoreRequest = REQUESTED.addAndGet(this, -emitted) > 0;
				}
				else {
					hasMoreRequest = true;
				}

				//let's exit early if we've transmitted the whole buffer and there's a terminal signal
				if (i == s && emitTerminalSignalIfAny()) {
					return;
				}

				//the only remaining early exit condition is if we're in slowpath and we've emitted
				//all the requested amount but there's still values in the buffer...
				//if the parent is configured to errorOnOverflow then we must terminate
				if (hasMoreData && !hasMoreRequest && parent.errorOnOverflow) {
					parent.remove(this);
					actual.onError(Exceptions.failWithOverflow("Can't deliver value due to lack of requests"));
					return;
				}

				//in all other cases, let's loop again in case of additional work, exit otherwise
				if (WIP.decrementAndGet(this) == 0) {
					return;
				}
			}
		}

		/**
		 * Attempt to terminate the subscriber if the publisher was marked as terminated.
		 * Note that if that is not the case, it is important to continue the drain loop
		 * since otherwise no downstream signal is going to be pushed yet we'd exit the loop early.
		 *
		 * @return true if the TestPublisher was terminated, false otherwise
		 */
		private boolean emitTerminalSignalIfAny() {
			if (parent.done && this.parent.values.size() == index) {
				parent.remove(this);

				final Throwable t = parent.error;
				if (t != null) {
					actual.onError(parent.error);
				}
				else {
					actual.onComplete();
				}
				return true;
			}
			return false;
		}
	}

	@Override
	public Flux flux() {
		return Flux.from(this);
	}

	@Override
	public boolean wasSubscribed() {
		return subscribeCount > 0;
	}

	@Override
	public long subscribeCount() {
		return subscribeCount;
	}

	@Override
	public boolean wasCancelled() {
		return cancelCount > 0;
	}

	@Override
	public boolean wasRequested() {
		return wasRequested;
	}

	@Override
	public Mono mono() {
		return Mono.from(this);
	}

	@Override
	public ColdTestPublisher assertMinRequested(long n) {
		ColdTestPublisherSubscription[] subs = subscribers;
		long minRequest = Stream.of(subs)
				.mapToLong(s -> s.requested)
				.min()
				.orElse(0);
		if (minRequest < n) {
			throw new AssertionError("Expected smallest requested amount to be >= " + n + "; got " + minRequest);
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertMaxRequested(long n) {
		ColdTestPublisherSubscription[] subs = subscribers;
		long maxRequest = Stream.of(subs)
				.mapToLong(s -> s.requested)
				.max()
				.orElse(0);
		if (maxRequest > n) {
			throw new AssertionError("Expected largest requested amount to be <= " + n + "; got " + maxRequest);
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertSubscribers() {
		ColdTestPublisherSubscription[] s = subscribers;
		if (s == EMPTY) {
			throw new AssertionError("Expected subscribers");
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertSubscribers(int n) {
		int sl = subscribers.length;
		if (sl != n) {
			throw new AssertionError("Expected " + n + " subscribers, got " + sl);
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertNoSubscribers() {
		int sl = subscribers.length;
		if (sl != 0) {
			throw new AssertionError("Expected no subscribers, got " + sl);
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertCancelled() {
		if (cancelCount == 0) {
			throw new AssertionError("Expected at least 1 cancellation");
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertCancelled(int n) {
		int cc = cancelCount;
		if (cc != n) {
			throw new AssertionError("Expected " + n + " cancellations, got " + cc);
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertNotCancelled() {
		if (cancelCount != 0) {
			throw new AssertionError("Expected no cancellation");
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertRequestOverflow() {
		if (!hasOverflown) {
			throw new AssertionError("Expected some request overflow");
		}
		return this;
	}

	@Override
	public ColdTestPublisher assertNoRequestOverflow() {
		if (hasOverflown) {
			throw new AssertionError("Unexpected request overflow");
		}
		return this;
	}

	@Override
	public ColdTestPublisher next(@Nullable T t) {
		if (!violations.contains(ALLOW_NULL)) {
			Objects.requireNonNull(t, "emitted values must be non-null");
		}

		values.add(t);
		for (ColdTestPublisherSubscription s : subscribers) {
			s.drain();
		}

		return this;
	}

	@Override
	public ColdTestPublisher error(Throwable t) {
		Objects.requireNonNull(t, "t");

		error = t;
		done = true;
		final ColdTestPublisherSubscription[] subs = SUBSCRIBERS.getAndSet(this, TERMINATED);
		for (ColdTestPublisherSubscription s : subs) {
			s.drain();
		}
		return this;
	}

	@Override
	public ColdTestPublisher complete() {
		done = true;
		error = null;
		final ColdTestPublisherSubscription[] subs = SUBSCRIBERS.getAndSet(this, TERMINATED);
		for (ColdTestPublisherSubscription s : subs) {
			s.drain();
		}
		return this;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy