1 /* 2 * Copyright (C) 2011 The Guava Authors 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.google.common.collect; 18 19 import com.google.common.annotations.GwtCompatible; 20 import com.google.common.base.Function; 21 import com.google.common.base.Functions; 22 import com.google.common.collect.testing.MapInterfaceTest; 23 24 import java.util.Collection; 25 import java.util.Iterator; 26 import java.util.Map; 27 import java.util.Set; 28 29 import javax.annotation.Nullable; 30 31 /** 32 * Tests for {@link Maps#transformValues} when the backing map's views 33 * have iterators that don't support {@code remove()}. 34 * 35 * @author Jared Levy 36 */ 37 @GwtCompatible 38 public class MapsTransformValuesUnmodifiableIteratorTest extends MapInterfaceTest<String, String> { 39 // TODO(jlevy): Move shared code of this class and MapsTransformValuesTest 40 // to a superclass. 41 MapsTransformValuesUnmodifiableIteratorTest()42 public MapsTransformValuesUnmodifiableIteratorTest() { 43 super(true, true, false /*supportsPut*/, true, true, false); 44 } 45 46 private static class UnmodifiableIteratorMap<K, V> extends ForwardingMap<K, V> { 47 final Map<K, V> delegate; 48 UnmodifiableIteratorMap(Map<K, V> delegate)49 UnmodifiableIteratorMap(Map<K, V> delegate) { 50 this.delegate = delegate; 51 } 52 delegate()53 @Override protected Map<K, V> delegate() { 54 return delegate; 55 } 56 keySet()57 @Override public Set<K> keySet() { 58 return new ForwardingSet<K>() { 59 @Override protected Set<K> delegate() { 60 return delegate.keySet(); 61 } 62 @Override public Iterator<K> iterator() { 63 return Iterators.unmodifiableIterator(delegate.keySet().iterator()); 64 } 65 @Override public boolean removeAll(Collection<?> c) { 66 return delegate.keySet().removeAll(c); 67 } 68 @Override public boolean retainAll(Collection<?> c) { 69 return delegate.keySet().retainAll(c); 70 } 71 }; 72 } 73 values()74 @Override public Collection<V> values() { 75 return new ForwardingCollection<V>() { 76 @Override protected Collection<V> delegate() { 77 return delegate.values(); 78 } 79 @Override public Iterator<V> iterator() { 80 return Iterators.unmodifiableIterator(delegate.values().iterator()); 81 } 82 @Override public boolean removeAll(Collection<?> c) { 83 return delegate.values().removeAll(c); 84 } 85 @Override public boolean retainAll(Collection<?> c) { 86 return delegate.values().retainAll(c); 87 } 88 }; 89 } 90 91 @Override public Set<Entry<K, V>> entrySet() { 92 return new ForwardingSet<Entry<K, V>>() { 93 @Override protected Set<Entry<K, V>> delegate() { 94 return delegate.entrySet(); 95 } 96 @Override public Iterator<Entry<K, V>> iterator() { 97 return Iterators.unmodifiableIterator(delegate.entrySet().iterator()); 98 } 99 @Override public boolean removeAll(Collection<?> c) { 100 return delegate.entrySet().removeAll(c); 101 } 102 @Override public boolean retainAll(Collection<?> c) { 103 return delegate.entrySet().retainAll(c); 104 } 105 }; 106 } 107 } 108 109 @Override protected Map<String, String> makeEmptyMap() { 110 Map<String, Integer> underlying = Maps.newHashMap(); 111 return Maps.transformValues( 112 new UnmodifiableIteratorMap<String, Integer>(underlying), Functions.toStringFunction()); 113 } 114 115 @Override protected Map<String, String> makePopulatedMap() { 116 Map<String, Integer> underlying = Maps.newHashMap(); 117 underlying.put("a", 1); 118 underlying.put("b", 2); 119 underlying.put("c", 3); 120 return Maps.transformValues( 121 new UnmodifiableIteratorMap<String, Integer>(underlying), Functions.toStringFunction()); 122 } 123 124 @Override protected String getKeyNotInPopulatedMap() 125 throws UnsupportedOperationException { 126 return "z"; 127 } 128 129 @Override protected String getValueNotInPopulatedMap() 130 throws UnsupportedOperationException { 131 return "26"; 132 } 133 134 /** Helper assertion comparing two maps */ 135 private void assertMapsEqual(Map<?, ?> expected, Map<?, ?> map) { 136 assertEquals(expected, map); 137 assertEquals(expected.hashCode(), map.hashCode()); 138 assertEquals(expected.entrySet(), map.entrySet()); 139 140 // Assert that expectedValues > mapValues and that 141 // mapValues > expectedValues; i.e. that expectedValues == mapValues. 142 Collection<?> expectedValues = expected.values(); 143 Collection<?> mapValues = map.values(); 144 assertEquals(expectedValues.size(), mapValues.size()); 145 assertTrue(expectedValues.containsAll(mapValues)); 146 assertTrue(mapValues.containsAll(expectedValues)); 147 } 148 149 public void testTransformEmptyMapEquality() { 150 Map<String, String> map = Maps.transformValues( 151 ImmutableMap.<String, Integer>of(), Functions.toStringFunction()); 152 assertMapsEqual(Maps.newHashMap(), map); 153 } 154 155 public void testTransformSingletonMapEquality() { 156 Map<String, String> map = Maps.transformValues( 157 ImmutableMap.of("a", 1), Functions.toStringFunction()); 158 Map<String, String> expected = ImmutableMap.of("a", "1"); 159 assertMapsEqual(expected, map); 160 assertEquals(expected.get("a"), map.get("a")); 161 } 162 163 public void testTransformIdentityFunctionEquality() { 164 Map<String, Integer> underlying = ImmutableMap.of("a", 1); 165 Map<String, Integer> map = Maps.transformValues( 166 underlying, Functions.<Integer>identity()); 167 assertMapsEqual(underlying, map); 168 } 169 170 public void testTransformPutEntryIsUnsupported() { 171 Map<String, String> map = Maps.transformValues( 172 ImmutableMap.of("a", 1), Functions.toStringFunction()); 173 try { 174 map.put("b", "2"); 175 fail(); 176 } catch (UnsupportedOperationException expected) { 177 } 178 179 try { 180 map.putAll(ImmutableMap.of("b", "2")); 181 fail(); 182 } catch (UnsupportedOperationException expected) { 183 } 184 185 try { 186 map.entrySet().iterator().next().setValue("one"); 187 fail(); 188 } catch (UnsupportedOperationException expected) { 189 } 190 } 191 192 public void testTransformRemoveEntry() { 193 Map<String, Integer> underlying = Maps.newHashMap(); 194 underlying.put("a", 1); 195 Map<String, String> map 196 = Maps.transformValues(underlying, Functions.toStringFunction()); 197 assertEquals("1", map.remove("a")); 198 assertNull(map.remove("b")); 199 } 200 201 public void testTransformEqualityOfMapsWithNullValues() { 202 Map<String, String> underlying = Maps.newHashMap(); 203 underlying.put("a", null); 204 underlying.put("b", ""); 205 206 Map<String, Boolean> map = Maps.transformValues(underlying, 207 new Function<String, Boolean>() { 208 @Override 209 public Boolean apply(@Nullable String from) { 210 return from == null; 211 } 212 } 213 ); 214 Map<String, Boolean> expected = ImmutableMap.of("a", true, "b", false); 215 assertMapsEqual(expected, map); 216 assertEquals(expected.get("a"), map.get("a")); 217 assertEquals(expected.containsKey("a"), map.containsKey("a")); 218 assertEquals(expected.get("b"), map.get("b")); 219 assertEquals(expected.containsKey("b"), map.containsKey("b")); 220 assertEquals(expected.get("c"), map.get("c")); 221 assertEquals(expected.containsKey("c"), map.containsKey("c")); 222 } 223 224 public void testTransformReflectsUnderlyingMap() { 225 Map<String, Integer> underlying = Maps.newHashMap(); 226 underlying.put("a", 1); 227 underlying.put("b", 2); 228 underlying.put("c", 3); 229 Map<String, String> map 230 = Maps.transformValues(underlying, Functions.toStringFunction()); 231 assertEquals(underlying.size(), map.size()); 232 233 underlying.put("d", 4); 234 assertEquals(underlying.size(), map.size()); 235 assertEquals("4", map.get("d")); 236 237 underlying.remove("c"); 238 assertEquals(underlying.size(), map.size()); 239 assertFalse(map.containsKey("c")); 240 241 underlying.clear(); 242 assertEquals(underlying.size(), map.size()); 243 } 244 245 public void testTransformChangesAreReflectedInUnderlyingMap() { 246 Map<String, Integer> underlying = Maps.newLinkedHashMap(); 247 underlying.put("a", 1); 248 underlying.put("b", 2); 249 underlying.put("c", 3); 250 underlying.put("d", 4); 251 underlying.put("e", 5); 252 underlying.put("f", 6); 253 underlying.put("g", 7); 254 Map<String, String> map 255 = Maps.transformValues(underlying, Functions.toStringFunction()); 256 257 map.remove("a"); 258 assertFalse(underlying.containsKey("a")); 259 260 Set<String> keys = map.keySet(); 261 keys.remove("b"); 262 assertFalse(underlying.containsKey("b")); 263 264 Iterator<String> keyIterator = keys.iterator(); 265 keyIterator.next(); 266 keyIterator.remove(); 267 assertFalse(underlying.containsKey("c")); 268 269 Collection<String> values = map.values(); 270 values.remove("4"); 271 assertFalse(underlying.containsKey("d")); 272 273 Iterator<String> valueIterator = values.iterator(); 274 valueIterator.next(); 275 valueIterator.remove(); 276 assertFalse(underlying.containsKey("e")); 277 278 Set<Map.Entry<String, String>> entries = map.entrySet(); 279 Map.Entry<String, String> firstEntry = entries.iterator().next(); 280 entries.remove(firstEntry); 281 assertFalse(underlying.containsKey("f")); 282 283 Iterator<Map.Entry<String, String>> entryIterator = entries.iterator(); 284 entryIterator.next(); 285 entryIterator.remove(); 286 assertFalse(underlying.containsKey("g")); 287 288 assertTrue(underlying.isEmpty()); 289 assertTrue(map.isEmpty()); 290 assertTrue(keys.isEmpty()); 291 assertTrue(values.isEmpty()); 292 assertTrue(entries.isEmpty()); 293 } 294 295 public void testTransformEquals() { 296 Map<String, Integer> underlying = ImmutableMap.of("a", 0, "b", 1, "c", 2); 297 Map<String, Integer> expected 298 = Maps.transformValues(underlying, Functions.<Integer>identity()); 299 300 assertMapsEqual(expected, expected); 301 302 Map<String, Integer> equalToUnderlying = Maps.newTreeMap(); 303 equalToUnderlying.putAll(underlying); 304 Map<String, Integer> map = Maps.transformValues( 305 equalToUnderlying, Functions.<Integer>identity()); 306 assertMapsEqual(expected, map); 307 308 map = Maps.transformValues(ImmutableMap.of("a", 1, "b", 2, "c", 3), 309 new Function<Integer, Integer>() { 310 @Override 311 public Integer apply(Integer from) { 312 return from - 1; 313 } 314 } 315 ); 316 assertMapsEqual(expected, map); 317 } 318 319 public void testTransformEntrySetContains() { 320 Map<String, Boolean> underlying = Maps.newHashMap(); 321 underlying.put("a", null); 322 underlying.put("b", true); 323 underlying.put(null, true); 324 325 Map<String, Boolean> map = Maps.transformValues( 326 underlying, new Function<Boolean, Boolean>() { 327 @Override 328 public Boolean apply(@Nullable Boolean from) { 329 return (from == null) ? true : null; 330 } 331 } 332 ); 333 334 Set<Map.Entry<String, Boolean>> entries = map.entrySet(); 335 assertTrue(entries.contains(Maps.immutableEntry("a", true))); 336 assertTrue(entries.contains(Maps.immutableEntry("b", (Boolean) null))); 337 assertTrue(entries.contains( 338 Maps.immutableEntry((String) null, (Boolean) null))); 339 340 assertFalse(entries.contains(Maps.immutableEntry("c", (Boolean) null))); 341 assertFalse(entries.contains(Maps.immutableEntry((String) null, true))); 342 } 343 344 @Override public void testKeySetRemoveAllNullFromEmpty() { 345 try { 346 super.testKeySetRemoveAllNullFromEmpty(); 347 } catch (RuntimeException tolerated) { 348 // GWT's HashMap.keySet().removeAll(null) doesn't throws NPE. 349 } 350 } 351 352 @Override public void testEntrySetRemoveAllNullFromEmpty() { 353 try { 354 super.testEntrySetRemoveAllNullFromEmpty(); 355 } catch (RuntimeException tolerated) { 356 // GWT's HashMap.entrySet().removeAll(null) doesn't throws NPE. 357 } 358 } 359 } 360