/* * Copyright (C) 2010 Google Inc. * * 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 libcore.java.util; import java.util.AbstractMap.SimpleEntry; import java.util.ConcurrentModificationException; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.Map.Entry; import java.util.NavigableMap; import java.util.SortedMap; import java.util.TreeMap; import junit.framework.TestCase; import libcore.util.SerializationTester; public class TreeMapTest extends TestCase { /** * Test that the entrySet() method produces correctly mutable entries. */ public void testEntrySetSetValue() { TreeMap<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); Iterator<Entry<String, String>> iterator = map.entrySet().iterator(); Entry<String, String> entryA = iterator.next(); assertEquals("a", entryA.setValue("x")); assertEquals("x", entryA.getValue()); assertEquals("x", map.get("A")); Entry<String, String> entryB = iterator.next(); assertEquals("b", entryB.setValue("y")); Entry<String, String> entryC = iterator.next(); assertEquals("c", entryC.setValue("z")); assertEquals("y", entryB.getValue()); assertEquals("y", map.get("B")); assertEquals("z", entryC.getValue()); assertEquals("z", map.get("C")); } /** * Test that the entrySet() method of a sub map produces correctly mutable * entries that propagate changes to the original map. */ public void testSubMapEntrySetSetValue() { TreeMap<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); map.put("D", "d"); NavigableMap<String, String> subMap = map.subMap("A", true, "C", true); Iterator<Entry<String, String>> iterator = subMap.entrySet().iterator(); Entry<String, String> entryA = iterator.next(); assertEquals("a", entryA.setValue("x")); assertEquals("x", entryA.getValue()); assertEquals("x", subMap.get("A")); assertEquals("x", map.get("A")); Entry<String, String> entryB = iterator.next(); assertEquals("b", entryB.setValue("y")); Entry<String, String> entryC = iterator.next(); assertEquals("c", entryC.setValue("z")); assertEquals("y", entryB.getValue()); assertEquals("y", subMap.get("B")); assertEquals("y", map.get("B")); assertEquals("z", entryC.getValue()); assertEquals("z", subMap.get("C")); assertEquals("z", map.get("C")); } /** * Test that an Entry given by any method except entrySet() is immutable. */ public void testExceptionsOnSetValue() { TreeMap<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); assertAllEntryMethodsReturnImmutableEntries(map); } /** * Test that an Entry given by any method except entrySet() of a sub map is immutable. */ public void testExceptionsOnSubMapSetValue() { TreeMap<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); map.put("D", "d"); assertAllEntryMethodsReturnImmutableEntries(map.subMap("A", true, "C", true)); } /** * Asserts that each NavigableMap method that returns an Entry (except entrySet()) returns an * immutable one. Assumes that the map contains at least entries with keys "A", "B" and "C". */ private void assertAllEntryMethodsReturnImmutableEntries(NavigableMap<String, String> map) { assertImmutable(map.ceilingEntry("B")); assertImmutable(map.firstEntry()); assertImmutable(map.floorEntry("D")); assertImmutable(map.higherEntry("A")); assertImmutable(map.lastEntry()); assertImmutable(map.lowerEntry("C")); assertImmutable(map.pollFirstEntry()); assertImmutable(map.pollLastEntry()); } private void assertImmutable(Entry<String, String> entry) { String initialValue = entry.getValue(); try { entry.setValue("x"); fail(); } catch (UnsupportedOperationException e) { } assertEquals(initialValue, entry.getValue()); } public void testConcurrentModificationDetection() { Map<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); Iterator<Entry<String,String>> iterator = map.entrySet().iterator(); iterator.next(); iterator.next(); iterator.remove(); map.put("D", "d"); try { iterator.next(); fail(); } catch (ConcurrentModificationException e) { } } public void testIteratorRemoves() { Map<String, String> map = new TreeMap<String, String>(); map.put("A", "a"); map.put("B", "b"); map.put("C", "c"); Iterator<Entry<String,String>> iterator = map.entrySet().iterator(); assertEquals("A", iterator.next().getKey()); assertEquals("B", iterator.next().getKey()); iterator.remove(); assertEquals("C", iterator.next().getKey()); iterator.remove(); assertFalse(iterator.hasNext()); } /** * Test that entry set contains and removal use the comparator rather than equals. */ public void testEntrySetUsesComparatorOnly() { Map<String,String> map = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); map.put("ABC", "a"); assertTrue(map.entrySet().contains(new SimpleEntry<String, String>("abc", "a"))); assertTrue(map.entrySet().remove(new SimpleEntry<String, String>("abc", "a"))); assertEquals(0, map.size()); } public void testMapConstructorPassingSortedMap() { Map<String,String> source = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); NavigableMap<String,String> copy = new TreeMap<String, String>(source); assertEquals(null, copy.comparator()); } public void testNullsWithNaturalOrder() { HashMap<String, String> copyFrom = new HashMap<String, String>(); copyFrom.put(null, "b"); try { new TreeMap<String, String>(copyFrom); fail(); // the RI doesn't throw if null is the only key } catch (NullPointerException expected) { } TreeMap<String,String> map = new TreeMap<String, String>(); try { map.put(null, "b"); fail(); } catch (NullPointerException e) { } try { map.descendingMap().put(null, "b"); fail(); } catch (NullPointerException e) { } try { map.subMap("a", "z").put(null, "b"); fail(); } catch (NullPointerException e) { } } public void testClassCastExceptions() { Map<Object, Object> map = new TreeMap<Object, Object>(); map.put("A", "a"); try { map.get(5); fail(); } catch (ClassCastException e) { } try { map.containsKey(5); fail(); } catch (ClassCastException e) { } try { map.remove(5); fail(); } catch (ClassCastException e) { } } public void testClassCastExceptionsOnBounds() { Map<Object, Object> map = new TreeMap<Object, Object>().subMap("a", "z"); try { map.get(5); fail(); } catch (ClassCastException e) { } try { map.containsKey(5); fail(); } catch (ClassCastException e) { } try { map.remove(5); fail(); } catch (ClassCastException e) { } } public void testClone() { TreeMap<String, String> map = new TreeMap<String, String>() { @Override public String toString() { return "x:" + super.toString(); } }; map.put("A", "a"); map.put("B", "b"); @SuppressWarnings("unchecked") TreeMap<String, String> clone = (TreeMap<String, String>) map.clone(); assertEquals(map.getClass(), clone.getClass()); assertEquals("x:{A=a, B=b}", map.toString()); } public void testEmptyMapSerialization() { String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" + "f6d70617261746f723b78707077040000000078"; TreeMap<String, String> map = new TreeMap<String, String>(); new SerializationTester<TreeMap<String, String>>(map, s).test(); } public void testSerializationWithComparator() { String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" + "f6d70617261746f723b78707372002a6a6176612e6c616e672e537472696e6724" + "43617365496e73656e736974697665436f6d70617261746f7277035c7d5c50e5c" + "e02000078707704000000027400016171007e00057400016271007e000678"; TreeMap<String, String> map = new TreeMap<String, String>( String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); new SerializationTester<NavigableMap<String, String>>(map, s) { @Override protected void verify(NavigableMap<String, String> deserialized) { assertEquals(0, deserialized.comparator().compare("X", "x")); } }.test(); } public void testSubMapSerialization() { String s = "aced0005737200216a6176612e7574696c2e547265654d617024417363656e646" + "96e675375624d61700cab946d1f0fab1c020000787200216a6176612e7574696c2" + "e547265654d6170244e6176696761626c655375624d6170e2d0a70e64210e08020" + "0075a000966726f6d53746172745a000b6869496e636c75736976655a000b6c6f4" + "96e636c75736976655a0005746f456e644c000268697400124c6a6176612f6c616" + "e672f4f626a6563743b4c00026c6f71007e00024c00016d7400134c6a6176612f7" + "574696c2f547265654d61703b7870000001007400016374000161737200116a617" + "6612e7574696c2e547265654d61700cc1f63e2d256ae60300014c000a636f6d706" + "17261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b78707" + "372002a6a6176612e6c616e672e537472696e672443617365496e73656e7369746" + "97665436f6d70617261746f7277035c7d5c50e5ce0200007870770400000004710" + "07e000671007e00067400016271007e000c71007e000571007e000574000164710" + "07e000d78"; TreeMap<String, String> map = new TreeMap<String, String>( String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); map.put("c", "c"); map.put("d", "d"); SortedMap<String, String> subMap = map.subMap("a", "c"); new SerializationTester<SortedMap<String, String>>(subMap, s) { @Override protected void verify(SortedMap<String, String> deserialized) { try { deserialized.put("e", "e"); fail(); } catch (IllegalArgumentException expected) { } } }.test(); } public void testNavigableSubMapSerialization() { String s = "aced0005737200216a6176612e7574696c2e547265654d617024417363656e646" + "96e675375624d61700cab946d1f0fab1c020000787200216a6176612e7574696c2" + "e547265654d6170244e6176696761626c655375624d6170e2d0a70e64210e08020" + "0075a000966726f6d53746172745a000b6869496e636c75736976655a000b6c6f4" + "96e636c75736976655a0005746f456e644c000268697400124c6a6176612f6c616" + "e672f4f626a6563743b4c00026c6f71007e00024c00016d7400134c6a6176612f7" + "574696c2f547265654d61703b7870000100007400016374000161737200116a617" + "6612e7574696c2e547265654d61700cc1f63e2d256ae60300014c000a636f6d706" + "17261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b78707" + "372002a6a6176612e6c616e672e537472696e672443617365496e73656e7369746" + "97665436f6d70617261746f7277035c7d5c50e5ce0200007870770400000004710" + "07e000671007e00067400016271007e000c71007e000571007e000574000164710" + "07e000d78"; TreeMap<String, String> map = new TreeMap<String, String>( String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); map.put("c", "c"); map.put("d", "d"); SortedMap<String, String> subMap = map.subMap("a", false, "c", true); new SerializationTester<SortedMap<String, String>>(subMap, s) { @Override protected void verify(SortedMap<String, String> deserialized) { try { deserialized.put("e", "e"); fail(); } catch (IllegalArgumentException expected) { } } }.test(); } public void testDescendingMapSerialization() { String s = "aced0005737200226a6176612e7574696c2e547265654d61702444657363656e6" + "4696e675375624d61700cab946d1f0f9d0c0200014c001172657665727365436f6" + "d70617261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b7" + "87200216a6176612e7574696c2e547265654d6170244e6176696761626c6553756" + "24d6170e2d0a70e64210e080200075a000966726f6d53746172745a000b6869496" + "e636c75736976655a000b6c6f496e636c75736976655a0005746f456e644c00026" + "8697400124c6a6176612f6c616e672f4f626a6563743b4c00026c6f71007e00034" + "c00016d7400134c6a6176612f7574696c2f547265654d61703b787001010101707" + "0737200116a6176612e7574696c2e547265654d61700cc1f63e2d256ae60300014" + "c000a636f6d70617261746f7271007e000178707372002a6a6176612e6c616e672" + "e537472696e672443617365496e73656e736974697665436f6d70617261746f727" + "7035c7d5c50e5ce02000078707704000000027400016171007e000a74000162710" + "07e000b78737200286a6176612e7574696c2e436f6c6c656374696f6e732452657" + "665727365436f6d70617261746f7232000003fa6c354d510200014c0003636d707" + "1007e0001787071007e0009"; TreeMap<String, String> map = new TreeMap<String, String>( String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); NavigableMap<String, String> descendingMap = map.descendingMap(); new SerializationTester<NavigableMap<String, String>>(descendingMap, s) { @Override protected void verify(NavigableMap<String, String> deserialized) { assertEquals("b", deserialized.navigableKeySet().first()); } }.test(); } public void testJava5SerializationWithComparator() { String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" + "f6d70617261746f723b78707372002a6a6176612e6c616e672e537472696e6724" + "43617365496e73656e736974697665436f6d70617261746f7277035c7d5c50e5c" + "e02000078707704000000027400016171007e00057400016271007e000678"; TreeMap<String, String> map = new TreeMap<String, String>( String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); new SerializationTester<TreeMap<String, String>>(map, s) { @Override protected void verify(TreeMap<String, String> deserialized) { assertEquals(0, deserialized.comparator().compare("X", "x")); } }.test(); } /** * On JDK5, this fails with a NullPointerException after deserialization! */ public void testJava5SubMapSerialization() { String s = "aced0005737200186a6176612e7574696c2e547265654d6170245375624d6170" + "a5818343a213c27f0200055a000966726f6d53746172745a0005746f456e644c0" + "00766726f6d4b65797400124c6a6176612f6c616e672f4f626a6563743b4c0006" + "7468697324307400134c6a6176612f7574696c2f547265654d61703b4c0005746" + "f4b657971007e00017870000074000161737200116a6176612e7574696c2e5472" + "65654d61700cc1f63e2d256ae60300014c000a636f6d70617261746f727400164" + "c6a6176612f7574696c2f436f6d70617261746f723b78707372002a6a6176612e" + "6c616e672e537472696e672443617365496e73656e736974697665436f6d70617" + "261746f7277035c7d5c50e5ce020000787077040000000471007e000471007e00" + "047400016271007e000a7400016371007e000b7400016471007e000c7871007e0" + "00b"; TreeMap<String, String> map = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); map.put("a", "a"); map.put("b", "b"); map.put("c", "c"); map.put("d", "d"); SortedMap<String, String> subMap = map.subMap("a", "c"); new SerializationTester<SortedMap<String, String>>(subMap, s) { @Override protected void verify(SortedMap<String, String> deserialized) { try { deserialized.put("e", "e"); fail(); } catch (IllegalArgumentException expected) { } } }.test(); } }