/** * 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.flowable; import static org.junit.Assert.*; import static org.mockito.ArgumentMatchers.any; import static org.mockito.Mockito.*; import java.util.*; import java.util.concurrent.atomic.*; import org.junit.Test; import org.reactivestreams.Subscriber; import io.reactivex.*; import io.reactivex.functions.*; import io.reactivex.internal.functions.Functions; import io.reactivex.internal.fuseable.QueueDisposable; import io.reactivex.subscribers.*; public class FlowableRangeLongTest { @Test public void testRangeStartAt2Count3() { Subscriber<Long> observer = TestHelper.mockSubscriber(); Flowable.rangeLong(2, 3).subscribe(observer); verify(observer, times(1)).onNext(2L); verify(observer, times(1)).onNext(3L); verify(observer, times(1)).onNext(4L); verify(observer, never()).onNext(5L); verify(observer, never()).onError(any(Throwable.class)); verify(observer, times(1)).onComplete(); } @Test public void testRangeUnsubscribe() { Subscriber<Long> observer = TestHelper.mockSubscriber(); final AtomicInteger count = new AtomicInteger(); Flowable.rangeLong(1, 1000).doOnNext(new Consumer<Long>() { @Override public void accept(Long t1) { count.incrementAndGet(); } }) .take(3).subscribe(observer); verify(observer, times(1)).onNext(1L); verify(observer, times(1)).onNext(2L); verify(observer, times(1)).onNext(3L); verify(observer, never()).onNext(4L); verify(observer, never()).onError(any(Throwable.class)); verify(observer, times(1)).onComplete(); assertEquals(3, count.get()); } @Test public void testRangeWithZero() { Flowable.rangeLong(1, 0); } @Test public void testRangeWithOverflow2() { Flowable.rangeLong(Long.MAX_VALUE, 0); } @Test public void testRangeWithOverflow3() { Flowable.rangeLong(1, Long.MAX_VALUE); } @Test(expected = IllegalArgumentException.class) public void testRangeWithOverflow4() { Flowable.rangeLong(2, Long.MAX_VALUE); } @Test public void testRangeWithOverflow5() { assertFalse(Flowable.rangeLong(Long.MIN_VALUE, 0).blockingIterable().iterator().hasNext()); } @Test public void testBackpressureViaRequest() { Flowable<Long> o = Flowable.rangeLong(1, Flowable.bufferSize()); TestSubscriber<Long> ts = new TestSubscriber<Long>(0L); ts.assertNoValues(); ts.request(1); o.subscribe(ts); ts.assertValue(1L); ts.request(2); ts.assertValues(1L, 2L, 3L); ts.request(3); ts.assertValues(1L, 2L, 3L, 4L, 5L, 6L); ts.request(Flowable.bufferSize()); ts.assertTerminated(); } @Test public void testNoBackpressure() { ArrayList<Long> list = new ArrayList<Long>(Flowable.bufferSize() * 2); for (long i = 1; i <= Flowable.bufferSize() * 2 + 1; i++) { list.add(i); } Flowable<Long> o = Flowable.rangeLong(1, list.size()); TestSubscriber<Long> ts = new TestSubscriber<Long>(0L); ts.assertNoValues(); ts.request(Long.MAX_VALUE); // infinite o.subscribe(ts); ts.assertValueSequence(list); ts.assertTerminated(); } void testWithBackpressureOneByOne(long start) { Flowable<Long> source = Flowable.rangeLong(start, 100); TestSubscriber<Long> ts = new TestSubscriber<Long>(0L); ts.request(1); source.subscribe(ts); List<Long> list = new ArrayList<Long>(100); for (long i = 0; i < 100; i++) { list.add(i + start); ts.request(1); } ts.assertValueSequence(list); ts.assertTerminated(); } void testWithBackpressureAllAtOnce(long start) { Flowable<Long> source = Flowable.rangeLong(start, 100); TestSubscriber<Long> ts = new TestSubscriber<Long>(0L); ts.request(100); source.subscribe(ts); List<Long> list = new ArrayList<Long>(100); for (long i = 0; i < 100; i++) { list.add(i + start); } ts.assertValueSequence(list); ts.assertTerminated(); } @Test public void testWithBackpressure1() { for (long i = 0; i < 100; i++) { testWithBackpressureOneByOne(i); } } @Test public void testWithBackpressureAllAtOnce() { for (long i = 0; i < 100; i++) { testWithBackpressureAllAtOnce(i); } } @Test public void testWithBackpressureRequestWayMore() { Flowable<Long> source = Flowable.rangeLong(50, 100); TestSubscriber<Long> ts = new TestSubscriber<Long>(0L); ts.request(150); source.subscribe(ts); List<Long> list = new ArrayList<Long>(100); for (long i = 0; i < 100; i++) { list.add(i + 50); } ts.request(50); // and then some ts.assertValueSequence(list); ts.assertTerminated(); } @Test public void testRequestOverflow() { final AtomicInteger count = new AtomicInteger(); int n = 10; Flowable.rangeLong(1, n).subscribe(new DefaultSubscriber<Long>() { @Override public void onStart() { request(2); } @Override public void onComplete() { //do nothing } @Override public void onError(Throwable e) { throw new RuntimeException(e); } @Override public void onNext(Long t) { count.incrementAndGet(); request(Long.MAX_VALUE - 1); }}); assertEquals(n, count.get()); } @Test public void testEmptyRangeSendsOnCompleteEagerlyWithRequestZero() { final AtomicBoolean completed = new AtomicBoolean(false); Flowable.rangeLong(1, 0).subscribe(new DefaultSubscriber<Long>() { @Override public void onStart() { // request(0); } @Override public void onComplete() { completed.set(true); } @Override public void onError(Throwable e) { } @Override public void onNext(Long t) { }}); assertTrue(completed.get()); } @Test(timeout = 1000) public void testNearMaxValueWithoutBackpressure() { TestSubscriber<Long> ts = new TestSubscriber<Long>(); Flowable.rangeLong(Long.MAX_VALUE - 1L, 2L).subscribe(ts); ts.assertComplete(); ts.assertNoErrors(); ts.assertValues(Long.MAX_VALUE - 1L, Long.MAX_VALUE); } @Test(timeout = 1000) public void testNearMaxValueWithBackpressure() { TestSubscriber<Long> ts = new TestSubscriber<Long>(3L); Flowable.rangeLong(Long.MAX_VALUE - 1L, 2L).subscribe(ts); ts.assertComplete(); ts.assertNoErrors(); ts.assertValues(Long.MAX_VALUE - 1L, Long.MAX_VALUE); } @Test public void negativeCount() { try { Flowable.rangeLong(1L, -1L); fail("Should have thrown IllegalArgumentException"); } catch (IllegalArgumentException ex) { assertEquals("count >= 0 required but it was -1", ex.getMessage()); } } @Test public void countOne() { Flowable.rangeLong(5495454L, 1L) .test() .assertResult(5495454L); } @Test public void fused() { TestSubscriber<Long> to = SubscriberFusion.newTest(QueueDisposable.ANY); Flowable.rangeLong(1, 2).subscribe(to); SubscriberFusion.assertFusion(to, QueueDisposable.SYNC) .assertResult(1L, 2L); } @Test public void fusedReject() { TestSubscriber<Long> to = SubscriberFusion.newTest(QueueDisposable.ASYNC); Flowable.rangeLong(1, 2).subscribe(to); SubscriberFusion.assertFusion(to, QueueDisposable.NONE) .assertResult(1L, 2L); } @Test public void disposed() { TestHelper.checkDisposed(Flowable.rangeLong(1, 2)); } @Test public void fusedClearIsEmpty() { TestHelper.checkFusedIsEmptyClear(Flowable.rangeLong(1, 2)); } @Test public void noOverflow() { Flowable.rangeLong(Long.MAX_VALUE - 1, 2); Flowable.rangeLong(Long.MIN_VALUE, 2); Flowable.rangeLong(Long.MIN_VALUE, Long.MAX_VALUE); } @Test public void conditionalNormal() { Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .test() .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void badRequest() { TestHelper.assertBadRequestReported(Flowable.rangeLong(1L, 5L)); TestHelper.assertBadRequestReported(Flowable.rangeLong(1L, 5L).filter(Functions.alwaysTrue())); } @Test public void conditionalNormalSlowpath() { Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .test(5) .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void conditionalSlowPathTakeExact() { Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .take(5) .test() .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void slowPathTakeExact() { Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .take(5) .test() .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void conditionalSlowPathRebatch() { Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .rebatchRequests(1) .test() .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void slowPathRebatch() { Flowable.rangeLong(1L, 5L) .rebatchRequests(1) .test() .assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void slowPathCancel() { TestSubscriber<Long> ts = new TestSubscriber<Long>(2L) { @Override public void onNext(Long t) { super.onNext(t); cancel(); onComplete(); } }; Flowable.rangeLong(1L, 5L) .subscribe(ts); ts.assertResult(1L); } @Test public void fastPathCancel() { TestSubscriber<Long> ts = new TestSubscriber<Long>() { @Override public void onNext(Long t) { super.onNext(t); cancel(); onComplete(); } }; Flowable.rangeLong(1L, 5L) .subscribe(ts); ts.assertResult(1L); } @Test public void conditionalSlowPathCancel() { TestSubscriber<Long> ts = new TestSubscriber<Long>(1L) { @Override public void onNext(Long t) { super.onNext(t); cancel(); onComplete(); } }; Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .subscribe(ts); ts.assertResult(1L); } @Test public void conditionalFastPathCancel() { TestSubscriber<Long> ts = new TestSubscriber<Long>() { @Override public void onNext(Long t) { super.onNext(t); cancel(); onComplete(); } }; Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .subscribe(ts); ts.assertResult(1L); } @Test public void conditionalRequestOneByOne() { TestSubscriber<Long> ts = new TestSubscriber<Long>(1L) { @Override public void onNext(Long t) { super.onNext(t); request(1); } }; Flowable.rangeLong(1L, 5L) .filter(new Predicate<Long>() { @Override public boolean test(Long v) throws Exception { return v % 2 == 0; } }) .subscribe(ts); ts.assertResult(2L, 4L); } @Test public void conditionalRequestOneByOne2() { TestSubscriber<Long> ts = new TestSubscriber<Long>(1L) { @Override public void onNext(Long t) { super.onNext(t); request(1); } }; Flowable.rangeLong(1L, 5L) .filter(Functions.alwaysTrue()) .subscribe(ts); ts.assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void fastPathCancelExact() { TestSubscriber<Long> ts = new TestSubscriber<Long>() { @Override public void onNext(Long t) { super.onNext(t); if (t == 5L) { cancel(); onComplete(); } } }; Flowable.rangeLong(1L, 5L) .subscribe(ts); ts.assertResult(1L, 2L, 3L, 4L, 5L); } @Test public void conditionalFastPathCancelExact() { TestSubscriber<Long> ts = new TestSubscriber<Long>() { @Override public void onNext(Long t) { super.onNext(t); if (t == 5L) { cancel(); onComplete(); } } }; Flowable.rangeLong(1L, 5L) .filter(new Predicate<Long>() { @Override public boolean test(Long v) throws Exception { return v % 2 == 0; } }) .subscribe(ts); ts.assertResult(2L, 4L); } }