001/*
002 * Copyright (C) 2009 The Guava Authors
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 * http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016
017package com.google.common.collect.testing.google;
018
019import static com.google.common.collect.testing.features.CollectionFeature.ALLOWS_NULL_VALUES;
020import static com.google.common.collect.testing.features.CollectionFeature.FAILS_FAST_ON_CONCURRENT_MODIFICATION;
021import static com.google.common.collect.testing.features.CollectionFeature.RESTRICTS_ELEMENTS;
022import static com.google.common.collect.testing.features.CollectionFeature.SUPPORTS_ADD;
023import static com.google.common.collect.testing.features.CollectionFeature.SUPPORTS_REMOVE;
024import static com.google.common.collect.testing.features.CollectionSize.SEVERAL;
025import static com.google.common.collect.testing.features.CollectionSize.ZERO;
026
027import com.google.common.annotations.GwtCompatible;
028import com.google.common.annotations.GwtIncompatible;
029import com.google.common.collect.Multiset;
030import com.google.common.collect.Multiset.Entry;
031import com.google.common.collect.testing.Helpers;
032import com.google.common.collect.testing.features.CollectionFeature;
033import com.google.common.collect.testing.features.CollectionSize;
034import java.lang.reflect.Method;
035import java.util.Arrays;
036import java.util.ConcurrentModificationException;
037import java.util.Iterator;
038import java.util.List;
039import org.junit.Ignore;
040
041/**
042 * Common superclass for {@link MultisetSetCountUnconditionallyTester} and {@link
043 * MultisetSetCountConditionallyTester}. It is used by those testers to test calls to the
044 * unconditional {@code setCount()} method and calls to the conditional {@code setCount()} method
045 * when the expected present count is correct.
046 *
047 * @author Chris Povirk
048 */
049@GwtCompatible(emulated = true)
050@Ignore // Affects only Android test runner, which respects JUnit 4 annotations on JUnit 3 tests.
051public abstract class AbstractMultisetSetCountTester<E> extends AbstractMultisetTester<E> {
052  /*
053   * TODO: consider adding MultisetFeatures.SUPPORTS_SET_COUNT. Currently we
054   * assume that using setCount() to increase the count is permitted iff add()
055   * is permitted and similarly for decrease/remove(). We assume that a
056   * setCount() no-op is permitted if either add() or remove() is permitted,
057   * though we also allow it to "succeed" if neither is permitted.
058   */
059
060  private void assertSetCount(E element, int count) {
061    setCountCheckReturnValue(element, count);
062
063    assertEquals(
064        "multiset.count() should return the value passed to setCount()",
065        count,
066        getMultiset().count(element));
067
068    int size = 0;
069    for (Multiset.Entry<E> entry : getMultiset().entrySet()) {
070      size += entry.getCount();
071    }
072    assertEquals(
073        "multiset.size() should be the sum of the counts of all entries",
074        size,
075        getMultiset().size());
076  }
077
078  /** Call the {@code setCount()} method under test, and check its return value. */
079  abstract void setCountCheckReturnValue(E element, int count);
080
081  /**
082   * Call the {@code setCount()} method under test, but do not check its return value. Callers
083   * should use this method over {@link #setCountCheckReturnValue(Object, int)} when they expect
084   * {@code setCount()} to throw an exception, as checking the return value could produce an
085   * incorrect error message like "setCount() should return the original count" instead of the
086   * message passed to a later invocation of {@code fail()}, like "setCount should throw
087   * UnsupportedOperationException."
088   */
089  abstract void setCountNoCheckReturnValue(E element, int count);
090
091  private void assertSetCountIncreasingFailure(E element, int count) {
092    try {
093      setCountNoCheckReturnValue(element, count);
094      fail("a call to multiset.setCount() to increase an element's count should throw");
095    } catch (UnsupportedOperationException expected) {
096    }
097  }
098
099  private void assertSetCountDecreasingFailure(E element, int count) {
100    try {
101      setCountNoCheckReturnValue(element, count);
102      fail("a call to multiset.setCount() to decrease an element's count should throw");
103    } catch (UnsupportedOperationException expected) {
104    }
105  }
106
107  // Unconditional setCount no-ops.
108
109  private void assertZeroToZero() {
110    assertSetCount(e3(), 0);
111  }
112
113  private void assertOneToOne() {
114    assertSetCount(e0(), 1);
115  }
116
117  private void assertThreeToThree() {
118    initThreeCopies();
119    assertSetCount(e0(), 3);
120  }
121
122  @CollectionFeature.Require(SUPPORTS_ADD)
123  public void testSetCount_zeroToZero_addSupported() {
124    assertZeroToZero();
125  }
126
127  @CollectionFeature.Require(SUPPORTS_REMOVE)
128  public void testSetCount_zeroToZero_removeSupported() {
129    assertZeroToZero();
130  }
131
132  @CollectionFeature.Require(absent = {SUPPORTS_ADD, SUPPORTS_REMOVE})
133  public void testSetCount_zeroToZero_unsupported() {
134    try {
135      assertZeroToZero();
136    } catch (UnsupportedOperationException tolerated) {
137    }
138  }
139
140  @CollectionSize.Require(absent = ZERO)
141  @CollectionFeature.Require(SUPPORTS_ADD)
142  public void testSetCount_oneToOne_addSupported() {
143    assertOneToOne();
144  }
145
146  @CollectionSize.Require(absent = ZERO)
147  @CollectionFeature.Require(SUPPORTS_REMOVE)
148  public void testSetCount_oneToOne_removeSupported() {
149    assertOneToOne();
150  }
151
152  @CollectionSize.Require(absent = ZERO)
153  @CollectionFeature.Require(absent = {SUPPORTS_ADD, SUPPORTS_REMOVE})
154  public void testSetCount_oneToOne_unsupported() {
155    try {
156      assertOneToOne();
157    } catch (UnsupportedOperationException tolerated) {
158    }
159  }
160
161  @CollectionSize.Require(SEVERAL)
162  @CollectionFeature.Require(SUPPORTS_ADD)
163  public void testSetCount_threeToThree_addSupported() {
164    assertThreeToThree();
165  }
166
167  @CollectionSize.Require(SEVERAL)
168  @CollectionFeature.Require(SUPPORTS_REMOVE)
169  public void testSetCount_threeToThree_removeSupported() {
170    assertThreeToThree();
171  }
172
173  @CollectionSize.Require(SEVERAL)
174  @CollectionFeature.Require(absent = {SUPPORTS_ADD, SUPPORTS_REMOVE})
175  public void testSetCount_threeToThree_unsupported() {
176    try {
177      assertThreeToThree();
178    } catch (UnsupportedOperationException tolerated) {
179    }
180  }
181
182  // Unconditional setCount size increases:
183
184  @CollectionFeature.Require(SUPPORTS_ADD)
185  public void testSetCount_zeroToOne_supported() {
186    assertSetCount(e3(), 1);
187  }
188
189  @CollectionFeature.Require({SUPPORTS_ADD, FAILS_FAST_ON_CONCURRENT_MODIFICATION})
190  public void testSetCountZeroToOneConcurrentWithIteration() {
191    try {
192      Iterator<E> iterator = collection.iterator();
193      assertSetCount(e3(), 1);
194      iterator.next();
195      fail("Expected ConcurrentModificationException");
196    } catch (ConcurrentModificationException expected) {
197      // success
198    }
199  }
200
201  @CollectionFeature.Require({SUPPORTS_ADD, FAILS_FAST_ON_CONCURRENT_MODIFICATION})
202  public void testSetCountZeroToOneConcurrentWithEntrySetIteration() {
203    try {
204      Iterator<Entry<E>> iterator = getMultiset().entrySet().iterator();
205      assertSetCount(e3(), 1);
206      iterator.next();
207      fail("Expected ConcurrentModificationException");
208    } catch (ConcurrentModificationException expected) {
209      // success
210    }
211  }
212
213  @CollectionFeature.Require(SUPPORTS_ADD)
214  public void testSetCount_zeroToThree_supported() {
215    assertSetCount(e3(), 3);
216  }
217
218  @CollectionSize.Require(absent = ZERO)
219  @CollectionFeature.Require(SUPPORTS_ADD)
220  public void testSetCount_oneToThree_supported() {
221    assertSetCount(e0(), 3);
222  }
223
224  @CollectionFeature.Require(absent = SUPPORTS_ADD)
225  public void testSetCount_zeroToOne_unsupported() {
226    assertSetCountIncreasingFailure(e3(), 1);
227  }
228
229  @CollectionFeature.Require(absent = SUPPORTS_ADD)
230  public void testSetCount_zeroToThree_unsupported() {
231    assertSetCountIncreasingFailure(e3(), 3);
232  }
233
234  @CollectionSize.Require(absent = ZERO)
235  @CollectionFeature.Require(absent = SUPPORTS_ADD)
236  public void testSetCount_oneToThree_unsupported() {
237    assertSetCountIncreasingFailure(e3(), 3);
238  }
239
240  // Unconditional setCount size decreases:
241
242  @CollectionSize.Require(absent = ZERO)
243  @CollectionFeature.Require(SUPPORTS_REMOVE)
244  public void testSetCount_oneToZero_supported() {
245    assertSetCount(e0(), 0);
246  }
247
248  @CollectionFeature.Require({SUPPORTS_REMOVE, FAILS_FAST_ON_CONCURRENT_MODIFICATION})
249  @CollectionSize.Require(absent = ZERO)
250  public void testSetCountOneToZeroConcurrentWithIteration() {
251    try {
252      Iterator<E> iterator = collection.iterator();
253      assertSetCount(e0(), 0);
254      iterator.next();
255      fail("Expected ConcurrentModificationException");
256    } catch (ConcurrentModificationException expected) {
257      // success
258    }
259  }
260
261  @CollectionFeature.Require({SUPPORTS_REMOVE, FAILS_FAST_ON_CONCURRENT_MODIFICATION})
262  @CollectionSize.Require(absent = ZERO)
263  public void testSetCountOneToZeroConcurrentWithEntrySetIteration() {
264    try {
265      Iterator<Entry<E>> iterator = getMultiset().entrySet().iterator();
266      assertSetCount(e0(), 0);
267      iterator.next();
268      fail("Expected ConcurrentModificationException");
269    } catch (ConcurrentModificationException expected) {
270      // success
271    }
272  }
273
274  @CollectionSize.Require(SEVERAL)
275  @CollectionFeature.Require(SUPPORTS_REMOVE)
276  public void testSetCount_threeToZero_supported() {
277    initThreeCopies();
278    assertSetCount(e0(), 0);
279  }
280
281  @CollectionSize.Require(SEVERAL)
282  @CollectionFeature.Require(SUPPORTS_REMOVE)
283  public void testSetCount_threeToOne_supported() {
284    initThreeCopies();
285    assertSetCount(e0(), 1);
286  }
287
288  @CollectionSize.Require(absent = ZERO)
289  @CollectionFeature.Require(absent = SUPPORTS_REMOVE)
290  public void testSetCount_oneToZero_unsupported() {
291    assertSetCountDecreasingFailure(e0(), 0);
292  }
293
294  @CollectionSize.Require(SEVERAL)
295  @CollectionFeature.Require(absent = SUPPORTS_REMOVE)
296  public void testSetCount_threeToZero_unsupported() {
297    initThreeCopies();
298    assertSetCountDecreasingFailure(e0(), 0);
299  }
300
301  @CollectionSize.Require(SEVERAL)
302  @CollectionFeature.Require(absent = SUPPORTS_REMOVE)
303  public void testSetCount_threeToOne_unsupported() {
304    initThreeCopies();
305    assertSetCountDecreasingFailure(e0(), 1);
306  }
307
308  // setCount with nulls:
309
310  @CollectionSize.Require(absent = ZERO)
311  @CollectionFeature.Require({SUPPORTS_REMOVE, ALLOWS_NULL_VALUES})
312  public void testSetCount_removeNull_nullSupported() {
313    initCollectionWithNullElement();
314    assertSetCount(null, 0);
315  }
316
317  @CollectionFeature.Require(
318    value = {SUPPORTS_ADD, ALLOWS_NULL_VALUES},
319    absent = RESTRICTS_ELEMENTS
320  )
321  public void testSetCount_addNull_nullSupported() {
322    assertSetCount(null, 1);
323  }
324
325  @CollectionFeature.Require(value = SUPPORTS_ADD, absent = ALLOWS_NULL_VALUES)
326  public void testSetCount_addNull_nullUnsupported() {
327    try {
328      setCountNoCheckReturnValue(null, 1);
329      fail("adding null with setCount() should throw NullPointerException");
330    } catch (NullPointerException expected) {
331    }
332  }
333
334  @CollectionFeature.Require(ALLOWS_NULL_VALUES)
335  public void testSetCount_noOpNull_nullSupported() {
336    try {
337      assertSetCount(null, 0);
338    } catch (UnsupportedOperationException tolerated) {
339    }
340  }
341
342  @CollectionFeature.Require(absent = ALLOWS_NULL_VALUES)
343  public void testSetCount_noOpNull_nullUnsupported() {
344    try {
345      assertSetCount(null, 0);
346    } catch (NullPointerException | UnsupportedOperationException tolerated) {
347    }
348  }
349
350  @CollectionSize.Require(absent = ZERO)
351  @CollectionFeature.Require(ALLOWS_NULL_VALUES)
352  public void testSetCount_existingNoNopNull_nullSupported() {
353    initCollectionWithNullElement();
354    try {
355      assertSetCount(null, 1);
356    } catch (UnsupportedOperationException tolerated) {
357    }
358  }
359
360  // Negative count.
361
362  @CollectionFeature.Require(SUPPORTS_REMOVE)
363  public void testSetCount_negative_removeSupported() {
364    try {
365      setCountNoCheckReturnValue(e3(), -1);
366      fail("calling setCount() with a negative count should throw IllegalArgumentException");
367    } catch (IllegalArgumentException expected) {
368    }
369  }
370
371  @CollectionFeature.Require(absent = SUPPORTS_REMOVE)
372  public void testSetCount_negative_removeUnsupported() {
373    try {
374      setCountNoCheckReturnValue(e3(), -1);
375      fail(
376          "calling setCount() with a negative count should throw "
377              + "IllegalArgumentException or UnsupportedOperationException");
378    } catch (IllegalArgumentException | UnsupportedOperationException expected) {
379    }
380  }
381
382  // TODO: test adding element of wrong type
383
384  /**
385   * Returns {@link Method} instances for the {@code setCount()} tests that assume multisets support
386   * duplicates so that the test of {@code Multisets.forSet()} can suppress them.
387   */
388  @GwtIncompatible // reflection
389  public static List<Method> getSetCountDuplicateInitializingMethods() {
390    return Arrays.asList(
391        getMethod("testSetCount_threeToThree_removeSupported"),
392        getMethod("testSetCount_threeToZero_supported"),
393        getMethod("testSetCount_threeToOne_supported"));
394  }
395
396  @GwtIncompatible // reflection
397  private static Method getMethod(String methodName) {
398    return Helpers.getMethod(AbstractMultisetSetCountTester.class, methodName);
399  }
400}