com.google.code.ssm.CacheImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2015 Jakub Białek
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
* documentation files (the "Software"), to deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the
* Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
* WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
package com.google.code.ssm;
import java.net.SocketAddress;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.TimeoutException;
import lombok.Getter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.Assert;
import com.google.code.ssm.api.format.SerializationType;
import com.google.code.ssm.providers.CacheClient;
import com.google.code.ssm.providers.CacheException;
import com.google.code.ssm.providers.CacheTranscoder;
import com.google.code.ssm.transcoders.JavaTranscoder;
import com.google.code.ssm.transcoders.JsonTranscoder;
import com.google.code.ssm.transcoders.LongToStringTranscoder;
/**
*
* @author Jakub Białek
* @since 2.0.0
*
*/
class CacheImpl implements Cache {
private static final Logger LOGGER = LoggerFactory.getLogger(CacheImpl.class);
@Getter
private final String name;
@Getter
private final Collection aliases;
@Getter
private final CacheProperties properties;
private final SerializationType defaultSerializationType;
private final JsonTranscoder jsonTranscoder;
private final JavaTranscoder javaTranscoder;
private final LongToStringTranscoder longToStringTranscoder = new LongToStringTranscoder();
private final CacheTranscoder customTranscoder;
private volatile CacheClient cacheClient;
CacheImpl(final String name, final Collection aliases, final CacheClient cacheClient,
final SerializationType defaultSerializationType, final JsonTranscoder jsonTranscoder, final JavaTranscoder javaTranscoder,
final CacheTranscoder customTranscoder, final CacheProperties properties) {
Assert.hasText(name, "'name' must not be null, empty, or blank");
Assert.notNull(aliases, "'aliases' cannot be null");
Assert.notNull(cacheClient, "'cacheClient' cannot be null");
Assert.notNull(defaultSerializationType, "'defaultSerializationType' cannot be null");
Assert.notNull(properties, "'cacheProperties' cannot be null");
validateTranscoder(SerializationType.JSON, jsonTranscoder, "jsonTranscoder");
validateTranscoder(SerializationType.JAVA, javaTranscoder, "javaTranscoder");
validateTranscoder(SerializationType.CUSTOM, customTranscoder, "customTranscoder");
this.name = name;
this.aliases = aliases;
this.cacheClient = cacheClient;
this.defaultSerializationType = defaultSerializationType;
this.jsonTranscoder = jsonTranscoder;
this.javaTranscoder = javaTranscoder;
this.customTranscoder = customTranscoder;
this.properties = properties;
}
@Override
public Collection getAvailableServers() {
return cacheClient.getAvailableServers();
}
@Override
public T get(final String cacheKey, final SerializationType serializationType) throws TimeoutException, CacheException {
switch (getSerializationType(serializationType)) {
case JAVA:
return get(cacheKey, SerializationType.JAVA, javaTranscoder);
case JSON:
return get(cacheKey, SerializationType.JSON, jsonTranscoder);
case PROVIDER:
return get(cacheKey, SerializationType.PROVIDER, null);
case CUSTOM:
return get(cacheKey, SerializationType.CUSTOM, customTranscoder);
default:
throw new IllegalArgumentException(String.format("Serialization type %s is not supported", serializationType));
}
}
@Override
@SuppressWarnings("unchecked")
public void set(final String cacheKey, final int expiration, final Object value, final SerializationType serializationType)
throws TimeoutException, CacheException {
switch (getSerializationType(serializationType)) {
case JAVA:
set(cacheKey, expiration, (T) value, SerializationType.JAVA, javaTranscoder);
break;
case JSON:
set(cacheKey, expiration, (T) value, SerializationType.JSON, jsonTranscoder);
break;
case PROVIDER:
set(cacheKey, expiration, (T) value, SerializationType.PROVIDER, null);
break;
case CUSTOM:
set(cacheKey, expiration, (T) value, SerializationType.CUSTOM, customTranscoder);
break;
default:
throw new IllegalArgumentException(String.format("Serialization type %s is not supported", serializationType));
}
}
@Override
public void setSilently(final String cacheKey, final int expiration, final Object value, final SerializationType serializationType) {
try {
set(cacheKey, expiration, value, serializationType);
} catch (TimeoutException e) {
warn(e, "Cannot set on key %s", cacheKey);
} catch (CacheException e) {
warn(e, "Cannot set on key %s", cacheKey);
}
}
@Override
public boolean add(final String cacheKey, final int expiration, final Object value, final SerializationType serializationType)
throws TimeoutException, CacheException {
final boolean added;
switch (getSerializationType(serializationType)) {
case JAVA:
added = add(cacheKey, expiration, value, SerializationType.JAVA, javaTranscoder);
break;
case JSON:
added = add(cacheKey, expiration, value, SerializationType.JSON, jsonTranscoder);
break;
case PROVIDER:
added = add(cacheKey, expiration, value, SerializationType.PROVIDER, null);
break;
case CUSTOM:
added = add(cacheKey, expiration, value, SerializationType.CUSTOM, customTranscoder);
break;
default:
throw new IllegalArgumentException(String.format("Serialization type %s is not supported", serializationType));
}
return added;
}
@Override
public boolean addSilently(final String cacheKey, final int expiration, final Object value, final SerializationType serializationType) {
try {
return add(cacheKey, expiration, value, serializationType);
} catch (TimeoutException e) {
warn(e, "Cannot add to key %s", cacheKey);
} catch (CacheException e) {
warn(e, "Cannot add to key %s", cacheKey);
}
return false;
}
@Override
public Map getBulk(final Collection keys, final SerializationType serializationType) throws TimeoutException,
CacheException {
switch (getSerializationType(serializationType)) {
case JAVA:
return getBulk(keys, SerializationType.JAVA, javaTranscoder);
case JSON:
return getBulk(keys, SerializationType.JSON, jsonTranscoder);
case PROVIDER:
return getBulk(keys, SerializationType.PROVIDER, null);
case CUSTOM:
return getBulk(keys, SerializationType.CUSTOM, customTranscoder);
default:
throw new IllegalArgumentException(String.format("Serialization type %s is not supported", serializationType));
}
}
@Override
public long decr(final String key, final int by) throws TimeoutException, CacheException {
return cacheClient.decr(key, by);
}
@Override
public boolean delete(final String key) throws TimeoutException, CacheException {
return cacheClient.delete(key);
}
@Override
public void delete(final Collection keys) throws TimeoutException, CacheException {
cacheClient.delete(keys);
}
@Override
public void flush() throws TimeoutException, CacheException {
cacheClient.flush();
}
@Override
public long incr(final String key, final int by, final long def) throws TimeoutException, CacheException {
return cacheClient.incr(key, by, def);
}
@Override
public long incr(final String key, final int by, final long def, final int exp) throws TimeoutException, CacheException {
return cacheClient.incr(key, by, def, exp);
}
@Override
public boolean isEnabled() {
return true;
}
@Override
public Long getCounter(final String cacheKey) throws TimeoutException, CacheException {
return cacheClient.get(cacheKey, longToStringTranscoder);
}
@Override
public void setCounter(final String cacheKey, final int expiration, final long value) throws TimeoutException, CacheException {
cacheClient.set(cacheKey, expiration, value, longToStringTranscoder);
}
@Override
public void shutdown() {
cacheClient.shutdown();
}
@Override
public Object getNativeClient() {
return cacheClient.getNativeClient();
}
void changeCacheClient(final CacheClient newCacheClient) {
if (newCacheClient != null) {
LOGGER.info("Replacing the cache client");
CacheClient oldCacheClient = cacheClient;
cacheClient = newCacheClient;
LOGGER.info("Cache client replaced");
LOGGER.info("Closing old cache client");
oldCacheClient.shutdown();
LOGGER.info("Old cache client closed");
}
}
@SuppressWarnings("unchecked")
private T get(final String cacheKey, final SerializationType serializationType, final CacheTranscoder cacheTranscoder)
throws TimeoutException, CacheException {
if (SerializationType.PROVIDER.equals(serializationType)) {
return (T) cacheClient.get(cacheKey);
}
if (cacheTranscoder == null) {
throw new IllegalArgumentException(String.format("Cannot use %s serialization because dedicated cache transcoder is null!",
serializationType));
}
return (T) cacheClient.get(cacheKey, cacheTranscoder);
}
private void set(final String cacheKey, final int expiration, final T value, final SerializationType serializationType,
final CacheTranscoder cacheTranscoder) throws TimeoutException, CacheException {
if (SerializationType.PROVIDER.equals(serializationType)) {
cacheClient.set(cacheKey, expiration, value);
return;
}
if (cacheTranscoder == null) {
throw new IllegalArgumentException(String.format("Cannot use %s serialization because dedicated cache transcoder is null!",
serializationType));
}
cacheClient.set(cacheKey, expiration, value, cacheTranscoder);
}
private boolean add(final String cacheKey, final int expiration, final Object value, final SerializationType serializationType,
final CacheTranscoder cacheTranscoder) throws TimeoutException, CacheException {
if (SerializationType.PROVIDER.equals(serializationType)) {
return cacheClient.add(cacheKey, expiration, value);
}
if (cacheTranscoder == null) {
throw new IllegalArgumentException(String.format("Cannot use %s serialization because dedicated cache transcoder is null!",
serializationType));
}
return cacheClient.add(cacheKey, expiration, value, cacheTranscoder);
}
private Map getBulk(final Collection keys, final SerializationType serializationType,
final CacheTranscoder cacheTranscoder) throws TimeoutException, CacheException {
if (SerializationType.PROVIDER.equals(serializationType)) {
return cacheClient.getBulk(keys);
}
if (cacheTranscoder == null) {
throw new IllegalArgumentException(String.format("Cannot use %s serialization because dedicated cache transcoder is null!",
serializationType));
}
return cacheClient.getBulk(keys, cacheTranscoder);
}
private SerializationType getSerializationType(final SerializationType serializationType) {
return (serializationType != null) ? serializationType : defaultSerializationType;
}
private void warn(final Exception e, final String format, final Object... args) {
if (LOGGER.isWarnEnabled()) {
LOGGER.warn(String.format(format, args), e);
}
}
private void validateTranscoder(final SerializationType serializationType, final CacheTranscoder cacheTranscoder,
final String transcoderName) {
if (defaultSerializationType == serializationType) {
Assert.notNull(cacheTranscoder,
String.format("'%s' cannot be null if default serialization type is set to %s", transcoderName, serializationType));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy