/** * 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.Mockito.*; import java.util.concurrent.*; import java.util.concurrent.atomic.AtomicBoolean; import org.junit.Test; import org.reactivestreams.Subscriber; import io.reactivex.*; import io.reactivex.schedulers.*; import io.reactivex.subscribers.TestSubscriber; public class FlowableToFutureTest { @Test public void testSuccess() throws Exception { @SuppressWarnings("unchecked") Future<Object> future = mock(Future.class); Object value = new Object(); when(future.get()).thenReturn(value); Subscriber<Object> o = TestHelper.mockSubscriber(); TestSubscriber<Object> ts = new TestSubscriber<Object>(o); Flowable.fromFuture(future).subscribe(ts); ts.dispose(); verify(o, times(1)).onNext(value); verify(o, times(1)).onComplete(); verify(o, never()).onError(any(Throwable.class)); verify(future, never()).cancel(anyBoolean()); } @Test public void testSuccessOperatesOnSuppliedScheduler() throws Exception { @SuppressWarnings("unchecked") Future<Object> future = mock(Future.class); Object value = new Object(); when(future.get()).thenReturn(value); Subscriber<Object> o = TestHelper.mockSubscriber(); TestScheduler scheduler = new TestScheduler(); TestSubscriber<Object> ts = new TestSubscriber<Object>(o); Flowable.fromFuture(future, scheduler).subscribe(ts); verify(o, never()).onNext(value); scheduler.triggerActions(); verify(o, times(1)).onNext(value); } @Test public void testFailure() throws Exception { @SuppressWarnings("unchecked") Future<Object> future = mock(Future.class); RuntimeException e = new RuntimeException(); when(future.get()).thenThrow(e); Subscriber<Object> o = TestHelper.mockSubscriber(); TestSubscriber<Object> ts = new TestSubscriber<Object>(o); Flowable.fromFuture(future).subscribe(ts); ts.dispose(); verify(o, never()).onNext(null); verify(o, never()).onComplete(); verify(o, times(1)).onError(e); verify(future, never()).cancel(anyBoolean()); } @Test public void testCancelledBeforeSubscribe() throws Exception { @SuppressWarnings("unchecked") Future<Object> future = mock(Future.class); CancellationException e = new CancellationException("unit test synthetic cancellation"); when(future.get()).thenThrow(e); Subscriber<Object> o = TestHelper.mockSubscriber(); TestSubscriber<Object> ts = new TestSubscriber<Object>(o); ts.dispose(); Flowable.fromFuture(future).subscribe(ts); ts.assertNoErrors(); ts.assertNotComplete(); } @Test public void testCancellationDuringFutureGet() throws Exception { Future<Object> future = new Future<Object>() { private AtomicBoolean isCancelled = new AtomicBoolean(false); private AtomicBoolean isDone = new AtomicBoolean(false); @Override public boolean cancel(boolean mayInterruptIfRunning) { isCancelled.compareAndSet(false, true); return true; } @Override public boolean isCancelled() { return isCancelled.get(); } @Override public boolean isDone() { return isCancelled() || isDone.get(); } @Override public Object get() throws InterruptedException, ExecutionException { Thread.sleep(500); isDone.compareAndSet(false, true); return "foo"; } @Override public Object get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return get(); } }; Subscriber<Object> o = TestHelper.mockSubscriber(); TestSubscriber<Object> ts = new TestSubscriber<Object>(o); Flowable<Object> futureObservable = Flowable.fromFuture(future); futureObservable.subscribeOn(Schedulers.computation()).subscribe(ts); Thread.sleep(100); ts.dispose(); ts.assertNoErrors(); ts.assertNoValues(); ts.assertNotComplete(); } @Test public void backpressure() { TestSubscriber<Integer> ts = new TestSubscriber<Integer>(0); FutureTask<Integer> f = new FutureTask<Integer>(new Runnable() { @Override public void run() { } }, 1); f.run(); Flowable.fromFuture(f).subscribe(ts); ts.assertNoValues(); ts.request(1); ts.assertValue(1); ts.assertNoErrors(); ts.assertComplete(); } @Test public void withTimeoutNoTimeout() { FutureTask<Integer> task = new FutureTask<Integer>(new Runnable() { @Override public void run() { } }, 1); task.run(); TestSubscriber<Integer> ts = TestSubscriber.create(); Flowable.fromFuture(task, 1, TimeUnit.SECONDS).subscribe(ts); ts.assertValue(1); ts.assertNoErrors(); ts.assertComplete(); } @Test public void withTimeoutTimeout() { FutureTask<Integer> task = new FutureTask<Integer>(new Runnable() { @Override public void run() { } }, 1); TestSubscriber<Integer> ts = TestSubscriber.create(); Flowable.fromFuture(task, 10, TimeUnit.MILLISECONDS).subscribe(ts); ts.assertNoValues(); ts.assertError(TimeoutException.class); ts.assertNotComplete(); } @Test public void withTimeoutNoTimeoutScheduler() { FutureTask<Integer> task = new FutureTask<Integer>(new Runnable() { @Override public void run() { } }, 1); TestSubscriber<Integer> ts = TestSubscriber.create(); Flowable.fromFuture(task, Schedulers.computation()).subscribe(ts); task.run(); ts.awaitTerminalEvent(5, TimeUnit.SECONDS); ts.assertValue(1); ts.assertNoErrors(); ts.assertComplete(); } }