ViewVC Help
View File | Revision Log | Show Annotations | Download File | Root Listing
root/jsr166/jsr166/src/test/jtreg/util/Spliterator/SpliteratorTraversingAndSplittingTest.java
Revision: 1.6
Committed: Fri Dec 9 04:46:51 2016 UTC (7 years, 6 months ago) by jsr166
Branch: MAIN
Changes since 1.5: +30 -1 lines
Log Message:
sync Spliterator tests from upstream

File Contents

# User Rev Content
1 jsr166 1.1 /*
2 jsr166 1.4 * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved.
3 jsr166 1.1 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4     *
5     * This code is free software; you can redistribute it and/or modify it
6     * under the terms of the GNU General Public License version 2 only, as
7     * published by the Free Software Foundation.
8     *
9     * This code is distributed in the hope that it will be useful, but WITHOUT
10     * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11     * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12     * version 2 for more details (a copy is included in the LICENSE file that
13     * accompanied this code).
14     *
15     * You should have received a copy of the GNU General Public License version
16     * 2 along with this work; if not, write to the Free Software Foundation,
17     * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18     *
19     * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20     * or visit www.oracle.com if you need additional information or have any
21     * questions.
22     */
23    
24     /**
25     * @test
26     * @summary Spliterator traversing and splitting tests
27     * @run testng SpliteratorTraversingAndSplittingTest
28 jsr166 1.6 * @bug 8020016 8071477 8072784 8169838
29 jsr166 1.1 */
30    
31     import org.testng.annotations.DataProvider;
32     import org.testng.annotations.Test;
33    
34 jsr166 1.6 import java.nio.CharBuffer;
35 jsr166 1.1 import java.util.AbstractCollection;
36     import java.util.AbstractList;
37     import java.util.AbstractSet;
38     import java.util.ArrayDeque;
39     import java.util.ArrayList;
40     import java.util.Arrays;
41 jsr166 1.6 import java.util.BitSet;
42 jsr166 1.1 import java.util.Collection;
43     import java.util.Collections;
44     import java.util.Comparator;
45     import java.util.Deque;
46     import java.util.HashMap;
47     import java.util.HashSet;
48     import java.util.IdentityHashMap;
49     import java.util.Iterator;
50     import java.util.LinkedHashMap;
51     import java.util.LinkedHashSet;
52     import java.util.LinkedList;
53     import java.util.List;
54 jsr166 1.5 import java.util.ListIterator;
55 jsr166 1.1 import java.util.Map;
56     import java.util.PriorityQueue;
57 jsr166 1.5 import java.util.RandomAccess;
58 jsr166 1.1 import java.util.Set;
59     import java.util.SortedSet;
60     import java.util.Spliterator;
61     import java.util.Spliterators;
62     import java.util.Stack;
63     import java.util.TreeMap;
64     import java.util.TreeSet;
65     import java.util.Vector;
66     import java.util.WeakHashMap;
67     import java.util.concurrent.ArrayBlockingQueue;
68     import java.util.concurrent.ConcurrentHashMap;
69     import java.util.concurrent.ConcurrentLinkedQueue;
70     import java.util.concurrent.ConcurrentSkipListMap;
71     import java.util.concurrent.ConcurrentSkipListSet;
72     import java.util.concurrent.CopyOnWriteArrayList;
73     import java.util.concurrent.CopyOnWriteArraySet;
74     import java.util.concurrent.LinkedBlockingDeque;
75     import java.util.concurrent.LinkedBlockingQueue;
76     import java.util.concurrent.LinkedTransferQueue;
77     import java.util.concurrent.PriorityBlockingQueue;
78     import java.util.function.Consumer;
79     import java.util.function.DoubleConsumer;
80     import java.util.function.Function;
81     import java.util.function.IntConsumer;
82     import java.util.function.LongConsumer;
83     import java.util.function.Supplier;
84     import java.util.function.UnaryOperator;
85 jsr166 1.6 import java.util.stream.IntStream;
86 jsr166 1.1
87 jsr166 1.6 import static java.util.stream.Collectors.toList;
88 jsr166 1.1 import static org.testng.Assert.*;
89     import static org.testng.Assert.assertEquals;
90    
91     @Test
92     public class SpliteratorTraversingAndSplittingTest {
93    
94 jsr166 1.3 private static final List<Integer> SIZES = Arrays.asList(0, 1, 10, 42);
95 jsr166 1.1
96     private static final String LOW = new String(new char[] {Character.MIN_LOW_SURROGATE});
97     private static final String HIGH = new String(new char[] {Character.MIN_HIGH_SURROGATE});
98     private static final String HIGH_LOW = HIGH + LOW;
99     private static final String CHAR_HIGH_LOW = "A" + HIGH_LOW;
100     private static final String HIGH_LOW_CHAR = HIGH_LOW + "A";
101     private static final String CHAR_HIGH_LOW_CHAR = "A" + HIGH_LOW + "A";
102    
103     private static final List<String> STRINGS = generateTestStrings();
104    
105     private static List<String> generateTestStrings() {
106     List<String> strings = new ArrayList<>();
107     for (int n : Arrays.asList(1, 2, 3, 16, 17)) {
108     strings.add(generate("A", n));
109     strings.add(generate(LOW, n));
110     strings.add(generate(HIGH, n));
111     strings.add(generate(HIGH_LOW, n));
112     strings.add(generate(CHAR_HIGH_LOW, n));
113     strings.add(generate(HIGH_LOW_CHAR, n));
114     strings.add(generate(CHAR_HIGH_LOW_CHAR, n));
115     }
116     return strings;
117     }
118    
119     private static String generate(String s, int n) {
120     StringBuilder sb = new StringBuilder();
121     for (int i = 0; i < n; i++) {
122     sb.append(s);
123     }
124     return sb.toString();
125     }
126    
127     private static class SpliteratorDataBuilder<T> {
128     List<Object[]> data;
129    
130     List<T> exp;
131    
132 jsr166 1.2 Map<T, T> mExp;
133 jsr166 1.1
134     SpliteratorDataBuilder(List<Object[]> data, List<T> exp) {
135     this.data = data;
136     this.exp = exp;
137     this.mExp = createMap(exp);
138     }
139    
140 jsr166 1.2 Map<T, T> createMap(List<T> l) {
141     Map<T, T> m = new LinkedHashMap<>();
142 jsr166 1.1 for (T t : l) {
143     m.put(t, t);
144     }
145     return m;
146     }
147    
148     void add(String description, Collection<?> expected, Supplier<Spliterator<?>> s) {
149     description = joiner(description).toString();
150     data.add(new Object[]{description, expected, s});
151     }
152    
153     void add(String description, Supplier<Spliterator<?>> s) {
154     add(description, exp, s);
155     }
156    
157     void addCollection(Function<Collection<T>, ? extends Collection<T>> c) {
158     add("new " + c.apply(Collections.<T>emptyList()).getClass().getName() + ".spliterator()",
159     () -> c.apply(exp).spliterator());
160     }
161    
162     void addList(Function<Collection<T>, ? extends List<T>> l) {
163     addCollection(l);
164 jsr166 1.4 addCollection(l.andThen(list -> list.subList(0, list.size())));
165 jsr166 1.1 }
166    
167 jsr166 1.2 void addMap(Function<Map<T, T>, ? extends Map<T, T>> m) {
168     String description = "new " + m.apply(Collections.<T, T>emptyMap()).getClass().getName();
169 jsr166 1.1 addMap(m, description);
170     }
171    
172 jsr166 1.2 void addMap(Function<Map<T, T>, ? extends Map<T, T>> m, String description) {
173 jsr166 1.1 add(description + ".keySet().spliterator()", () -> m.apply(mExp).keySet().spliterator());
174     add(description + ".values().spliterator()", () -> m.apply(mExp).values().spliterator());
175     add(description + ".entrySet().spliterator()", mExp.entrySet(), () -> m.apply(mExp).entrySet().spliterator());
176     }
177    
178     StringBuilder joiner(String description) {
179     return new StringBuilder(description).
180     append(" {").
181     append("size=").append(exp.size()).
182     append("}");
183     }
184     }
185    
186     static Object[][] spliteratorDataProvider;
187    
188     @DataProvider(name = "Spliterator<Integer>")
189     public static Object[][] spliteratorDataProvider() {
190     if (spliteratorDataProvider != null) {
191     return spliteratorDataProvider;
192     }
193    
194     List<Object[]> data = new ArrayList<>();
195     for (int size : SIZES) {
196     List<Integer> exp = listIntRange(size);
197     SpliteratorDataBuilder<Integer> db = new SpliteratorDataBuilder<>(data, exp);
198    
199     // Direct spliterator methods
200    
201     db.add("Spliterators.spliterator(Collection, ...)",
202     () -> Spliterators.spliterator(exp, 0));
203    
204     db.add("Spliterators.spliterator(Iterator, ...)",
205     () -> Spliterators.spliterator(exp.iterator(), exp.size(), 0));
206    
207     db.add("Spliterators.spliteratorUnknownSize(Iterator, ...)",
208     () -> Spliterators.spliteratorUnknownSize(exp.iterator(), 0));
209    
210     db.add("Spliterators.spliterator(Spliterators.iteratorFromSpliterator(Spliterator ), ...)",
211     () -> Spliterators.spliterator(Spliterators.iterator(exp.spliterator()), exp.size(), 0));
212    
213     db.add("Spliterators.spliterator(T[], ...)",
214     () -> Spliterators.spliterator(exp.toArray(new Integer[0]), 0));
215    
216     db.add("Arrays.spliterator(T[], ...)",
217     () -> Arrays.spliterator(exp.toArray(new Integer[0])));
218    
219     class SpliteratorFromIterator extends Spliterators.AbstractSpliterator<Integer> {
220     Iterator<Integer> it;
221    
222     SpliteratorFromIterator(Iterator<Integer> it, long est) {
223     super(est, Spliterator.SIZED);
224     this.it = it;
225     }
226    
227     @Override
228     public boolean tryAdvance(Consumer<? super Integer> action) {
229     if (action == null)
230     throw new NullPointerException();
231     if (it.hasNext()) {
232     action.accept(it.next());
233     return true;
234     }
235     else {
236     return false;
237     }
238     }
239     }
240     db.add("new Spliterators.AbstractSpliterator()",
241     () -> new SpliteratorFromIterator(exp.iterator(), exp.size()));
242    
243     // Collections
244    
245     // default method implementations
246    
247     class AbstractCollectionImpl extends AbstractCollection<Integer> {
248     Collection<Integer> c;
249    
250     AbstractCollectionImpl(Collection<Integer> c) {
251     this.c = c;
252     }
253    
254     @Override
255     public Iterator<Integer> iterator() {
256     return c.iterator();
257     }
258    
259     @Override
260     public int size() {
261     return c.size();
262     }
263     }
264     db.addCollection(
265     c -> new AbstractCollectionImpl(c));
266    
267     class AbstractListImpl extends AbstractList<Integer> {
268     List<Integer> l;
269    
270     AbstractListImpl(Collection<Integer> c) {
271     this.l = new ArrayList<>(c);
272     }
273    
274     @Override
275     public Integer get(int index) {
276     return l.get(index);
277     }
278    
279     @Override
280     public int size() {
281     return l.size();
282     }
283     }
284     db.addCollection(
285     c -> new AbstractListImpl(c));
286    
287     class AbstractSetImpl extends AbstractSet<Integer> {
288     Set<Integer> s;
289    
290     AbstractSetImpl(Collection<Integer> c) {
291     this.s = new HashSet<>(c);
292     }
293    
294     @Override
295     public Iterator<Integer> iterator() {
296     return s.iterator();
297     }
298    
299     @Override
300     public int size() {
301     return s.size();
302     }
303     }
304     db.addCollection(
305     c -> new AbstractSetImpl(c));
306    
307     class AbstractSortedSetImpl extends AbstractSet<Integer> implements SortedSet<Integer> {
308     SortedSet<Integer> s;
309    
310     AbstractSortedSetImpl(Collection<Integer> c) {
311     this.s = new TreeSet<>(c);
312     }
313    
314     @Override
315     public Iterator<Integer> iterator() {
316     return s.iterator();
317     }
318    
319     @Override
320     public int size() {
321     return s.size();
322     }
323    
324     @Override
325     public Comparator<? super Integer> comparator() {
326     return s.comparator();
327     }
328    
329     @Override
330     public SortedSet<Integer> subSet(Integer fromElement, Integer toElement) {
331     return s.subSet(fromElement, toElement);
332     }
333    
334     @Override
335     public SortedSet<Integer> headSet(Integer toElement) {
336     return s.headSet(toElement);
337     }
338    
339     @Override
340     public SortedSet<Integer> tailSet(Integer fromElement) {
341     return s.tailSet(fromElement);
342     }
343    
344     @Override
345     public Integer first() {
346     return s.first();
347     }
348    
349     @Override
350     public Integer last() {
351     return s.last();
352     }
353    
354     @Override
355     public Spliterator<Integer> spliterator() {
356     return SortedSet.super.spliterator();
357     }
358     }
359     db.addCollection(
360     c -> new AbstractSortedSetImpl(c));
361    
362     class IterableWrapper implements Iterable<Integer> {
363     final Iterable<Integer> it;
364    
365     IterableWrapper(Iterable<Integer> it) {
366     this.it = it;
367     }
368    
369     @Override
370     public Iterator<Integer> iterator() {
371     return it.iterator();
372     }
373     }
374     db.add("new Iterable.spliterator()",
375     () -> new IterableWrapper(exp).spliterator());
376    
377     //
378    
379     db.add("Arrays.asList().spliterator()",
380     () -> Spliterators.spliterator(Arrays.asList(exp.toArray(new Integer[0])), 0));
381    
382     db.addList(ArrayList::new);
383    
384     db.addList(LinkedList::new);
385    
386     db.addList(Vector::new);
387    
388 jsr166 1.5 class AbstractRandomAccessListImpl extends AbstractList<Integer> implements RandomAccess {
389     Integer[] ia;
390    
391     AbstractRandomAccessListImpl(Collection<Integer> c) {
392     this.ia = c.toArray(new Integer[c.size()]);
393     }
394    
395     @Override
396     public Integer get(int index) {
397     return ia[index];
398     }
399    
400     @Override
401     public int size() {
402     return ia.length;
403     }
404     }
405     db.addList(AbstractRandomAccessListImpl::new);
406    
407     class RandomAccessListImpl implements List<Integer>, RandomAccess {
408     Integer[] ia;
409     List<Integer> l;
410    
411     RandomAccessListImpl(Collection<Integer> c) {
412     this.ia = c.toArray(new Integer[c.size()]);
413     this.l = Arrays.asList(ia);
414     }
415    
416     @Override
417     public Integer get(int index) {
418     return ia[index];
419     }
420    
421     @Override
422     public Integer set(int index, Integer element) {
423     throw new UnsupportedOperationException();
424     }
425    
426     @Override
427     public void add(int index, Integer element) {
428     throw new UnsupportedOperationException();
429     }
430    
431     @Override
432     public Integer remove(int index) {
433     throw new UnsupportedOperationException();
434     }
435    
436     @Override
437     public int indexOf(Object o) {
438     return l.indexOf(o);
439     }
440    
441     @Override
442     public int lastIndexOf(Object o) {
443     return Arrays.asList(ia).lastIndexOf(o);
444     }
445    
446     @Override
447     public ListIterator<Integer> listIterator() {
448     return l.listIterator();
449     }
450    
451     @Override
452     public ListIterator<Integer> listIterator(int index) {
453     return l.listIterator(index);
454     }
455    
456     @Override
457     public List<Integer> subList(int fromIndex, int toIndex) {
458     return l.subList(fromIndex, toIndex);
459     }
460    
461     @Override
462     public int size() {
463     return ia.length;
464     }
465    
466     @Override
467     public boolean isEmpty() {
468     return size() != 0;
469     }
470    
471     @Override
472     public boolean contains(Object o) {
473     return l.contains(o);
474     }
475    
476     @Override
477     public Iterator<Integer> iterator() {
478     return l.iterator();
479     }
480    
481     @Override
482     public Object[] toArray() {
483     return l.toArray();
484     }
485    
486     @Override
487     public <T> T[] toArray(T[] a) {
488     return l.toArray(a);
489     }
490    
491     @Override
492     public boolean add(Integer integer) {
493     throw new UnsupportedOperationException();
494     }
495    
496     @Override
497     public boolean remove(Object o) {
498     throw new UnsupportedOperationException();
499     }
500    
501     @Override
502     public boolean containsAll(Collection<?> c) {
503     return l.containsAll(c);
504     }
505    
506     @Override
507     public boolean addAll(Collection<? extends Integer> c) {
508     throw new UnsupportedOperationException();
509     }
510    
511     @Override
512     public boolean addAll(int index, Collection<? extends Integer> c) {
513     throw new UnsupportedOperationException();
514     }
515    
516     @Override
517     public boolean removeAll(Collection<?> c) {
518     throw new UnsupportedOperationException();
519     }
520    
521     @Override
522     public boolean retainAll(Collection<?> c) {
523     throw new UnsupportedOperationException();
524     }
525    
526     @Override
527     public void clear() {
528     throw new UnsupportedOperationException();
529     }
530     }
531     db.addList(RandomAccessListImpl::new);
532 jsr166 1.1
533     db.addCollection(HashSet::new);
534    
535     db.addCollection(LinkedHashSet::new);
536    
537     db.addCollection(TreeSet::new);
538    
539    
540     db.addCollection(c -> { Stack<Integer> s = new Stack<>(); s.addAll(c); return s;});
541    
542     db.addCollection(PriorityQueue::new);
543    
544     db.addCollection(ArrayDeque::new);
545    
546    
547     db.addCollection(ConcurrentSkipListSet::new);
548    
549     if (size > 0) {
550     db.addCollection(c -> {
551     ArrayBlockingQueue<Integer> abq = new ArrayBlockingQueue<>(size);
552     abq.addAll(c);
553     return abq;
554     });
555     }
556    
557     db.addCollection(PriorityBlockingQueue::new);
558    
559     db.addCollection(LinkedBlockingQueue::new);
560    
561     db.addCollection(LinkedTransferQueue::new);
562    
563     db.addCollection(ConcurrentLinkedQueue::new);
564    
565     db.addCollection(LinkedBlockingDeque::new);
566    
567     db.addCollection(CopyOnWriteArrayList::new);
568    
569     db.addCollection(CopyOnWriteArraySet::new);
570    
571     if (size == 0) {
572     db.addCollection(c -> Collections.<Integer>emptySet());
573     db.addList(c -> Collections.<Integer>emptyList());
574     }
575     else if (size == 1) {
576     db.addCollection(c -> Collections.singleton(exp.get(0)));
577     db.addCollection(c -> Collections.singletonList(exp.get(0)));
578     }
579    
580     {
581     Integer[] ai = new Integer[size];
582     Arrays.fill(ai, 1);
583     db.add(String.format("Collections.nCopies(%d, 1)", exp.size()),
584     Arrays.asList(ai),
585     () -> Collections.nCopies(exp.size(), 1).spliterator());
586     }
587    
588     // Collections.synchronized/unmodifiable/checked wrappers
589     db.addCollection(Collections::unmodifiableCollection);
590     db.addCollection(c -> Collections.unmodifiableSet(new HashSet<>(c)));
591     db.addCollection(c -> Collections.unmodifiableSortedSet(new TreeSet<>(c)));
592     db.addList(c -> Collections.unmodifiableList(new ArrayList<>(c)));
593     db.addMap(Collections::unmodifiableMap);
594     db.addMap(m -> Collections.unmodifiableSortedMap(new TreeMap<>(m)));
595    
596     db.addCollection(Collections::synchronizedCollection);
597     db.addCollection(c -> Collections.synchronizedSet(new HashSet<>(c)));
598     db.addCollection(c -> Collections.synchronizedSortedSet(new TreeSet<>(c)));
599     db.addList(c -> Collections.synchronizedList(new ArrayList<>(c)));
600     db.addMap(Collections::synchronizedMap);
601     db.addMap(m -> Collections.synchronizedSortedMap(new TreeMap<>(m)));
602    
603     db.addCollection(c -> Collections.checkedCollection(c, Integer.class));
604     db.addCollection(c -> Collections.checkedQueue(new ArrayDeque<>(c), Integer.class));
605     db.addCollection(c -> Collections.checkedSet(new HashSet<>(c), Integer.class));
606     db.addCollection(c -> Collections.checkedSortedSet(new TreeSet<>(c), Integer.class));
607     db.addList(c -> Collections.checkedList(new ArrayList<>(c), Integer.class));
608     db.addMap(c -> Collections.checkedMap(c, Integer.class, Integer.class));
609     db.addMap(m -> Collections.checkedSortedMap(new TreeMap<>(m), Integer.class, Integer.class));
610    
611     // Maps
612    
613     db.addMap(HashMap::new);
614    
615     db.addMap(m -> {
616     // Create a Map ensuring that for large sizes
617     // buckets will contain 2 or more entries
618     HashMap<Integer, Integer> cm = new HashMap<>(1, m.size() + 1);
619     // Don't use putAll which inflates the table by
620     // m.size() * loadFactor, thus creating a very sparse
621     // map for 1000 entries defeating the purpose of this test,
622     // in addition it will cause the split until null test to fail
623     // because the number of valid splits is larger than the
624     // threshold
625     for (Map.Entry<Integer, Integer> e : m.entrySet())
626     cm.put(e.getKey(), e.getValue());
627     return cm;
628     }, "new java.util.HashMap(1, size + 1)");
629    
630     db.addMap(LinkedHashMap::new);
631    
632     db.addMap(IdentityHashMap::new);
633    
634     db.addMap(WeakHashMap::new);
635    
636     db.addMap(m -> {
637     // Create a Map ensuring that for large sizes
638     // buckets will be consist of 2 or more entries
639     WeakHashMap<Integer, Integer> cm = new WeakHashMap<>(1, m.size() + 1);
640     for (Map.Entry<Integer, Integer> e : m.entrySet())
641     cm.put(e.getKey(), e.getValue());
642     return cm;
643     }, "new java.util.WeakHashMap(1, size + 1)");
644    
645     // @@@ Descending maps etc
646     db.addMap(TreeMap::new);
647    
648     db.addMap(ConcurrentHashMap::new);
649    
650     db.addMap(ConcurrentSkipListMap::new);
651    
652     if (size == 0) {
653     db.addMap(m -> Collections.<Integer, Integer>emptyMap());
654     }
655     else if (size == 1) {
656     db.addMap(m -> Collections.singletonMap(exp.get(0), exp.get(0)));
657     }
658     }
659    
660     return spliteratorDataProvider = data.toArray(new Object[0][]);
661     }
662    
663     private static List<Integer> listIntRange(int upTo) {
664     List<Integer> exp = new ArrayList<>();
665     for (int i = 0; i < upTo; i++)
666     exp.add(i);
667     return Collections.unmodifiableList(exp);
668     }
669    
670     @Test(dataProvider = "Spliterator<Integer>")
671     @SuppressWarnings({"unchecked", "rawtypes"})
672     public void testNullPointerException(String description, Collection exp, Supplier<Spliterator> s) {
673     executeAndCatch(NullPointerException.class, () -> s.get().forEachRemaining(null));
674     executeAndCatch(NullPointerException.class, () -> s.get().tryAdvance(null));
675     }
676    
677     @Test(dataProvider = "Spliterator<Integer>")
678     @SuppressWarnings({"unchecked", "rawtypes"})
679     public void testForEach(String description, Collection exp, Supplier<Spliterator> s) {
680     testForEach(exp, s, (Consumer<Object> b) -> b);
681     }
682    
683     @Test(dataProvider = "Spliterator<Integer>")
684     @SuppressWarnings({"unchecked", "rawtypes"})
685     public void testTryAdvance(String description, Collection exp, Supplier<Spliterator> s) {
686     testTryAdvance(exp, s, (Consumer<Object> b) -> b);
687     }
688    
689     @Test(dataProvider = "Spliterator<Integer>")
690     @SuppressWarnings({"unchecked", "rawtypes"})
691     public void testMixedTryAdvanceForEach(String description, Collection exp, Supplier<Spliterator> s) {
692     testMixedTryAdvanceForEach(exp, s, (Consumer<Object> b) -> b);
693     }
694    
695     @Test(dataProvider = "Spliterator<Integer>")
696     @SuppressWarnings({"unchecked", "rawtypes"})
697     public void testSplitAfterFullTraversal(String description, Collection exp, Supplier<Spliterator> s) {
698     testSplitAfterFullTraversal(s, (Consumer<Object> b) -> b);
699     }
700    
701     @Test(dataProvider = "Spliterator<Integer>")
702     @SuppressWarnings({"unchecked", "rawtypes"})
703     public void testSplitOnce(String description, Collection exp, Supplier<Spliterator> s) {
704     testSplitOnce(exp, s, (Consumer<Object> b) -> b);
705     }
706    
707     @Test(dataProvider = "Spliterator<Integer>")
708     @SuppressWarnings({"unchecked", "rawtypes"})
709     public void testSplitSixDeep(String description, Collection exp, Supplier<Spliterator> s) {
710     testSplitSixDeep(exp, s, (Consumer<Object> b) -> b);
711     }
712    
713     @Test(dataProvider = "Spliterator<Integer>")
714     @SuppressWarnings({"unchecked", "rawtypes"})
715     public void testSplitUntilNull(String description, Collection exp, Supplier<Spliterator> s) {
716     testSplitUntilNull(exp, s, (Consumer<Object> b) -> b);
717     }
718    
719     //
720    
721     private static class SpliteratorOfIntDataBuilder {
722     List<Object[]> data;
723    
724     List<Integer> exp;
725    
726     SpliteratorOfIntDataBuilder(List<Object[]> data, List<Integer> exp) {
727     this.data = data;
728     this.exp = exp;
729     }
730    
731     void add(String description, List<Integer> expected, Supplier<Spliterator.OfInt> s) {
732     description = joiner(description).toString();
733     data.add(new Object[]{description, expected, s});
734     }
735    
736     void add(String description, Supplier<Spliterator.OfInt> s) {
737     add(description, exp, s);
738     }
739    
740     StringBuilder joiner(String description) {
741     return new StringBuilder(description).
742     append(" {").
743     append("size=").append(exp.size()).
744     append("}");
745     }
746     }
747    
748     private static class SpliteratorOfIntCharDataBuilder {
749     List<Object[]> data;
750    
751     String s;
752    
753     List<Integer> expChars;
754    
755     List<Integer> expCodePoints;
756    
757     SpliteratorOfIntCharDataBuilder(List<Object[]> data, String s) {
758     this.data = data;
759     this.s = s;
760     this.expChars = transform(s, false);
761     this.expCodePoints = transform(s, true);
762     }
763    
764     static List<Integer> transform(String s, boolean toCodePoints) {
765     List<Integer> l = new ArrayList<>();
766    
767     if (!toCodePoints) {
768     for (int i = 0; i < s.length(); i++) {
769     l.add((int) s.charAt(i));
770     }
771     }
772     else {
773     for (int i = 0; i < s.length();) {
774     char c1 = s.charAt(i++);
775     int cp = c1;
776     if (Character.isHighSurrogate(c1) && i < s.length()) {
777     char c2 = s.charAt(i);
778     if (Character.isLowSurrogate(c2)) {
779     i++;
780     cp = Character.toCodePoint(c1, c2);
781     }
782     }
783     l.add(cp);
784     }
785     }
786     return l;
787     }
788    
789     void add(String description, Function<String, CharSequence> f) {
790     description = description.replace("%s", s);
791     {
792     Supplier<Spliterator.OfInt> supplier = () -> f.apply(s).chars().spliterator();
793     data.add(new Object[]{description + ".chars().spliterator()", expChars, supplier});
794     }
795     {
796     Supplier<Spliterator.OfInt> supplier = () -> f.apply(s).codePoints().spliterator();
797     data.add(new Object[]{description + ".codePoints().spliterator()", expCodePoints, supplier});
798     }
799     }
800     }
801    
802     static Object[][] spliteratorOfIntDataProvider;
803    
804     @DataProvider(name = "Spliterator.OfInt")
805     public static Object[][] spliteratorOfIntDataProvider() {
806     if (spliteratorOfIntDataProvider != null) {
807     return spliteratorOfIntDataProvider;
808     }
809    
810     List<Object[]> data = new ArrayList<>();
811     for (int size : SIZES) {
812     int exp[] = arrayIntRange(size);
813     SpliteratorOfIntDataBuilder db = new SpliteratorOfIntDataBuilder(data, listIntRange(size));
814    
815     db.add("Spliterators.spliterator(int[], ...)",
816     () -> Spliterators.spliterator(exp, 0));
817    
818     db.add("Arrays.spliterator(int[], ...)",
819     () -> Arrays.spliterator(exp));
820    
821     db.add("Spliterators.spliterator(PrimitiveIterator.OfInt, ...)",
822     () -> Spliterators.spliterator(Spliterators.iterator(Arrays.spliterator(exp)), exp.length, 0));
823    
824     db.add("Spliterators.spliteratorUnknownSize(PrimitiveIterator.OfInt, ...)",
825     () -> Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(exp)), 0));
826    
827     class IntSpliteratorFromArray extends Spliterators.AbstractIntSpliterator {
828     int[] a;
829     int index = 0;
830    
831     IntSpliteratorFromArray(int[] a) {
832     super(a.length, Spliterator.SIZED);
833     this.a = a;
834     }
835    
836     @Override
837     public boolean tryAdvance(IntConsumer action) {
838     if (action == null)
839     throw new NullPointerException();
840     if (index < a.length) {
841     action.accept(a[index++]);
842     return true;
843     }
844     else {
845     return false;
846     }
847     }
848     }
849     db.add("new Spliterators.AbstractIntAdvancingSpliterator()",
850     () -> new IntSpliteratorFromArray(exp));
851     }
852    
853     // Class for testing default methods
854     class CharSequenceImpl implements CharSequence {
855     final String s;
856    
857     public CharSequenceImpl(String s) {
858     this.s = s;
859     }
860    
861     @Override
862     public int length() {
863     return s.length();
864     }
865    
866     @Override
867     public char charAt(int index) {
868     return s.charAt(index);
869     }
870    
871     @Override
872     public CharSequence subSequence(int start, int end) {
873     return s.subSequence(start, end);
874     }
875    
876     @Override
877     public String toString() {
878     return s;
879     }
880     }
881    
882     for (String string : STRINGS) {
883     SpliteratorOfIntCharDataBuilder cdb = new SpliteratorOfIntCharDataBuilder(data, string);
884     cdb.add("\"%s\"", s -> s);
885     cdb.add("new CharSequenceImpl(\"%s\")", CharSequenceImpl::new);
886     cdb.add("new StringBuilder(\"%s\")", StringBuilder::new);
887     cdb.add("new StringBuffer(\"%s\")", StringBuffer::new);
888 jsr166 1.6 cdb.add("CharBuffer.wrap(\"%s\".toCharArray())", s -> CharBuffer.wrap(s.toCharArray()));
889 jsr166 1.1 }
890    
891 jsr166 1.6
892     Object[][] bitStreamTestcases = new Object[][] {
893     { "none", IntStream.empty().toArray() },
894     { "index 0", IntStream.of(0).toArray() },
895     { "index 255", IntStream.of(255).toArray() },
896     { "index 0 and 255", IntStream.of(0, 255).toArray() },
897     { "every bit", IntStream.range(0, 255).toArray() },
898     { "step 2", IntStream.range(0, 255).map(f -> f * 2).toArray() },
899     { "step 3", IntStream.range(0, 255).map(f -> f * 3).toArray() },
900     { "step 5", IntStream.range(0, 255).map(f -> f * 5).toArray() },
901     { "step 7", IntStream.range(0, 255).map(f -> f * 7).toArray() },
902     { "1, 10, 100, 1000", IntStream.of(1, 10, 100, 1000).toArray() },
903     };
904     for (Object[] tc : bitStreamTestcases) {
905     String description = (String)tc[0];
906     int[] exp = (int[])tc[1];
907     SpliteratorOfIntDataBuilder db = new SpliteratorOfIntDataBuilder(
908     data, IntStream.of(exp).boxed().collect(toList()));
909    
910     db.add("BitSet.stream.spliterator() {" + description + "}", () ->
911     IntStream.of(exp).collect(BitSet::new, BitSet::set, BitSet::or).
912     stream().spliterator()
913     );
914     }
915 jsr166 1.1 return spliteratorOfIntDataProvider = data.toArray(new Object[0][]);
916     }
917    
918     private static int[] arrayIntRange(int upTo) {
919     int[] exp = new int[upTo];
920     for (int i = 0; i < upTo; i++)
921     exp[i] = i;
922     return exp;
923     }
924    
925     private static UnaryOperator<Consumer<Integer>> intBoxingConsumer() {
926     class BoxingAdapter implements Consumer<Integer>, IntConsumer {
927     private final Consumer<Integer> b;
928    
929     BoxingAdapter(Consumer<Integer> b) {
930     this.b = b;
931     }
932    
933     @Override
934     public void accept(Integer value) {
935     throw new IllegalStateException();
936     }
937    
938     @Override
939     public void accept(int value) {
940     b.accept(value);
941     }
942     }
943    
944     return b -> new BoxingAdapter(b);
945     }
946    
947     @Test(dataProvider = "Spliterator.OfInt")
948     public void testIntNullPointerException(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
949     executeAndCatch(NullPointerException.class, () -> s.get().forEachRemaining((IntConsumer) null));
950     executeAndCatch(NullPointerException.class, () -> s.get().tryAdvance((IntConsumer) null));
951     }
952    
953     @Test(dataProvider = "Spliterator.OfInt")
954     public void testIntForEach(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
955     testForEach(exp, s, intBoxingConsumer());
956     }
957    
958     @Test(dataProvider = "Spliterator.OfInt")
959     public void testIntTryAdvance(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
960     testTryAdvance(exp, s, intBoxingConsumer());
961     }
962    
963     @Test(dataProvider = "Spliterator.OfInt")
964     public void testIntMixedTryAdvanceForEach(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
965     testMixedTryAdvanceForEach(exp, s, intBoxingConsumer());
966     }
967    
968     @Test(dataProvider = "Spliterator.OfInt")
969     public void testIntSplitAfterFullTraversal(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
970     testSplitAfterFullTraversal(s, intBoxingConsumer());
971     }
972    
973     @Test(dataProvider = "Spliterator.OfInt")
974     public void testIntSplitOnce(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
975     testSplitOnce(exp, s, intBoxingConsumer());
976     }
977    
978     @Test(dataProvider = "Spliterator.OfInt")
979     public void testIntSplitSixDeep(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
980     testSplitSixDeep(exp, s, intBoxingConsumer());
981     }
982    
983     @Test(dataProvider = "Spliterator.OfInt")
984     public void testIntSplitUntilNull(String description, Collection<Integer> exp, Supplier<Spliterator.OfInt> s) {
985     testSplitUntilNull(exp, s, intBoxingConsumer());
986     }
987    
988     //
989    
990     private static class SpliteratorOfLongDataBuilder {
991     List<Object[]> data;
992    
993     List<Long> exp;
994    
995     SpliteratorOfLongDataBuilder(List<Object[]> data, List<Long> exp) {
996     this.data = data;
997     this.exp = exp;
998     }
999    
1000     void add(String description, List<Long> expected, Supplier<Spliterator.OfLong> s) {
1001     description = joiner(description).toString();
1002     data.add(new Object[]{description, expected, s});
1003     }
1004    
1005     void add(String description, Supplier<Spliterator.OfLong> s) {
1006     add(description, exp, s);
1007     }
1008    
1009     StringBuilder joiner(String description) {
1010     return new StringBuilder(description).
1011     append(" {").
1012     append("size=").append(exp.size()).
1013     append("}");
1014     }
1015     }
1016    
1017     static Object[][] spliteratorOfLongDataProvider;
1018    
1019     @DataProvider(name = "Spliterator.OfLong")
1020     public static Object[][] spliteratorOfLongDataProvider() {
1021     if (spliteratorOfLongDataProvider != null) {
1022     return spliteratorOfLongDataProvider;
1023     }
1024    
1025     List<Object[]> data = new ArrayList<>();
1026     for (int size : SIZES) {
1027     long exp[] = arrayLongRange(size);
1028     SpliteratorOfLongDataBuilder db = new SpliteratorOfLongDataBuilder(data, listLongRange(size));
1029    
1030     db.add("Spliterators.spliterator(long[], ...)",
1031     () -> Spliterators.spliterator(exp, 0));
1032    
1033     db.add("Arrays.spliterator(long[], ...)",
1034     () -> Arrays.spliterator(exp));
1035    
1036     db.add("Spliterators.spliterator(PrimitiveIterator.OfLong, ...)",
1037     () -> Spliterators.spliterator(Spliterators.iterator(Arrays.spliterator(exp)), exp.length, 0));
1038    
1039     db.add("Spliterators.spliteratorUnknownSize(PrimitiveIterator.OfLong, ...)",
1040     () -> Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(exp)), 0));
1041    
1042     class LongSpliteratorFromArray extends Spliterators.AbstractLongSpliterator {
1043     long[] a;
1044     int index = 0;
1045    
1046     LongSpliteratorFromArray(long[] a) {
1047     super(a.length, Spliterator.SIZED);
1048     this.a = a;
1049     }
1050    
1051     @Override
1052     public boolean tryAdvance(LongConsumer action) {
1053     if (action == null)
1054     throw new NullPointerException();
1055     if (index < a.length) {
1056     action.accept(a[index++]);
1057     return true;
1058     }
1059     else {
1060     return false;
1061     }
1062     }
1063     }
1064     db.add("new Spliterators.AbstractLongAdvancingSpliterator()",
1065     () -> new LongSpliteratorFromArray(exp));
1066     }
1067    
1068     return spliteratorOfLongDataProvider = data.toArray(new Object[0][]);
1069     }
1070    
1071     private static List<Long> listLongRange(int upTo) {
1072     List<Long> exp = new ArrayList<>();
1073     for (long i = 0; i < upTo; i++)
1074     exp.add(i);
1075     return Collections.unmodifiableList(exp);
1076     }
1077    
1078     private static long[] arrayLongRange(int upTo) {
1079     long[] exp = new long[upTo];
1080     for (int i = 0; i < upTo; i++)
1081     exp[i] = i;
1082     return exp;
1083     }
1084    
1085     private static UnaryOperator<Consumer<Long>> longBoxingConsumer() {
1086     class BoxingAdapter implements Consumer<Long>, LongConsumer {
1087     private final Consumer<Long> b;
1088    
1089     BoxingAdapter(Consumer<Long> b) {
1090     this.b = b;
1091     }
1092    
1093     @Override
1094     public void accept(Long value) {
1095     throw new IllegalStateException();
1096     }
1097    
1098     @Override
1099     public void accept(long value) {
1100     b.accept(value);
1101     }
1102     }
1103    
1104     return b -> new BoxingAdapter(b);
1105     }
1106    
1107     @Test(dataProvider = "Spliterator.OfLong")
1108     public void testLongNullPointerException(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1109     executeAndCatch(NullPointerException.class, () -> s.get().forEachRemaining((LongConsumer) null));
1110     executeAndCatch(NullPointerException.class, () -> s.get().tryAdvance((LongConsumer) null));
1111     }
1112    
1113     @Test(dataProvider = "Spliterator.OfLong")
1114     public void testLongForEach(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1115     testForEach(exp, s, longBoxingConsumer());
1116     }
1117    
1118     @Test(dataProvider = "Spliterator.OfLong")
1119     public void testLongTryAdvance(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1120     testTryAdvance(exp, s, longBoxingConsumer());
1121     }
1122    
1123     @Test(dataProvider = "Spliterator.OfLong")
1124     public void testLongMixedTryAdvanceForEach(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1125     testMixedTryAdvanceForEach(exp, s, longBoxingConsumer());
1126     }
1127    
1128     @Test(dataProvider = "Spliterator.OfLong")
1129     public void testLongSplitAfterFullTraversal(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1130     testSplitAfterFullTraversal(s, longBoxingConsumer());
1131     }
1132    
1133     @Test(dataProvider = "Spliterator.OfLong")
1134     public void testLongSplitOnce(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1135     testSplitOnce(exp, s, longBoxingConsumer());
1136     }
1137    
1138     @Test(dataProvider = "Spliterator.OfLong")
1139     public void testLongSplitSixDeep(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1140     testSplitSixDeep(exp, s, longBoxingConsumer());
1141     }
1142    
1143     @Test(dataProvider = "Spliterator.OfLong")
1144     public void testLongSplitUntilNull(String description, Collection<Long> exp, Supplier<Spliterator.OfLong> s) {
1145     testSplitUntilNull(exp, s, longBoxingConsumer());
1146     }
1147    
1148     //
1149    
1150     private static class SpliteratorOfDoubleDataBuilder {
1151     List<Object[]> data;
1152    
1153     List<Double> exp;
1154    
1155     SpliteratorOfDoubleDataBuilder(List<Object[]> data, List<Double> exp) {
1156     this.data = data;
1157     this.exp = exp;
1158     }
1159    
1160     void add(String description, List<Double> expected, Supplier<Spliterator.OfDouble> s) {
1161     description = joiner(description).toString();
1162     data.add(new Object[]{description, expected, s});
1163     }
1164    
1165     void add(String description, Supplier<Spliterator.OfDouble> s) {
1166     add(description, exp, s);
1167     }
1168    
1169     StringBuilder joiner(String description) {
1170     return new StringBuilder(description).
1171     append(" {").
1172     append("size=").append(exp.size()).
1173     append("}");
1174     }
1175     }
1176    
1177     static Object[][] spliteratorOfDoubleDataProvider;
1178    
1179     @DataProvider(name = "Spliterator.OfDouble")
1180     public static Object[][] spliteratorOfDoubleDataProvider() {
1181     if (spliteratorOfDoubleDataProvider != null) {
1182     return spliteratorOfDoubleDataProvider;
1183     }
1184    
1185     List<Object[]> data = new ArrayList<>();
1186     for (int size : SIZES) {
1187     double exp[] = arrayDoubleRange(size);
1188     SpliteratorOfDoubleDataBuilder db = new SpliteratorOfDoubleDataBuilder(data, listDoubleRange(size));
1189    
1190     db.add("Spliterators.spliterator(double[], ...)",
1191     () -> Spliterators.spliterator(exp, 0));
1192    
1193     db.add("Arrays.spliterator(double[], ...)",
1194     () -> Arrays.spliterator(exp));
1195    
1196     db.add("Spliterators.spliterator(PrimitiveIterator.OfDouble, ...)",
1197     () -> Spliterators.spliterator(Spliterators.iterator(Arrays.spliterator(exp)), exp.length, 0));
1198    
1199     db.add("Spliterators.spliteratorUnknownSize(PrimitiveIterator.OfDouble, ...)",
1200     () -> Spliterators.spliteratorUnknownSize(Spliterators.iterator(Arrays.spliterator(exp)), 0));
1201    
1202     class DoubleSpliteratorFromArray extends Spliterators.AbstractDoubleSpliterator {
1203     double[] a;
1204     int index = 0;
1205    
1206     DoubleSpliteratorFromArray(double[] a) {
1207     super(a.length, Spliterator.SIZED);
1208     this.a = a;
1209     }
1210    
1211     @Override
1212     public boolean tryAdvance(DoubleConsumer action) {
1213     if (action == null)
1214     throw new NullPointerException();
1215     if (index < a.length) {
1216     action.accept(a[index++]);
1217     return true;
1218     }
1219     else {
1220     return false;
1221     }
1222     }
1223     }
1224     db.add("new Spliterators.AbstractDoubleAdvancingSpliterator()",
1225     () -> new DoubleSpliteratorFromArray(exp));
1226     }
1227    
1228     return spliteratorOfDoubleDataProvider = data.toArray(new Object[0][]);
1229     }
1230    
1231     private static List<Double> listDoubleRange(int upTo) {
1232     List<Double> exp = new ArrayList<>();
1233     for (double i = 0; i < upTo; i++)
1234     exp.add(i);
1235     return Collections.unmodifiableList(exp);
1236     }
1237    
1238     private static double[] arrayDoubleRange(int upTo) {
1239     double[] exp = new double[upTo];
1240     for (int i = 0; i < upTo; i++)
1241     exp[i] = i;
1242     return exp;
1243     }
1244    
1245     private static UnaryOperator<Consumer<Double>> doubleBoxingConsumer() {
1246     class BoxingAdapter implements Consumer<Double>, DoubleConsumer {
1247     private final Consumer<Double> b;
1248    
1249     BoxingAdapter(Consumer<Double> b) {
1250     this.b = b;
1251     }
1252    
1253     @Override
1254     public void accept(Double value) {
1255     throw new IllegalStateException();
1256     }
1257    
1258     @Override
1259     public void accept(double value) {
1260     b.accept(value);
1261     }
1262     }
1263    
1264     return b -> new BoxingAdapter(b);
1265     }
1266    
1267     @Test(dataProvider = "Spliterator.OfDouble")
1268     public void testDoubleNullPointerException(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1269     executeAndCatch(NullPointerException.class, () -> s.get().forEachRemaining((DoubleConsumer) null));
1270     executeAndCatch(NullPointerException.class, () -> s.get().tryAdvance((DoubleConsumer) null));
1271     }
1272    
1273     @Test(dataProvider = "Spliterator.OfDouble")
1274     public void testDoubleForEach(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1275     testForEach(exp, s, doubleBoxingConsumer());
1276     }
1277    
1278     @Test(dataProvider = "Spliterator.OfDouble")
1279     public void testDoubleTryAdvance(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1280     testTryAdvance(exp, s, doubleBoxingConsumer());
1281     }
1282    
1283     @Test(dataProvider = "Spliterator.OfDouble")
1284     public void testDoubleMixedTryAdvanceForEach(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1285     testMixedTryAdvanceForEach(exp, s, doubleBoxingConsumer());
1286     }
1287    
1288     @Test(dataProvider = "Spliterator.OfDouble")
1289     public void testDoubleSplitAfterFullTraversal(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1290     testSplitAfterFullTraversal(s, doubleBoxingConsumer());
1291     }
1292    
1293     @Test(dataProvider = "Spliterator.OfDouble")
1294     public void testDoubleSplitOnce(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1295     testSplitOnce(exp, s, doubleBoxingConsumer());
1296     }
1297    
1298     @Test(dataProvider = "Spliterator.OfDouble")
1299     public void testDoubleSplitSixDeep(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1300     testSplitSixDeep(exp, s, doubleBoxingConsumer());
1301     }
1302    
1303     @Test(dataProvider = "Spliterator.OfDouble")
1304     public void testDoubleSplitUntilNull(String description, Collection<Double> exp, Supplier<Spliterator.OfDouble> s) {
1305     testSplitUntilNull(exp, s, doubleBoxingConsumer());
1306     }
1307    
1308     //
1309    
1310     private static <T, S extends Spliterator<T>> void testForEach(
1311     Collection<T> exp,
1312     Supplier<S> supplier,
1313     UnaryOperator<Consumer<T>> boxingAdapter) {
1314     S spliterator = supplier.get();
1315     long sizeIfKnown = spliterator.getExactSizeIfKnown();
1316     boolean isOrdered = spliterator.hasCharacteristics(Spliterator.ORDERED);
1317    
1318     ArrayList<T> fromForEach = new ArrayList<>();
1319     spliterator = supplier.get();
1320     Consumer<T> addToFromForEach = boxingAdapter.apply(fromForEach::add);
1321     spliterator.forEachRemaining(addToFromForEach);
1322    
1323     // Assert that forEach now produces no elements
1324     spliterator.forEachRemaining(boxingAdapter.apply(e -> fail("Spliterator.forEach produced an element after spliterator exhausted: " + e)));
1325     // Assert that tryAdvance now produce no elements
1326     spliterator.tryAdvance(boxingAdapter.apply(e -> fail("Spliterator.tryAdvance produced an element after spliterator exhausted: " + e)));
1327    
1328     // assert that size, tryAdvance, and forEach are consistent
1329     if (sizeIfKnown >= 0) {
1330     assertEquals(sizeIfKnown, exp.size());
1331     }
1332     assertEquals(fromForEach.size(), exp.size());
1333    
1334     assertContents(fromForEach, exp, isOrdered);
1335     }
1336    
1337     private static <T, S extends Spliterator<T>> void testTryAdvance(
1338     Collection<T> exp,
1339     Supplier<S> supplier,
1340     UnaryOperator<Consumer<T>> boxingAdapter) {
1341     S spliterator = supplier.get();
1342     long sizeIfKnown = spliterator.getExactSizeIfKnown();
1343     boolean isOrdered = spliterator.hasCharacteristics(Spliterator.ORDERED);
1344    
1345     spliterator = supplier.get();
1346     ArrayList<T> fromTryAdvance = new ArrayList<>();
1347     Consumer<T> addToFromTryAdvance = boxingAdapter.apply(fromTryAdvance::add);
1348     while (spliterator.tryAdvance(addToFromTryAdvance)) { }
1349    
1350     // Assert that forEach now produces no elements
1351     spliterator.forEachRemaining(boxingAdapter.apply(e -> fail("Spliterator.forEach produced an element after spliterator exhausted: " + e)));
1352     // Assert that tryAdvance now produce no elements
1353     spliterator.tryAdvance(boxingAdapter.apply(e -> fail("Spliterator.tryAdvance produced an element after spliterator exhausted: " + e)));
1354    
1355     // assert that size, tryAdvance, and forEach are consistent
1356     if (sizeIfKnown >= 0) {
1357     assertEquals(sizeIfKnown, exp.size());
1358     }
1359     assertEquals(fromTryAdvance.size(), exp.size());
1360    
1361     assertContents(fromTryAdvance, exp, isOrdered);
1362     }
1363    
1364     private static <T, S extends Spliterator<T>> void testMixedTryAdvanceForEach(
1365     Collection<T> exp,
1366     Supplier<S> supplier,
1367     UnaryOperator<Consumer<T>> boxingAdapter) {
1368     S spliterator = supplier.get();
1369     long sizeIfKnown = spliterator.getExactSizeIfKnown();
1370     boolean isOrdered = spliterator.hasCharacteristics(Spliterator.ORDERED);
1371    
1372     // tryAdvance first few elements, then forEach rest
1373     ArrayList<T> dest = new ArrayList<>();
1374     spliterator = supplier.get();
1375     Consumer<T> addToDest = boxingAdapter.apply(dest::add);
1376     for (int i = 0; i < 10 && spliterator.tryAdvance(addToDest); i++) { }
1377     spliterator.forEachRemaining(addToDest);
1378    
1379     // Assert that forEach now produces no elements
1380     spliterator.forEachRemaining(boxingAdapter.apply(e -> fail("Spliterator.forEach produced an element after spliterator exhausted: " + e)));
1381     // Assert that tryAdvance now produce no elements
1382     spliterator.tryAdvance(boxingAdapter.apply(e -> fail("Spliterator.tryAdvance produced an element after spliterator exhausted: " + e)));
1383    
1384     if (sizeIfKnown >= 0) {
1385     assertEquals(sizeIfKnown, dest.size());
1386     }
1387     assertEquals(dest.size(), exp.size());
1388    
1389     if (isOrdered) {
1390     assertEquals(dest, exp);
1391     }
1392     else {
1393     assertContentsUnordered(dest, exp);
1394     }
1395     }
1396    
1397     private static <T, S extends Spliterator<T>> void testSplitAfterFullTraversal(
1398     Supplier<S> supplier,
1399     UnaryOperator<Consumer<T>> boxingAdapter) {
1400     // Full traversal using tryAdvance
1401     Spliterator<T> spliterator = supplier.get();
1402     while (spliterator.tryAdvance(boxingAdapter.apply(e -> { }))) { }
1403     Spliterator<T> split = spliterator.trySplit();
1404     assertNull(split);
1405    
1406     // Full traversal using forEach
1407     spliterator = supplier.get();
1408     spliterator.forEachRemaining(boxingAdapter.apply(e -> {
1409     }));
1410     split = spliterator.trySplit();
1411     assertNull(split);
1412    
1413     // Full traversal using tryAdvance then forEach
1414     spliterator = supplier.get();
1415     spliterator.tryAdvance(boxingAdapter.apply(e -> { }));
1416     spliterator.forEachRemaining(boxingAdapter.apply(e -> {
1417     }));
1418     split = spliterator.trySplit();
1419     assertNull(split);
1420     }
1421    
1422     private static <T, S extends Spliterator<T>> void testSplitOnce(
1423     Collection<T> exp,
1424     Supplier<S> supplier,
1425     UnaryOperator<Consumer<T>> boxingAdapter) {
1426     S spliterator = supplier.get();
1427     long sizeIfKnown = spliterator.getExactSizeIfKnown();
1428     boolean isOrdered = spliterator.hasCharacteristics(Spliterator.ORDERED);
1429    
1430     ArrayList<T> fromSplit = new ArrayList<>();
1431     Spliterator<T> s1 = supplier.get();
1432     Spliterator<T> s2 = s1.trySplit();
1433     long s1Size = s1.getExactSizeIfKnown();
1434     long s2Size = (s2 != null) ? s2.getExactSizeIfKnown() : 0;
1435     Consumer<T> addToFromSplit = boxingAdapter.apply(fromSplit::add);
1436     if (s2 != null)
1437     s2.forEachRemaining(addToFromSplit);
1438     s1.forEachRemaining(addToFromSplit);
1439    
1440     if (sizeIfKnown >= 0) {
1441     assertEquals(sizeIfKnown, fromSplit.size());
1442     if (s1Size >= 0 && s2Size >= 0)
1443     assertEquals(sizeIfKnown, s1Size + s2Size);
1444     }
1445     assertContents(fromSplit, exp, isOrdered);
1446     }
1447    
1448     private static <T, S extends Spliterator<T>> void testSplitSixDeep(
1449     Collection<T> exp,
1450     Supplier<S> supplier,
1451     UnaryOperator<Consumer<T>> boxingAdapter) {
1452     S spliterator = supplier.get();
1453     boolean isOrdered = spliterator.hasCharacteristics(Spliterator.ORDERED);
1454    
1455     for (int depth=0; depth < 6; depth++) {
1456     List<T> dest = new ArrayList<>();
1457     spliterator = supplier.get();
1458    
1459     assertRootSpliterator(spliterator);
1460    
1461     // verify splitting with forEach
1462     visit(depth, 0, dest, spliterator, boxingAdapter, spliterator.characteristics(), false);
1463     assertContents(dest, exp, isOrdered);
1464    
1465     // verify splitting with tryAdvance
1466     dest.clear();
1467     spliterator = supplier.get();
1468     visit(depth, 0, dest, spliterator, boxingAdapter, spliterator.characteristics(), true);
1469     assertContents(dest, exp, isOrdered);
1470     }
1471     }
1472    
1473     private static <T, S extends Spliterator<T>> void visit(int depth, int curLevel,
1474     List<T> dest, S spliterator, UnaryOperator<Consumer<T>> boxingAdapter,
1475     int rootCharacteristics, boolean useTryAdvance) {
1476     if (curLevel < depth) {
1477     long beforeSize = spliterator.getExactSizeIfKnown();
1478     Spliterator<T> split = spliterator.trySplit();
1479     if (split != null) {
1480     assertSpliterator(split, rootCharacteristics);
1481     assertSpliterator(spliterator, rootCharacteristics);
1482    
1483     if ((rootCharacteristics & Spliterator.SUBSIZED) != 0 &&
1484     (rootCharacteristics & Spliterator.SIZED) != 0) {
1485     assertEquals(beforeSize, split.estimateSize() + spliterator.estimateSize());
1486     }
1487     visit(depth, curLevel + 1, dest, split, boxingAdapter, rootCharacteristics, useTryAdvance);
1488     }
1489     visit(depth, curLevel + 1, dest, spliterator, boxingAdapter, rootCharacteristics, useTryAdvance);
1490     }
1491     else {
1492     long sizeIfKnown = spliterator.getExactSizeIfKnown();
1493     if (useTryAdvance) {
1494     Consumer<T> addToDest = boxingAdapter.apply(dest::add);
1495     int count = 0;
1496     while (spliterator.tryAdvance(addToDest)) {
1497     ++count;
1498     }
1499    
1500     if (sizeIfKnown >= 0)
1501     assertEquals(sizeIfKnown, count);
1502    
1503     // Assert that forEach now produces no elements
1504     spliterator.forEachRemaining(boxingAdapter.apply(e -> fail("Spliterator.forEach produced an element after spliterator exhausted: " + e)));
1505    
1506     Spliterator<T> split = spliterator.trySplit();
1507     assertNull(split);
1508     }
1509     else {
1510     List<T> leafDest = new ArrayList<>();
1511     Consumer<T> addToLeafDest = boxingAdapter.apply(leafDest::add);
1512     spliterator.forEachRemaining(addToLeafDest);
1513    
1514     if (sizeIfKnown >= 0)
1515     assertEquals(sizeIfKnown, leafDest.size());
1516    
1517     // Assert that forEach now produces no elements
1518     spliterator.tryAdvance(boxingAdapter.apply(e -> fail("Spliterator.tryAdvance produced an element after spliterator exhausted: " + e)));
1519    
1520     Spliterator<T> split = spliterator.trySplit();
1521     assertNull(split);
1522    
1523     dest.addAll(leafDest);
1524     }
1525     }
1526     }
1527    
1528     private static <T, S extends Spliterator<T>> void testSplitUntilNull(
1529     Collection<T> exp,
1530     Supplier<S> supplier,
1531     UnaryOperator<Consumer<T>> boxingAdapter) {
1532     Spliterator<T> s = supplier.get();
1533     boolean isOrdered = s.hasCharacteristics(Spliterator.ORDERED);
1534     assertRootSpliterator(s);
1535    
1536     List<T> splits = new ArrayList<>();
1537     Consumer<T> c = boxingAdapter.apply(splits::add);
1538    
1539     testSplitUntilNull(new SplitNode<T>(c, s));
1540     assertContents(splits, exp, isOrdered);
1541     }
1542    
1543     private static class SplitNode<T> {
1544     // Constant for every node
1545     final Consumer<T> c;
1546     final int rootCharacteristics;
1547    
1548     final Spliterator<T> s;
1549    
1550     SplitNode(Consumer<T> c, Spliterator<T> s) {
1551     this(c, s.characteristics(), s);
1552     }
1553    
1554     private SplitNode(Consumer<T> c, int rootCharacteristics, Spliterator<T> s) {
1555     this.c = c;
1556     this.rootCharacteristics = rootCharacteristics;
1557     this.s = s;
1558     }
1559    
1560     SplitNode<T> fromSplit(Spliterator<T> split) {
1561     return new SplitNode<>(c, rootCharacteristics, split);
1562     }
1563     }
1564    
1565     /**
1566     * Set the maximum stack capacity to 0.25MB. This should be more than enough to detect a bad spliterator
1567     * while not unduly disrupting test infrastructure given the test data sizes that are used are small.
1568     * Note that j.u.c.ForkJoinPool sets the max queue size to 64M (1 << 26).
1569     */
1570     private static final int MAXIMUM_STACK_CAPACITY = 1 << 18; // 0.25MB
1571    
1572     private static <T> void testSplitUntilNull(SplitNode<T> e) {
1573     // Use an explicit stack to avoid a StackOverflowException when testing a Spliterator
1574     // that when repeatedly split produces a right-balanced (and maybe degenerate) tree, or
1575     // for a spliterator that is badly behaved.
1576     Deque<SplitNode<T>> stack = new ArrayDeque<>();
1577     stack.push(e);
1578    
1579     int iteration = 0;
1580     while (!stack.isEmpty()) {
1581     assertTrue(iteration++ < MAXIMUM_STACK_CAPACITY, "Exceeded maximum stack modification count of 1 << 18");
1582    
1583     e = stack.pop();
1584     Spliterator<T> parentAndRightSplit = e.s;
1585    
1586     long parentEstimateSize = parentAndRightSplit.estimateSize();
1587     assertTrue(parentEstimateSize >= 0,
1588     String.format("Split size estimate %d < 0", parentEstimateSize));
1589    
1590     long parentSize = parentAndRightSplit.getExactSizeIfKnown();
1591     Spliterator<T> leftSplit = parentAndRightSplit.trySplit();
1592     if (leftSplit == null) {
1593     parentAndRightSplit.forEachRemaining(e.c);
1594     continue;
1595     }
1596    
1597     assertSpliterator(leftSplit, e.rootCharacteristics);
1598     assertSpliterator(parentAndRightSplit, e.rootCharacteristics);
1599    
1600     if (parentEstimateSize != Long.MAX_VALUE && leftSplit.estimateSize() > 0 && parentAndRightSplit.estimateSize() > 0) {
1601     assertTrue(leftSplit.estimateSize() < parentEstimateSize,
1602     String.format("Left split size estimate %d >= parent split size estimate %d", leftSplit.estimateSize(), parentEstimateSize));
1603     assertTrue(parentAndRightSplit.estimateSize() < parentEstimateSize,
1604     String.format("Right split size estimate %d >= parent split size estimate %d", leftSplit.estimateSize(), parentEstimateSize));
1605     }
1606     else {
1607     assertTrue(leftSplit.estimateSize() <= parentEstimateSize,
1608     String.format("Left split size estimate %d > parent split size estimate %d", leftSplit.estimateSize(), parentEstimateSize));
1609     assertTrue(parentAndRightSplit.estimateSize() <= parentEstimateSize,
1610     String.format("Right split size estimate %d > parent split size estimate %d", leftSplit.estimateSize(), parentEstimateSize));
1611     }
1612    
1613     long leftSize = leftSplit.getExactSizeIfKnown();
1614     long rightSize = parentAndRightSplit.getExactSizeIfKnown();
1615     if (parentSize >= 0 && leftSize >= 0 && rightSize >= 0)
1616     assertEquals(parentSize, leftSize + rightSize,
1617     String.format("exact left split size %d + exact right split size %d != parent exact split size %d",
1618     leftSize, rightSize, parentSize));
1619    
1620     // Add right side to stack first so left side is popped off first
1621     stack.push(e.fromSplit(parentAndRightSplit));
1622     stack.push(e.fromSplit(leftSplit));
1623     }
1624     }
1625    
1626     private static void assertRootSpliterator(Spliterator<?> s) {
1627     assertFalse(s.hasCharacteristics(Spliterator.SIZED | Spliterator.CONCURRENT),
1628     "Root spliterator should not be SIZED and CONCURRENT");
1629    
1630     assertSpliterator(s);
1631     }
1632    
1633     private static void assertSpliterator(Spliterator<?> s, int rootCharacteristics) {
1634     if ((rootCharacteristics & Spliterator.SUBSIZED) != 0) {
1635     assertTrue(s.hasCharacteristics(Spliterator.SUBSIZED),
1636     "Child split is not SUBSIZED when root split is SUBSIZED");
1637     }
1638     assertSpliterator(s);
1639     }
1640    
1641     private static void assertSpliterator(Spliterator<?> s) {
1642     if (s.hasCharacteristics(Spliterator.SUBSIZED)) {
1643     assertTrue(s.hasCharacteristics(Spliterator.SIZED));
1644     }
1645     if (s.hasCharacteristics(Spliterator.SIZED)) {
1646     assertTrue(s.estimateSize() != Long.MAX_VALUE);
1647     assertTrue(s.getExactSizeIfKnown() >= 0);
1648     }
1649     try {
1650     s.getComparator();
1651     assertTrue(s.hasCharacteristics(Spliterator.SORTED));
1652     } catch (IllegalStateException e) {
1653     assertFalse(s.hasCharacteristics(Spliterator.SORTED));
1654     }
1655     }
1656    
1657     private static<T> void assertContents(Collection<T> actual, Collection<T> expected, boolean isOrdered) {
1658     if (isOrdered) {
1659     assertEquals(actual, expected);
1660     }
1661     else {
1662     assertContentsUnordered(actual, expected);
1663     }
1664     }
1665    
1666     private static<T> void assertContentsUnordered(Iterable<T> actual, Iterable<T> expected) {
1667     assertEquals(toBoxedMultiset(actual), toBoxedMultiset(expected));
1668     }
1669    
1670     private static <T> Map<T, Integer> toBoxedMultiset(Iterable<T> c) {
1671     Map<T, Integer> result = new HashMap<>();
1672     c.forEach(e -> {
1673     if (result.containsKey(e)) result.put(e, result.get(e) + 1);
1674     else result.put(e, 1);
1675     });
1676     return result;
1677     }
1678    
1679     private void executeAndCatch(Class<? extends Exception> expected, Runnable r) {
1680     Exception caught = null;
1681     try {
1682     r.run();
1683     }
1684     catch (Exception e) {
1685     caught = e;
1686     }
1687    
1688     assertNotNull(caught,
1689     String.format("No Exception was thrown, expected an Exception of %s to be thrown",
1690     expected.getName()));
1691     assertTrue(expected.isInstance(caught),
1692     String.format("Exception thrown %s not an instance of %s",
1693     caught.getClass().getName(), expected.getName()));
1694     }
1695    
1696     }