/**
* Copyright (c) 2016-present, RxJava Contributors.
*
* 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 io.reactivex.internal.operators.completable;
import static org.junit.Assert.*;
import java.util.*;
import org.junit.Test;
import org.reactivestreams.Subscriber;
import io.reactivex.*;
import io.reactivex.disposables.*;
import io.reactivex.exceptions.*;
import io.reactivex.functions.Function;
import io.reactivex.internal.subscriptions.BooleanSubscription;
import io.reactivex.observers.TestObserver;
import io.reactivex.plugins.RxJavaPlugins;
import io.reactivex.processors.PublishProcessor;
import io.reactivex.schedulers.Schedulers;
public class CompletableMergeTest {
@Test
public void invalidPrefetch() {
try {
Completable.merge(Flowable.just(Completable.complete()), -99);
fail("Should have thrown IllegalArgumentExceptio");
} catch (IllegalArgumentException ex) {
assertEquals("maxConcurrency > 0 required but it was -99", ex.getMessage());
}
}
@Test
public void cancelAfterFirst() {
final TestObserver<Void> to = new TestObserver<Void>();
Completable.mergeArray(new Completable() {
@Override
protected void subscribeActual(CompletableObserver s) {
s.onSubscribe(Disposables.empty());
s.onComplete();
to.cancel();
}
}, Completable.complete())
.subscribe(to);
to.assertEmpty();
}
@Test
public void cancelAfterFirstDelayError() {
final TestObserver<Void> to = new TestObserver<Void>();
Completable.mergeArrayDelayError(new Completable() {
@Override
protected void subscribeActual(CompletableObserver s) {
s.onSubscribe(Disposables.empty());
s.onComplete();
to.cancel();
}
}, Completable.complete())
.subscribe(to);
to.assertEmpty();
}
@Test
public void onErrorAfterComplete() {
List<Throwable> errors = TestHelper.trackPluginErrors();
try {
final CompletableObserver[] co = { null };
Completable.mergeArrayDelayError(Completable.complete(), new Completable() {
@Override
protected void subscribeActual(CompletableObserver s) {
s.onSubscribe(Disposables.empty());
s.onComplete();
co[0] = s;
}
})
.test()
.assertResult();
co[0].onError(new TestException());
TestHelper.assertUndeliverable(errors, 0, TestException.class);
} finally {
RxJavaPlugins.reset();
}
}
@Test
public void completeAfterMain() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeArray(Completable.complete(), pp.ignoreElements())
.test();
pp.onComplete();
to.assertResult();
}
@Test
public void completeAfterMainDelayError() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeArrayDelayError(Completable.complete(), pp.ignoreElements())
.test();
pp.onComplete();
to.assertResult();
}
@Test
public void errorAfterMainDelayError() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeArrayDelayError(Completable.complete(), pp.ignoreElements())
.test();
pp.onError(new TestException());
to.assertFailure(TestException.class);
}
@Test
public void dispose() {
TestHelper.checkDisposed(Completable.merge(Flowable.just(Completable.complete())));
}
@Test
public void disposePropagates() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.merge(Flowable.just(pp.ignoreElements())).test();
assertTrue(pp.hasSubscribers());
to.cancel();
assertFalse(pp.hasSubscribers());
to.assertEmpty();
}
@Test
public void innerComplete() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.merge(Flowable.just(pp.ignoreElements())).test();
pp.onComplete();
to.assertResult();
}
@Test
public void innerError() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.merge(Flowable.just(pp.ignoreElements())).test();
pp.onError(new TestException());
to.assertFailure(TestException.class);
}
@Test
public void innerErrorDelayError() {
PublishProcessor<Integer> pp = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeDelayError(Flowable.just(pp.ignoreElements())).test();
pp.onError(new TestException());
to.assertFailure(TestException.class);
}
@Test
public void mainErrorInnerErrorRace() {
for (int i = 0; i < 500; i++) {
List<Throwable> errors = TestHelper.trackPluginErrors();
try {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.merge(pp1.map(new Function<Integer, Completable>() {
@Override
public Completable apply(Integer v) throws Exception {
return pp2.ignoreElements();
}
})).test();
pp1.onNext(1);
final Throwable ex1 = new TestException();
final Throwable ex2 = new TestException();
Runnable r1 = new Runnable() {
@Override
public void run() {
pp1.onError(ex1);
}
};
Runnable r2 = new Runnable() {
@Override
public void run() {
pp2.onError(ex2);
}
};
TestHelper.race(r1, r2, Schedulers.single());
Throwable ex = to.errors().get(0);
if (ex instanceof CompositeException) {
to.assertSubscribed().assertNoValues().assertNotComplete();
errors = TestHelper.compositeList(ex);
TestHelper.assertError(errors, 0, TestException.class);
TestHelper.assertError(errors, 1, TestException.class);
} else {
to.assertFailure(TestException.class);
if (!errors.isEmpty()) {
TestHelper.assertUndeliverable(errors, 0, TestException.class);
}
}
} finally {
RxJavaPlugins.reset();
}
}
}
@Test
public void mainErrorInnerErrorDelayedRace() {
for (int i = 0; i < 500; i++) {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeDelayError(pp1.map(new Function<Integer, Completable>() {
@Override
public Completable apply(Integer v) throws Exception {
return pp2.ignoreElements();
}
})).test();
pp1.onNext(1);
final Throwable ex1 = new TestException();
Runnable r1 = new Runnable() {
@Override
public void run() {
pp1.onError(ex1);
}
};
final Throwable ex2 = new TestException();
Runnable r2 = new Runnable() {
@Override
public void run() {
pp2.onError(ex2);
}
};
TestHelper.race(r1, r2, Schedulers.single());
to.assertFailure(CompositeException.class);
List<Throwable> errors = TestHelper.compositeList(to.errors().get(0));
TestHelper.assertError(errors, 0, TestException.class);
TestHelper.assertError(errors, 1, TestException.class);
}
}
@Test
public void maxConcurrencyOne() {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.merge(Flowable.just(pp1.ignoreElements(), pp2.ignoreElements()), 1)
.test();
assertTrue(pp1.hasSubscribers());
assertFalse(pp2.hasSubscribers());
pp1.onComplete();
assertTrue(pp2.hasSubscribers());
pp2.onComplete();
to.assertResult();
}
@Test
public void maxConcurrencyOneDelayError() {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeDelayError(Flowable.just(pp1.ignoreElements(), pp2.ignoreElements()), 1)
.test();
assertTrue(pp1.hasSubscribers());
assertFalse(pp2.hasSubscribers());
pp1.onComplete();
assertTrue(pp2.hasSubscribers());
pp2.onComplete();
to.assertResult();
}
@Test
public void maxConcurrencyOneDelayErrorFirst() {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeDelayError(Flowable.just(pp1.ignoreElements(), pp2.ignoreElements()), 1)
.test();
assertTrue(pp1.hasSubscribers());
assertFalse(pp2.hasSubscribers());
pp1.onError(new TestException());
assertTrue(pp2.hasSubscribers());
pp2.onComplete();
to.assertFailure(TestException.class);
}
@Test
public void maxConcurrencyOneDelayMainErrors() {
final PublishProcessor<PublishProcessor<Integer>> pp0 = PublishProcessor.create();
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeDelayError(
pp0.map(new Function<PublishProcessor<Integer>, Completable>() {
@Override
public Completable apply(PublishProcessor<Integer> v) throws Exception {
return v.ignoreElements();
}
}), 1)
.test();
pp0.onNext(pp1);
assertTrue(pp1.hasSubscribers());
assertFalse(pp2.hasSubscribers());
pp1.onComplete();
pp0.onNext(pp2);
pp0.onError(new TestException());
assertTrue(pp2.hasSubscribers());
pp2.onComplete();
to.assertFailure(TestException.class);
}
@Test
public void mainDoubleOnError() {
List<Throwable> errors = TestHelper.trackPluginErrors();
try {
Completable.mergeDelayError(new Flowable<Completable>() {
@Override
protected void subscribeActual(Subscriber<? super Completable> s) {
s.onSubscribe(new BooleanSubscription());
s.onNext(Completable.complete());
s.onError(new TestException("First"));
s.onError(new TestException("Second"));
}
})
.test()
.assertFailureAndMessage(TestException.class, "First");
TestHelper.assertUndeliverable(errors, 0, TestException.class, "Second");
} finally {
RxJavaPlugins.reset();
}
}
@Test
public void innerDoubleOnError() {
List<Throwable> errors = TestHelper.trackPluginErrors();
try {
final CompletableObserver[] o = { null };
Completable.mergeDelayError(Flowable.just(new Completable() {
@Override
protected void subscribeActual(CompletableObserver s) {
s.onSubscribe(Disposables.empty());
s.onError(new TestException("First"));
o[0] = s;
}
}))
.test()
.assertFailureAndMessage(TestException.class, "First");
o[0].onError(new TestException("Second"));
TestHelper.assertUndeliverable(errors, 0, TestException.class, "Second");
} finally {
RxJavaPlugins.reset();
}
}
@Test
public void innerIsDisposed() {
final TestObserver<Void> to = new TestObserver<Void>();
Completable.mergeDelayError(Flowable.just(new Completable() {
@Override
protected void subscribeActual(CompletableObserver s) {
s.onSubscribe(Disposables.empty());
assertFalse(((Disposable)s).isDisposed());
to.dispose();
assertTrue(((Disposable)s).isDisposed());
}
}))
.subscribe(to);
}
@Test
public void mergeArrayInnerErrorRace() {
for (int i = 0; i < 500; i++) {
List<Throwable> errors = TestHelper.trackPluginErrors();
try {
final PublishProcessor<Integer> pp1 = PublishProcessor.create();
final PublishProcessor<Integer> pp2 = PublishProcessor.create();
TestObserver<Void> to = Completable.mergeArray(pp1.ignoreElements(), pp2.ignoreElements()).test();
pp1.onNext(1);
final Throwable ex1 = new TestException();
final Throwable ex2 = new TestException();
Runnable r1 = new Runnable() {
@Override
public void run() {
pp1.onError(ex1);
}
};
Runnable r2 = new Runnable() {
@Override
public void run() {
pp2.onError(ex2);
}
};
TestHelper.race(r1, r2, Schedulers.single());
to.assertFailure(TestException.class);
if (!errors.isEmpty()) {
TestHelper.assertUndeliverable(errors, 0, TestException.class);
}
} finally {
RxJavaPlugins.reset();
}
}
}
@Test
public void delayErrorIterableCancel() {
Completable.mergeDelayError(Arrays.asList(Completable.complete()))
.test(true)
.assertEmpty();
}
@Test
public void delayErrorIterableCancelAfterHasNext() {
final TestObserver<Void> to = new TestObserver<Void>();
Completable.mergeDelayError(new Iterable<Completable>() {
@Override
public Iterator<Completable> iterator() {
return new Iterator<Completable>() {
@Override
public boolean hasNext() {
to.cancel();
return true;
}
@Override
public Completable next() {
return Completable.complete();
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
}
})
.subscribe(to);
to.assertEmpty();
}
@Test
public void delayErrorIterableCancelAfterNext() {
final TestObserver<Void> to = new TestObserver<Void>();
Completable.mergeDelayError(new Iterable<Completable>() {
@Override
public Iterator<Completable> iterator() {
return new Iterator<Completable>() {
@Override
public boolean hasNext() {
return true;
}
@Override
public Completable next() {
to.cancel();
return Completable.complete();
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
}
})
.subscribe(to);
to.assertEmpty();
}
}