blob: 1688bade0936427e128be01db8ec7abc83a3c082 [file] [log] [blame]
/*
* 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 com.google.common.collect.Iterables.getOnlyElement;
import static com.google.common.collect.Iterables.unmodifiableIterable;
import static com.google.common.collect.Sets.newHashSet;
import static com.google.common.collect.testing.features.CollectionFeature.ALLOWS_NULL_QUERIES;
import static com.google.common.collect.testing.features.CollectionFeature.SERIALIZABLE;
import static java.lang.reflect.Proxy.newProxyInstance;
import static java.util.Arrays.asList;
import com.google.common.annotations.GwtCompatible;
import com.google.common.annotations.GwtIncompatible;
import com.google.common.collect.testing.Helpers;
import com.google.common.collect.testing.ListTestSuiteBuilder;
import com.google.common.collect.testing.MinimalCollection;
import com.google.common.collect.testing.MinimalIterable;
import com.google.common.collect.testing.features.CollectionSize;
import com.google.common.collect.testing.google.ListGenerators.BuilderAddAllListGenerator;
import com.google.common.collect.testing.google.ListGenerators.BuilderReversedListGenerator;
import com.google.common.collect.testing.google.ListGenerators.ImmutableListHeadSubListGenerator;
import com.google.common.collect.testing.google.ListGenerators.ImmutableListMiddleSubListGenerator;
import com.google.common.collect.testing.google.ListGenerators.ImmutableListOfGenerator;
import com.google.common.collect.testing.google.ListGenerators.ImmutableListTailSubListGenerator;
import com.google.common.collect.testing.google.ListGenerators.UnhashableElementsImmutableListGenerator;
import com.google.common.collect.testing.testers.ListHashCodeTester;
import com.google.common.testing.NullPointerTester;
import com.google.common.testing.SerializableTester;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.CopyOnWriteArrayList;
import junit.framework.Test;
import junit.framework.TestCase;
import junit.framework.TestSuite;
/**
* Unit test for {@link ImmutableList}.
*
* @author Kevin Bourrillion
* @author George van den Driessche
* @author Jared Levy
*/
@GwtCompatible(emulated = true)
public class ImmutableListTest extends TestCase {
@GwtIncompatible // suite
public static Test suite() {
TestSuite suite = new TestSuite();
suite.addTest(
ListTestSuiteBuilder.using(new ImmutableListOfGenerator())
.named("ImmutableList")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new BuilderAddAllListGenerator())
.named("ImmutableList, built with Builder.add")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new BuilderAddAllListGenerator())
.named("ImmutableList, built with Builder.addAll")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new BuilderReversedListGenerator())
.named("ImmutableList, reversed")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new ImmutableListHeadSubListGenerator())
.named("ImmutableList, head subList")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new ImmutableListTailSubListGenerator())
.named("ImmutableList, tail subList")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new ImmutableListMiddleSubListGenerator())
.named("ImmutableList, middle subList")
.withFeatures(CollectionSize.ANY, SERIALIZABLE, ALLOWS_NULL_QUERIES)
.createTestSuite());
suite.addTest(
ListTestSuiteBuilder.using(new UnhashableElementsImmutableListGenerator())
.suppressing(ListHashCodeTester.getHashCodeMethod())
.named("ImmutableList, unhashable values")
.withFeatures(CollectionSize.ANY, ALLOWS_NULL_QUERIES)
.createTestSuite());
return suite;
}
public static class CreationTests extends TestCase {
public void testCreation_noArgs() {
List<String> list = ImmutableList.of();
assertEquals(Collections.emptyList(), list);
}
public void testCreation_oneElement() {
List<String> list = ImmutableList.of("a");
assertEquals(Collections.singletonList("a"), list);
}
public void testCreation_twoElements() {
List<String> list = ImmutableList.of("a", "b");
assertEquals(Lists.newArrayList("a", "b"), list);
}
public void testCreation_threeElements() {
List<String> list = ImmutableList.of("a", "b", "c");
assertEquals(Lists.newArrayList("a", "b", "c"), list);
}
public void testCreation_fourElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d");
assertEquals(Lists.newArrayList("a", "b", "c", "d"), list);
}
public void testCreation_fiveElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e"), list);
}
public void testCreation_sixElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f"), list);
}
public void testCreation_sevenElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f", "g");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f", "g"), list);
}
public void testCreation_eightElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h"), list);
}
public void testCreation_nineElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i"), list);
}
public void testCreation_tenElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i", "j");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i", "j"), list);
}
public void testCreation_elevenElements() {
List<String> list = ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k");
assertEquals(Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k"), list);
}
// Varargs versions
public void testCreation_twelveElements() {
List<String> list =
ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l");
assertEquals(
Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l"), list);
}
public void testCreation_thirteenElements() {
List<String> list =
ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m");
assertEquals(
Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m"),
list);
}
public void testCreation_fourteenElements() {
List<String> list =
ImmutableList.of("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n");
assertEquals(
Lists.newArrayList("a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n"),
list);
}
public void testCreation_singletonNull() {
try {
ImmutableList.of((String) null);
fail();
} catch (NullPointerException expected) {
}
}
public void testCreation_withNull() {
try {
ImmutableList.of("a", null, "b");
fail();
} catch (NullPointerException expected) {
}
}
public void testCreation_generic() {
List<String> a = ImmutableList.of("a");
// only verify that there is no compile warning
ImmutableList<List<String>> unused = ImmutableList.of(a, a);
}
public void testCreation_arrayOfArray() {
String[] array = new String[] {"a"};
List<String[]> list = ImmutableList.<String[]>of(array);
assertEquals(Collections.singletonList(array), list);
}
public void testCopyOf_emptyArray() {
String[] array = new String[0];
List<String> list = ImmutableList.copyOf(array);
assertEquals(Collections.emptyList(), list);
}
public void testCopyOf_arrayOfOneElement() {
String[] array = new String[] {"a"};
List<String> list = ImmutableList.copyOf(array);
assertEquals(Collections.singletonList("a"), list);
}
public void testCopyOf_nullArray() {
try {
ImmutableList.copyOf((String[]) null);
fail();
} catch (NullPointerException expected) {
}
}
public void testCopyOf_arrayContainingOnlyNull() {
String[] array = new String[] {null};
try {
ImmutableList.copyOf(array);
fail();
} catch (NullPointerException expected) {
}
}
public void testCopyOf_collection_empty() {
// "<String>" is required to work around a javac 1.5 bug.
Collection<String> c = MinimalCollection.<String>of();
List<String> list = ImmutableList.copyOf(c);
assertEquals(Collections.emptyList(), list);
}
public void testCopyOf_collection_oneElement() {
Collection<String> c = MinimalCollection.of("a");
List<String> list = ImmutableList.copyOf(c);
assertEquals(Collections.singletonList("a"), list);
}
public void testCopyOf_collection_general() {
Collection<String> c = MinimalCollection.of("a", "b", "a");
List<String> list = ImmutableList.copyOf(c);
assertEquals(asList("a", "b", "a"), list);
List<String> mutableList = asList("a", "b");
list = ImmutableList.copyOf(mutableList);
mutableList.set(0, "c");
assertEquals(asList("a", "b"), list);
}
public void testCopyOf_collectionContainingNull() {
Collection<String> c = MinimalCollection.of("a", null, "b");
try {
ImmutableList.copyOf(c);
fail();
} catch (NullPointerException expected) {
}
}
public void testCopyOf_iterator_empty() {
Iterator<String> iterator = Iterators.emptyIterator();
List<String> list = ImmutableList.copyOf(iterator);
assertEquals(Collections.emptyList(), list);
}
public void testCopyOf_iterator_oneElement() {
Iterator<String> iterator = Iterators.singletonIterator("a");
List<String> list = ImmutableList.copyOf(iterator);
assertEquals(Collections.singletonList("a"), list);
}
public void testCopyOf_iterator_general() {
Iterator<String> iterator = asList("a", "b", "a").iterator();
List<String> list = ImmutableList.copyOf(iterator);
assertEquals(asList("a", "b", "a"), list);
}
public void testCopyOf_iteratorContainingNull() {
Iterator<String> iterator = asList("a", null, "b").iterator();
try {
ImmutableList.copyOf(iterator);
fail();
} catch (NullPointerException expected) {
}
}
public void testCopyOf_iteratorNull() {
try {
ImmutableList.copyOf((Iterator<String>) null);
fail();
} catch (NullPointerException expected) {
}
}
public void testCopyOf_concurrentlyMutating() {
List<String> sample = Lists.newArrayList("a", "b", "c");
for (int delta : new int[] {-1, 0, 1}) {
for (int i = 0; i < sample.size(); i++) {
Collection<String> misleading = Helpers.misleadingSizeCollection(delta);
List<String> expected = sample.subList(0, i);
misleading.addAll(expected);
assertEquals(expected, ImmutableList.copyOf(misleading));
assertEquals(expected, ImmutableList.copyOf((Iterable<String>) misleading));
}
}
}
private static class CountingIterable implements Iterable<String> {
int count = 0;
@Override
public Iterator<String> iterator() {
count++;
return asList("a", "b", "a").iterator();
}
}
public void testCopyOf_plainIterable() {
CountingIterable iterable = new CountingIterable();
List<String> list = ImmutableList.copyOf(iterable);
assertEquals(asList("a", "b", "a"), list);
}
public void testCopyOf_plainIterable_iteratesOnce() {
CountingIterable iterable = new CountingIterable();
ImmutableList<String> unused = ImmutableList.copyOf(iterable);
assertEquals(1, iterable.count);
}
public void testCopyOf_shortcut_empty() {
Collection<String> c = ImmutableList.of();
assertSame(c, ImmutableList.copyOf(c));
}
public void testCopyOf_shortcut_singleton() {
Collection<String> c = ImmutableList.of("a");
assertSame(c, ImmutableList.copyOf(c));
}
public void testCopyOf_shortcut_immutableList() {
Collection<String> c = ImmutableList.of("a", "b", "c");
assertSame(c, ImmutableList.copyOf(c));
}
public void testBuilderAddArrayHandlesNulls() {
String[] elements = {"a", null, "b"};
ImmutableList.Builder<String> builder = ImmutableList.builder();
try {
builder.add(elements);
fail("Expected NullPointerException");
} catch (NullPointerException expected) {
}
ImmutableList<String> result = builder.build();
/*
* Maybe it rejects all elements, or maybe it adds "a" before failing.
* Either way is fine with us.
*/
if (result.isEmpty()) {
return;
}
assertTrue(ImmutableList.of("a").equals(result));
assertEquals(1, result.size());
}
public void testBuilderAddCollectionHandlesNulls() {
List<String> elements = Arrays.asList("a", null, "b");
ImmutableList.Builder<String> builder = ImmutableList.builder();
try {
builder.addAll(elements);
fail("Expected NullPointerException");
} catch (NullPointerException expected) {
}
ImmutableList<String> result = builder.build();
assertEquals(ImmutableList.of("a"), result);
assertEquals(1, result.size());
}
public void testSortedCopyOf_natural() {
Collection<Integer> c = MinimalCollection.of(4, 16, 10, -1, 5);
ImmutableList<Integer> list = ImmutableList.sortedCopyOf(c);
assertEquals(asList(-1, 4, 5, 10, 16), list);
}
public void testSortedCopyOf_natural_empty() {
Collection<Integer> c = MinimalCollection.of();
ImmutableList<Integer> list = ImmutableList.sortedCopyOf(c);
assertEquals(asList(), list);
}
public void testSortedCopyOf_natural_singleton() {
Collection<Integer> c = MinimalCollection.of(100);
ImmutableList<Integer> list = ImmutableList.sortedCopyOf(c);
assertEquals(asList(100), list);
}
public void testSortedCopyOf_natural_containsNull() {
Collection<Integer> c = MinimalCollection.of(1, 3, null, 2);
try {
ImmutableList.sortedCopyOf(c);
fail("Expected NPE");
} catch (NullPointerException expected) {
}
}
public void testSortedCopyOf() {
Collection<String> c = MinimalCollection.of("a", "b", "A", "c");
List<String> list = ImmutableList.sortedCopyOf(String.CASE_INSENSITIVE_ORDER, c);
assertEquals(asList("a", "A", "b", "c"), list);
}
public void testSortedCopyOf_empty() {
Collection<String> c = MinimalCollection.of();
List<String> list = ImmutableList.sortedCopyOf(String.CASE_INSENSITIVE_ORDER, c);
assertEquals(asList(), list);
}
public void testSortedCopyOf_singleton() {
Collection<String> c = MinimalCollection.of("a");
List<String> list = ImmutableList.sortedCopyOf(String.CASE_INSENSITIVE_ORDER, c);
assertEquals(asList("a"), list);
}
public void testSortedCopyOf_containsNull() {
Collection<String> c = MinimalCollection.of("a", "b", "A", null, "c");
try {
ImmutableList.sortedCopyOf(String.CASE_INSENSITIVE_ORDER, c);
fail("Expected NPE");
} catch (NullPointerException expected) {
}
}
}
@GwtIncompatible // reflection
public static class ConcurrentTests extends TestCase {
enum WrapWithIterable {
WRAP,
NO_WRAP
}
private static void runConcurrentlyMutatedTest(
Collection<Integer> initialContents,
Iterable<ListFrobber> actionsToPerformConcurrently,
WrapWithIterable wrap) {
ConcurrentlyMutatedList<Integer> concurrentlyMutatedList =
newConcurrentlyMutatedList(initialContents, actionsToPerformConcurrently);
Iterable<Integer> iterableToCopy =
wrap == WrapWithIterable.WRAP
? unmodifiableIterable(concurrentlyMutatedList)
: concurrentlyMutatedList;
ImmutableList<Integer> copyOfIterable = ImmutableList.copyOf(iterableToCopy);
assertTrue(concurrentlyMutatedList.getAllStates().contains(copyOfIterable));
}
private static void runConcurrentlyMutatedTest(WrapWithIterable wrap) {
/*
* TODO: Iterate over many array sizes and all possible operation lists,
* performing adds and removes in different ways.
*/
runConcurrentlyMutatedTest(elements(), ops(add(1), add(2)), wrap);
runConcurrentlyMutatedTest(elements(), ops(add(1), nop()), wrap);
runConcurrentlyMutatedTest(elements(), ops(add(1), remove()), wrap);
runConcurrentlyMutatedTest(elements(), ops(nop(), add(1)), wrap);
runConcurrentlyMutatedTest(elements(1), ops(remove(), nop()), wrap);
runConcurrentlyMutatedTest(elements(1), ops(remove(), add(2)), wrap);
runConcurrentlyMutatedTest(elements(1, 2), ops(remove(), remove()), wrap);
runConcurrentlyMutatedTest(elements(1, 2), ops(remove(), nop()), wrap);
runConcurrentlyMutatedTest(elements(1, 2), ops(remove(), add(3)), wrap);
runConcurrentlyMutatedTest(elements(1, 2), ops(nop(), remove()), wrap);
runConcurrentlyMutatedTest(elements(1, 2, 3), ops(remove(), remove()), wrap);
}
private static ImmutableList<Integer> elements(Integer... elements) {
return ImmutableList.copyOf(elements);
}
private static ImmutableList<ListFrobber> ops(ListFrobber... elements) {
return ImmutableList.copyOf(elements);
}
public void testCopyOf_concurrentlyMutatedList() {
runConcurrentlyMutatedTest(WrapWithIterable.NO_WRAP);
}
public void testCopyOf_concurrentlyMutatedIterable() {
runConcurrentlyMutatedTest(WrapWithIterable.WRAP);
}
/** An operation to perform on a list. */
interface ListFrobber {
void perform(List<Integer> list);
}
static ListFrobber add(final int element) {
return new ListFrobber() {
@Override
public void perform(List<Integer> list) {
list.add(0, element);
}
};
}
static ListFrobber remove() {
return new ListFrobber() {
@Override
public void perform(List<Integer> list) {
list.remove(0);
}
};
}
static ListFrobber nop() {
return new ListFrobber() {
@Override
public void perform(List<Integer> list) {}
};
}
/** A list that mutates itself after every call to each of its {@link List} methods. */
interface ConcurrentlyMutatedList<E> extends List<E> {
/**
* The elements of a {@link ConcurrentlyMutatedList} are added and removed over time. This
* method returns every state that the list has passed through at some point.
*/
Set<List<E>> getAllStates();
}
/**
* Returns a {@link ConcurrentlyMutatedList} that performs the given operations as its
* concurrent modifications. The mutations occur in the same thread as the triggering method
* call.
*/
private static ConcurrentlyMutatedList<Integer> newConcurrentlyMutatedList(
final Collection<Integer> initialContents,
final Iterable<ListFrobber> actionsToPerformConcurrently) {
InvocationHandler invocationHandler =
new InvocationHandler() {
final CopyOnWriteArrayList<Integer> delegate =
new CopyOnWriteArrayList<>(initialContents);
final Method getAllStatesMethod =
getOnlyElement(asList(ConcurrentlyMutatedList.class.getDeclaredMethods()));
final Iterator<ListFrobber> remainingActions = actionsToPerformConcurrently.iterator();
final Set<List<Integer>> allStates = newHashSet();
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
return method.equals(getAllStatesMethod)
? getAllStates()
: invokeListMethod(method, args);
}
private Set<List<Integer>> getAllStates() {
return allStates;
}
private Object invokeListMethod(Method method, Object[] args) throws Throwable {
try {
Object returnValue = method.invoke(delegate, args);
mutateDelegate();
return returnValue;
} catch (InvocationTargetException e) {
throw e.getCause();
} catch (IllegalAccessException e) {
throw new AssertionError(e);
}
}
private void mutateDelegate() {
allStates.add(ImmutableList.copyOf(delegate));
remainingActions.next().perform(delegate);
allStates.add(ImmutableList.copyOf(delegate));
}
};
@SuppressWarnings("unchecked")
ConcurrentlyMutatedList<Integer> list =
(ConcurrentlyMutatedList<Integer>)
newProxyInstance(
ImmutableListTest.CreationTests.class.getClassLoader(),
new Class[] {ConcurrentlyMutatedList.class},
invocationHandler);
return list;
}
}
public static class BasicTests extends TestCase {
@GwtIncompatible // NullPointerTester
public void testNullPointers() {
NullPointerTester tester = new NullPointerTester();
tester.testAllPublicStaticMethods(ImmutableList.class);
tester.testAllPublicInstanceMethods(ImmutableList.of(1, 2, 3));
}
@GwtIncompatible // SerializableTester
public void testSerialization_empty() {
Collection<String> c = ImmutableList.of();
assertSame(c, SerializableTester.reserialize(c));
}
@GwtIncompatible // SerializableTester
public void testSerialization_singleton() {
Collection<String> c = ImmutableList.of("a");
SerializableTester.reserializeAndAssert(c);
}
@GwtIncompatible // SerializableTester
public void testSerialization_multiple() {
Collection<String> c = ImmutableList.of("a", "b", "c");
SerializableTester.reserializeAndAssert(c);
}
public void testEquals_immutableList() {
Collection<String> c = ImmutableList.of("a", "b", "c");
assertTrue(c.equals(ImmutableList.of("a", "b", "c")));
assertFalse(c.equals(ImmutableList.of("a", "c", "b")));
assertFalse(c.equals(ImmutableList.of("a", "b")));
assertFalse(c.equals(ImmutableList.of("a", "b", "c", "d")));
}
public void testBuilderAdd() {
ImmutableList<String> list =
new ImmutableList.Builder<String>().add("a").add("b").add("a").add("c").build();
assertEquals(asList("a", "b", "a", "c"), list);
}
@GwtIncompatible("Builder impl")
public void testBuilderForceCopy() {
ImmutableList.Builder<Integer> builder = ImmutableList.builder();
Object[] prevArray = null;
for (int i = 0; i < 10; i++) {
builder.add(i);
assertNotSame(builder.contents, prevArray);
prevArray = builder.contents;
ImmutableList<Integer> unused = builder.build();
}
}
@GwtIncompatible
public void testBuilderExactlySizedReusesArray() {
ImmutableList.Builder<Integer> builder = ImmutableList.builderWithExpectedSize(10);
Object[] builderArray = builder.contents;
for (int i = 0; i < 10; i++) {
builder.add(i);
}
Object[] builderArrayAfterAdds = builder.contents;
RegularImmutableList<Integer> list = (RegularImmutableList<Integer>) builder.build();
Object[] listInternalArray = list.array;
assertSame(builderArray, builderArrayAfterAdds);
assertSame(builderArray, listInternalArray);
}
public void testBuilderAdd_varargs() {
ImmutableList<String> list =
new ImmutableList.Builder<String>().add("a", "b", "a", "c").build();
assertEquals(asList("a", "b", "a", "c"), list);
}
public void testBuilderAddAll_iterable() {
List<String> a = asList("a", "b");
List<String> b = asList("c", "d");
ImmutableList<String> list = new ImmutableList.Builder<String>().addAll(a).addAll(b).build();
assertEquals(asList("a", "b", "c", "d"), list);
b.set(0, "f");
assertEquals(asList("a", "b", "c", "d"), list);
}
public void testBuilderAddAll_iterator() {
List<String> a = asList("a", "b");
List<String> b = asList("c", "d");
ImmutableList<String> list =
new ImmutableList.Builder<String>().addAll(a.iterator()).addAll(b.iterator()).build();
assertEquals(asList("a", "b", "c", "d"), list);
b.set(0, "f");
assertEquals(asList("a", "b", "c", "d"), list);
}
public void testComplexBuilder() {
List<Integer> colorElem = asList(0x00, 0x33, 0x66, 0x99, 0xCC, 0xFF);
ImmutableList.Builder<Integer> webSafeColorsBuilder = ImmutableList.builder();
for (Integer red : colorElem) {
for (Integer green : colorElem) {
for (Integer blue : colorElem) {
webSafeColorsBuilder.add((red << 16) + (green << 8) + blue);
}
}
}
ImmutableList<Integer> webSafeColors = webSafeColorsBuilder.build();
assertEquals(216, webSafeColors.size());
Integer[] webSafeColorArray = webSafeColors.toArray(new Integer[webSafeColors.size()]);
assertEquals(0x000000, (int) webSafeColorArray[0]);
assertEquals(0x000033, (int) webSafeColorArray[1]);
assertEquals(0x000066, (int) webSafeColorArray[2]);
assertEquals(0x003300, (int) webSafeColorArray[6]);
assertEquals(0x330000, (int) webSafeColorArray[36]);
assertEquals(0x000066, (int) webSafeColors.get(2));
assertEquals(0x003300, (int) webSafeColors.get(6));
ImmutableList<Integer> addedColor = webSafeColorsBuilder.add(0x00BFFF).build();
assertEquals(
"Modifying the builder should not have changed any already" + " built sets",
216,
webSafeColors.size());
assertEquals("the new array should be one bigger than webSafeColors", 217, addedColor.size());
Integer[] appendColorArray = addedColor.toArray(new Integer[addedColor.size()]);
assertEquals(0x00BFFF, (int) appendColorArray[216]);
}
public void testBuilderAddHandlesNullsCorrectly() {
ImmutableList.Builder<String> builder = ImmutableList.builder();
try {
builder.add((String) null);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
try {
builder.add((String[]) null);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
try {
builder.add("a", null, "b");
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
}
public void testBuilderAddAllHandlesNullsCorrectly() {
ImmutableList.Builder<String> builder = ImmutableList.builder();
try {
builder.addAll((Iterable<String>) null);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
try {
builder.addAll((Iterator<String>) null);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
builder = ImmutableList.builder();
List<String> listWithNulls = asList("a", null, "b");
try {
builder.addAll(listWithNulls);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
builder = ImmutableList.builder();
Iterator<String> iteratorWithNulls = asList("a", null, "b").iterator();
try {
builder.addAll(iteratorWithNulls);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
Iterable<String> iterableWithNulls = MinimalIterable.of("a", null, "b");
try {
builder.addAll(iterableWithNulls);
fail("expected NullPointerException");
} catch (NullPointerException expected) {
}
}
public void testAsList() {
ImmutableList<String> list = ImmutableList.of("a", "b");
assertSame(list, list.asList());
}
@GwtIncompatible("builder internals")
public void testReusedBuilder() {
ImmutableList.Builder<String> builder = new ImmutableList.Builder<String>();
for (int i = 0; i < 10; i++) {
builder.add("foo");
}
builder.add("bar");
RegularImmutableList<String> list = (RegularImmutableList<String>) builder.build();
builder.add("baz");
assertTrue(list.array != builder.contents);
}
}
}