org.ff4j.cache.FF4jCacheManagerRedisLettuce Maven / Gradle / Ivy
package org.ff4j.cache;
/*-
* #%L
* ff4j-store-redis
* %%
* Copyright (C) 2013 - 2023 FF4J
* %%
* 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.
* #L%
*/
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import io.lettuce.core.api.sync.RedisKeyCommands;
import org.ff4j.core.Feature;
import org.ff4j.property.Property;
import org.ff4j.redis.RedisKeysBuilder;
import org.ff4j.utils.Util;
import org.ff4j.utils.json.FeatureJsonParser;
import org.ff4j.utils.json.PropertyJsonParser;
import io.lettuce.core.KeyScanCursor;
import io.lettuce.core.RedisClient;
import io.lettuce.core.ScanArgs;
import io.lettuce.core.api.sync.RedisCommands;
import io.lettuce.core.cluster.RedisClusterClient;
import io.lettuce.core.cluster.api.sync.RedisAdvancedClusterCommands;
/**
* Implementation of ditributed cache to limit overhead, with REDIS (JEDIS).
*
* @author Cedrick LUNVEN
*/
public class FF4jCacheManagerRedisLettuce implements FF4JCacheManager {
/** time to live for cache on top of store. */
protected int timeToLive = 900000000;
/** Lettuce client. */
private RedisCommands redisCommands;
/** Support the cluster based redis deployment. */
private RedisAdvancedClusterCommands redisCommandsCluster;
/** Default key builder. */
private RedisKeysBuilder keyBuilder = new RedisKeysBuilder();
/**
* Public void.
*/
public FF4jCacheManagerRedisLettuce(RedisClient redisClient) {
this(redisClient, new RedisKeysBuilder());
}
public FF4jCacheManagerRedisLettuce(RedisClient redisClient, RedisKeysBuilder keyBuilder) {
this.redisCommands = redisClient.connect().sync();
this.keyBuilder = keyBuilder;
}
public FF4jCacheManagerRedisLettuce(RedisClusterClient redisClusterClient) {
this(redisClusterClient, new RedisKeysBuilder());
}
public FF4jCacheManagerRedisLettuce(RedisClusterClient redisClusterClient, RedisKeysBuilder keyBuilder) {
this.redisCommandsCluster = redisClusterClient.connect().sync();
this.keyBuilder = keyBuilder;
}
/** {@inheritDoc} */
@Override
public Set listCachedFeatureNames() {
return getKeys(keyBuilder.getKeyFeature("*"));
}
/** {@inheritDoc} */
@Override
public String getCacheProviderName() {
return "REDIS";
}
/** {@inheritDoc} */
@Override
public void clearFeatures() {
try {
Set matchingKeys = getKeys(keyBuilder.getKeyFeature("*"));
if (!matchingKeys.isEmpty()) {
if (null != redisCommands) {
redisCommands.del(matchingKeys.toArray(new String[matchingKeys.size()]));
} else {
redisCommandsCluster.del(matchingKeys.toArray(new String[matchingKeys.size()]));
}
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public void clearProperties() {
try {
Set matchingKeys = getKeys(keyBuilder.getKeyProperty("*"));
if (!matchingKeys.isEmpty()) {
if (null != redisCommands) {
redisCommands.del(matchingKeys.toArray(new String[matchingKeys.size()]));
} else {
redisCommandsCluster.del(matchingKeys.toArray(new String[matchingKeys.size()]));
}
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public void evictFeature(String uid) {
Util.assertParamHasLength(uid, " feature identifier");
try {
if (null != redisCommands) {
redisCommands.del(keyBuilder.getKeyFeature(uid));
} else {
redisCommandsCluster.del(keyBuilder.getKeyFeature(uid));
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public void evictProperty(String propertyName) {
Util.assertParamHasLength(propertyName, " property name");
try {
if (propertyName == null || propertyName.isEmpty()) {
throw new IllegalArgumentException("PropertyName cannot be null nor empty");
}
if (null != redisCommands) {
redisCommands.del(keyBuilder.getKeyProperty(propertyName));
} else {
redisCommandsCluster.del(keyBuilder.getKeyProperty(propertyName));
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public void putFeature(Feature fp) {
Util.assertNotNull(fp);
try {
if (null != redisCommands) {
redisCommands.set(keyBuilder.getKeyFeature(fp.getUid()), fp.toJson());
redisCommands.expire(keyBuilder.getKeyFeature(fp.getUid()), getTimeToLive());
} else {
redisCommandsCluster.set(keyBuilder.getKeyFeature(fp.getUid()), fp.toJson());
redisCommandsCluster.expire(keyBuilder.getKeyFeature(fp.getUid()), getTimeToLive());
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public void putProperty(Property> property) {
Util.assertNotNull(property);
try {
if (null != redisCommands) {
redisCommands.set(keyBuilder.getKeyProperty(property.getName()), property.toJson());
redisCommands.expire(keyBuilder.getKeyProperty(property.getName()), getTimeToLive());
} else {
redisCommandsCluster.set(keyBuilder.getKeyProperty(property.getName()), property.toJson());
redisCommandsCluster.expire(keyBuilder.getKeyProperty(property.getName()), getTimeToLive());
}
} catch(RuntimeException re) {
onException(re);
}
}
/** {@inheritDoc} */
@Override
public Feature getFeature(String uid) {
Util.assertParamHasLength(uid, "feature uid");
String value = "";
try {
if (null != redisCommands) {
value = redisCommands.get(keyBuilder.getKeyFeature(uid));
} else {
value = redisCommandsCluster.get(keyBuilder.getKeyFeature(uid));
}
if (value != null) {
return FeatureJsonParser.parseFeature(value);
}
} catch(RuntimeException re) {
onException(re);
}
return null;
}
/** {@inheritDoc} */
@Override
public Property> getProperty(String propertyName) {
Util.assertParamHasLength(propertyName, "property name");
String value = "";
try {
if (null != redisCommands) {
value = redisCommands.get(keyBuilder.getKeyProperty(propertyName));
} else {
value = redisCommandsCluster.get(keyBuilder.getKeyProperty(propertyName));
}
if (value != null) {
return PropertyJsonParser.parseProperty(value);
}
} catch(RuntimeException re) {
onException(re);
}
return null;
}
/** {@inheritDoc} */
public Set listCachedPropertyNames() {
List keys = new ArrayList<>();
try {
if (null != redisCommands) {
keys = redisCommands.keys(keyBuilder.getKeyProperty("*"));
} else {
keys = redisCommandsCluster.keys(keyBuilder.getKeyProperty("*"));
}
} catch(RuntimeException re) {
onException(re);
}
return new HashSet(keys);
}
/** {@inheritDoc} */
public Object getFeatureNativeCache() {
return (null != redisCommands) ? redisCommands : redisCommandsCluster;
}
/** {@inheritDoc} */
public Object getPropertyNativeCache() {
return (null != redisCommands) ? redisCommands : redisCommandsCluster;
}
private Set getKeys(String pattern) {
try {
RedisKeyCommands redisKeyCommands = (null != redisCommands ) ? redisCommands : redisCommandsCluster;
KeyScanCursor ksc = redisKeyCommands.scan(new ScanArgs().match(pattern));
Set matchingKeys = new HashSet<>(ksc.getKeys());
while (!ksc.isFinished()) {
ksc = redisKeyCommands.scan(ksc);
matchingKeys.addAll(ksc.getKeys());
}
return matchingKeys;
} catch (RuntimeException re) {
onException(re);
}
return null;
}
/**
* Getter accessor for attribute 'timeToLive'.
*
* @return
* current value of 'timeToLive'
*/
public int getTimeToLive() {
return timeToLive;
}
/**
* Setter accessor for attribute 'timeToLive'.
*
* @param timeToLive
* new value for 'timeToLive '
*/
public void setTimeToLive(int timeToLive) {
this.timeToLive = timeToLive;
}
/**
* Getter accessor for attribute 'keyBuilder'.
*
* @return
* current value of 'keyBuilder'
*/
public RedisKeysBuilder getKeyBuilder() {
return keyBuilder;
}
}