1 /*
2  * Copyright (C) 2008 The Guava Authors
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.google.common.collect;
18 
19 import static com.google.common.collect.testing.IteratorFeature.UNMODIFIABLE;
20 import static com.google.common.truth.Truth.assertThat;
21 import static java.util.Arrays.asList;
22 
23 import com.google.common.annotations.GwtCompatible;
24 import com.google.common.annotations.GwtIncompatible;
25 import com.google.common.collect.testing.Helpers;
26 import com.google.common.collect.testing.IteratorTester;
27 import com.google.common.collect.testing.MinimalCollection;
28 import com.google.common.collect.testing.MinimalIterable;
29 
30 import junit.framework.TestCase;
31 
32 import java.util.Collection;
33 import java.util.Collections;
34 import java.util.Iterator;
35 import java.util.List;
36 import java.util.Set;
37 
38 /**
39  * Base class for {@link ImmutableSet} and  {@link ImmutableSortedSet} tests.
40  *
41  * @author Kevin Bourrillion
42  * @author Jared Levy
43  */
44 @GwtCompatible(emulated = true)
45 public abstract class AbstractImmutableSetTest extends TestCase {
46 
47   protected abstract Set<String> of();
48   protected abstract Set<String> of(String e);
49   protected abstract Set<String> of(String e1, String e2);
50   protected abstract Set<String> of(String e1, String e2, String e3);
51   protected abstract Set<String> of(String e1, String e2, String e3, String e4);
52   protected abstract Set<String> of(String e1, String e2, String e3, String e4,
53       String e5);
54   protected abstract Set<String> of(String e1, String e2, String e3, String e4,
55       String e5, String e6, String... rest);
56   protected abstract Set<String> copyOf(String[] elements);
57   protected abstract Set<String> copyOf(Collection<String> elements);
58   protected abstract Set<String> copyOf(Iterable<String> elements);
59   protected abstract Set<String> copyOf(Iterator<String> elements);
60 
61   public void testCreation_noArgs() {
62     Set<String> set = of();
63     assertEquals(Collections.<String>emptySet(), set);
64     assertSame(of(), set);
65   }
66 
67   public void testCreation_oneElement() {
68     Set<String> set = of("a");
69     assertEquals(Collections.singleton("a"), set);
70   }
71 
72   public void testCreation_twoElements() {
73     Set<String> set = of("a", "b");
74     assertEquals(Sets.newHashSet("a", "b"), set);
75   }
76 
77   public void testCreation_threeElements() {
78     Set<String> set = of("a", "b", "c");
79     assertEquals(Sets.newHashSet("a", "b", "c"), set);
80   }
81 
82   public void testCreation_fourElements() {
83     Set<String> set = of("a", "b", "c", "d");
84     assertEquals(Sets.newHashSet("a", "b", "c", "d"), set);
85   }
86 
87   public void testCreation_fiveElements() {
88     Set<String> set = of("a", "b", "c", "d", "e");
89     assertEquals(Sets.newHashSet("a", "b", "c", "d", "e"), set);
90   }
91 
92   public void testCreation_sixElements() {
93     Set<String> set = of("a", "b", "c", "d", "e", "f");
94     assertEquals(Sets.newHashSet("a", "b", "c", "d", "e", "f"), set);
95   }
96 
97   public void testCreation_sevenElements() {
98     Set<String> set = of("a", "b", "c", "d", "e", "f", "g");
99     assertEquals(Sets.newHashSet("a", "b", "c", "d", "e", "f", "g"), set);
100   }
101 
102   public void testCreation_eightElements() {
103     Set<String> set = of("a", "b", "c", "d", "e", "f", "g", "h");
104     assertEquals(Sets.newHashSet("a", "b", "c", "d", "e", "f", "g", "h"), set);
105   }
106 
107   public void testCopyOf_emptyArray() {
108     String[] array = new String[0];
109     Set<String> set = copyOf(array);
110     assertEquals(Collections.<String>emptySet(), set);
111     assertSame(of(), set);
112   }
113 
114   public void testCopyOf_arrayOfOneElement() {
115     String[] array = new String[] { "a" };
116     Set<String> set = copyOf(array);
117     assertEquals(Collections.singleton("a"), set);
118   }
119 
120   public void testCopyOf_nullArray() {
121     try {
122       copyOf((String[]) null);
123       fail();
124     } catch (NullPointerException expected) {
125     }
126   }
127 
128   public void testCopyOf_arrayContainingOnlyNull() {
129     String[] array = new String[] { null };
130     try {
131       copyOf(array);
132       fail();
133     } catch (NullPointerException expected) {
134     }
135   }
136 
137   public void testCopyOf_collection_empty() {
138     // "<String>" is required to work around a javac 1.5 bug.
139     Collection<String> c = MinimalCollection.<String>of();
140     Set<String> set = copyOf(c);
141     assertEquals(Collections.<String>emptySet(), set);
142     assertSame(of(), set);
143   }
144 
145   public void testCopyOf_collection_oneElement() {
146     Collection<String> c = MinimalCollection.of("a");
147     Set<String> set = copyOf(c);
148     assertEquals(Collections.singleton("a"), set);
149   }
150 
151   public void testCopyOf_collection_oneElementRepeated() {
152     Collection<String> c = MinimalCollection.of("a", "a", "a");
153     Set<String> set = copyOf(c);
154     assertEquals(Collections.singleton("a"), set);
155   }
156 
157   public void testCopyOf_collection_general() {
158     Collection<String> c = MinimalCollection.of("a", "b", "a");
159     Set<String> set = copyOf(c);
160     assertEquals(2, set.size());
161     assertTrue(set.contains("a"));
162     assertTrue(set.contains("b"));
163   }
164 
165   public void testCopyOf_collectionContainingNull() {
166     Collection<String> c = MinimalCollection.of("a", null, "b");
167     try {
168       copyOf(c);
169       fail();
170     } catch (NullPointerException expected) {
171     }
172   }
173 
174   public void testCopyOf_iterator_empty() {
175     Iterator<String> iterator = Iterators.emptyIterator();
176     Set<String> set = copyOf(iterator);
177     assertEquals(Collections.<String>emptySet(), set);
178     assertSame(of(), set);
179   }
180 
181   public void testCopyOf_iterator_oneElement() {
182     Iterator<String> iterator = Iterators.singletonIterator("a");
183     Set<String> set = copyOf(iterator);
184     assertEquals(Collections.singleton("a"), set);
185   }
186 
187   public void testCopyOf_iterator_oneElementRepeated() {
188     Iterator<String> iterator = Iterators.forArray("a", "a", "a");
189     Set<String> set = copyOf(iterator);
190     assertEquals(Collections.singleton("a"), set);
191   }
192 
193   public void testCopyOf_iterator_general() {
194     Iterator<String> iterator = Iterators.forArray("a", "b", "a");
195     Set<String> set = copyOf(iterator);
196     assertEquals(2, set.size());
197     assertTrue(set.contains("a"));
198     assertTrue(set.contains("b"));
199   }
200 
201   public void testCopyOf_iteratorContainingNull() {
202     Iterator<String> c = Iterators.forArray("a", null, "b");
203     try {
204       copyOf(c);
205       fail();
206     } catch (NullPointerException expected) {
207     }
208   }
209 
210   private static class CountingIterable implements Iterable<String> {
211     int count = 0;
212     @Override
213     public Iterator<String> iterator() {
214       count++;
215       return Iterators.forArray("a", "b", "a");
216     }
217   }
218 
219   public void testCopyOf_plainIterable() {
220     CountingIterable iterable = new CountingIterable();
221     Set<String> set = copyOf(iterable);
222     assertEquals(2, set.size());
223     assertTrue(set.contains("a"));
224     assertTrue(set.contains("b"));
225   }
226 
227   public void testCopyOf_plainIterable_iteratesOnce() {
228     CountingIterable iterable = new CountingIterable();
229     copyOf(iterable);
230     assertEquals(1, iterable.count);
231   }
232 
233   public void testCopyOf_shortcut_empty() {
234     Collection<String> c = of();
235     assertEquals(Collections.<String>emptySet(), copyOf(c));
236     assertSame(c, copyOf(c));
237   }
238 
239   public void testCopyOf_shortcut_singleton() {
240     Collection<String> c = of("a");
241     assertEquals(Collections.singleton("a"), copyOf(c));
242     assertSame(c, copyOf(c));
243   }
244 
245   public void testCopyOf_shortcut_sameType() {
246     Collection<String> c = of("a", "b", "c");
247     assertSame(c, copyOf(c));
248   }
249 
250   public void testToString() {
251     Set<String> set = of("a", "b", "c", "d", "e", "f", "g");
252     assertEquals("[a, b, c, d, e, f, g]", set.toString());
253   }
254 
255   @GwtIncompatible("slow (~40s)")
256   public void testIterator_oneElement() {
257     new IteratorTester<String>(5, UNMODIFIABLE, Collections.singleton("a"),
258         IteratorTester.KnownOrder.KNOWN_ORDER) {
259       @Override protected Iterator<String> newTargetIterator() {
260         return of("a").iterator();
261       }
262     }.test();
263   }
264 
265   @GwtIncompatible("slow (~30s)")
266   public void testIterator_general() {
267     new IteratorTester<String>(5, UNMODIFIABLE, asList("a", "b", "c"),
268         IteratorTester.KnownOrder.KNOWN_ORDER) {
269       @Override protected Iterator<String> newTargetIterator() {
270         return of("a", "b", "c").iterator();
271       }
272     }.test();
273   }
274 
275   public void testContainsAll_sameType() {
276     Collection<String> c = of("a", "b", "c");
277     assertFalse(c.containsAll(of("a", "b", "c", "d")));
278     assertFalse(c.containsAll(of("a", "d")));
279     assertTrue(c.containsAll(of("a", "c")));
280     assertTrue(c.containsAll(of("a", "b", "c")));
281   }
282 
283   public void testEquals_sameType() {
284     Collection<String> c = of("a", "b", "c");
285     assertTrue(c.equals(of("a", "b", "c")));
286     assertFalse(c.equals(of("a", "b", "d")));
287   }
288 
289   abstract <E extends Comparable<E>> ImmutableSet.Builder<E> builder();
290 
291   public void testBuilderWithNonDuplicateElements() {
292     ImmutableSet<String> set = this.<String>builder()
293         .add("a")
294         .add("b", "c")
295         .add("d", "e", "f")
296         .add("g", "h", "i", "j")
297         .build();
298     assertThat(set).has().exactly(
299         "a", "b", "c", "d", "e", "f", "g", "h", "i", "j").inOrder();
300   }
301 
302   public void testReuseBuilderWithNonDuplicateElements() {
303     ImmutableSet.Builder<String> builder = this.<String>builder()
304         .add("a")
305         .add("b");
306     assertThat(builder.build()).has().exactly("a", "b").inOrder();
307     builder.add("c", "d");
308     assertThat(builder.build()).has().exactly("a", "b", "c", "d").inOrder();
309   }
310 
311   public void testBuilderWithDuplicateElements() {
312     ImmutableSet<String> set = this.<String>builder()
313         .add("a")
314         .add("a", "a")
315         .add("a", "a", "a")
316         .add("a", "a", "a", "a")
317         .build();
318     assertTrue(set.contains("a"));
319     assertFalse(set.contains("b"));
320     assertEquals(1, set.size());
321   }
322 
323   public void testReuseBuilderWithDuplicateElements() {
324     ImmutableSet.Builder<String> builder = this.<String>builder()
325         .add("a")
326         .add("a", "a")
327         .add("b");
328     assertThat(builder.build()).has().exactly("a", "b").inOrder();
329     builder.add("a", "b", "c", "c");
330     assertThat(builder.build()).has().exactly("a", "b", "c").inOrder();
331   }
332 
333   public void testBuilderAddAll() {
334     List<String> a = asList("a", "b", "c");
335     List<String> b = asList("c", "d", "e");
336     ImmutableSet<String> set = this.<String>builder()
337         .addAll(a)
338         .addAll(b)
339         .build();
340     assertThat(set).has().exactly("a", "b", "c", "d", "e").inOrder();
341   }
342 
343   static final int LAST_COLOR_ADDED = 0x00BFFF;
344 
345   public void testComplexBuilder() {
346     List<Integer> colorElem = asList(0x00, 0x33, 0x66, 0x99, 0xCC, 0xFF);
347     // javac won't compile this without "this.<Integer>"
348     ImmutableSet.Builder<Integer> webSafeColorsBuilder
349         = this.<Integer>builder();
350     for (Integer red : colorElem) {
351       for (Integer green : colorElem) {
352         for (Integer blue : colorElem) {
353           webSafeColorsBuilder.add((red << 16) + (green << 8) + blue);
354         }
355       }
356     }
357     ImmutableSet<Integer> webSafeColors = webSafeColorsBuilder.build();
358     assertEquals(216, webSafeColors.size());
359     Integer[] webSafeColorArray =
360         webSafeColors.toArray(new Integer[webSafeColors.size()]);
361     assertEquals(0x000000, (int) webSafeColorArray[0]);
362     assertEquals(0x000033, (int) webSafeColorArray[1]);
363     assertEquals(0x000066, (int) webSafeColorArray[2]);
364     assertEquals(0x003300, (int) webSafeColorArray[6]);
365     assertEquals(0x330000, (int) webSafeColorArray[36]);
366     ImmutableSet<Integer> addedColor
367         = webSafeColorsBuilder.add(LAST_COLOR_ADDED).build();
368     assertEquals(
369         "Modifying the builder should not have changed any already built sets",
370         216, webSafeColors.size());
371     assertEquals("the new array should be one bigger than webSafeColors",
372         217, addedColor.size());
373     Integer[] appendColorArray =
374         addedColor.toArray(new Integer[addedColor.size()]);
375     assertEquals(
376         getComplexBuilderSetLastElement(), (int) appendColorArray[216]);
377   }
378 
379   abstract int getComplexBuilderSetLastElement();
380 
381   public void testBuilderAddHandlesNullsCorrectly() {
382     ImmutableSet.Builder<String> builder = this.<String>builder();
383     try {
384       builder.add((String) null);
385       fail("expected NullPointerException");  // COV_NF_LINE
386     } catch (NullPointerException expected) {
387     }
388 
389     builder = this.<String>builder();
390     try {
391       builder.add((String[]) null);
392       fail("expected NullPointerException");  // COV_NF_LINE
393     } catch (NullPointerException expected) {
394     }
395 
396     builder = this.<String>builder();
397     try {
398       builder.add("a", (String) null);
399       fail("expected NullPointerException");  // COV_NF_LINE
400     } catch (NullPointerException expected) {
401     }
402 
403     builder = this.<String>builder();
404     try {
405       builder.add("a", "b", (String) null);
406       fail("expected NullPointerException");  // COV_NF_LINE
407     } catch (NullPointerException expected) {
408     }
409 
410     builder = this.<String>builder();
411     try {
412       builder.add("a", "b", "c", null);
413       fail("expected NullPointerException");  // COV_NF_LINE
414     } catch (NullPointerException expected) {
415     }
416 
417     builder = this.<String>builder();
418     try {
419       builder.add("a", "b", null, "c");
420       fail("expected NullPointerException");  // COV_NF_LINE
421     } catch (NullPointerException expected) {
422     }
423   }
424 
425   public void testBuilderAddAllHandlesNullsCorrectly() {
426     ImmutableSet.Builder<String> builder = this.<String>builder();
427     try {
428       builder.addAll((Iterable<String>) null);
429       fail("expected NullPointerException");  // COV_NF_LINE
430     } catch (NullPointerException expected) {
431     }
432 
433     try {
434       builder.addAll((Iterator<String>) null);
435       fail("expected NullPointerException");  // COV_NF_LINE
436     } catch (NullPointerException expected) {
437     }
438 
439     builder = this.<String>builder();
440     List<String> listWithNulls = asList("a", null, "b");
441     try {
442       builder.addAll(listWithNulls);
443       fail("expected NullPointerException");  // COV_NF_LINE
444     } catch (NullPointerException expected) {
445     }
446 
447     Iterable<String> iterableWithNulls = MinimalIterable.of("a", null, "b");
448     try {
449       builder.addAll(iterableWithNulls);
450       fail("expected NullPointerException");  // COV_NF_LINE
451     } catch (NullPointerException expected) {
452     }
453   }
454 
455   /**
456    * Verify thread safety by using a collection whose size() may be inconsistent
457    * with the actual number of elements.  Tests using this method might fail in
458    * GWT because the GWT emulations might count on size() during copy.  It is
459    * safe to do so in GWT because javascript is single-threaded.
460    */
461   // TODO(benyu): turn this into a test once all copyOf(Collection) are
462   // thread-safe
463   @GwtIncompatible("GWT is single threaded")
464   void verifyThreadSafe() {
465     List<String> sample = Lists.newArrayList("a", "b", "c");
466     for (int delta : new int[] {-1, 0, 1}) {
467       for (int i = 0; i < sample.size(); i++) {
468         Collection<String> misleading = Helpers.misleadingSizeCollection(delta);
469         List<String> expected = sample.subList(0, i);
470         misleading.addAll(expected);
471         assertEquals("delta: " + delta + " sample size: " + i,
472             Sets.newHashSet(expected), copyOf(misleading));
473       }
474     }
475   }
476 }
477