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

hu.akarnokd.rxjava2.subscribers.TestSubscriber Maven / Gradle / Ivy

/**
 * Copyright 2015 David Karnok and Netflix, Inc.
 * 
 * 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
 * 
 * http://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 hu.akarnokd.rxjava2.subscribers;

import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.*;

import org.reactivestreams.*;

import hu.akarnokd.rxjava2.Notification;
import hu.akarnokd.rxjava2.disposables.Disposable;
import hu.akarnokd.rxjava2.exceptions.CompositeException;
import hu.akarnokd.rxjava2.internal.functions.Objects;
import hu.akarnokd.rxjava2.internal.subscribers.EmptySubscriber;
import hu.akarnokd.rxjava2.internal.subscriptions.SubscriptionHelper;
import hu.akarnokd.rxjava2.internal.util.BackpressureHelper;

/**
 * A subscriber that records events and allows making assertions about them.
 *
 * 

You can override the onSubscribe, onNext, onError, onComplete, request and * cancel methods but not the others (this is by desing). * *

The TestSubscriber implements Disposable for convenience where dispose calls cancel. * *

When calling the default request method, you are requesting on behalf of the * wrapped actual subscriber. * * @param the value type */ public class TestSubscriber implements Subscriber, Subscription, Disposable { /** The actual subscriber to forward events to. */ private final Subscriber actual; /** The initial request amount if not null. */ private final Long initialRequest; /** The latch that indicates an onError or onCompleted has been called. */ private final CountDownLatch done; /** The list of values received. */ private final List values; /** The list of errors received. */ private final List errors; /** The number of completions. */ private long completions; /** The last thread seen by the subscriber. */ private Thread lastThread; /** Makes sure the incoming Subscriptions get cancelled immediately. */ private volatile boolean cancelled; /** Holds the current subscription if any. */ private volatile Subscription subscription; /** Updater for subscription. */ @SuppressWarnings("rawtypes") private static final AtomicReferenceFieldUpdater SUBSCRIPTION = AtomicReferenceFieldUpdater.newUpdater(TestSubscriber.class, Subscription.class, "subscription"); /** Holds the requested amount until a subscription arrives. */ @SuppressWarnings("unused") private volatile long missedRequested; /** Updater for subscription. */ @SuppressWarnings("rawtypes") private static final AtomicLongFieldUpdater MISSED_REQUESTED = AtomicLongFieldUpdater.newUpdater(TestSubscriber.class, "missedRequested"); /** Indicates a cancelled subscription. */ private static final Subscription CANCELLED = new Subscription() { @Override public void request(long n) { } @Override public void cancel() { } }; /** * Constructs a non-forwarding TestSubscriber with an initial request value of Long.MAX_VALUE. */ public TestSubscriber() { this(EmptySubscriber.INSTANCE_NOERROR, Long.MAX_VALUE); } /** * Constructs a non-forwarding TestSubscriber with the specified initial request value. *

The TestSubscriber doesn't validate the initialRequest value so one can * test sources with invalid values as well. * @param initialRequest the initial request value if not null */ public TestSubscriber(Long initialRequest) { this(EmptySubscriber.INSTANCE_NOERROR, initialRequest); } /** * Constructs a forwarding TestSubscriber but leaves the requesting to the wrapped subscriber. * @param actual the actual Subscriber to forward events to */ public TestSubscriber(Subscriber actual) { this(actual, null); } /** * Constructs a forwarding TestSubscriber with the specified initial request value. *

The TestSubscriber doesn't validate the initialRequest value so one can * test sources with invalid values as well. * @param actual the actual Subscriber to forward events to * @param initialRequest the initial request value if not null */ public TestSubscriber(Subscriber actual, Long initialRequest) { this.actual = actual; this.initialRequest = initialRequest; this.values = new ArrayList(); this.errors = new ArrayList(); this.done = new CountDownLatch(1); } @Override public void onSubscribe(Subscription s) { lastThread = Thread.currentThread(); if (s == null) { errors.add(new NullPointerException("onSubscribe received a null Subscription")); return; } if (!SUBSCRIPTION.compareAndSet(this, null, s)) { s.cancel(); if (subscription != CANCELLED) { errors.add(new NullPointerException("onSubscribe received multiple subscriptions: " + s)); } return; } if (cancelled) { s.cancel(); } actual.onSubscribe(s); if (cancelled) { return; } if (initialRequest != null) { s.request(initialRequest); } long mr = MISSED_REQUESTED.getAndSet(this, 0L); if (mr != 0L) { s.request(mr); } } @Override public void onNext(T t) { lastThread = Thread.currentThread(); values.add(t); if (t == null) { errors.add(new NullPointerException("onNext received a null Subscription")); } actual.onNext(t); } @Override public void onError(Throwable t) { try { lastThread = Thread.currentThread(); errors.add(t); if (t == null) { errors.add(new NullPointerException("onError received a null Subscription")); } actual.onError(t); } finally { done.countDown(); } } @Override public void onComplete() { try { lastThread = Thread.currentThread(); completions++; actual.onComplete(); } finally { done.countDown(); } } @Override public void request(long n) { if (SubscriptionHelper.validateRequest(n)) { return; } Subscription s = subscription; if (s != null) { s.request(n); } else { BackpressureHelper.add(MISSED_REQUESTED, this, n); s = subscription; if (s != null) { long mr = MISSED_REQUESTED.getAndSet(this, 0L); if (mr != 0L) { s.request(mr); } } } } @Override public void cancel() { if (!cancelled) { cancelled = true; Subscription s = subscription; if (s != CANCELLED) { s = SUBSCRIPTION.getAndSet(this, CANCELLED); if (s != CANCELLED && s != null) { s.cancel(); } } } } /** * Returns true if this TestSubscriber has been cancelled. * @return true if this TestSubscriber has been cancelled */ public final boolean isCancelled() { return cancelled; } @Override public final void dispose() { cancel(); } // state retrieval methods /** * Returns the last thread which called the onXXX methods of this TestSubscriber. * @return the last thread which called the onXXX methods */ public final Thread lastThread() { return lastThread; } /** * Returns a shared list of received onNext values. * @return a list of received onNext values */ public final List values() { return values; } /** * Returns a shared list of received onError exceptions. * @return a list of received events onError exceptions */ public final List errors() { return errors; } /** * Returns the number of times onComplete was called. * @return the number of times onComplete was called */ public final long completions() { return completions; } /** * Returns true if TestSubscriber received any onError or onComplete events. * @return true if TestSubscriber received any onError or onComplete events */ public final boolean isTerminated() { return done.getCount() == 0; } /** * Returns the number of onNext values received. * @return the number of onNext values received */ public final int valueCount() { return values.size(); } /** * Returns the number of onError exceptions received. * @return the number of onError exceptions received */ public final int errorCount() { return errors.size(); } /** * Returns true if this TestSubscriber received a subscription. * @return true if this TestSubscriber received a subscription */ public final boolean hasSubscription() { return subscription != null; } /** * Awaits until this TestSubscriber receives an onError or onComplete events. * @throws InterruptedException if the current thread is interrupted while waiting * @see #awaitTerminalEvent() */ public final void await() throws InterruptedException { if (done.getCount() == 0) { return; } done.await(); } /** * Awaits the specified amount of time or until this TestSubscriber * receives an onError or onComplete events, whichever happens first. * @param time the waiting time * @param unit the time unit of the waiting time * @return true if the TestSubscriber terminated, false if timeout happened * @throws InterruptedException if the current thread is interrupted while waiting * @see #awaitTerminalEvent(long, TimeUnit) */ public final boolean await(long time, TimeUnit unit) throws InterruptedException { if (done.getCount() == 0) { return true; } return done.await(time, unit); } // assertion methods /** * Fail with the given message and add the sequence of errors as suppressed ones. *

Note this is delibarately the only fail method. Most of the times an assertion * would fail but it is possible it was due to an exception somewhere. This construct * will capture those potential errors and report it along with the original failure. * * @param message the message to use * @param errors the sequence of errors to add as suppressed exception */ private void fail(String prefix, String message, Iterable errors) { AssertionError ae = new AssertionError(prefix + message); CompositeException ce = new CompositeException(); for (Throwable e : errors) { if (e == null) { ce.suppress(new NullPointerException("Throwable was null!")); } else { ce.suppress(e); } }; if (!ce.isEmpty()) { ae.initCause(ce); } throw ae; } /** * Assert that this TestSubscriber received exactly one onComplete event. */ public void assertComplete() { String prefix = ""; /* * This creates a happens-before relation with the possible completion of the TestSubscriber. * Don't move it after the instance reads or into fail()! */ if (done.getCount() != 0) { prefix = "Subscriber still running! "; } long c = completions; if (c == 0) { fail(prefix, "Not completed", errors); } else if (c > 1) { fail(prefix, "Multiple completions: " + c, errors); } } /** * Assert that this TestSubscriber has not received any onComplete event. */ public void assertNotComplete() { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } long c = completions; if (c == 1) { fail(prefix, "Completed!", errors); } else if (c > 1) { fail(prefix, "Multiple completions: " + c, errors); } } /** * Assert that this TestSubscriber has not received any onError event. */ public void assertNoErrors() { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = errors.size(); if (s != 0) { fail(prefix, "Error(s) present: " + errors, errors); } } /** * Assert that this TestSubscriber received exactly the specified onError event value. * *

The comparison is performed via Objects.equals(); since most exceptions don't * implement equals(), this assertion may fail. Use the {@link #assertError(Class)} * overload to test against the class of an error instead of an instance of an error. * @param error the error to check * @see #assertError(Class) */ public void assertError(Throwable error) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = errors.size(); if (s == 0) { fail(prefix, "No errors", Collections.emptyList()); } if (errors.contains(error)) { if (s != 1) { fail(prefix, "Error present but other errors as well", errors); } } else { fail(prefix, "Error not present", errors); } } /** * Asserts that this TestSubscriber received exactly one onError event which is an * instance of the specified errorClass class. * @param errorClass the error class to expect */ public void assertError(Class errorClass) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = errors.size(); if (s == 0) { fail(prefix, "No errors", Collections.emptyList()); } boolean found = false; for (Throwable e : errors) { if (errorClass.isInstance(e)) { found = true; break; } } if (found) { if (s != 1) { fail(prefix, "Error present but other errors as well", errors); } } else { fail(prefix, "Error not present", errors); } } /** * Assert that this TestSubscriber received exactly one onNext value which is equal to * the given value with respect to Objects.equals. * @param value the value to expect */ public final void assertValue(T value) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = values.size(); if (s != 1) { fail(prefix, "Expected: " + valueAndClass(value) + ", Actual: " + values, errors); } T v = values.get(0); if (!Objects.equals(value, v)) { fail(prefix, "Expected: " + valueAndClass(value) + ", Actual: " + valueAndClass(v), errors); } } /** Appends the class name to a non-null value. */ static String valueAndClass(Object o) { if (o != null) { return o + " (class: " + o.getClass().getSimpleName() + ")"; } return "null"; } /** * Assert that this TestSubscriber received the specified number onNext events. * @param count the expected number of onNext events */ public final void assertValueCount(int count) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = values.size(); if (s != count) { fail(prefix, "Value counts differ; Expected: " + count + ", Actual: " + s, errors); } } /** * Assert that this TestSubscriber has not received any onNext events. */ public final void assertNoValues() { assertValueCount(0); } /** * Assert that the TestSubscriber received only the specified values in the specified order. * @param values the values expected * @see #assertValueSet(Collection) */ public final void assertValues(T... values) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = this.values.size(); if (s != values.length) { fail(prefix, "Value count differs; Expected: " + values.length + " " + Arrays.toString(values) + ", Actual: " + s + " " + this.values, errors); } for (int i = 0; i < s; i++) { T v = this.values.get(i); T u = values[i]; if (!Objects.equals(u, v)) { fail(prefix, "Values at position " + i + " differ; Expected: " + valueAndClass(u) + ", Actual: " + valueAndClass(v), errors); } } } /** * Assert that the TestSubscriber received only the specified values in any order. *

This helps asserting when the order of the values is not guaranteed, i.e., when merging * asynchronous streams. * * @param values the collection of values expected in any order */ public final void assertValueSet(Collection values) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = this.values.size(); if (s != values.size()) { fail(prefix, "Value count differs; Expected: " + values.size() + " " + values + ", Actual: " + s + " " + this.values, errors); } for (int i = 0; i < s; i++) { T v = this.values.get(i); if (!values.contains(v)) { fail(prefix, "Value not in the expected collection: " + valueAndClass(v), errors); } } } /** * Assert that the TestSubscriber received only the specified sequence of values in the same order. * @param sequence the sequence of expected values in order */ public final void assertValueSequence(Iterable sequence) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int i = 0; Iterator vit = values.iterator(); Iterator it = sequence.iterator(); boolean itNext = false; boolean vitNext = false; while ((itNext = it.hasNext()) && (vitNext = vit.hasNext())) { T v = it.next(); T u = vit.next(); if (!Objects.equals(u, v)) { fail(prefix, "Values at position " + i + " differ; Expected: " + valueAndClass(u) + ", Actual: " + valueAndClass(v), errors); } i++; } if (itNext && !vitNext) { fail(prefix, "More values received than expected (" + i + ")", errors); } if (!itNext && !vitNext) { fail(prefix, "Fever values received than expected (" + i + ")", errors); } } /** * Assert that the TestSubscriber terminated (i.e., the terminal latch reached zero). */ public final void assertTerminated() { if (done.getCount() != 0) { fail("", "Subscriber still running!", errors); } long c = completions; if (c > 1) { fail("", "Terminated with multiple completions: " + c, errors); } int s = errors.size(); if (s > 1) { fail("", "Terminated with multiple errors: " + s, errors); } if (c != 0 && s != 0) { fail("", "Terminated with multiple completions and errors: " + c, errors); } } /** * Assert that the TestSubscriber has not terminated (i.e., the terminal latch is still non-zero). */ public final void assertNotTerminated() { if (done.getCount() == 0) { fail("", "Subscriber terminated!", errors); } } /** * Assert that the onSubscribe method was called exactly once. */ public final void assertSubscribed() { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } if (subscription == null) { fail(prefix, "Not subscribed!", errors); } } /** * Assert that the onSubscribe method hasn't been called at all. */ public final void assertNotSubscribed() { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } if (subscription != null) { fail(prefix, "Subscribed!", errors); } else if (!errors.isEmpty()) { fail(prefix, "Not subscribed but errors found", errors); } } /** * Waits until the any terminal event has been received by this TestSubscriber * or returns false if the wait has been interrupted. * @return true if the TestSubscriber terminated, false if the wait has been interrupted */ public boolean awaitTerminalEvent() { try { await(); return true; } catch (InterruptedException ex) { Thread.currentThread().interrupt(); return false; } } /** * Awaits the specified amount of time or until this TestSubscriber * receives an onError or onComplete events, whichever happens first. * @param duration the waiting time * @param unit the time unit of the waiting time * @return true if the TestSubscriber terminated, false if timeout or interrupt happened */ public boolean awaitTerminalEvent(long duration, TimeUnit unit) { try { return await(duration, unit); } catch (InterruptedException ex) { Thread.currentThread().interrupt(); return false; } } public void assertErrorMessage(String message) { String prefix = ""; if (done.getCount() != 0) { prefix = "Subscriber still running! "; } int s = errors.size(); if (s == 0) { fail(prefix, "No errors", Collections.emptyList()); } else if (s == 1) { Throwable e = errors.get(0); if (e == null) { fail(prefix, "Error is null", Collections.emptyList()); } String errorMessage = e.getMessage(); if (!Objects.equals(message, errorMessage)) { fail(prefix, "Error message differs; Expected: " + message + ", Actual: " + errorMessage, Collections.singletonList(e)); } } else { fail(prefix, "Multiple errors", errors); } } /** * Returns a list of 3 other lists: the first inner list contains the plain * values received; the second list contains the potential errors * and the final list contains the potential completions as Notifications. * * @return a list of (values, errors, completion-notifications) */ @SuppressWarnings({ "rawtypes", "unchecked" }) public List> getEvents() { List> result = new ArrayList>(); result.add((List)values()); result.add((List)errors()); List completeList = new ArrayList(); for (long i = 0; i < completions; i++) { completeList.add(Notification.complete()); } result.add(completeList); return result; } }