1 /* 2 * Copyright (C) 2011 The Guava Authors 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except 5 * in compliance with the License. You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software distributed under the License 10 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express 11 * or implied. See the License for the specific language governing permissions and limitations under 12 * the License. 13 */ 14 15 package com.google.common.cache; 16 17 import static com.google.common.cache.CacheTesting.checkEmpty; 18 import static com.google.common.cache.TestingCacheLoaders.identityLoader; 19 import static java.util.Arrays.asList; 20 import static java.util.concurrent.TimeUnit.DAYS; 21 import static java.util.concurrent.TimeUnit.SECONDS; 22 23 import com.google.common.base.Function; 24 import com.google.common.cache.CacheBuilderFactory.DurationSpec; 25 import com.google.common.cache.LocalCache.Strength; 26 import com.google.common.collect.ImmutableSet; 27 import com.google.common.collect.Iterables; 28 import com.google.common.collect.Maps; 29 import com.google.common.testing.EqualsTester; 30 31 import junit.framework.TestCase; 32 33 import java.util.Collection; 34 import java.util.Map.Entry; 35 import java.util.Set; 36 import java.util.concurrent.ExecutionException; 37 38 /** 39 * {@link LoadingCache} tests that deal with empty caches. 40 * 41 * @author mike nonemacher 42 */ 43 44 public class EmptyCachesTest extends TestCase { 45 testEmpty()46 public void testEmpty() { 47 for (LoadingCache<Object, Object> cache : caches()) { 48 checkEmpty(cache); 49 } 50 } 51 testInvalidate_empty()52 public void testInvalidate_empty() { 53 for (LoadingCache<Object, Object> cache : caches()) { 54 cache.getUnchecked("a"); 55 cache.getUnchecked("b"); 56 cache.invalidate("a"); 57 cache.invalidate("b"); 58 cache.invalidate(0); 59 checkEmpty(cache); 60 } 61 } 62 testInvalidateAll_empty()63 public void testInvalidateAll_empty() { 64 for (LoadingCache<Object, Object> cache : caches()) { 65 cache.getUnchecked("a"); 66 cache.getUnchecked("b"); 67 cache.getUnchecked("c"); 68 cache.invalidateAll(); 69 checkEmpty(cache); 70 } 71 } 72 testEquals_null()73 public void testEquals_null() { 74 for (LoadingCache<Object, Object> cache : caches()) { 75 assertFalse(cache.equals(null)); 76 } 77 } 78 testEqualsAndHashCode_different()79 public void testEqualsAndHashCode_different() { 80 for (CacheBuilder<Object, Object> builder : cacheFactory().buildAllPermutations()) { 81 // all caches should be different: instance equality 82 new EqualsTester() 83 .addEqualityGroup(builder.build(identityLoader())) 84 .addEqualityGroup(builder.build(identityLoader())) 85 .addEqualityGroup(builder.build(identityLoader())) 86 .testEquals(); 87 } 88 } 89 testGet_null()90 public void testGet_null() throws ExecutionException { 91 for (LoadingCache<Object, Object> cache : caches()) { 92 try { 93 cache.get(null); 94 fail("Expected NullPointerException"); 95 } catch (NullPointerException e) { 96 // expected 97 } 98 checkEmpty(cache); 99 } 100 } 101 testGetUnchecked_null()102 public void testGetUnchecked_null() { 103 for (LoadingCache<Object, Object> cache : caches()) { 104 try { 105 cache.getUnchecked(null); 106 fail("Expected NullPointerException"); 107 } catch (NullPointerException e) { 108 // expected 109 } 110 checkEmpty(cache); 111 } 112 } 113 114 /* ---------------- Key Set -------------- */ 115 testKeySet_nullToArray()116 public void testKeySet_nullToArray() { 117 for (LoadingCache<Object, Object> cache : caches()) { 118 Set<Object> keys = cache.asMap().keySet(); 119 try { 120 keys.toArray(null); 121 fail(); 122 } catch (NullPointerException e) { 123 // expected 124 } 125 checkEmpty(cache); 126 } 127 } 128 testKeySet_addNotSupported()129 public void testKeySet_addNotSupported() { 130 for (LoadingCache<Object, Object> cache : caches()) { 131 try { 132 cache.asMap().keySet().add(1); 133 fail(); 134 } catch (UnsupportedOperationException e) { 135 // expected 136 } 137 138 try { 139 cache.asMap().keySet().addAll(asList(1, 2)); 140 fail(); 141 } catch (UnsupportedOperationException e) { 142 // expected 143 } 144 } 145 } 146 testKeySet_clear()147 public void testKeySet_clear() { 148 for (LoadingCache<Object, Object> cache : caches()) { 149 warmUp(cache, 0, 100); 150 151 Set<Object> keys = cache.asMap().keySet(); 152 keys.clear(); 153 checkEmpty(keys); 154 checkEmpty(cache); 155 } 156 } 157 testKeySet_empty_remove()158 public void testKeySet_empty_remove() { 159 for (LoadingCache<Object, Object> cache : caches()) { 160 Set<Object> keys = cache.asMap().keySet(); 161 assertFalse(keys.remove(null)); 162 assertFalse(keys.remove(6)); 163 assertFalse(keys.remove(-6)); 164 assertFalse(keys.removeAll(asList(null, 0, 15, 1500))); 165 assertFalse(keys.retainAll(asList(null, 0, 15, 1500))); 166 checkEmpty(keys); 167 checkEmpty(cache); 168 } 169 } 170 testKeySet_remove()171 public void testKeySet_remove() { 172 for (LoadingCache<Object, Object> cache : caches()) { 173 cache.getUnchecked(1); 174 cache.getUnchecked(2); 175 176 Set<Object> keys = cache.asMap().keySet(); 177 // We don't know whether these are still in the cache, so we can't assert on the return 178 // values of these removes, but the cache should be empty after the removes, regardless. 179 keys.remove(1); 180 keys.remove(2); 181 assertFalse(keys.remove(null)); 182 assertFalse(keys.remove(6)); 183 assertFalse(keys.remove(-6)); 184 assertFalse(keys.removeAll(asList(null, 0, 15, 1500))); 185 assertFalse(keys.retainAll(asList(null, 0, 15, 1500))); 186 checkEmpty(keys); 187 checkEmpty(cache); 188 } 189 } 190 191 /* ---------------- Values -------------- */ 192 testValues_nullToArray()193 public void testValues_nullToArray() { 194 for (LoadingCache<Object, Object> cache : caches()) { 195 Collection<Object> values = cache.asMap().values(); 196 try { 197 values.toArray(null); 198 fail(); 199 } catch (NullPointerException e) { 200 // expected 201 } 202 checkEmpty(cache); 203 } 204 } 205 testValues_addNotSupported()206 public void testValues_addNotSupported() { 207 for (LoadingCache<Object, Object> cache : caches()) { 208 try { 209 cache.asMap().values().add(1); 210 fail(); 211 } catch (UnsupportedOperationException e) { 212 // expected 213 } 214 215 try { 216 cache.asMap().values().addAll(asList(1, 2)); 217 fail(); 218 } catch (UnsupportedOperationException e) { 219 // expected 220 } 221 } 222 } 223 testValues_clear()224 public void testValues_clear() { 225 for (LoadingCache<Object, Object> cache : caches()) { 226 warmUp(cache, 0, 100); 227 228 Collection<Object> values = cache.asMap().values(); 229 values.clear(); 230 checkEmpty(values); 231 checkEmpty(cache); 232 } 233 } 234 testValues_empty_remove()235 public void testValues_empty_remove() { 236 for (LoadingCache<Object, Object> cache : caches()) { 237 Collection<Object> values = cache.asMap().values(); 238 assertFalse(values.remove(null)); 239 assertFalse(values.remove(6)); 240 assertFalse(values.remove(-6)); 241 assertFalse(values.removeAll(asList(null, 0, 15, 1500))); 242 assertFalse(values.retainAll(asList(null, 0, 15, 1500))); 243 checkEmpty(values); 244 checkEmpty(cache); 245 } 246 } 247 testValues_remove()248 public void testValues_remove() { 249 for (LoadingCache<Object, Object> cache : caches()) { 250 cache.getUnchecked(1); 251 cache.getUnchecked(2); 252 253 Collection<Object> values = cache.asMap().keySet(); 254 // We don't know whether these are still in the cache, so we can't assert on the return 255 // values of these removes, but the cache should be empty after the removes, regardless. 256 values.remove(1); 257 values.remove(2); 258 assertFalse(values.remove(null)); 259 assertFalse(values.remove(6)); 260 assertFalse(values.remove(-6)); 261 assertFalse(values.removeAll(asList(null, 0, 15, 1500))); 262 assertFalse(values.retainAll(asList(null, 0, 15, 1500))); 263 checkEmpty(values); 264 checkEmpty(cache); 265 } 266 } 267 268 /* ---------------- Entry Set -------------- */ 269 testEntrySet_nullToArray()270 public void testEntrySet_nullToArray() { 271 for (LoadingCache<Object, Object> cache : caches()) { 272 Set<Entry<Object, Object>> entries = cache.asMap().entrySet(); 273 try { 274 entries.toArray(null); 275 fail(); 276 } catch (NullPointerException e) { 277 // expected 278 } 279 checkEmpty(cache); 280 } 281 } 282 testEntrySet_addNotSupported()283 public void testEntrySet_addNotSupported() { 284 for (LoadingCache<Object, Object> cache : caches()) { 285 try { 286 cache.asMap().entrySet().add(entryOf(1, 1)); 287 fail(); 288 } catch (UnsupportedOperationException e) { 289 // expected 290 } 291 292 try { 293 cache.asMap().values().addAll(asList(entryOf(1, 1), entryOf(2, 2))); 294 fail(); 295 } catch (UnsupportedOperationException e) { 296 // expected 297 } 298 } 299 } 300 testEntrySet_clear()301 public void testEntrySet_clear() { 302 for (LoadingCache<Object, Object> cache : caches()) { 303 warmUp(cache, 0, 100); 304 305 Set<Entry<Object, Object>> entrySet = cache.asMap().entrySet(); 306 entrySet.clear(); 307 checkEmpty(entrySet); 308 checkEmpty(cache); 309 } 310 } 311 testEntrySet_empty_remove()312 public void testEntrySet_empty_remove() { 313 for (LoadingCache<Object, Object> cache : caches()) { 314 Set<Entry<Object, Object>> entrySet = cache.asMap().entrySet(); 315 assertFalse(entrySet.remove(null)); 316 assertFalse(entrySet.remove(entryOf(6, 6))); 317 assertFalse(entrySet.remove(entryOf(-6, -6))); 318 assertFalse(entrySet.removeAll(asList(null, entryOf(0, 0), entryOf(15, 15)))); 319 assertFalse(entrySet.retainAll(asList(null, entryOf(0, 0), entryOf(15, 15)))); 320 checkEmpty(entrySet); 321 checkEmpty(cache); 322 } 323 } 324 testEntrySet_remove()325 public void testEntrySet_remove() { 326 for (LoadingCache<Object, Object> cache : caches()) { 327 cache.getUnchecked(1); 328 cache.getUnchecked(2); 329 330 Set<Entry<Object, Object>> entrySet = cache.asMap().entrySet(); 331 // We don't know whether these are still in the cache, so we can't assert on the return 332 // values of these removes, but the cache should be empty after the removes, regardless. 333 entrySet.remove(entryOf(1, 1)); 334 entrySet.remove(entryOf(2, 2)); 335 assertFalse(entrySet.remove(null)); 336 assertFalse(entrySet.remove(entryOf(1, 1))); 337 assertFalse(entrySet.remove(entryOf(6, 6))); 338 assertFalse(entrySet.removeAll(asList(null, entryOf(1, 1), entryOf(15, 15)))); 339 assertFalse(entrySet.retainAll(asList(null, entryOf(1, 1), entryOf(15, 15)))); 340 checkEmpty(entrySet); 341 checkEmpty(cache); 342 } 343 } 344 345 /* ---------------- Local utilities -------------- */ 346 347 /** 348 * Most of the tests in this class run against every one of these caches. 349 */ caches()350 private Iterable<LoadingCache<Object, Object>> caches() { 351 // lots of different ways to configure a LoadingCache 352 CacheBuilderFactory factory = cacheFactory(); 353 return Iterables.transform(factory.buildAllPermutations(), 354 new Function<CacheBuilder<Object, Object>, LoadingCache<Object, Object>>() { 355 @Override public LoadingCache<Object, Object> apply( 356 CacheBuilder<Object, Object> builder) { 357 return builder.build(identityLoader()); 358 } 359 }); 360 } 361 362 private CacheBuilderFactory cacheFactory() { 363 return new CacheBuilderFactory() 364 .withKeyStrengths(ImmutableSet.of(Strength.STRONG, Strength.WEAK)) 365 .withValueStrengths(ImmutableSet.copyOf(Strength.values())) 366 .withConcurrencyLevels(ImmutableSet.of(1, 4, 16, 64)) 367 .withMaximumSizes(ImmutableSet.of(0, 1, 10, 100, 1000)) 368 .withInitialCapacities(ImmutableSet.of(0, 1, 10, 100, 1000)) 369 .withExpireAfterWrites(ImmutableSet.of( 370 DurationSpec.of(0, SECONDS), 371 DurationSpec.of(1, SECONDS), 372 DurationSpec.of(1, DAYS))) 373 .withExpireAfterAccesses(ImmutableSet.of( 374 DurationSpec.of(0, SECONDS), 375 DurationSpec.of(1, SECONDS), 376 DurationSpec.of(1, DAYS))) 377 .withRefreshes(ImmutableSet.of( 378 DurationSpec.of(1, SECONDS), 379 DurationSpec.of(1, DAYS))); 380 } 381 382 private void warmUp(LoadingCache<Object, Object> cache, int minimum, int maximum) { 383 for (int i = minimum; i < maximum; i++) { 384 cache.getUnchecked(i); 385 } 386 } 387 388 private Entry<Object, Object> entryOf(Object key, Object value) { 389 return Maps.immutableEntry(key, value); 390 } 391 } 392