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.base.Preconditions.checkArgument;
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.ListTestSuiteBuilder;
26 import com.google.common.collect.testing.MinimalCollection;
27 import com.google.common.collect.testing.SetTestSuiteBuilder;
28 import com.google.common.collect.testing.TestStringListGenerator;
29 import com.google.common.collect.testing.TestStringSetGenerator;
30 import com.google.common.collect.testing.features.CollectionFeature;
31 import com.google.common.collect.testing.features.CollectionSize;
32 import com.google.common.collect.testing.google.MultisetTestSuiteBuilder;
33 import com.google.common.collect.testing.google.TestStringMultisetGenerator;
34 import com.google.common.collect.testing.google.UnmodifiableCollectionTests;
35 import com.google.common.testing.EqualsTester;
36 import com.google.common.testing.NullPointerTester;
37 import com.google.common.testing.SerializableTester;
38 
39 import junit.framework.Test;
40 import junit.framework.TestCase;
41 import junit.framework.TestSuite;
42 
43 import java.util.ArrayList;
44 import java.util.Collection;
45 import java.util.HashSet;
46 import java.util.Iterator;
47 import java.util.List;
48 import java.util.Set;
49 
50 /**
51  * Tests for {@link ImmutableMultiset}.
52  *
53  * @author Jared Levy
54  */
55 @GwtCompatible(emulated = true)
56 public class ImmutableMultisetTest extends TestCase {
57 
58   @GwtIncompatible("suite") // TODO(cpovirk): add to collect/gwt/suites
59   public static Test suite() {
60     TestSuite suite = new TestSuite();
61     suite.addTestSuite(ImmutableMultisetTest.class);
62 
63     suite.addTest(MultisetTestSuiteBuilder.using(
64         new TestStringMultisetGenerator() {
65           @Override protected Multiset<String> create(String[] elements) {
66             return ImmutableMultiset.copyOf(elements);
67           }
68         })
69         .named("ImmutableMultiset")
70         .withFeatures(CollectionSize.ANY,
71             CollectionFeature.SERIALIZABLE_INCLUDING_VIEWS,
72             CollectionFeature.ALLOWS_NULL_QUERIES)
73         .createTestSuite());
74 
75     suite.addTest(SetTestSuiteBuilder.using(new TestStringSetGenerator() {
76           @Override protected Set<String> create(String[] elements) {
77             return ImmutableMultiset.copyOf(elements).elementSet();
78           }
79         })
80         .named("ImmutableMultiset, element set")
81         .withFeatures(CollectionSize.ANY,
82             CollectionFeature.SERIALIZABLE,
83             CollectionFeature.ALLOWS_NULL_QUERIES)
84         .createTestSuite());
85 
86     suite.addTest(ListTestSuiteBuilder.using(new TestStringListGenerator() {
87           @Override protected List<String> create(String[] elements) {
88             return ImmutableMultiset.copyOf(elements).asList();
89           }
90 
91           @Override
92           public List<String> order(List<String> insertionOrder) {
93             List<String> order = new ArrayList<String>();
94             for (String s : insertionOrder) {
95               int index = order.indexOf(s);
96               if (index == -1) {
97                 order.add(s);
98               } else {
99                 order.add(index, s);
100               }
101             }
102             return order;
103           }
104         })
105         .named("ImmutableMultiset.asList")
106         .withFeatures(CollectionSize.ANY,
107             CollectionFeature.SERIALIZABLE,
108             CollectionFeature.ALLOWS_NULL_QUERIES)
109         .createTestSuite());
110 
111     suite.addTest(ListTestSuiteBuilder.using(new TestStringListGenerator() {
112           @Override protected List<String> create(String[] elements) {
113             Set<String> set = new HashSet<String>();
114             ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
115             for (String s : elements) {
116               checkArgument(set.add(s));
117               builder.addCopies(s, 2);
118             }
119             ImmutableSet<String> elementSet = (ImmutableSet<String>) builder.build().elementSet();
120             return elementSet.asList();
121           }
122         })
123         .named("ImmutableMultiset.elementSet.asList")
124         .withFeatures(CollectionSize.ANY,
125             CollectionFeature.REJECTS_DUPLICATES_AT_CREATION,
126             CollectionFeature.SERIALIZABLE,
127             CollectionFeature.ALLOWS_NULL_QUERIES)
128         .createTestSuite());
129 
130     return suite;
131   }
132 
133   public void testCreation_noArgs() {
134     Multiset<String> multiset = ImmutableMultiset.of();
135     assertTrue(multiset.isEmpty());
136   }
137 
138   public void testCreation_oneElement() {
139     Multiset<String> multiset = ImmutableMultiset.of("a");
140     assertEquals(HashMultiset.create(asList("a")), multiset);
141   }
142 
143   public void testCreation_twoElements() {
144     Multiset<String> multiset = ImmutableMultiset.of("a", "b");
145     assertEquals(HashMultiset.create(asList("a", "b")), multiset);
146   }
147 
148   public void testCreation_threeElements() {
149     Multiset<String> multiset = ImmutableMultiset.of("a", "b", "c");
150     assertEquals(HashMultiset.create(asList("a", "b", "c")), multiset);
151   }
152 
153   public void testCreation_fourElements() {
154     Multiset<String> multiset = ImmutableMultiset.of("a", "b", "c", "d");
155     assertEquals(HashMultiset.create(asList("a", "b", "c", "d")), multiset);
156   }
157 
158   public void testCreation_fiveElements() {
159     Multiset<String> multiset = ImmutableMultiset.of("a", "b", "c", "d", "e");
160     assertEquals(HashMultiset.create(asList("a", "b", "c", "d", "e")),
161         multiset);
162   }
163 
164   public void testCreation_sixElements() {
165     Multiset<String> multiset = ImmutableMultiset.of(
166         "a", "b", "c", "d", "e", "f");
167     assertEquals(HashMultiset.create(asList("a", "b", "c", "d", "e", "f")),
168         multiset);
169   }
170 
171   public void testCreation_sevenElements() {
172     Multiset<String> multiset = ImmutableMultiset.of(
173         "a", "b", "c", "d", "e", "f", "g");
174     assertEquals(
175         HashMultiset.create(asList("a", "b", "c", "d", "e", "f", "g")),
176         multiset);
177   }
178 
179   public void testCreation_emptyArray() {
180     String[] array = new String[0];
181     Multiset<String> multiset = ImmutableMultiset.copyOf(array);
182     assertTrue(multiset.isEmpty());
183   }
184 
185   public void testCreation_arrayOfOneElement() {
186     String[] array = new String[] { "a" };
187     Multiset<String> multiset = ImmutableMultiset.copyOf(array);
188     assertEquals(HashMultiset.create(asList("a")), multiset);
189   }
190 
191   public void testCreation_arrayOfArray() {
192     String[] array = new String[] { "a" };
193     Multiset<String[]> multiset = ImmutableMultiset.<String[]>of(array);
194     Multiset<String[]> expected = HashMultiset.create();
195     expected.add(array);
196     assertEquals(expected, multiset);
197   }
198 
199   public void testCreation_arrayContainingOnlyNull() {
200     String[] array = new String[] { null };
201     try {
202       ImmutableMultiset.copyOf(array);
203       fail();
204     } catch (NullPointerException expected) {}
205   }
206 
207   public void testCopyOf_collection_empty() {
208     // "<String>" is required to work around a javac 1.5 bug.
209     Collection<String> c = MinimalCollection.<String>of();
210     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
211     assertTrue(multiset.isEmpty());
212   }
213 
214   public void testCopyOf_collection_oneElement() {
215     Collection<String> c = MinimalCollection.of("a");
216     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
217     assertEquals(HashMultiset.create(asList("a")), multiset);
218   }
219 
220   public void testCopyOf_collection_general() {
221     Collection<String> c = MinimalCollection.of("a", "b", "a");
222     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
223     assertEquals(HashMultiset.create(asList("a", "b", "a")), multiset);
224   }
225 
226   public void testCopyOf_collectionContainingNull() {
227     Collection<String> c = MinimalCollection.of("a", null, "b");
228     try {
229       ImmutableMultiset.copyOf(c);
230       fail();
231     } catch (NullPointerException expected) {}
232   }
233 
234   public void testCopyOf_multiset_empty() {
235     Multiset<String> c = HashMultiset.create();
236     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
237     assertTrue(multiset.isEmpty());
238   }
239 
240   public void testCopyOf_multiset_oneElement() {
241     Multiset<String> c = HashMultiset.create(asList("a"));
242     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
243     assertEquals(HashMultiset.create(asList("a")), multiset);
244   }
245 
246   public void testCopyOf_multiset_general() {
247     Multiset<String> c = HashMultiset.create(asList("a", "b", "a"));
248     Multiset<String> multiset = ImmutableMultiset.copyOf(c);
249     assertEquals(HashMultiset.create(asList("a", "b", "a")), multiset);
250   }
251 
252   public void testCopyOf_multisetContainingNull() {
253     Multiset<String> c = HashMultiset.create(asList("a", null, "b"));
254     try {
255       ImmutableMultiset.copyOf(c);
256       fail();
257     } catch (NullPointerException expected) {}
258   }
259 
260   public void testCopyOf_iterator_empty() {
261     Iterator<String> iterator = Iterators.emptyIterator();
262     Multiset<String> multiset = ImmutableMultiset.copyOf(iterator);
263     assertTrue(multiset.isEmpty());
264   }
265 
266   public void testCopyOf_iterator_oneElement() {
267     Iterator<String> iterator = Iterators.singletonIterator("a");
268     Multiset<String> multiset = ImmutableMultiset.copyOf(iterator);
269     assertEquals(HashMultiset.create(asList("a")), multiset);
270   }
271 
272   public void testCopyOf_iterator_general() {
273     Iterator<String> iterator = asList("a", "b", "a").iterator();
274     Multiset<String> multiset = ImmutableMultiset.copyOf(iterator);
275     assertEquals(HashMultiset.create(asList("a", "b", "a")), multiset);
276   }
277 
278   public void testCopyOf_iteratorContainingNull() {
279     Iterator<String> iterator = asList("a", null, "b").iterator();
280     try {
281       ImmutableMultiset.copyOf(iterator);
282       fail();
283     } catch (NullPointerException expected) {}
284   }
285 
286   private static class CountingIterable implements Iterable<String> {
287     int count = 0;
288     @Override
289     public Iterator<String> iterator() {
290       count++;
291       return asList("a", "b", "a").iterator();
292     }
293   }
294 
295   public void testCopyOf_plainIterable() {
296     CountingIterable iterable = new CountingIterable();
297     Multiset<String> multiset = ImmutableMultiset.copyOf(iterable);
298     assertEquals(HashMultiset.create(asList("a", "b", "a")), multiset);
299     assertEquals(1, iterable.count);
300   }
301 
302   public void testCopyOf_shortcut_empty() {
303     Collection<String> c = ImmutableMultiset.of();
304     assertSame(c, ImmutableMultiset.copyOf(c));
305   }
306 
307   public void testCopyOf_shortcut_singleton() {
308     Collection<String> c = ImmutableMultiset.of("a");
309     assertSame(c, ImmutableMultiset.copyOf(c));
310   }
311 
312   public void testCopyOf_shortcut_immutableMultiset() {
313     Collection<String> c = ImmutableMultiset.of("a", "b", "c");
314     assertSame(c, ImmutableMultiset.copyOf(c));
315   }
316 
317   public void testBuilderAdd() {
318     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
319         .add("a")
320         .add("b")
321         .add("a")
322         .add("c")
323         .build();
324     assertEquals(HashMultiset.create(asList("a", "b", "a", "c")), multiset);
325   }
326 
327   public void testBuilderAddAll() {
328     List<String> a = asList("a", "b");
329     List<String> b = asList("c", "d");
330     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
331         .addAll(a)
332         .addAll(b)
333         .build();
334     assertEquals(HashMultiset.create(asList("a", "b", "c", "d")), multiset);
335   }
336 
337   public void testBuilderAddAllMultiset() {
338     Multiset<String> a = HashMultiset.create(asList("a", "b", "b"));
339     Multiset<String> b = HashMultiset.create(asList("c", "b"));
340     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
341         .addAll(a)
342         .addAll(b)
343         .build();
344     assertEquals(
345         HashMultiset.create(asList("a", "b", "b", "b", "c")), multiset);
346   }
347 
348   public void testBuilderAddAllIterator() {
349     Iterator<String> iterator = asList("a", "b", "a", "c").iterator();
350     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
351         .addAll(iterator)
352         .build();
353     assertEquals(HashMultiset.create(asList("a", "b", "a", "c")), multiset);
354   }
355 
356   public void testBuilderAddCopies() {
357     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
358         .addCopies("a", 2)
359         .addCopies("b", 3)
360         .addCopies("c", 0)
361         .build();
362     assertEquals(
363         HashMultiset.create(asList("a", "a", "b", "b", "b")), multiset);
364   }
365 
366   public void testBuilderSetCount() {
367     ImmutableMultiset<String> multiset = new ImmutableMultiset.Builder<String>()
368         .add("a")
369         .setCount("a", 2)
370         .setCount("b", 3)
371         .build();
372     assertEquals(
373         HashMultiset.create(asList("a", "a", "b", "b", "b")), multiset);
374   }
375 
376   public void testBuilderAddHandlesNullsCorrectly() {
377     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
378     try {
379       builder.add((String) null);
380       fail("expected NullPointerException");
381     } catch (NullPointerException expected) {}
382   }
383 
384   public void testBuilderAddAllHandlesNullsCorrectly() {
385     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
386     try {
387       builder.addAll((Collection<String>) null);
388       fail("expected NullPointerException");
389     } catch (NullPointerException expected) {}
390 
391     builder = ImmutableMultiset.builder();
392     List<String> listWithNulls = asList("a", null, "b");
393     try {
394       builder.addAll(listWithNulls);
395       fail("expected NullPointerException");
396     } catch (NullPointerException expected) {}
397 
398     builder = ImmutableMultiset.builder();
399     Multiset<String> multisetWithNull
400         = LinkedHashMultiset.create(asList("a", null, "b"));
401     try {
402       builder.addAll(multisetWithNull);
403       fail("expected NullPointerException");
404     } catch (NullPointerException expected) {}
405   }
406 
407   public void testBuilderAddCopiesHandlesNullsCorrectly() {
408     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
409     try {
410       builder.addCopies(null, 2);
411       fail("expected NullPointerException");
412     } catch (NullPointerException expected) {}
413   }
414 
415   public void testBuilderAddCopiesIllegal() {
416     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
417     try {
418       builder.addCopies("a", -2);
419       fail("expected IllegalArgumentException");
420     } catch (IllegalArgumentException expected) {}
421   }
422 
423   public void testBuilderSetCountHandlesNullsCorrectly() {
424     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
425     try {
426       builder.setCount(null, 2);
427       fail("expected NullPointerException");
428     } catch (NullPointerException expected) {}
429   }
430 
431   public void testBuilderSetCountIllegal() {
432     ImmutableMultiset.Builder<String> builder = ImmutableMultiset.builder();
433     try {
434       builder.setCount("a", -2);
435       fail("expected IllegalArgumentException");
436     } catch (IllegalArgumentException expected) {}
437   }
438 
439   @GwtIncompatible("NullPointerTester")
440   public void testNullPointers() {
441     NullPointerTester tester = new NullPointerTester();
442     tester.testAllPublicStaticMethods(ImmutableMultiset.class);
443   }
444 
445   @GwtIncompatible("SerializableTester")
446   public void testSerialization_empty() {
447     Collection<String> c = ImmutableMultiset.of();
448     assertSame(c, SerializableTester.reserialize(c));
449   }
450 
451   @GwtIncompatible("SerializableTester")
452   public void testSerialization_multiple() {
453     Collection<String> c = ImmutableMultiset.of("a", "b", "a");
454     Collection<String> copy = SerializableTester.reserializeAndAssert(c);
455     assertThat(copy).has().exactly("a", "a", "b").inOrder();
456   }
457 
458   @GwtIncompatible("SerializableTester")
459   public void testSerialization_elementSet() {
460     Multiset<String> c = ImmutableMultiset.of("a", "b", "a");
461     Collection<String> copy =
462         LenientSerializableTester.reserializeAndAssertLenient(c.elementSet());
463     assertThat(copy).has().exactly("a", "b").inOrder();
464   }
465 
466   @GwtIncompatible("SerializableTester")
467   public void testSerialization_entrySet() {
468     Multiset<String> c = ImmutableMultiset.of("a", "b", "c");
469     SerializableTester.reserializeAndAssert(c.entrySet());
470   }
471 
472   public void testEquals_immutableMultiset() {
473     Collection<String> c = ImmutableMultiset.of("a", "b", "a");
474     assertEquals(c, ImmutableMultiset.of("a", "b", "a"));
475     assertEquals(c, ImmutableMultiset.of("a", "a", "b"));
476     assertThat(c).isNotEqualTo(ImmutableMultiset.of("a", "b"));
477     assertThat(c).isNotEqualTo(ImmutableMultiset.of("a", "b", "c", "d"));
478   }
479 
480   public void testIterationOrder() {
481     Collection<String> c = ImmutableMultiset.of("a", "b", "a");
482     assertThat(c).has().exactly("a", "a", "b").inOrder();
483   }
484 
485   public void testMultisetWrites() {
486     Multiset<String> multiset = ImmutableMultiset.of("a", "b", "a");
487     UnmodifiableCollectionTests.assertMultisetIsUnmodifiable(multiset, "test");
488   }
489 
490   public void testAsList() {
491     ImmutableMultiset<String> multiset
492         = ImmutableMultiset.of("a", "a", "b", "b", "b");
493     ImmutableList<String> list = multiset.asList();
494     assertEquals(ImmutableList.of("a", "a", "b", "b", "b"), list);
495     assertEquals(2, list.indexOf("b"));
496     assertEquals(4, list.lastIndexOf("b"));
497   }
498 
499   @GwtIncompatible("SerializableTester")
500   public void testSerialization_asList() {
501     ImmutableMultiset<String> multiset
502         = ImmutableMultiset.of("a", "a", "b", "b", "b");
503     SerializableTester.reserializeAndAssert(multiset.asList());
504   }
505 
506   public void testEquals() {
507     new EqualsTester()
508         .addEqualityGroup(ImmutableMultiset.of(), ImmutableMultiset.of())
509         .addEqualityGroup(ImmutableMultiset.of(1), ImmutableMultiset.of(1))
510         .addEqualityGroup(ImmutableMultiset.of(1, 1), ImmutableMultiset.of(1, 1))
511         .addEqualityGroup(ImmutableMultiset.of(1, 2, 1), ImmutableMultiset.of(2, 1, 1))
512         .testEquals();
513   }
514 }
515