org.javabits.yar.guice.CacheContainer Maven / Gradle / Ivy
/*
* Copyright 2013 Romain Gilles
*
* 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.javabits.yar.guice;
import com.google.common.base.Function;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.Iterables;
import javax.annotation.Nullable;
import java.lang.reflect.Type;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutionException;
import static com.google.common.base.Throwables.propagate;
import static java.util.Collections.unmodifiableList;
import static org.javabits.yar.guice.CacheContainer.KeyConversionStrategies.NO_TYPE_ERASURE;
import static org.javabits.yar.guice.KeyEvent.newKeyEvent;
/**
* This class is responsible to maintain a multi-map of {@link Type}s
* associated to values. It's implementation is based on Guava {@code LoadingCache}.
* {@link KeyListener}{@literal } can be added. They will be
* triggered on {@code CacheLoader.load(Type)} for addition, and on
* {@link #invalidate(java.lang.reflect.Type)}, {@link #invalidateAll(Iterable)}
* for removal.
*
* @param the type of the multi-values associated to a {@link Type}
*
* @author Romain Gilles
*/
class CacheContainer implements Container {
private final LoadingCache> loadingCache;
private final Collection> keyListeners;
private final Function keyConversionStrategy;
enum KeyConversionStrategies implements Function {
NO_TYPE_ERASURE() {
@Nullable
@Override
public Type apply(@Nullable Type type) {
return type;
}
},
TYPE_ERASURE() {
@Nullable
@Override
public Type apply(@Nullable Type type) {
return Reflections.getRawType(type);
}
}
}
static CacheContainer newConcurrentContainer() {
return newConcurrentContainer(new CopyOnWriteArrayList>(), NO_TYPE_ERASURE);
}
private static CacheContainer newConcurrentContainer(final Collection> typeListeners, Function keyConversionStrategy) {
return new CacheContainer<>(CacheBuilder.newBuilder().build(new CacheLoader>() {
@Override
public List load(Type key) {
for (KeyListener typeListener : typeListeners) {
typeListener.keyAdded(newKeyEvent(key));
}
return new CopyOnWriteArrayList<>();
}
}), typeListeners, keyConversionStrategy);
}
static CacheContainer newNonConcurrentContainer(Function keyConversionStrategy) {
return new CacheContainer<>(keyConversionStrategy);
}
private CacheContainer(Function keyConversionStrategy) {
final CopyOnWriteArrayList> keyListeners = new CopyOnWriteArrayList<>();
this.keyListeners = keyListeners;
this.loadingCache = CacheBuilder.newBuilder().build(new CacheLoader>() {
@Override
public List load(Type key) {
for (KeyListener keyListener : keyListeners) {
keyListener.keyAdded(newKeyEvent(key));
}
return new CopyOnWriteArrayList<>();
}
});
this.keyConversionStrategy = keyConversionStrategy;
}
private CacheContainer(LoadingCache> loadingCache, Collection> keyListeners, Function keyConversionStrategy) {
this.loadingCache = loadingCache;
this.keyListeners = keyListeners;
this.keyConversionStrategy = keyConversionStrategy;
}
@Override
public List getAll(Type key) {
try {
key = keyConversionStrategy.apply(key);
return unmodifiableList(loadingCache.get(key));
} catch (ExecutionException e) {
throw propagate(e);
}
}
@Nullable
@Override
public V getFirst(Type key) {
key = keyConversionStrategy.apply(key);
return Iterables.getFirst(getAll(key), null);
}
@Override
public boolean put(Type key, V value) {
key = keyConversionStrategy.apply(key);
try {
List valueList = loadingCache.get(key);
return valueList.add(value);
} catch (ExecutionException e) {
throw propagate(e);
}
}
@Override
public boolean remove(Type key, V value) {
key = keyConversionStrategy.apply(key);
try {
return loadingCache.get(key).remove(value);
} catch (ExecutionException e) {
throw propagate(e);
}
}
@Override
public void invalidate(Type key) {
key = keyConversionStrategy.apply(key);
loadingCache.invalidate(key);
fireKeyRemoved(key);
}
private void fireKeyRemoved(Type key) {
for (KeyListener keyListener : keyListeners) {
keyListener.keyRemoved(newKeyEvent(key));
}
}
public void invalidateAll(Iterable keys) {
loadingCache.invalidateAll(keys);
for (Type key : keys) {
key = keyConversionStrategy.apply(key);
fireKeyRemoved(key);
}
}
@Override
public Map> asMap() {
return loadingCache.asMap();
}
@Override
public void addKeyListener(KeyListener keyListener) {
keyListeners.add(keyListener);
}
@Override
public void removeKeyListener(KeyListener keyListener) {
keyListeners.add(keyListener);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy