/* * Copyright Terracotta, 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 org.ehcache.internal.tier; import org.ehcache.core.spi.function.Function; import org.ehcache.core.spi.store.Store; import org.ehcache.core.spi.store.StoreAccessException; import org.ehcache.core.spi.store.tiering.CachingTier; import org.ehcache.spi.test.After; import org.ehcache.spi.test.Before; import org.ehcache.spi.test.LegalSPITesterException; import org.ehcache.spi.test.SPITest; import java.util.HashSet; import java.util.Set; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicBoolean; import static org.hamcrest.Matchers.is; import static org.junit.Assert.assertThat; /** * CachingTierInvalidate */ public class CachingTierInvalidate<K, V> extends CachingTierTester<K, V> { private CachingTier<K, V> tier; public CachingTierInvalidate(CachingTierFactory<K, V> factory) { super(factory); } @After public void tearDown() { if (tier != null) { factory.disposeOf(tier); tier = null; } } @SPITest public void invalidateKey() throws LegalSPITesterException { tier = factory.newCachingTier(); final K key = factory.createKey(42L); final V value = factory.createValue(42L); try { // install value tier.getOrComputeIfAbsent(key, new Function<K, Store.ValueHolder<V>>() { @Override public Store.ValueHolder<V> apply(K k) { return wrap(value); } }); // register invalidation listener final AtomicBoolean invalidated = new AtomicBoolean(false); tier.setInvalidationListener(new CachingTier.InvalidationListener<K, V>() { @Override public void onInvalidation(K key, Store.ValueHolder<V> valueHolder) { assertThat(valueHolder.value(), is(value)); invalidated.set(true); } }); tier.invalidate(key); assertThat(invalidated.get(), is(true)); } catch (StoreAccessException e) { throw new LegalSPITesterException("Warning - exception thrown", e); } } @SPITest public void invalidateTriggeredByEviction() throws LegalSPITesterException { tier = factory.newCachingTier(5L); try { // register invalidation listener final AtomicBoolean invalidated = new AtomicBoolean(false); tier.setInvalidationListener(new CachingTier.InvalidationListener<K, V>() { @Override public void onInvalidation(K key, Store.ValueHolder<V> valueHolder) { invalidated.set(true); } }); // install values for (int i = 0; i < 20; i++) { final K key = factory.createKey(i^3); final V value = factory.createValue(i^5); tier.getOrComputeIfAbsent(key, new Function<K, Store.ValueHolder<V>>() { @Override public Store.ValueHolder<V> apply(K k) { return wrap(value); } }); } assertThat(invalidated.get(), is(true)); } catch (StoreAccessException e) { throw new LegalSPITesterException("Warning - exception thrown", e); } } @SPITest public void invalidateAll() throws LegalSPITesterException { tier = factory.newCachingTier(20L); try { // register invalidation listener final Set<K> invalidatedKeys = new HashSet<K>(); tier.setInvalidationListener(new CachingTier.InvalidationListener<K, V>() { @Override public void onInvalidation(K key, Store.ValueHolder<V> valueHolder) { invalidatedKeys.add(key); } }); // install values final Set<K> generatedKeys = new HashSet<K>(); for (int i = 0; i < 5; i++) { final K key = factory.createKey(i^3); final V value = factory.createValue(i^5); generatedKeys.add(key); tier.getOrComputeIfAbsent(key, new Function<K, Store.ValueHolder<V>>() { @Override public Store.ValueHolder<V> apply(K k) { return wrap(value); } }); } tier.invalidateAll(); assertThat(invalidatedKeys, is(generatedKeys)); } catch (StoreAccessException e) { throw new LegalSPITesterException("Warning - exception thrown", e); } } private Store.ValueHolder<V> wrap(final V value) { return new Store.ValueHolder<V>() { @Override public V value() { return value; } @Override public long creationTime(TimeUnit unit) { return 0L; } @Override public long expirationTime(TimeUnit unit) { return 0L; } @Override public boolean isExpired(long expirationTime, TimeUnit unit) { return false; } @Override public long lastAccessTime(TimeUnit unit) { return 0L; } @Override public float hitRate(long now, TimeUnit unit) { return 0L; } @Override public long hits() { return 0L; } @Override public long getId() { return -1L; } }; } }