Support for bean refs in caching SpEL expressions
Issue: SPR-13182
This commit is contained in:
parent
1ce9788a66
commit
9b5e47026c
|
|
@ -705,7 +705,8 @@ public abstract class CacheAspectSupport extends AbstractCacheInvoker
|
|||
|
||||
private EvaluationContext createEvaluationContext(Object result) {
|
||||
return evaluator.createEvaluationContext(
|
||||
this.caches, this.metadata.method, this.args, this.target, this.metadata.targetClass, result);
|
||||
this.caches, this.metadata.method, this.args, this.target, this.metadata.targetClass,
|
||||
result, applicationContext);
|
||||
}
|
||||
|
||||
protected Collection<? extends Cache> getCaches() {
|
||||
|
|
|
|||
|
|
@ -22,8 +22,10 @@ import java.util.Map;
|
|||
import java.util.concurrent.ConcurrentHashMap;
|
||||
|
||||
import org.springframework.aop.support.AopUtils;
|
||||
import org.springframework.beans.factory.BeanFactory;
|
||||
import org.springframework.cache.Cache;
|
||||
import org.springframework.context.expression.AnnotatedElementKey;
|
||||
import org.springframework.context.expression.BeanFactoryResolver;
|
||||
import org.springframework.context.expression.CachedExpressionEvaluator;
|
||||
import org.springframework.core.DefaultParameterNameDiscoverer;
|
||||
import org.springframework.core.ParameterNameDiscoverer;
|
||||
|
|
@ -75,12 +77,12 @@ class ExpressionEvaluator extends CachedExpressionEvaluator {
|
|||
|
||||
/**
|
||||
* Create an {@link EvaluationContext} without a return value.
|
||||
* @see #createEvaluationContext(Collection, Method, Object[], Object, Class, Object)
|
||||
* @see #createEvaluationContext(Collection, Method, Object[], Object, Class, Object, BeanFactory)
|
||||
*/
|
||||
public EvaluationContext createEvaluationContext(Collection<? extends Cache> caches,
|
||||
Method method, Object[] args, Object target, Class<?> targetClass) {
|
||||
Method method, Object[] args, Object target, Class<?> targetClass, BeanFactory beanFactory) {
|
||||
|
||||
return createEvaluationContext(caches, method, args, target, targetClass, NO_RESULT);
|
||||
return createEvaluationContext(caches, method, args, target, targetClass, NO_RESULT, beanFactory);
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -95,7 +97,8 @@ class ExpressionEvaluator extends CachedExpressionEvaluator {
|
|||
* @return the evaluation context
|
||||
*/
|
||||
public EvaluationContext createEvaluationContext(Collection<? extends Cache> caches,
|
||||
Method method, Object[] args, Object target, Class<?> targetClass, Object result) {
|
||||
Method method, Object[] args, Object target, Class<?> targetClass, Object result,
|
||||
BeanFactory beanFactory) {
|
||||
|
||||
CacheExpressionRootObject rootObject = new CacheExpressionRootObject(caches,
|
||||
method, args, target, targetClass);
|
||||
|
|
@ -108,6 +111,9 @@ class ExpressionEvaluator extends CachedExpressionEvaluator {
|
|||
else if (result != NO_RESULT) {
|
||||
evaluationContext.setVariable(RESULT_VARIABLE, result);
|
||||
}
|
||||
if (beanFactory != null) {
|
||||
evaluationContext.setBeanResolver(new BeanFactoryResolver(beanFactory));
|
||||
}
|
||||
return evaluationContext;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -2,6 +2,7 @@ package org.springframework.cache.config;
|
|||
|
||||
import java.util.concurrent.atomic.AtomicLong;
|
||||
|
||||
import org.junit.After;
|
||||
import org.junit.Test;
|
||||
|
||||
import org.springframework.cache.Cache;
|
||||
|
|
@ -21,29 +22,37 @@ import org.springframework.context.annotation.Import;
|
|||
import static org.springframework.cache.CacheTestUtils.*;
|
||||
|
||||
/**
|
||||
* Tests that represent real use cases with advanced configuration
|
||||
* Tests that represent real use cases with advanced configuration.
|
||||
*
|
||||
* @author Stephane Nicoll
|
||||
*/
|
||||
public class EnableCachingIntegrationTests {
|
||||
|
||||
private ConfigurableApplicationContext context;
|
||||
|
||||
@After
|
||||
public void closeContext() {
|
||||
if (this.context != null) {
|
||||
this.context.close();
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void fooServiceWithInterface() {
|
||||
ConfigurableApplicationContext context = new AnnotationConfigApplicationContext(FooConfig.class);
|
||||
FooService service = context.getBean(FooService.class);
|
||||
fooGetSimple(context, service);
|
||||
this.context = new AnnotationConfigApplicationContext(FooConfig.class);
|
||||
FooService service = this.context.getBean(FooService.class);
|
||||
fooGetSimple(service);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void fooServiceWithInterfaceCglib() {
|
||||
ConfigurableApplicationContext context = new AnnotationConfigApplicationContext(FooConfigCglib.class);
|
||||
FooService service = context.getBean(FooService.class);
|
||||
fooGetSimple(context, service);
|
||||
this.context = new AnnotationConfigApplicationContext(FooConfigCglib.class);
|
||||
FooService service = this.context.getBean(FooService.class);
|
||||
fooGetSimple(service);
|
||||
}
|
||||
|
||||
private void fooGetSimple(ApplicationContext context, FooService service) {
|
||||
CacheManager cacheManager = context.getBean(CacheManager.class);
|
||||
|
||||
Cache cache = cacheManager.getCache("testCache");
|
||||
private void fooGetSimple(FooService service) {
|
||||
Cache cache = getCache();
|
||||
|
||||
Object key = new Object();
|
||||
assertCacheMiss(key, cache);
|
||||
|
|
@ -52,6 +61,21 @@ public class EnableCachingIntegrationTests {
|
|||
assertCacheHit(key, value, cache);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void beanCondition() {
|
||||
this.context = new AnnotationConfigApplicationContext(BeanConditionConfig.class);
|
||||
Cache cache = getCache();
|
||||
FooService service = context.getBean(FooService.class);
|
||||
|
||||
Object key = new Object();
|
||||
service.getWithCondition(key);
|
||||
assertCacheMiss(key, cache);
|
||||
}
|
||||
|
||||
private Cache getCache() {
|
||||
return this.context.getBean(CacheManager.class).getCache("testCache");
|
||||
}
|
||||
|
||||
@Configuration
|
||||
static class SharedConfig extends CachingConfigurerSupport {
|
||||
@Override
|
||||
|
|
@ -81,8 +105,10 @@ public class EnableCachingIntegrationTests {
|
|||
}
|
||||
}
|
||||
|
||||
private static interface FooService {
|
||||
public Object getSimple(Object key);
|
||||
private interface FooService {
|
||||
Object getSimple(Object key);
|
||||
|
||||
Object getWithCondition(Object key);
|
||||
}
|
||||
|
||||
@CacheConfig(cacheNames = "testCache")
|
||||
|
|
@ -94,6 +120,35 @@ public class EnableCachingIntegrationTests {
|
|||
public Object getSimple(Object key) {
|
||||
return counter.getAndIncrement();
|
||||
}
|
||||
|
||||
@Override
|
||||
@Cacheable(condition = "@bar.enabled")
|
||||
public Object getWithCondition(Object key) {
|
||||
return counter.getAndIncrement();
|
||||
}
|
||||
}
|
||||
|
||||
@Configuration
|
||||
@Import(FooConfig.class)
|
||||
@EnableCaching
|
||||
static class BeanConditionConfig {
|
||||
|
||||
@Bean
|
||||
public Bar bar() {
|
||||
return new Bar(false);
|
||||
}
|
||||
|
||||
static class Bar {
|
||||
private final boolean enabled;
|
||||
|
||||
public Bar(boolean enabled) {
|
||||
this.enabled = enabled;
|
||||
}
|
||||
|
||||
public boolean isEnabled() {
|
||||
return enabled;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
|||
|
|
@ -23,11 +23,15 @@ import java.util.Iterator;
|
|||
|
||||
import org.junit.Test;
|
||||
|
||||
import org.springframework.beans.factory.BeanFactory;
|
||||
import org.springframework.beans.factory.config.BeanDefinition;
|
||||
import org.springframework.beans.factory.support.RootBeanDefinition;
|
||||
import org.springframework.cache.annotation.AnnotationCacheOperationSource;
|
||||
import org.springframework.cache.annotation.Cacheable;
|
||||
import org.springframework.cache.annotation.Caching;
|
||||
import org.springframework.cache.concurrent.ConcurrentMapCache;
|
||||
import org.springframework.context.expression.AnnotatedElementKey;
|
||||
import org.springframework.context.support.StaticApplicationContext;
|
||||
import org.springframework.expression.EvaluationContext;
|
||||
import org.springframework.expression.spel.standard.SpelExpressionParser;
|
||||
import org.springframework.util.ReflectionUtils;
|
||||
|
|
@ -75,7 +79,8 @@ public class ExpressionEvaluatorTests {
|
|||
Object[] args = new Object[] { new Object(), new Object() };
|
||||
Collection<ConcurrentMapCache> caches = Collections.singleton(new ConcurrentMapCache("test"));
|
||||
|
||||
EvaluationContext evalCtx = eval.createEvaluationContext(caches, method, args, target, target.getClass());
|
||||
EvaluationContext evalCtx = eval.createEvaluationContext(caches, method, args,
|
||||
target, target.getClass(), null);
|
||||
Collection<CacheOperation> ops = getOps("multipleCaching");
|
||||
|
||||
Iterator<CacheOperation> it = ops.iterator();
|
||||
|
|
@ -122,14 +127,29 @@ public class ExpressionEvaluatorTests {
|
|||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void resolveBeanReference() throws Exception {
|
||||
StaticApplicationContext applicationContext = new StaticApplicationContext();
|
||||
BeanDefinition beanDefinition = new RootBeanDefinition(String.class);
|
||||
applicationContext.registerBeanDefinition("myBean", beanDefinition);
|
||||
applicationContext.refresh();
|
||||
|
||||
EvaluationContext context = createEvaluationContext(ExpressionEvaluator.NO_RESULT, applicationContext);
|
||||
Object value = new SpelExpressionParser().parseExpression("@myBean.class.getName()").getValue(context);
|
||||
assertThat(value, is(String.class.getName()));
|
||||
}
|
||||
|
||||
private EvaluationContext createEvaluationContext(Object result) {
|
||||
return createEvaluationContext(result, null);
|
||||
}
|
||||
|
||||
private EvaluationContext createEvaluationContext(Object result, BeanFactory beanFactory) {
|
||||
AnnotatedClass target = new AnnotatedClass();
|
||||
Method method = ReflectionUtils.findMethod(AnnotatedClass.class, "multipleCaching", Object.class,
|
||||
Object.class);
|
||||
Object[] args = new Object[] { new Object(), new Object() };
|
||||
Collection<ConcurrentMapCache> caches = Collections.singleton(new ConcurrentMapCache("test"));
|
||||
EvaluationContext context = eval.createEvaluationContext(caches, method, args, target, target.getClass(), result);
|
||||
return context;
|
||||
return eval.createEvaluationContext(caches, method, args, target, target.getClass(), result, beanFactory);
|
||||
}
|
||||
|
||||
|
||||
|
|
|
|||
Loading…
Reference in New Issue