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 java.util.Arrays.asList; 20 21 import com.google.common.annotations.GwtIncompatible; 22 import com.google.common.base.Objects; 23 import com.google.common.base.Predicate; 24 import com.google.common.base.Predicates; 25 import com.google.common.collect.testing.features.CollectionFeature; 26 import com.google.common.collect.testing.features.CollectionSize; 27 import com.google.common.collect.testing.google.MultisetTestSuiteBuilder; 28 import com.google.common.collect.testing.google.SortedMultisetTestSuiteBuilder; 29 import com.google.common.collect.testing.google.TestStringMultisetGenerator; 30 31 import junit.framework.Test; 32 import junit.framework.TestCase; 33 import junit.framework.TestSuite; 34 35 import java.util.ArrayList; 36 import java.util.Collections; 37 import java.util.List; 38 39 /** 40 * Collection tests on wrappers from {@link Multisets}. 41 * 42 * @author Jared Levy 43 */ 44 @GwtIncompatible("suite") // TODO(cpovirk): set up collect/gwt/suites version 45 public class MultisetsCollectionTest extends TestCase { suite()46 public static Test suite() { 47 TestSuite suite = new TestSuite(); 48 49 suite.addTest(MultisetTestSuiteBuilder.using( 50 unmodifiableMultisetGenerator()) 51 .withFeatures(CollectionSize.ANY, CollectionFeature.KNOWN_ORDER, 52 CollectionFeature.SERIALIZABLE, 53 CollectionFeature.ALLOWS_NULL_QUERIES) 54 .named("Multisets.unmodifiableMultiset[LinkedHashMultiset]") 55 .createTestSuite()); 56 57 suite.addTest(SortedMultisetTestSuiteBuilder.using(unmodifiableSortedMultisetGenerator()) 58 .withFeatures(CollectionSize.ANY, CollectionFeature.KNOWN_ORDER, 59 CollectionFeature.ALLOWS_NULL_QUERIES) 60 .named("Multisets.unmodifiableMultiset[TreeMultiset]") 61 .createTestSuite()); 62 63 suite.addTest(MultisetTestSuiteBuilder.using(unionGenerator()) 64 .withFeatures(CollectionSize.ANY, 65 CollectionFeature.ALLOWS_NULL_VALUES) 66 .named("Multisets.union") 67 .createTestSuite()); 68 69 suite.addTest(MultisetTestSuiteBuilder.using(intersectionGenerator()) 70 .withFeatures(CollectionSize.ANY, 71 CollectionFeature.ALLOWS_NULL_VALUES, 72 CollectionFeature.KNOWN_ORDER) 73 .named("Multisets.intersection") 74 .createTestSuite()); 75 76 suite.addTest(MultisetTestSuiteBuilder.using(sumGenerator()) 77 .withFeatures(CollectionSize.ANY, 78 CollectionFeature.ALLOWS_NULL_VALUES) 79 .named("Multisets.sum") 80 .createTestSuite()); 81 82 suite.addTest(MultisetTestSuiteBuilder.using(differenceGenerator()) 83 .withFeatures(CollectionSize.ANY, 84 CollectionFeature.ALLOWS_NULL_VALUES, 85 CollectionFeature.KNOWN_ORDER) 86 .named("Multisets.difference") 87 .createTestSuite()); 88 89 suite.addTest(MultisetTestSuiteBuilder.using(filteredGenerator()) 90 .withFeatures(CollectionSize.ANY, 91 CollectionFeature.ALLOWS_NULL_VALUES, 92 CollectionFeature.KNOWN_ORDER, 93 CollectionFeature.SUPPORTS_ADD, 94 CollectionFeature.SUPPORTS_REMOVE) 95 .named("Multiset.filter[Multiset, Predicate]") 96 .createTestSuite()); 97 98 return suite; 99 } 100 unmodifiableMultisetGenerator()101 private static TestStringMultisetGenerator unmodifiableMultisetGenerator() { 102 return new TestStringMultisetGenerator() { 103 @Override protected Multiset<String> create(String[] elements) { 104 return Multisets.unmodifiableMultiset( 105 LinkedHashMultiset.create(asList(elements))); 106 } 107 @Override public List<String> order(List<String> insertionOrder) { 108 List<String> order = new ArrayList<String>(); 109 for (String s : insertionOrder) { 110 int index = order.indexOf(s); 111 if (index == -1) { 112 order.add(s); 113 } else { 114 order.add(index, s); 115 } 116 } 117 return order; 118 } 119 }; 120 } 121 122 private static TestStringMultisetGenerator unmodifiableSortedMultisetGenerator() { 123 return new TestStringMultisetGenerator() { 124 @Override protected Multiset<String> create(String[] elements) { 125 return Multisets.unmodifiableSortedMultiset( 126 TreeMultiset.create(asList(elements))); 127 } 128 @Override public List<String> order(List<String> insertionOrder) { 129 Collections.sort(insertionOrder); 130 return insertionOrder; 131 } 132 }; 133 } 134 135 private static TestStringMultisetGenerator unionGenerator() { 136 return new TestStringMultisetGenerator() { 137 @Override 138 protected Multiset<String> create(String[] elements) { 139 Multiset<String> multiset1 = LinkedHashMultiset.create(); 140 Multiset<String> multiset2 = LinkedHashMultiset.create(); 141 for (int i = 0; i < elements.length; i++) { 142 String element = elements[i]; 143 if (multiset1.contains(element) || 144 multiset2.contains(element)) { 145 // add to both; the one already containing it will have more 146 multiset1.add(element); 147 multiset2.add(element); 148 } else if (i % 2 == 0) { 149 multiset1.add(elements[i]); 150 } else { 151 multiset2.add(elements[i]); 152 } 153 } 154 return Multisets.union(multiset1, multiset2); 155 } 156 }; 157 } 158 159 private static TestStringMultisetGenerator intersectionGenerator() { 160 return new TestStringMultisetGenerator() { 161 @Override protected Multiset<String> create(String[] elements) { 162 Multiset<String> multiset1 = LinkedHashMultiset.create(); 163 Multiset<String> multiset2 = LinkedHashMultiset.create(); 164 multiset1.add("only1"); 165 multiset2.add("only2"); 166 for (int i = 0; i < elements.length; i++) { 167 multiset1.add(elements[i]); 168 multiset2.add(elements[elements.length - 1 - i]); 169 } 170 if (elements.length > 0) { 171 multiset1.add(elements[0]); 172 } 173 if (elements.length > 1) { 174 /* 175 * When a test requests a multiset with duplicates, our plan of 176 * "add an extra item 0 to A and an extra item 1 to B" really means 177 * "add an extra item 0 to A and B," which isn't what we want. 178 */ 179 if (!Objects.equal(elements[0], elements[1])) { 180 multiset2.add(elements[1], 2); 181 } 182 } 183 return Multisets.intersection(multiset1, multiset2); 184 } 185 }; 186 } 187 188 private static TestStringMultisetGenerator sumGenerator() { 189 return new TestStringMultisetGenerator() { 190 @Override protected Multiset<String> create(String[] elements) { 191 Multiset<String> multiset1 = LinkedHashMultiset.create(); 192 Multiset<String> multiset2 = LinkedHashMultiset.create(); 193 for (int i = 0; i < elements.length; i++) { 194 // add to either; sum should contain all 195 if (i % 2 == 0) { 196 multiset1.add(elements[i]); 197 } else { 198 multiset2.add(elements[i]); 199 } 200 } 201 return Multisets.sum(multiset1, multiset2); 202 } 203 }; 204 } 205 206 private static TestStringMultisetGenerator differenceGenerator() { 207 return new TestStringMultisetGenerator() { 208 @Override protected Multiset<String> create(String[] elements) { 209 Multiset<String> multiset1 = LinkedHashMultiset.create(); 210 Multiset<String> multiset2 = LinkedHashMultiset.create(); 211 multiset1.add("equalIn1"); 212 multiset1.add("fewerIn1"); 213 multiset2.add("equalIn1"); 214 multiset2.add("fewerIn1", 3); 215 multiset2.add("onlyIn2", 2); 216 for (int i = 0; i < elements.length; i++) { 217 // add 1 more copy of each element to multiset1 than multiset2 218 multiset1.add(elements[i], i + 2); 219 multiset2.add(elements[i], i + 1); 220 } 221 return Multisets.difference(multiset1, multiset2); 222 } 223 }; 224 } 225 226 private static final Multiset<String> ELEMENTS_TO_FILTER_OUT = ImmutableMultiset.of( 227 "foobar", "bazfoo", "foobar", "foobar"); 228 229 private static final Predicate<String> PREDICATE = 230 Predicates.not(Predicates.in(ELEMENTS_TO_FILTER_OUT)); 231 232 private static TestStringMultisetGenerator filteredGenerator() { 233 return new TestStringMultisetGenerator() { 234 @Override 235 protected Multiset<String> create(String[] elements) { 236 Multiset<String> multiset = LinkedHashMultiset.create(); 237 Collections.addAll(multiset, elements); 238 multiset.addAll(ELEMENTS_TO_FILTER_OUT); 239 return Multisets.filter(multiset, PREDICATE); 240 } 241 242 @Override 243 public List<String> order(List<String> insertionOrder) { 244 return Lists.newArrayList(LinkedHashMultiset.create(insertionOrder)); 245 } 246 }; 247 } 248 } 249