/**
* 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.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.*;
import java.util.concurrent.TimeUnit;
import org.junit.*;
import org.mockito.InOrder;
import org.reactivestreams.*;
import io.reactivex.*;
import io.reactivex.exceptions.*;
import io.reactivex.internal.subscriptions.BooleanSubscription;
import io.reactivex.processors.*;
import io.reactivex.schedulers.*;
import io.reactivex.subscribers.TestSubscriber;
public class FlowableSampleTest {
private TestScheduler scheduler;
private Scheduler.Worker innerScheduler;
private Subscriber<Long> observer;
private Subscriber<Object> observer2;
@Before
// due to mocking
public void before() {
scheduler = new TestScheduler();
innerScheduler = scheduler.createWorker();
observer = TestHelper.mockSubscriber();
observer2 = TestHelper.mockSubscriber();
}
@Test
public void testSample() {
Flowable<Long> source = Flowable.unsafeCreate(new Publisher<Long>() {
@Override
public void subscribe(final Subscriber<? super Long> observer1) {
observer1.onSubscribe(new BooleanSubscription());
innerScheduler.schedule(new Runnable() {
@Override
public void run() {
observer1.onNext(1L);
}
}, 1, TimeUnit.SECONDS);
innerScheduler.schedule(new Runnable() {
@Override
public void run() {
observer1.onNext(2L);
}
}, 2, TimeUnit.SECONDS);
innerScheduler.schedule(new Runnable() {
@Override
public void run() {
observer1.onComplete();
}
}, 3, TimeUnit.SECONDS);
}
});
Flowable<Long> sampled = source.sample(400L, TimeUnit.MILLISECONDS, scheduler);
sampled.subscribe(observer);
InOrder inOrder = inOrder(observer);
scheduler.advanceTimeTo(800L, TimeUnit.MILLISECONDS);
verify(observer, never()).onNext(any(Long.class));
verify(observer, never()).onComplete();
verify(observer, never()).onError(any(Throwable.class));
scheduler.advanceTimeTo(1200L, TimeUnit.MILLISECONDS);
inOrder.verify(observer, times(1)).onNext(1L);
verify(observer, never()).onNext(2L);
verify(observer, never()).onComplete();
verify(observer, never()).onError(any(Throwable.class));
scheduler.advanceTimeTo(1600L, TimeUnit.MILLISECONDS);
inOrder.verify(observer, never()).onNext(1L);
verify(observer, never()).onNext(2L);
verify(observer, never()).onComplete();
verify(observer, never()).onError(any(Throwable.class));
scheduler.advanceTimeTo(2000L, TimeUnit.MILLISECONDS);
inOrder.verify(observer, never()).onNext(1L);
inOrder.verify(observer, times(1)).onNext(2L);
verify(observer, never()).onComplete();
verify(observer, never()).onError(any(Throwable.class));
scheduler.advanceTimeTo(3000L, TimeUnit.MILLISECONDS);
inOrder.verify(observer, never()).onNext(1L);
inOrder.verify(observer, never()).onNext(2L);
verify(observer, times(1)).onComplete();
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerNormal() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
source.onNext(2);
sampler.onNext(1);
source.onNext(3);
source.onNext(4);
sampler.onNext(2);
source.onComplete();
sampler.onNext(3);
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, never()).onNext(1);
inOrder.verify(observer2, times(1)).onNext(2);
inOrder.verify(observer2, never()).onNext(3);
inOrder.verify(observer2, times(1)).onNext(4);
inOrder.verify(observer2, times(1)).onComplete();
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerNoDuplicates() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
source.onNext(2);
sampler.onNext(1);
sampler.onNext(1);
source.onNext(3);
source.onNext(4);
sampler.onNext(2);
sampler.onNext(2);
source.onComplete();
sampler.onNext(3);
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, never()).onNext(1);
inOrder.verify(observer2, times(1)).onNext(2);
inOrder.verify(observer2, never()).onNext(3);
inOrder.verify(observer2, times(1)).onNext(4);
inOrder.verify(observer2, times(1)).onComplete();
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerTerminatingEarly() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
source.onNext(2);
sampler.onNext(1);
sampler.onComplete();
source.onNext(3);
source.onNext(4);
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, never()).onNext(1);
inOrder.verify(observer2, times(1)).onNext(2);
inOrder.verify(observer2, times(1)).onComplete();
inOrder.verify(observer2, never()).onNext(any());
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerEmitAndTerminate() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
source.onNext(2);
sampler.onNext(1);
source.onNext(3);
source.onComplete();
sampler.onNext(2);
sampler.onComplete();
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, never()).onNext(1);
inOrder.verify(observer2, times(1)).onNext(2);
inOrder.verify(observer2, never()).onNext(3);
inOrder.verify(observer2, times(1)).onComplete();
inOrder.verify(observer2, never()).onNext(any());
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerEmptySource() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onComplete();
sampler.onNext(1);
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, times(1)).onComplete();
verify(observer2, never()).onNext(any());
verify(observer, never()).onError(any(Throwable.class));
}
@Test
public void sampleWithSamplerSourceThrows() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
source.onError(new RuntimeException("Forced failure!"));
sampler.onNext(1);
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, times(1)).onError(any(Throwable.class));
verify(observer2, never()).onNext(any());
verify(observer, never()).onComplete();
}
@Test
public void sampleWithSamplerThrows() {
PublishProcessor<Integer> source = PublishProcessor.create();
PublishProcessor<Integer> sampler = PublishProcessor.create();
Flowable<Integer> m = source.sample(sampler);
m.subscribe(observer2);
source.onNext(1);
sampler.onNext(1);
sampler.onError(new RuntimeException("Forced failure!"));
InOrder inOrder = inOrder(observer2);
inOrder.verify(observer2, times(1)).onNext(1);
inOrder.verify(observer2, times(1)).onError(any(RuntimeException.class));
verify(observer, never()).onComplete();
}
@Test
public void testSampleUnsubscribe() {
final Subscription s = mock(Subscription.class);
Flowable<Integer> o = Flowable.unsafeCreate(
new Publisher<Integer>() {
@Override
public void subscribe(Subscriber<? super Integer> subscriber) {
subscriber.onSubscribe(s);
}
}
);
o.throttleLast(1, TimeUnit.MILLISECONDS).subscribe().dispose();
verify(s).cancel();
}
@Test
public void dispose() {
TestHelper.checkDisposed(PublishProcessor.create().sample(1, TimeUnit.SECONDS, new TestScheduler()));
TestHelper.checkDisposed(PublishProcessor.create().sample(Flowable.never()));
}
@Test
public void error() {
Flowable.error(new TestException())
.sample(1, TimeUnit.SECONDS)
.test()
.assertFailure(TestException.class);
}
@Test
public void backpressureOverflow() {
BehaviorProcessor.createDefault(1)
.sample(1, TimeUnit.MILLISECONDS)
.test(0L)
.awaitDone(5, TimeUnit.SECONDS)
.assertFailure(MissingBackpressureException.class);
}
@Test
public void backpressureOverflowWithOtherPublisher() {
BehaviorProcessor.createDefault(1)
.sample(Flowable.timer(1, TimeUnit.MILLISECONDS))
.test(0L)
.awaitDone(5, TimeUnit.SECONDS)
.assertFailure(MissingBackpressureException.class);
}
@Test
public void emitLastTimed() {
Flowable.just(1)
.sample(1, TimeUnit.DAYS, true)
.test()
.assertResult(1);
}
@Test
public void emitLastTimedEmpty() {
Flowable.empty()
.sample(1, TimeUnit.DAYS, true)
.test()
.assertResult();
}
@Test
public void emitLastTimedCustomScheduler() {
Flowable.just(1)
.sample(1, TimeUnit.DAYS, Schedulers.single(), true)
.test()
.assertResult(1);
}
@Test
public void emitLastTimedRunCompleteRace() {
for (int i = 0; i < 1000; i++) {
final TestScheduler scheduler = new TestScheduler();
final PublishProcessor<Integer> pp = PublishProcessor.create();
TestSubscriber<Integer> ts = pp.sample(1, TimeUnit.SECONDS, scheduler, true)
.test();
pp.onNext(1);
Runnable r1 = new Runnable() {
@Override
public void run() {
pp.onComplete();
}
};
Runnable r2 = new Runnable() {
@Override
public void run() {
scheduler.advanceTimeBy(1, TimeUnit.SECONDS);
}
};
TestHelper.race(r1, r2);
ts.assertResult(1);
}
}
@Test
public void emitLastOther() {
Flowable.just(1)
.sample(Flowable.timer(1, TimeUnit.DAYS), true)
.test()
.assertResult(1);
}
@Test
public void emitLastOtherEmpty() {
Flowable.empty()
.sample(Flowable.timer(1, TimeUnit.DAYS), true)
.test()
.assertResult();
}
@Test
public void emitLastOtherRunCompleteRace() {
for (int i = 0; i < 1000; i++) {
final PublishProcessor<Integer> pp = PublishProcessor.create();
final PublishProcessor<Integer> sampler = PublishProcessor.create();
TestSubscriber<Integer> ts = pp.sample(sampler, true)
.test();
pp.onNext(1);
Runnable r1 = new Runnable() {
@Override
public void run() {
pp.onComplete();
}
};
Runnable r2 = new Runnable() {
@Override
public void run() {
sampler.onNext(1);
}
};
TestHelper.race(r1, r2);
ts.assertResult(1);
}
}
@Test
public void emitLastOtherCompleteCompleteRace() {
for (int i = 0; i < 1000; i++) {
final PublishProcessor<Integer> pp = PublishProcessor.create();
final PublishProcessor<Integer> sampler = PublishProcessor.create();
TestSubscriber<Integer> ts = pp.sample(sampler, true).test();
pp.onNext(1);
Runnable r1 = new Runnable() {
@Override
public void run() {
pp.onComplete();
}
};
Runnable r2 = new Runnable() {
@Override
public void run() {
sampler.onComplete();
}
};
TestHelper.race(r1, r2);
ts.assertResult(1);
}
}
}