From 5c72f9cd8fd61684ffe271355c2d5c82fc7268c3 Mon Sep 17 00:00:00 2001 From: oliviarla Date: Fri, 23 Aug 2024 18:30:03 +0900 Subject: [PATCH] INTERNAL: add builder and transaction for ArcusCacheManager --- pom.xml | 18 ++++ .../arcus/spring/cache/ArcusCacheManager.java | 98 ++++++++++++++++++- .../cache/ArcusCacheManagerBuilderTest.java | 75 ++++++++++++++ 3 files changed, 189 insertions(+), 2 deletions(-) create mode 100644 src/test/java/com/navercorp/arcus/spring/cache/ArcusCacheManagerBuilderTest.java diff --git a/pom.xml b/pom.xml index 7c10435..7255ffb 100644 --- a/pom.xml +++ b/pom.xml @@ -131,6 +131,24 @@ provided + + org.springframework + spring-context-support + ${spring.version} + + + + commons-logging + commons-logging + + + provided + + + org.springframework + spring-tx + ${spring.version} + org.springframework spring-test diff --git a/src/main/java/com/navercorp/arcus/spring/cache/ArcusCacheManager.java b/src/main/java/com/navercorp/arcus/spring/cache/ArcusCacheManager.java index adc095b..3a46de2 100644 --- a/src/main/java/com/navercorp/arcus/spring/cache/ArcusCacheManager.java +++ b/src/main/java/com/navercorp/arcus/spring/cache/ArcusCacheManager.java @@ -19,8 +19,12 @@ import java.util.ArrayList; import java.util.Collection; +import java.util.Collections; +import java.util.LinkedHashMap; import java.util.List; import java.util.Map; +import java.util.Optional; +import java.util.Set; import net.spy.memcached.ArcusClient; import net.spy.memcached.ArcusClientPool; @@ -28,13 +32,14 @@ import org.springframework.beans.factory.DisposableBean; import org.springframework.cache.Cache; -import org.springframework.cache.support.AbstractCacheManager; +import org.springframework.cache.transaction.AbstractTransactionSupportingCacheManager; +import org.springframework.util.Assert; /** * 스프링 CacheManager의 Arcus 구현체. * 미리 정의하지 않은 이름의 캐시에 대해 get 요청을 받으면 (SimpleCacheManager와 다르게) 기본 설정으로 새 캐시를 생성하고 저장합니다. */ -public class ArcusCacheManager extends AbstractCacheManager implements DisposableBean { +public class ArcusCacheManager extends AbstractTransactionSupportingCacheManager implements DisposableBean { private final ArcusClientPool client; protected ArcusCacheConfiguration defaultConfiguration; protected Map initialCacheConfigs; @@ -85,6 +90,17 @@ public ArcusCacheManager( this.internalClient = true; } + public static ArcusCacheManagerBuilder builder(ArcusClientPool arcusClientPool) { + return new ArcusCacheManagerBuilder(arcusClientPool); + } + + public static ArcusCacheManagerBuilder builder(String adminAddress, + String serviceCode, + ConnectionFactoryBuilder connectionFactoryBuilder, + int poolSize) { + return new ArcusCacheManagerBuilder(adminAddress, serviceCode, connectionFactoryBuilder, poolSize); + } + @Override protected Collection loadCaches() { List caches = new ArrayList(initialCacheConfigs.size()); @@ -117,4 +133,82 @@ public void destroy() { client.shutdown(); } } + + public static class ArcusCacheManagerBuilder { + private final ArcusClientPool arcusClientPool; + private final boolean internalClient; + private final Map initialCaches = new LinkedHashMap<>(); + private boolean enableTransactions; + private ArcusCacheConfiguration defaultConfiguration = new ArcusCacheConfiguration(); + + private ArcusCacheManagerBuilder(ArcusClientPool arcusClientPool) { + this.arcusClientPool = arcusClientPool; + this.internalClient = false; + } + + private ArcusCacheManagerBuilder(String adminAddress, + String serviceCode, + ConnectionFactoryBuilder connectionFactoryBuilder, + int poolSize) { + this.arcusClientPool = ArcusClient.createArcusClientPool( + adminAddress, serviceCode, connectionFactoryBuilder, poolSize); + this.internalClient = true; + } + + public ArcusCacheManagerBuilder cacheDefaults(ArcusCacheConfiguration defaultCacheConfiguration) { + this.defaultConfiguration = defaultCacheConfiguration; + return this; + } + + public ArcusCacheManagerBuilder initialCacheNames(Set cacheNames) { + Assert.notNull(cacheNames, "Cache names must not be null"); + + cacheNames.forEach(cacheName -> initialCaches.put(cacheName, defaultConfiguration)); + return this; + } + + public ArcusCacheManagerBuilder transactionAware() { + this.enableTransactions = true; + return this; + } + + public ArcusCacheManagerBuilder withCacheConfiguration(String cacheName, ArcusCacheConfiguration cacheConfiguration) { + Assert.notNull(cacheName, "Cache name must not be null"); + Assert.notNull(cacheConfiguration, "Cache configuration must not be null"); + + this.initialCaches.put(cacheName, cacheConfiguration); + return this; + } + + public ArcusCacheManagerBuilder withInitialCacheConfigurations(Map cacheConfigurations) { + Assert.notNull(cacheConfigurations, "Cache configurations must not be null"); + + this.initialCaches.putAll(cacheConfigurations); + return this; + } + + public Optional getCacheConfigurationFor(String cacheName) { + return Optional.ofNullable(this.initialCaches.get(cacheName)); + } + + public Set getConfiguredCaches() { + return Collections.unmodifiableSet(this.initialCaches.keySet()); + } + + /** + * Create new instance of {@link ArcusCacheManager} with configuration options applied. + * + * @return new instance of {@link ArcusCacheManager}. + */ + public ArcusCacheManager build() { + Assert.state(arcusClientPool != null, "ArcusClient must not be null"); + + ArcusCacheManager cacheManager = new ArcusCacheManager(arcusClientPool, defaultConfiguration, initialCaches); + cacheManager.internalClient = this.internalClient; + cacheManager.setTransactionAware(this.enableTransactions); + + return cacheManager; + } + + } } diff --git a/src/test/java/com/navercorp/arcus/spring/cache/ArcusCacheManagerBuilderTest.java b/src/test/java/com/navercorp/arcus/spring/cache/ArcusCacheManagerBuilderTest.java new file mode 100644 index 0000000..bf92d1e --- /dev/null +++ b/src/test/java/com/navercorp/arcus/spring/cache/ArcusCacheManagerBuilderTest.java @@ -0,0 +1,75 @@ +package com.navercorp.arcus.spring.cache; + +import java.util.Collections; + +import net.spy.memcached.ArcusClient; +import net.spy.memcached.ArcusClientPool; + +import org.junit.jupiter.api.Test; + +import org.springframework.cache.Cache; +import org.springframework.cache.transaction.TransactionAwareCacheDecorator; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertInstanceOf; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertThrows; + +class ArcusCacheManagerBuilderTest { + + private final ArcusClientPool arcusClientPool = ArcusClient.createArcusClientPool("localhost:2181", "test", 4); + + @Test + void testMissingCacheMadeByDefaultCacheConfig() { + ArcusCacheConfiguration configuration = new ArcusCacheConfiguration(); + configuration.setServiceId("TEST-"); + ArcusCacheManager cm = ArcusCacheManager.builder(arcusClientPool).cacheDefaults(configuration).build(); + cm.afterPropertiesSet(); + + ArcusCache missingCache = (ArcusCache) cm.getMissingCache("new-cache"); + assertNotNull(missingCache); + assertEquals(configuration.getServiceId(), missingCache.getServiceId()); + assertEquals(configuration.getPrefix(), missingCache.getPrefix()); + assertEquals(configuration.getExpireSeconds(), missingCache.getExpireSeconds()); + } + + @Test + void testSettingDifferentDefaultCacheConfiguration() { + ArcusCacheConfiguration withPrefix = new ArcusCacheConfiguration(); + withPrefix.setPrefix("prefix"); + ArcusCacheConfiguration withoutPrefix = new ArcusCacheConfiguration(); + + ArcusCacheManager cm = ArcusCacheManager.builder(arcusClientPool) + .cacheDefaults(withPrefix) + .initialCacheNames(Collections.singleton("first-cache")) + .cacheDefaults(withoutPrefix) + .initialCacheNames(Collections.singleton("second-cache")) + .build(); + + cm.afterPropertiesSet(); + + ArcusCache firstCache = (ArcusCache) cm.getCache("first-cache"); + assertNotNull(firstCache); + assertEquals(withPrefix.getPrefix(), firstCache.getPrefix()); + ArcusCache secondCache = (ArcusCache) cm.getCache("second-cache"); + assertNotNull(secondCache); + assertNull(withoutPrefix.getPrefix()); + assertEquals(withoutPrefix.getPrefix(), secondCache.getPrefix()); + } + + @Test + void testTransactionAwareCacheManager() { + Cache cache = ArcusCacheManager.builder(arcusClientPool) + .transactionAware() + .build() + .getCache("decorated-cache"); + + assertInstanceOf(TransactionAwareCacheDecorator.class, cache); + } + + @Test + void testArcusClientNull() { + assertThrows(IllegalStateException.class, () -> ArcusCacheManager.builder(null).build()); + } +}