All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.edgar615.util.spring.cache.CacheWithRedisAutoConfiguration Maven / Gradle / Ivy

There is a newer version: 0.2.11
Show newest version
package com.github.edgar615.util.spring.cache;

import com.github.benmanes.caffeine.cache.Caffeine;
import java.time.Duration;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.cache.Cache;
import org.springframework.cache.CacheManager;
import org.springframework.cache.caffeine.CaffeineCache;
import org.springframework.cache.support.CompositeCacheManager;
import org.springframework.cache.support.SimpleCacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.cache.RedisCacheWriter;
import org.springframework.data.redis.connection.RedisConnectionFactory;

/**
 * Created by Administrator on 2017/10/7.
 *
 * @ConditionalOnBean:当容器里有指定的Bean 的条件下。
 * @ConditionalOnClass:当类路径下有指定的类的条件下。
 * @ConditionalOnExpression:基于SpEL 表达式作为判断条件。
 * @ConditionalOnJava:基于JVM 版本作为判断条件。
 * @ConditionalOnJndi:在JNDI 存在的条件下查找指定的位置。
 * @ConditionalOnMissingBean:当容器里没有指定Bean 的情况下。
 * @ConditionalOnMissingClass:当类路径下没有指定的类的条件下。
 * @ConditionalOnNotWebApplication:当前项目不是Web 项目的条件下。
 * @ConditionalOnProperty:指定的属性是否有指定的值。
 * @ConditionalOnResource:类路径是否有指定的值。
 * @ConditionalOnSingleCandidate:当指定Bean 在容器中只有一个,或者虽然有多个但是指定首选的Bean。
 * @ConditionalOnWebApplication:当前项目是Web 项目的条件下。
 */
@Configuration
@EnableConfigurationProperties({CacheProperties.class})
public class CacheWithRedisAutoConfiguration {

  @Bean
  @ConditionalOnProperty(name = "caching.enabled", matchIfMissing = false, havingValue = "true")
  @ConditionalOnMissingBean(CacheManager.class)
//  @ConditionalOnBean(RedisConnectionFactory.class)
  public CacheManager cacheManager(RedisConnectionFactory redisCacheConfiguration,
      CacheProperties properties) {
    List caches = new ArrayList<>();
    List caffeine = properties.getCaffeine().getSpec().entrySet().stream()
        .map(spec -> new CaffeineCache(spec.getKey(), Caffeine.from(spec.getValue()).build()))
        .collect(Collectors.toList());
    caches.addAll(caffeine);
    SimpleCacheManager caffeineManager = new SimpleCacheManager();
    caffeineManager.setCaches(caches);
    caffeineManager.initializeCaches();
    List cacheManagers = new ArrayList<>();
    cacheManagers.add(caffeineManager);

    //定义 redis缓存
    RedisCacheManager redisCacheManager = create(redisCacheConfiguration, properties);
    cacheManagers.add(redisCacheManager);

    //要先定义好缓存,然后才能定义二级缓存
    List l2Cache = properties.getL2Cache().getSpec().entrySet().stream()
        .map(spec -> new L2Cache(spec.getKey(), findByName(caches, spec.getValue().getL1()),
            findByName(caches, spec.getValue().getL2()), true))
        .collect(Collectors.toList());
    if (!l2Cache.isEmpty()) {
      SimpleCacheManager l2CacheManager = new SimpleCacheManager();
      l2CacheManager.setCaches(l2Cache);
      l2CacheManager.initializeCaches();
      cacheManagers.add(l2CacheManager);
    }
    CompositeCacheManager compositeCacheManager = new CompositeCacheManager();
    compositeCacheManager.setCacheManagers(cacheManagers);
    compositeCacheManager.afterPropertiesSet();
    compositeCacheManager.setFallbackToNoOpCache(true);
    return compositeCacheManager;
  }

  @Bean
  @ConditionalOnBean(CacheManager.class)
  public CacheEvictEventListener cacheEvictEventListener(CacheManager cacheManager) {
    return new CacheEvictEventListener(cacheManager);
  }

  private Cache findByName(List caches, String name) {
    Optional optional = caches.stream().filter(c -> name.equals(c.getName()))
        .findFirst();
    if (optional.isPresent()) {
      return optional.get();
    }
    throw new NullPointerException("cache:" + name);
  }

  public RedisCacheManager create(RedisConnectionFactory redisConnectionFactory,
      CacheProperties cacheProperties) {
    Map initialCacheConfigurations = new HashMap<>();
    for (String cacheName : cacheProperties.getRedis().getSpec().keySet()) {
      RedisCacheConfiguration configuration = redisCacheConfiguration(cacheProperties.getRedis().getSpec().get(cacheName));
      initialCacheConfigurations.put(cacheName, configuration);
    }
    RedisCacheManager redisCacheManager = new RedisCacheManager(RedisCacheWriter.nonLockingRedisCacheWriter(redisConnectionFactory),
        RedisCacheConfiguration.defaultCacheConfig(), initialCacheConfigurations, false);
    redisCacheManager.initializeCaches();
    return redisCacheManager;
  }

  private RedisCacheConfiguration redisCacheConfiguration(RedisCacheSpec spec) {
    RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig();
    if (spec.getTimeToLive() > 0) {
      config = config.entryTtl(Duration.ofMillis(spec.getTimeToLive()));
    }
    if (spec.getKeyPrefix() != null) {
      config = config.prefixKeysWith(spec.getKeyPrefix());
    }
    if (!spec.isCacheNullValues()) {
      config = config.disableCachingNullValues();
    }
    if (!spec.isUseKeyPrefix()) {
      config = config.disableKeyPrefix();
    }
    return config;
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy