Index: third_party/protobuf/java/src/test/java/com/google/protobuf/LazyStringArrayListTest.java |
diff --git a/third_party/protobuf/java/src/test/java/com/google/protobuf/LazyStringArrayListTest.java b/third_party/protobuf/java/src/test/java/com/google/protobuf/LazyStringArrayListTest.java |
new file mode 100644 |
index 0000000000000000000000000000000000000000..0f42ac506075b394e2024f754b6208cafdbee361 |
--- /dev/null |
+++ b/third_party/protobuf/java/src/test/java/com/google/protobuf/LazyStringArrayListTest.java |
@@ -0,0 +1,362 @@ |
+// Protocol Buffers - Google's data interchange format |
+// Copyright 2008 Google Inc. All rights reserved. |
+// https://developers.google.com/protocol-buffers/ |
+// |
+// Redistribution and use in source and binary forms, with or without |
+// modification, are permitted provided that the following conditions are |
+// met: |
+// |
+// * Redistributions of source code must retain the above copyright |
+// notice, this list of conditions and the following disclaimer. |
+// * Redistributions in binary form must reproduce the above |
+// copyright notice, this list of conditions and the following disclaimer |
+// in the documentation and/or other materials provided with the |
+// distribution. |
+// * Neither the name of Google Inc. nor the names of its |
+// contributors may be used to endorse or promote products derived from |
+// this software without specific prior written permission. |
+// |
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
+ |
+package com.google.protobuf; |
+ |
+import static java.util.Arrays.asList; |
+ |
+import junit.framework.TestCase; |
+ |
+import java.util.ArrayList; |
+import java.util.ConcurrentModificationException; |
+import java.util.Iterator; |
+import java.util.List; |
+ |
+/** |
+ * Tests for {@link LazyStringArrayList}. |
+ * |
+ * @author jonp@google.com (Jon Perlow) |
+ */ |
+public class LazyStringArrayListTest extends TestCase { |
+ |
+ private static String STRING_A = "A"; |
+ private static String STRING_B = "B"; |
+ private static String STRING_C = "C"; |
+ |
+ private static ByteString BYTE_STRING_A = ByteString.copyFromUtf8("A"); |
+ private static ByteString BYTE_STRING_B = ByteString.copyFromUtf8("B"); |
+ private static ByteString BYTE_STRING_C = ByteString.copyFromUtf8("C"); |
+ |
+ public void testJustStrings() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.add(STRING_A); |
+ list.add(STRING_B); |
+ list.add(STRING_C); |
+ |
+ assertEquals(3, list.size()); |
+ assertSame(STRING_A, list.get(0)); |
+ assertSame(STRING_B, list.get(1)); |
+ assertSame(STRING_C, list.get(2)); |
+ |
+ list.set(1, STRING_C); |
+ assertSame(STRING_C, list.get(1)); |
+ |
+ list.remove(1); |
+ assertSame(STRING_A, list.get(0)); |
+ assertSame(STRING_C, list.get(1)); |
+ |
+ List<ByteString> byteStringList = list.asByteStringList(); |
+ assertEquals(BYTE_STRING_A, byteStringList.get(0)); |
+ assertEquals(BYTE_STRING_C, byteStringList.get(1)); |
+ |
+ // Underlying list should be transformed. |
+ assertSame(byteStringList.get(0), list.getByteString(0)); |
+ assertSame(byteStringList.get(1), list.getByteString(1)); |
+ } |
+ |
+ public void testJustByteString() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.add(BYTE_STRING_A); |
+ list.add(BYTE_STRING_B); |
+ list.add(BYTE_STRING_C); |
+ |
+ assertEquals(3, list.size()); |
+ assertSame(BYTE_STRING_A, list.getByteString(0)); |
+ assertSame(BYTE_STRING_B, list.getByteString(1)); |
+ assertSame(BYTE_STRING_C, list.getByteString(2)); |
+ |
+ list.remove(1); |
+ assertSame(BYTE_STRING_A, list.getByteString(0)); |
+ assertSame(BYTE_STRING_C, list.getByteString(1)); |
+ |
+ List<ByteString> byteStringList = list.asByteStringList(); |
+ assertSame(BYTE_STRING_A, byteStringList.get(0)); |
+ assertSame(BYTE_STRING_C, byteStringList.get(1)); |
+ } |
+ |
+ public void testConversionBackAndForth() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.add(STRING_A); |
+ list.add(BYTE_STRING_B); |
+ list.add(BYTE_STRING_C); |
+ |
+ // String a should be the same because it was originally a string |
+ assertSame(STRING_A, list.get(0)); |
+ |
+ // String b and c should be different because the string has to be computed |
+ // from the ByteString |
+ String bPrime = list.get(1); |
+ assertNotSame(STRING_B, bPrime); |
+ assertEquals(STRING_B, bPrime); |
+ String cPrime = list.get(2); |
+ assertNotSame(STRING_C, cPrime); |
+ assertEquals(STRING_C, cPrime); |
+ |
+ // String c and c should stay the same once cached. |
+ assertSame(bPrime, list.get(1)); |
+ assertSame(cPrime, list.get(2)); |
+ |
+ // ByteString needs to be computed from string for both a and b |
+ ByteString aPrimeByteString = list.getByteString(0); |
+ assertEquals(BYTE_STRING_A, aPrimeByteString); |
+ ByteString bPrimeByteString = list.getByteString(1); |
+ assertNotSame(BYTE_STRING_B, bPrimeByteString); |
+ assertEquals(BYTE_STRING_B, list.getByteString(1)); |
+ |
+ // Once cached, ByteString should stay cached. |
+ assertSame(aPrimeByteString, list.getByteString(0)); |
+ assertSame(bPrimeByteString, list.getByteString(1)); |
+ } |
+ |
+ public void testCopyConstructorCopiesByReference() { |
+ LazyStringArrayList list1 = new LazyStringArrayList(); |
+ list1.add(STRING_A); |
+ list1.add(BYTE_STRING_B); |
+ list1.add(BYTE_STRING_C); |
+ |
+ LazyStringArrayList list2 = new LazyStringArrayList(list1); |
+ assertEquals(3, list2.size()); |
+ assertSame(STRING_A, list2.get(0)); |
+ assertSame(BYTE_STRING_B, list2.getByteString(1)); |
+ assertSame(BYTE_STRING_C, list2.getByteString(2)); |
+ } |
+ |
+ public void testListCopyConstructor() { |
+ List<String> list1 = new ArrayList<String>(); |
+ list1.add(STRING_A); |
+ list1.add(STRING_B); |
+ list1.add(STRING_C); |
+ |
+ LazyStringArrayList list2 = new LazyStringArrayList(list1); |
+ assertEquals(3, list2.size()); |
+ assertSame(STRING_A, list2.get(0)); |
+ assertSame(STRING_B, list2.get(1)); |
+ assertSame(STRING_C, list2.get(2)); |
+ } |
+ |
+ public void testAddAllCopiesByReferenceIfPossible() { |
+ LazyStringArrayList list1 = new LazyStringArrayList(); |
+ list1.add(STRING_A); |
+ list1.add(BYTE_STRING_B); |
+ list1.add(BYTE_STRING_C); |
+ |
+ LazyStringArrayList list2 = new LazyStringArrayList(); |
+ list2.addAll(list1); |
+ |
+ assertEquals(3, list2.size()); |
+ assertSame(STRING_A, list2.get(0)); |
+ assertSame(BYTE_STRING_B, list2.getByteString(1)); |
+ assertSame(BYTE_STRING_C, list2.getByteString(2)); |
+ } |
+ |
+ public void testModificationWithIteration() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.addAll(asList(STRING_A, STRING_B, STRING_C)); |
+ Iterator<String> iterator = list.iterator(); |
+ assertEquals(3, list.size()); |
+ assertEquals(STRING_A, list.get(0)); |
+ assertEquals(STRING_A, iterator.next()); |
+ |
+ // Does not structurally modify. |
+ iterator = list.iterator(); |
+ list.set(0, STRING_B); |
+ iterator.next(); |
+ |
+ list.remove(0); |
+ try { |
+ iterator.next(); |
+ fail(); |
+ } catch (ConcurrentModificationException e) { |
+ // expected |
+ } |
+ |
+ iterator = list.iterator(); |
+ list.add(0, STRING_C); |
+ try { |
+ iterator.next(); |
+ fail(); |
+ } catch (ConcurrentModificationException e) { |
+ // expected |
+ } |
+ } |
+ |
+ public void testMakeImmutable() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.add(STRING_A); |
+ list.add(STRING_B); |
+ list.add(STRING_C); |
+ list.makeImmutable(); |
+ assertGenericListImmutable(list, STRING_A); |
+ |
+ // LazyStringArrayList has extra methods not covered in the generic |
+ // assertion. |
+ |
+ try { |
+ list.add(BYTE_STRING_A.toByteArray()); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.add(BYTE_STRING_A); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.addAllByteArray(asList(BYTE_STRING_A.toByteArray())); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.addAllByteString(asList(BYTE_STRING_A)); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.mergeFrom(new LazyStringArrayList()); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.set(0, BYTE_STRING_A.toByteArray()); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.set(0, BYTE_STRING_A); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ } |
+ |
+ public void testImmutabilityPropagation() { |
+ LazyStringArrayList list = new LazyStringArrayList(); |
+ list.add(STRING_A); |
+ list.makeImmutable(); |
+ |
+ assertGenericListImmutable(list.asByteStringList(), BYTE_STRING_A); |
+ |
+ // Arrays use reference equality so need to retrieve the underlying value |
+ // to properly test deep immutability. |
+ List<byte[]> byteArrayList = list.asByteArrayList(); |
+ assertGenericListImmutable(byteArrayList, byteArrayList.get(0)); |
+ } |
+ |
+ private static <T> void assertGenericListImmutable(List<T> list, T value) { |
+ try { |
+ list.add(value); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.add(0, value); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.addAll(asList(value)); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.addAll(0, asList(value)); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.clear(); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.remove(0); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.remove(value); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.removeAll(asList(value)); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.retainAll(asList()); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.retainAll(asList()); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ |
+ try { |
+ list.set(0, value); |
+ fail(); |
+ } catch (UnsupportedOperationException e) { |
+ // expected |
+ } |
+ } |
+} |