|
26 | 26 | import org.mockito.Mockito;
|
27 | 27 | import org.reactivestreams.*;
|
28 | 28 |
|
| 29 | +import com.google.common.base.Ticker; |
29 | 30 | import com.google.common.cache.*;
|
30 | 31 |
|
31 | 32 | import io.reactivex.*;
|
@@ -1947,6 +1948,127 @@ public void run() throws Exception {
|
1947 | 1948 | }
|
1948 | 1949 | };
|
1949 | 1950 | }
|
| 1951 | + |
| 1952 | + private static final class TestTicker extends Ticker { |
| 1953 | + long tick = 0; |
| 1954 | + |
| 1955 | + @Override |
| 1956 | + public long read() { |
| 1957 | + return tick; |
| 1958 | + } |
| 1959 | + } |
| 1960 | + |
| 1961 | + @Test |
| 1962 | + public void testGroupByEvictionCancellationOfSource5933() { |
| 1963 | + PublishProcessor<Integer> source = PublishProcessor.create(); |
| 1964 | + final TestTicker testTicker = new TestTicker(); |
| 1965 | + |
| 1966 | + Function<Consumer<Object>, Map<Integer, Object>> mapFactory = new Function<Consumer<Object>, Map<Integer, Object>>() { |
| 1967 | + @Override |
| 1968 | + public Map<Integer, Object> apply(final Consumer<Object> action) throws Exception { |
| 1969 | + return CacheBuilder.newBuilder() // |
| 1970 | + .expireAfterAccess(5, TimeUnit.SECONDS).removalListener(new RemovalListener<Object, Object>() { |
| 1971 | + @Override |
| 1972 | + public void onRemoval(RemovalNotification<Object, Object> notification) { |
| 1973 | + try { |
| 1974 | + action.accept(notification.getValue()); |
| 1975 | + } catch (Exception ex) { |
| 1976 | + throw new RuntimeException(ex); |
| 1977 | + } |
| 1978 | + } |
| 1979 | + }).ticker(testTicker) // |
| 1980 | + .<Integer, Object>build().asMap(); |
| 1981 | + } |
| 1982 | + }; |
| 1983 | + |
| 1984 | + final List<String> list = new CopyOnWriteArrayList<String>(); |
| 1985 | + Flowable<Integer> stream = source // |
| 1986 | + .doOnCancel(new Action() { |
| 1987 | + @Override |
| 1988 | + public void run() throws Exception { |
| 1989 | + list.add("Source canceled"); |
| 1990 | + } |
| 1991 | + }) |
| 1992 | + .<Integer, Integer>groupBy(Functions.<Integer>identity(), Functions.<Integer>identity(), false, |
| 1993 | + Flowable.bufferSize(), mapFactory) // |
| 1994 | + .flatMap(new Function<GroupedFlowable<Integer, Integer>, Publisher<? extends Integer>>() { |
| 1995 | + @Override |
| 1996 | + public Publisher<? extends Integer> apply(GroupedFlowable<Integer, Integer> group) |
| 1997 | + throws Exception { |
| 1998 | + return group // |
| 1999 | + .doOnComplete(new Action() { |
| 2000 | + @Override |
| 2001 | + public void run() throws Exception { |
| 2002 | + list.add("Group completed"); |
| 2003 | + } |
| 2004 | + }).doOnCancel(new Action() { |
| 2005 | + @Override |
| 2006 | + public void run() throws Exception { |
| 2007 | + list.add("Group canceled"); |
| 2008 | + } |
| 2009 | + }); |
| 2010 | + } |
| 2011 | + }); |
| 2012 | + TestSubscriber<Integer> ts = stream // |
| 2013 | + .doOnCancel(new Action() { |
| 2014 | + @Override |
| 2015 | + public void run() throws Exception { |
| 2016 | + list.add("Outer group by canceled"); |
| 2017 | + } |
| 2018 | + }).test(); |
| 2019 | + |
| 2020 | + // Send 3 in the same group and wait for them to be seen |
| 2021 | + source.onNext(1); |
| 2022 | + source.onNext(1); |
| 2023 | + source.onNext(1); |
| 2024 | + ts.awaitCount(3); |
| 2025 | + |
| 2026 | + // Advance time far enough to evict the group. |
| 2027 | + // NOTE -- Comment this line out to make the test "pass". |
| 2028 | + testTicker.tick = TimeUnit.SECONDS.toNanos(6); |
| 2029 | + |
| 2030 | + // Send more data in the group (triggering eviction and recreation) |
| 2031 | + source.onNext(1); |
| 2032 | + |
| 2033 | + // Wait for the last 2 and then cancel the subscription |
| 2034 | + ts.awaitCount(4); |
| 2035 | + ts.cancel(); |
| 2036 | + |
| 2037 | + // Observe the result. Note that right now the result differs depending on whether eviction occurred or |
| 2038 | + // not. The observed sequence in that case is: Group completed, Outer group by canceled., Group canceled. |
| 2039 | + // The addition of the "Group completed" is actually fine, but the fact that the cancel doesn't reach the |
| 2040 | + // source seems like a bug. Commenting out the setting of "tick" above will produce the "expected" sequence. |
| 2041 | + System.out.println(list); |
| 2042 | + assertTrue(list.contains("Source canceled")); |
| 2043 | + assertEquals(Arrays.asList( |
| 2044 | + "Group completed", // this is here when eviction occurs |
| 2045 | + "Outer group by canceled", |
| 2046 | + "Group canceled", |
| 2047 | + "Source canceled" // This is *not* here when eviction occurs |
| 2048 | + ), list); |
| 2049 | + } |
| 2050 | + |
| 2051 | + @Test |
| 2052 | + public void testCancellationOfUpstreamWhenGroupedFlowableCompletes() { |
| 2053 | + final AtomicBoolean cancelled = new AtomicBoolean(); |
| 2054 | + Flowable.just(1).repeat().doOnCancel(new Action() { |
| 2055 | + @Override |
| 2056 | + public void run() throws Exception { |
| 2057 | + cancelled.set(true); |
| 2058 | + } |
| 2059 | + }) |
| 2060 | + .groupBy(Functions.<Integer>identity(), Functions.<Integer>identity()) // |
| 2061 | + .flatMap(new Function<GroupedFlowable<Integer, Integer>, Publisher<? extends Object>>() { |
| 2062 | + @Override |
| 2063 | + public Publisher<? extends Object> apply(GroupedFlowable<Integer, Integer> g) throws Exception { |
| 2064 | + return g.first(0).toFlowable(); |
| 2065 | + } |
| 2066 | + }) |
| 2067 | + .take(4) // |
| 2068 | + .test() // |
| 2069 | + .assertComplete(); |
| 2070 | + assertTrue(cancelled.get()); |
| 2071 | + } |
1950 | 2072 |
|
1951 | 2073 | //not thread safe
|
1952 | 2074 | private static final class SingleThreadEvictingHashMap<K,V> implements Map<K,V> {
|
|
0 commit comments