/* * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package org.openjdk.tests.java.util.stream; import org.testng.annotations.Test; import java.util.*; import java.util.Spliterators; import java.util.concurrent.atomic.AtomicInteger; import java.util.function.BiFunction; import java.util.function.Consumer; import java.util.function.Function; import java.util.function.Supplier; import java.util.stream.*; import static java.util.stream.LambdaTestHelpers.*; /** * SortedOpTest * * @author Brian Goetz */ @Test public class SortedOpTest extends OpTestCase { public void testRefStreamTooLarge() { Function<LongStream, Stream<Long>> f = s -> // Clear the SORTED flag s.mapToObj(i -> i) .sorted(); testStreamTooLarge(f, Stream::findFirst); } public void testIntStreamTooLarge() { Function<LongStream, IntStream> f = s -> // Clear the SORTED flag s.mapToInt(i -> (int) i) .sorted(); testStreamTooLarge(f, IntStream::findFirst); } public void testLongStreamTooLarge() { Function<LongStream, LongStream> f = s -> // Clear the SORTED flag s.map(i -> i) .sorted(); testStreamTooLarge(f, LongStream::findFirst); } public void testDoubleStreamTooLarge() { Function<LongStream, DoubleStream> f = s -> // Clear the SORTED flag s.mapToDouble(i -> (double) i) .sorted(); testStreamTooLarge(f, DoubleStream::findFirst); } <T, S extends BaseStream<T, S>> void testStreamTooLarge(Function<LongStream, S> s, Function<S, ?> terminal) { // Set up conditions for a large input > maximum array size Supplier<LongStream> input = () -> LongStream.range(0, 1L + Integer.MAX_VALUE); // Transformation functions List<Function<LongStream, LongStream>> transforms = Arrays.asList( ls -> ls, ls -> ls.parallel(), // Clear the SIZED flag ls -> ls.limit(Long.MAX_VALUE), ls -> ls.limit(Long.MAX_VALUE).parallel()); for (Function<LongStream, LongStream> transform : transforms) { RuntimeException caught = null; try { terminal.apply(s.apply(transform.apply(input.get()))); } catch (RuntimeException e) { caught = e; } assertNotNull(caught, "Expected an instance of exception IllegalArgumentException but no exception thrown"); assertTrue(caught instanceof IllegalArgumentException, String.format("Expected an instance of exception IllegalArgumentException but got %s", caught)); } } public void testSorted() { assertCountSum(countTo(0).stream().sorted(), 0, 0); assertCountSum(countTo(10).stream().sorted(), 10, 55); assertCountSum(countTo(10).stream().sorted(cInteger.reversed()), 10, 55); List<Integer> to10 = countTo(10); assertSorted(to10.stream().sorted(cInteger.reversed()).iterator(), cInteger.reversed()); Collections.reverse(to10); assertSorted(to10.stream().sorted().iterator()); Spliterator<Integer> s = to10.stream().sorted().spliterator(); assertTrue(s.hasCharacteristics(Spliterator.SORTED)); s = to10.stream().sorted(cInteger.reversed()).spliterator(); assertFalse(s.hasCharacteristics(Spliterator.SORTED)); } @Test(groups = { "serialization-hostile" }) public void testSequentialShortCircuitTerminal() { // The sorted op for sequential evaluation will buffer all elements when // accepting then at the end sort those elements and push those elements // downstream // A peek operation is added in-between the sorted() and terminal // operation that counts the number of calls to its consumer and // asserts that the number of calls is at most the required quantity List<Integer> l = Arrays.asList(5, 4, 3, 2, 1); Function<Integer, Stream<Integer>> knownSize = i -> assertNCallsOnly( l.stream().sorted(), Stream::peek, i); Function<Integer, Stream<Integer>> unknownSize = i -> assertNCallsOnly (unknownSizeStream(l).sorted(), Stream::peek, i); // Find assertEquals(knownSize.apply(1).findFirst(), Optional.of(1)); assertEquals(knownSize.apply(1).findAny(), Optional.of(1)); assertEquals(unknownSize.apply(1).findFirst(), Optional.of(1)); assertEquals(unknownSize.apply(1).findAny(), Optional.of(1)); // Match assertEquals(knownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(knownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(knownSize.apply(2).allMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(unknownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).allMatch(i -> i == 2), false); } private <T> Stream<T> unknownSizeStream(List<T> l) { return StreamSupport.stream(Spliterators.spliteratorUnknownSize(l.iterator(), 0), false); } @Test(dataProvider = "StreamTestData<Integer>", dataProviderClass = StreamTestDataProvider.class) public void testOps(String name, TestData.OfRef<Integer> data) { Collection<Integer> result = exerciseOpsInt(data, Stream::sorted, IntStream::sorted, LongStream::sorted, DoubleStream::sorted); assertSorted(result.iterator()); assertContentsUnordered(data, result); result = exerciseOps(data, s -> s.sorted(cInteger.reversed())); assertSorted(result.iterator(), cInteger.reversed()); assertContentsUnordered(data, result); } @Test(dataProvider = "StreamTestData<Integer>", dataProviderClass = StreamTestDataProvider.class) public void testSortSort(String name, TestData.OfRef<Integer> data) { // For parallel cases ensure the size is known Collection<Integer> result = withData(data) .stream(s -> s.sorted().sorted(), new CollectorOps.TestParallelSizedOp<Integer>()) .exercise(); assertSorted(result); assertContentsUnordered(data, result); result = withData(data) .stream(s -> s.sorted(cInteger.reversed()).sorted(cInteger.reversed()), new CollectorOps.TestParallelSizedOp<Integer>()) .exercise(); assertSorted(result, cInteger.reversed()); assertContentsUnordered(data, result); result = withData(data) .stream(s -> s.sorted().sorted(cInteger.reversed()), new CollectorOps.TestParallelSizedOp<Integer>()) .exercise(); assertSorted(result, cInteger.reversed()); assertContentsUnordered(data, result); result = withData(data) .stream(s -> s.sorted(cInteger.reversed()).sorted(), new CollectorOps.TestParallelSizedOp<Integer>()) .exercise(); assertSorted(result); assertContentsUnordered(data, result); } // @Test(groups = { "serialization-hostile" }) public void testIntSequentialShortCircuitTerminal() { int[] a = new int[]{5, 4, 3, 2, 1}; Function<Integer, IntStream> knownSize = i -> assertNCallsOnly( Arrays.stream(a).sorted(), (s, c) -> s.peek(c::accept), i); Function<Integer, IntStream> unknownSize = i -> assertNCallsOnly (unknownSizeIntStream(a).sorted(), (s, c) -> s.peek(c::accept), i); // Find assertEquals(knownSize.apply(1).findFirst(), OptionalInt.of(1)); assertEquals(knownSize.apply(1).findAny(), OptionalInt.of(1)); assertEquals(unknownSize.apply(1).findFirst(), OptionalInt.of(1)); assertEquals(unknownSize.apply(1).findAny(), OptionalInt.of(1)); // Match assertEquals(knownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(knownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(knownSize.apply(2).allMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(unknownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).allMatch(i -> i == 2), false); } private IntStream unknownSizeIntStream(int[] a) { return StreamSupport.intStream(Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(a)), 0), false); } @Test(dataProvider = "IntStreamTestData", dataProviderClass = IntStreamTestDataProvider.class) public void testIntOps(String name, TestData.OfInt data) { Collection<Integer> result = exerciseOps(data, s -> s.sorted()); assertSorted(result); assertContentsUnordered(data, result); } @Test(dataProvider = "IntStreamTestData", dataProviderClass = IntStreamTestDataProvider.class) public void testIntSortSort(String name, TestData.OfInt data) { // For parallel cases ensure the size is known Collection<Integer> result = withData(data) .stream(s -> s.sorted().sorted(), new CollectorOps.TestParallelSizedOp.OfInt()) .exercise(); assertSorted(result); assertContentsUnordered(data, result); } // @Test(groups = { "serialization-hostile" }) public void testLongSequentialShortCircuitTerminal() { long[] a = new long[]{5, 4, 3, 2, 1}; Function<Integer, LongStream> knownSize = i -> assertNCallsOnly( Arrays.stream(a).sorted(), (s, c) -> s.peek(c::accept), i); Function<Integer, LongStream> unknownSize = i -> assertNCallsOnly (unknownSizeLongStream(a).sorted(), (s, c) -> s.peek(c::accept), i); // Find assertEquals(knownSize.apply(1).findFirst(), OptionalLong.of(1)); assertEquals(knownSize.apply(1).findAny(), OptionalLong.of(1)); assertEquals(unknownSize.apply(1).findFirst(), OptionalLong.of(1)); assertEquals(unknownSize.apply(1).findAny(), OptionalLong.of(1)); // Match assertEquals(knownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(knownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(knownSize.apply(2).allMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).anyMatch(i -> i == 2), true); assertEquals(unknownSize.apply(2).noneMatch(i -> i == 2), false); assertEquals(unknownSize.apply(2).allMatch(i -> i == 2), false); } private LongStream unknownSizeLongStream(long[] a) { return StreamSupport.longStream(Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(a)), 0), false); } @Test(dataProvider = "LongStreamTestData", dataProviderClass = LongStreamTestDataProvider.class) public void testLongOps(String name, TestData.OfLong data) { Collection<Long> result = exerciseOps(data, s -> s.sorted()); assertSorted(result); assertContentsUnordered(data, result); } @Test(dataProvider = "LongStreamTestData", dataProviderClass = LongStreamTestDataProvider.class) public void testLongSortSort(String name, TestData.OfLong data) { // For parallel cases ensure the size is known Collection<Long> result = withData(data) .stream(s -> s.sorted().sorted(), new CollectorOps.TestParallelSizedOp.OfLong()) .exercise(); assertSorted(result); assertContentsUnordered(data, result); } // @Test(groups = { "serialization-hostile" }) public void testDoubleSequentialShortCircuitTerminal() { double[] a = new double[]{5.0, 4.0, 3.0, 2.0, 1.0}; Function<Integer, DoubleStream> knownSize = i -> assertNCallsOnly( Arrays.stream(a).sorted(), (s, c) -> s.peek(c::accept), i); Function<Integer, DoubleStream> unknownSize = i -> assertNCallsOnly (unknownSizeDoubleStream(a).sorted(), (s, c) -> s.peek(c::accept), i); // Find assertEquals(knownSize.apply(1).findFirst(), OptionalDouble.of(1)); assertEquals(knownSize.apply(1).findAny(), OptionalDouble.of(1)); assertEquals(unknownSize.apply(1).findFirst(), OptionalDouble.of(1)); assertEquals(unknownSize.apply(1).findAny(), OptionalDouble.of(1)); // Match assertEquals(knownSize.apply(2).anyMatch(i -> i == 2.0), true); assertEquals(knownSize.apply(2).noneMatch(i -> i == 2.0), false); assertEquals(knownSize.apply(2).allMatch(i -> i == 2.0), false); assertEquals(unknownSize.apply(2).anyMatch(i -> i == 2.0), true); assertEquals(unknownSize.apply(2).noneMatch(i -> i == 2.0), false); assertEquals(unknownSize.apply(2).allMatch(i -> i == 2.0), false); } private DoubleStream unknownSizeDoubleStream(double[] a) { return StreamSupport.doubleStream(Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(a)), 0), false); } @Test(dataProvider = "DoubleStreamTestData", dataProviderClass = DoubleStreamTestDataProvider.class) public void testDoubleOps(String name, TestData.OfDouble data) { Collection<Double> result = exerciseOps(data, s -> s.sorted()); assertSorted(result); assertContentsUnordered(data, result); } @Test(dataProvider = "DoubleStreamTestData", dataProviderClass = DoubleStreamTestDataProvider.class) public void testDoubleSortSort(String name, TestData.OfDouble data) { // For parallel cases ensure the size is known Collection<Double> result = withData(data) .stream(s -> s.sorted().sorted(), new CollectorOps.TestParallelSizedOp.OfDouble()) .exercise(); assertSorted(result); assertContentsUnordered(data, result); } /** * Interpose a consumer that asserts it is called at most N times. */ <T, S extends BaseStream<T, S>, R> S assertNCallsOnly(S s, BiFunction<S, Consumer<T>, S> pf, int n) { AtomicInteger boxedInt = new AtomicInteger(); return pf.apply(s, i -> { assertFalse(boxedInt.incrementAndGet() > n, "Intermediate op called more than " + n + " time(s)"); }); } }