/*
* Copyright (C) 2007 The Guava Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.common.collect;
import static java.util.Arrays.asList;
import static org.truth0.Truth.ASSERT;
import com.google.common.annotations.GwtCompatible;
import com.google.common.annotations.GwtIncompatible;
import com.google.common.testing.SerializableTester;
import junit.framework.TestCase;
import java.util.AbstractCollection;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.RandomAccess;
import java.util.Set;
import java.util.SortedSet;
/**
* Tests for {@code Constraints}.
*
* @author Mike Bostock
* @author Jared Levy
*/
@GwtCompatible(emulated = true)
public class ConstraintsTest extends TestCase {
private static final String TEST_ELEMENT = "test";
private static final class TestElementException
extends IllegalArgumentException {
private static final long serialVersionUID = 0;
}
private static final Constraint<String> TEST_CONSTRAINT
= new Constraint<String>() {
@Override
public String checkElement(String element) {
if (TEST_ELEMENT.equals(element)) {
throw new TestElementException();
}
return element;
}
};
public void testNotNull() {
Constraint<? super String> constraint = Constraints.notNull();
assertSame(TEST_ELEMENT, constraint.checkElement(TEST_ELEMENT));
try {
constraint.checkElement(null);
fail("NullPointerException expected");
} catch (NullPointerException expected) {}
assertEquals("Not null", constraint.toString());
}
public void testConstrainedCollectionLegal() {
Collection<String> collection = Lists.newArrayList("foo", "bar");
Collection<String> constrained = Constraints.constrainedCollection(
collection, TEST_CONSTRAINT);
collection.add(TEST_ELEMENT);
constrained.add("qux");
constrained.addAll(asList("cat", "dog"));
/* equals and hashCode aren't defined for Collection */
ASSERT.that(collection).has()
.exactly("foo", "bar", TEST_ELEMENT, "qux", "cat", "dog").inOrder();
ASSERT.that(constrained).has()
.exactly("foo", "bar", TEST_ELEMENT, "qux", "cat", "dog").inOrder();
}
public void testConstrainedCollectionIllegal() {
Collection<String> collection = Lists.newArrayList("foo", "bar");
Collection<String> constrained = Constraints.constrainedCollection(
collection, TEST_CONSTRAINT);
try {
constrained.add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
ASSERT.that(constrained).has().exactly("foo", "bar").inOrder();
ASSERT.that(collection).has().exactly("foo", "bar").inOrder();
}
public void testConstrainedSetLegal() {
Set<String> set = Sets.newLinkedHashSet(asList("foo", "bar"));
Set<String> constrained = Constraints.constrainedSet(set, TEST_CONSTRAINT);
set.add(TEST_ELEMENT);
constrained.add("qux");
constrained.addAll(asList("cat", "dog"));
assertTrue(set.equals(constrained));
assertTrue(constrained.equals(set));
assertEquals(set.toString(), constrained.toString());
assertEquals(set.hashCode(), constrained.hashCode());
ASSERT.that(set).has().exactly("foo", "bar", TEST_ELEMENT, "qux", "cat", "dog").inOrder();
ASSERT.that(constrained).has()
.exactly("foo", "bar", TEST_ELEMENT, "qux", "cat", "dog").inOrder();
}
public void testConstrainedSetIllegal() {
Set<String> set = Sets.newLinkedHashSet(asList("foo", "bar"));
Set<String> constrained = Constraints.constrainedSet(set, TEST_CONSTRAINT);
try {
constrained.add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
ASSERT.that(constrained).has().exactly("foo", "bar").inOrder();
ASSERT.that(set).has().exactly("foo", "bar").inOrder();
}
public void testConstrainedSortedSetLegal() {
SortedSet<String> sortedSet = Sets.newTreeSet(asList("foo", "bar"));
SortedSet<String> constrained = Constraints.constrainedSortedSet(
sortedSet, TEST_CONSTRAINT);
sortedSet.add(TEST_ELEMENT);
constrained.add("qux");
constrained.addAll(asList("cat", "dog"));
assertTrue(sortedSet.equals(constrained));
assertTrue(constrained.equals(sortedSet));
assertEquals(sortedSet.toString(), constrained.toString());
assertEquals(sortedSet.hashCode(), constrained.hashCode());
ASSERT.that(sortedSet).has().exactly("bar", "cat", "dog", "foo", "qux", TEST_ELEMENT).inOrder();
ASSERT.that(constrained).has()
.exactly("bar", "cat", "dog", "foo", "qux", TEST_ELEMENT).inOrder();
assertNull(constrained.comparator());
assertEquals("bar", constrained.first());
assertEquals(TEST_ELEMENT, constrained.last());
}
public void testConstrainedSortedSetIllegal() {
SortedSet<String> sortedSet = Sets.newTreeSet(asList("foo", "bar"));
SortedSet<String> constrained = Constraints.constrainedSortedSet(
sortedSet, TEST_CONSTRAINT);
try {
constrained.add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.subSet("bar", "foo").add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.headSet("bar").add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.tailSet("foo").add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
ASSERT.that(constrained).has().exactly("bar", "foo").inOrder();
ASSERT.that(sortedSet).has().exactly("bar", "foo").inOrder();
}
public void testConstrainedListLegal() {
List<String> list = Lists.newArrayList("foo", "bar");
List<String> constrained = Constraints.constrainedList(
list, TEST_CONSTRAINT);
list.add(TEST_ELEMENT);
constrained.add("qux");
constrained.addAll(asList("cat", "dog"));
constrained.add(1, "cow");
constrained.addAll(4, asList("box", "fan"));
constrained.set(2, "baz");
assertTrue(list.equals(constrained));
assertTrue(constrained.equals(list));
assertEquals(list.toString(), constrained.toString());
assertEquals(list.hashCode(), constrained.hashCode());
ASSERT.that(list).has().exactly(
"foo", "cow", "baz", TEST_ELEMENT, "box", "fan", "qux", "cat", "dog").inOrder();
ASSERT.that(constrained).has().exactly(
"foo", "cow", "baz", TEST_ELEMENT, "box", "fan", "qux", "cat", "dog").inOrder();
ListIterator<String> iterator = constrained.listIterator();
iterator.next();
iterator.set("sun");
constrained.listIterator(2).add("sky");
ASSERT.that(list).has().exactly(
"sun", "cow", "sky", "baz", TEST_ELEMENT, "box", "fan", "qux", "cat", "dog").inOrder();
ASSERT.that(constrained).has().exactly(
"sun", "cow", "sky", "baz", TEST_ELEMENT, "box", "fan", "qux", "cat", "dog").inOrder();
assertTrue(constrained instanceof RandomAccess);
}
public void testConstrainedListRandomAccessFalse() {
List<String> list = Lists.newLinkedList(asList("foo", "bar"));
List<String> constrained = Constraints.constrainedList(
list, TEST_CONSTRAINT);
list.add(TEST_ELEMENT);
constrained.add("qux");
assertFalse(constrained instanceof RandomAccess);
}
public void testConstrainedListIllegal() {
List<String> list = Lists.newArrayList("foo", "bar");
List<String> constrained = Constraints.constrainedList(
list, TEST_CONSTRAINT);
try {
constrained.add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.listIterator().add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.listIterator(1).add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.listIterator().set(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.listIterator(1).set(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.subList(0, 1).add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.add(1, TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.set(1, TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(1, asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
ASSERT.that(constrained).has().exactly("foo", "bar").inOrder();
ASSERT.that(list).has().exactly("foo", "bar").inOrder();
}
public void testConstrainedMultisetLegal() {
Multiset<String> multiset = HashMultiset.create(asList("foo", "bar"));
Multiset<String> constrained = Constraints.constrainedMultiset(
multiset, TEST_CONSTRAINT);
multiset.add(TEST_ELEMENT);
constrained.add("qux");
constrained.addAll(asList("cat", "dog"));
constrained.add("cow", 2);
assertTrue(multiset.equals(constrained));
assertTrue(constrained.equals(multiset));
assertEquals(multiset.toString(), constrained.toString());
assertEquals(multiset.hashCode(), constrained.hashCode());
ASSERT.that(multiset).has().exactly(
"foo", "bar", TEST_ELEMENT, "qux", "cat", "dog", "cow", "cow");
ASSERT.that(constrained).has().exactly(
"foo", "bar", TEST_ELEMENT, "qux", "cat", "dog", "cow", "cow");
assertEquals(1, constrained.count("foo"));
assertEquals(1, constrained.remove("foo", 3));
assertEquals(2, constrained.setCount("cow", 0));
ASSERT.that(multiset).has().exactly("bar", TEST_ELEMENT, "qux", "cat", "dog");
ASSERT.that(constrained).has().exactly("bar", TEST_ELEMENT, "qux", "cat", "dog");
}
public void testConstrainedMultisetIllegal() {
Multiset<String> multiset = HashMultiset.create(asList("foo", "bar"));
Multiset<String> constrained = Constraints.constrainedMultiset(
multiset, TEST_CONSTRAINT);
try {
constrained.add(TEST_ELEMENT);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.add(TEST_ELEMENT, 2);
fail("TestElementException expected");
} catch (TestElementException expected) {}
try {
constrained.addAll(asList("baz", TEST_ELEMENT));
fail("TestElementException expected");
} catch (TestElementException expected) {}
ASSERT.that(constrained).has().exactly("foo", "bar");
ASSERT.that(multiset).has().exactly("foo", "bar");
}
public void testNefariousAddAll() {
List<String> list = Lists.newArrayList("foo", "bar");
List<String> constrained = Constraints.constrainedList(
list, TEST_CONSTRAINT);
Collection<String> onceIterable = onceIterableCollection("baz");
constrained.addAll(onceIterable);
ASSERT.that(constrained).has().exactly("foo", "bar", "baz").inOrder();
ASSERT.that(list).has().exactly("foo", "bar", "baz").inOrder();
}
/**
* Returns a "nefarious" collection, which permits only one call to
* iterator(). This verifies that the constrained collection uses a defensive
* copy instead of potentially checking the elements in one snapshot and
* adding the elements from another.
*
* @param element the element to be contained in the collection
*/
static <E> Collection<E> onceIterableCollection(final E element) {
return new AbstractCollection<E>() {
boolean iteratorCalled;
@Override public int size() {
/*
* We could make the collection empty, but that seems more likely to
* trigger special cases (so maybe we should test both empty and
* nonempty...).
*/
return 1;
}
@Override public Iterator<E> iterator() {
assertFalse("Expected only one call to iterator()", iteratorCalled);
iteratorCalled = true;
return Collections.singleton(element).iterator();
}
};
}
@GwtIncompatible("SerializableTester")
public void testSerialization() {
// TODO: Test serialization of constrained collections.
assertSame(Constraints.notNull(),
SerializableTester.reserialize(Constraints.notNull()));
}
}