/*
* Copyright 2013-2016 the original author or authors.
*
* 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.springframework.cloud.aws.cache.config.annotation;
import com.amazonaws.services.elasticache.AmazonElastiCache;
import com.amazonaws.services.elasticache.model.CacheCluster;
import com.amazonaws.services.elasticache.model.DescribeCacheClustersRequest;
import com.amazonaws.services.elasticache.model.DescribeCacheClustersResult;
import com.amazonaws.services.elasticache.model.Endpoint;
import org.junit.After;
import org.junit.Test;
import org.mockito.Mockito;
import org.springframework.cache.Cache;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurer;
import org.springframework.cloud.aws.cache.config.TestMemcacheServer;
import org.springframework.cloud.aws.cache.memcached.SimpleSpringMemcached;
import org.springframework.cloud.aws.core.env.stack.ListableStackResourceFactory;
import org.springframework.cloud.aws.core.env.stack.StackResource;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.util.ReflectionUtils;
import java.lang.reflect.Field;
import java.util.Arrays;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
public class ElastiCacheCachingConfigurationTest {
private AnnotationConfigApplicationContext context;
@After
public void tearDown() throws Exception {
if (this.context != null) {
this.context.close();
}
}
@Test
public void enableElasticache_configuredWithExplicitCluster_configuresExplicitlyConfiguredCaches() throws Exception {
//Arrange
//Act
this.context = new AnnotationConfigApplicationContext(ApplicationConfigurationWithExplicitStackConfiguration.class);
//Assert
CacheManager cacheManager = this.context.getBean(CachingConfigurer.class).cacheManager();
assertEquals(2, cacheManager.getCacheNames().size());
Cache firstCache = cacheManager.getCache("firstCache");
assertNotNull(firstCache.getName());
assertEquals(0, getExpirationFromCache(firstCache));
Cache secondCache = cacheManager.getCache("secondCache");
assertNotNull(secondCache.getName());
assertEquals(0, getExpirationFromCache(secondCache));
}
@Test
public void enableElasticache_configuredWithExplicitClusterAndExpiration_configuresExplicitlyConfiguredCachesWithCustomExpirationTimes() throws Exception {
//Arrange
//Act
this.context = new AnnotationConfigApplicationContext(ApplicationConfigurationWithExplicitStackConfigurationAndExpiryTime.class);
//Assert
CacheManager cacheManager = this.context.getBean(CachingConfigurer.class).cacheManager();
assertEquals(2, cacheManager.getCacheNames().size());
Cache firstCache = cacheManager.getCache("firstCache");
assertNotNull(firstCache.getName());
assertEquals(23, getExpirationFromCache(firstCache));
Cache secondCache = cacheManager.getCache("secondCache");
assertNotNull(secondCache.getName());
assertEquals(42, getExpirationFromCache(secondCache));
}
@Test
public void enableElasticache_configuredWithExplicitClusterAndExpiration_configuresExplicitlyConfiguredCachesWithMixedExpirationTimes() throws Exception {
//Arrange
//Act
this.context = new AnnotationConfigApplicationContext(ApplicationConfigurationWithExplicitStackConfigurationAndMixedExpiryTime.class);
//Assert
CacheManager cacheManager = this.context.getBean(CachingConfigurer.class).cacheManager();
assertEquals(2, cacheManager.getCacheNames().size());
Cache firstCache = cacheManager.getCache("firstCache");
assertNotNull(firstCache.getName());
assertEquals(12, getExpirationFromCache(firstCache));
Cache secondCache = cacheManager.getCache("secondCache");
assertNotNull(secondCache.getName());
assertEquals(42, getExpirationFromCache(secondCache));
}
@Test
public void enableElasticache_configuredWithoutExplicitCluster_configuresImplicitlyConfiguredCaches() throws Exception {
//Arrange
//Act
this.context = new AnnotationConfigApplicationContext(ApplicationConfigurationWithNoExplicitStackConfiguration.class);
//Assert
CacheManager cacheManager = this.context.getBean(CachingConfigurer.class).cacheManager();
assertEquals(2, cacheManager.getCacheNames().size());
Cache firstCache = cacheManager.getCache("sampleCacheOneLogical");
assertNotNull(firstCache.getName());
assertEquals(0, getExpirationFromCache(firstCache));
Cache secondCache = cacheManager.getCache("sampleCacheTwoLogical");
assertNotNull(secondCache.getName());
assertEquals(0, getExpirationFromCache(secondCache));
}
@Test
public void enableElasticache_configuredWithoutExplicitClusterButDefaultExpiryTime_configuresImplicitlyConfiguredCachesWithDefaultExpiryTimeOnAllCaches() throws Exception {
//Arrange
//Act
this.context = new AnnotationConfigApplicationContext(ApplicationConfigurationWithNoExplicitStackConfigurationAndDefaultExpiration.class);
//Assert
CacheManager cacheManager = this.context.getBean(CachingConfigurer.class).cacheManager();
assertEquals(2, cacheManager.getCacheNames().size());
Cache firstCache = cacheManager.getCache("sampleCacheOneLogical");
assertNotNull(firstCache.getName());
assertEquals(23, getExpirationFromCache(firstCache));
Cache secondCache = cacheManager.getCache("sampleCacheTwoLogical");
assertNotNull(secondCache.getName());
assertEquals(23, getExpirationFromCache(secondCache));
}
@EnableElastiCache({@CacheClusterConfig(name = "firstCache"), @CacheClusterConfig(name = "secondCache")})
public static class ApplicationConfigurationWithExplicitStackConfiguration {
@Bean
public AmazonElastiCache amazonElastiCache() {
AmazonElastiCache amazonElastiCache = Mockito.mock(AmazonElastiCache.class);
int port = TestMemcacheServer.startServer();
DescribeCacheClustersRequest describeCacheClustersRequest = new DescribeCacheClustersRequest().withCacheClusterId("firstCache");
describeCacheClustersRequest.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(describeCacheClustersRequest)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
DescribeCacheClustersRequest secondCache = new DescribeCacheClustersRequest().withCacheClusterId("secondCache");
secondCache.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(secondCache)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
return amazonElastiCache;
}
}
@EnableElastiCache({@CacheClusterConfig(name = "firstCache", expiration = 23), @CacheClusterConfig(name = "secondCache", expiration = 42)})
public static class ApplicationConfigurationWithExplicitStackConfigurationAndExpiryTime {
@Bean
public AmazonElastiCache amazonElastiCache() {
AmazonElastiCache amazonElastiCache = Mockito.mock(AmazonElastiCache.class);
int port = TestMemcacheServer.startServer();
DescribeCacheClustersRequest firstCache = new DescribeCacheClustersRequest().withCacheClusterId("firstCache");
firstCache.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(firstCache)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
DescribeCacheClustersRequest secondCache = new DescribeCacheClustersRequest().withCacheClusterId("secondCache");
secondCache.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(secondCache)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
return amazonElastiCache;
}
}
@EnableElastiCache(value = {@CacheClusterConfig(name = "firstCache"), @CacheClusterConfig(name = "secondCache", expiration = 42)}, defaultExpiration = 12)
public static class ApplicationConfigurationWithExplicitStackConfigurationAndMixedExpiryTime {
@Bean
public AmazonElastiCache amazonElastiCache() {
AmazonElastiCache amazonElastiCache = Mockito.mock(AmazonElastiCache.class);
int port = TestMemcacheServer.startServer();
DescribeCacheClustersRequest firstCache = new DescribeCacheClustersRequest().withCacheClusterId("firstCache");
firstCache.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(firstCache)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
DescribeCacheClustersRequest secondCache = new DescribeCacheClustersRequest().withCacheClusterId("secondCache");
secondCache.setShowCacheNodeInfo(true);
Mockito.when(amazonElastiCache.describeCacheClusters(secondCache)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
return amazonElastiCache;
}
}
@EnableElastiCache
public static class ApplicationConfigurationWithNoExplicitStackConfiguration {
@Bean
public AmazonElastiCache amazonElastiCache() {
AmazonElastiCache amazonElastiCache = Mockito.mock(AmazonElastiCache.class);
int port = TestMemcacheServer.startServer();
DescribeCacheClustersRequest sampleCacheOneLogical = new DescribeCacheClustersRequest().withCacheClusterId("sampleCacheOneLogical");
sampleCacheOneLogical.setShowCacheNodeInfo(Boolean.TRUE);
Mockito.when(amazonElastiCache.describeCacheClusters(sampleCacheOneLogical)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
DescribeCacheClustersRequest sampleCacheTwoLogical = new DescribeCacheClustersRequest().withCacheClusterId("sampleCacheTwoLogical");
sampleCacheTwoLogical.setShowCacheNodeInfo(Boolean.TRUE);
Mockito.when(amazonElastiCache.describeCacheClusters(sampleCacheTwoLogical)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
return amazonElastiCache;
}
@Bean
public ListableStackResourceFactory stackResourceFactory() {
ListableStackResourceFactory resourceFactory = Mockito.mock(ListableStackResourceFactory.class);
Mockito.when(resourceFactory.resourcesByType("AWS::ElastiCache::CacheCluster")).thenReturn(Arrays.asList(
new StackResource("sampleCacheOneLogical", "sampleCacheOne", "AWS::ElastiCache::CacheCluster"),
new StackResource("sampleCacheTwoLogical", "sampleCacheTwo", "AWS::ElastiCache::CacheCluster")));
return resourceFactory;
}
}
@EnableElastiCache(defaultExpiration = 23)
public static class ApplicationConfigurationWithNoExplicitStackConfigurationAndDefaultExpiration {
@Bean
public AmazonElastiCache amazonElastiCache() {
AmazonElastiCache amazonElastiCache = Mockito.mock(AmazonElastiCache.class);
int port = TestMemcacheServer.startServer();
DescribeCacheClustersRequest sampleCacheOneLogical = new DescribeCacheClustersRequest().withCacheClusterId("sampleCacheOneLogical");
sampleCacheOneLogical.setShowCacheNodeInfo(Boolean.TRUE);
Mockito.when(amazonElastiCache.describeCacheClusters(sampleCacheOneLogical)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
DescribeCacheClustersRequest sampleCacheTwoLogical = new DescribeCacheClustersRequest().withCacheClusterId("sampleCacheTwoLogical");
sampleCacheTwoLogical.setShowCacheNodeInfo(Boolean.TRUE);
Mockito.when(amazonElastiCache.describeCacheClusters(sampleCacheTwoLogical)).
thenReturn(new DescribeCacheClustersResult().withCacheClusters(new CacheCluster().
withConfigurationEndpoint(new Endpoint().withAddress("localhost").withPort(port)).
withEngine("memcached")));
return amazonElastiCache;
}
@Bean
public ListableStackResourceFactory stackResourceFactory() {
ListableStackResourceFactory resourceFactory = Mockito.mock(ListableStackResourceFactory.class);
Mockito.when(resourceFactory.resourcesByType("AWS::ElastiCache::CacheCluster")).thenReturn(Arrays.asList(
new StackResource("sampleCacheOneLogical", "sampleCacheOne", "AWS::ElastiCache::CacheCluster"),
new StackResource("sampleCacheTwoLogical", "sampleCacheTwo", "AWS::ElastiCache::CacheCluster")));
return resourceFactory;
}
}
private static int getExpirationFromCache(Cache cache) throws IllegalAccessException {
assertTrue(cache instanceof SimpleSpringMemcached);
Field expiration = ReflectionUtils.findField(SimpleSpringMemcached.class, "expiration");
assertNotNull(expiration);
ReflectionUtils.makeAccessible(expiration);
return expiration.getInt(cache);
}
}