|
|
|
|
@ -1,5 +1,5 @@
@@ -1,5 +1,5 @@
|
|
|
|
|
/* |
|
|
|
|
* Copyright 2002-2013 the original author or authors. |
|
|
|
|
* Copyright 2002-2014 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. |
|
|
|
|
@ -20,6 +20,7 @@ import java.lang.reflect.Method;
@@ -20,6 +20,7 @@ import java.lang.reflect.Method;
|
|
|
|
|
import java.util.ArrayList; |
|
|
|
|
import java.util.Collection; |
|
|
|
|
import java.util.Collections; |
|
|
|
|
import java.util.LinkedList; |
|
|
|
|
import java.util.List; |
|
|
|
|
import java.util.Set; |
|
|
|
|
|
|
|
|
|
@ -29,7 +30,6 @@ import org.apache.commons.logging.LogFactory;
@@ -29,7 +30,6 @@ import org.apache.commons.logging.LogFactory;
|
|
|
|
|
import org.springframework.aop.framework.AopProxyUtils; |
|
|
|
|
import org.springframework.beans.factory.InitializingBean; |
|
|
|
|
import org.springframework.cache.Cache; |
|
|
|
|
import org.springframework.cache.Cache.ValueWrapper; |
|
|
|
|
import org.springframework.cache.CacheManager; |
|
|
|
|
import org.springframework.cache.support.SimpleValueWrapper; |
|
|
|
|
import org.springframework.expression.EvaluationContext; |
|
|
|
|
@ -64,6 +64,7 @@ import org.springframework.util.StringUtils;
@@ -64,6 +64,7 @@ import org.springframework.util.StringUtils;
|
|
|
|
|
* @author Chris Beams |
|
|
|
|
* @author Phillip Webb |
|
|
|
|
* @author Sam Brannen |
|
|
|
|
* @author Stephane Nicoll |
|
|
|
|
* @since 3.1 |
|
|
|
|
*/ |
|
|
|
|
public abstract class CacheAspectSupport implements InitializingBean { |
|
|
|
|
@ -194,15 +195,20 @@ public abstract class CacheAspectSupport implements InitializingBean {
@@ -194,15 +195,20 @@ public abstract class CacheAspectSupport implements InitializingBean {
|
|
|
|
|
// Process any early evictions
|
|
|
|
|
processCacheEvicts(contexts.get(CacheEvictOperation.class), true, ExpressionEvaluator.NO_RESULT); |
|
|
|
|
|
|
|
|
|
// Collect puts from any @Cachable miss
|
|
|
|
|
List<CachePutRequest> cachePutRequests = new ArrayList<CachePutRequest>(); |
|
|
|
|
collectPutRequests(contexts.get(CacheableOperation.class), ExpressionEvaluator.NO_RESULT, cachePutRequests, true); |
|
|
|
|
// Check if we have a cached item matching the conditions
|
|
|
|
|
Cache.ValueWrapper cacheHit = findCachedItem(contexts.get(CacheableOperation.class)); |
|
|
|
|
|
|
|
|
|
ValueWrapper result = null; |
|
|
|
|
// Collect puts from any @Cacheable miss, if no cached item is found
|
|
|
|
|
List<CachePutRequest> cachePutRequests = new LinkedList<CachePutRequest>(); |
|
|
|
|
if (cacheHit == null) { |
|
|
|
|
collectPutRequests(contexts.get(CacheableOperation.class), ExpressionEvaluator.NO_RESULT, cachePutRequests); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
Cache.ValueWrapper result = null; |
|
|
|
|
|
|
|
|
|
// We only attempt to get a cached result if there are no put requests
|
|
|
|
|
// If there are no put requests, just use the cache hit
|
|
|
|
|
if (cachePutRequests.isEmpty() && contexts.get(CachePutOperation.class).isEmpty()) { |
|
|
|
|
result = findCachedResult(contexts.get(CacheableOperation.class)); |
|
|
|
|
result = cacheHit; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Invoke the method if don't have a cache hit
|
|
|
|
|
@ -211,7 +217,7 @@ public abstract class CacheAspectSupport implements InitializingBean {
@@ -211,7 +217,7 @@ public abstract class CacheAspectSupport implements InitializingBean {
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Collect any explicit @CachePuts
|
|
|
|
|
collectPutRequests(contexts.get(CachePutOperation.class), result.get(), cachePutRequests, false); |
|
|
|
|
collectPutRequests(contexts.get(CachePutOperation.class), result.get(), cachePutRequests); |
|
|
|
|
|
|
|
|
|
// Process any collected put requests, either from @CachePut or a @Cacheable miss
|
|
|
|
|
for (CachePutRequest cachePutRequest : cachePutRequests) { |
|
|
|
|
@ -257,39 +263,42 @@ public abstract class CacheAspectSupport implements InitializingBean {
@@ -257,39 +263,42 @@ public abstract class CacheAspectSupport implements InitializingBean {
|
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private void collectPutRequests(Collection<CacheOperationContext> contexts, |
|
|
|
|
Object result, Collection<CachePutRequest> putRequests, boolean whenNotInCache) { |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* Find a cached item only for {@link CacheableOperation} that passes the condition. |
|
|
|
|
* @param contexts the cacheable operations |
|
|
|
|
* @return a {@link Cache.ValueWrapper} holding the cached item, |
|
|
|
|
* or {@code null} if none is found |
|
|
|
|
*/ |
|
|
|
|
private Cache.ValueWrapper findCachedItem(Collection<CacheOperationContext> contexts) { |
|
|
|
|
Object result = ExpressionEvaluator.NO_RESULT; |
|
|
|
|
for (CacheOperationContext context : contexts) { |
|
|
|
|
if (isConditionPassing(context, result)) { |
|
|
|
|
Object key = generateKey(context, result); |
|
|
|
|
if (!whenNotInCache || findInAnyCaches(contexts, key) == null) { |
|
|
|
|
putRequests.add(new CachePutRequest(context, key)); |
|
|
|
|
Cache.ValueWrapper cached = findInCaches(context, key); |
|
|
|
|
if (cached != null) { |
|
|
|
|
return cached; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return null; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private Cache.ValueWrapper findCachedResult(Collection<CacheOperationContext> contexts) { |
|
|
|
|
ValueWrapper result = null; |
|
|
|
|
for (CacheOperationContext context : contexts) { |
|
|
|
|
if (isConditionPassing(context, ExpressionEvaluator.NO_RESULT)) { |
|
|
|
|
if (result == null) { |
|
|
|
|
result = findInCaches(context, generateKey(context, ExpressionEvaluator.NO_RESULT)); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return result; |
|
|
|
|
} |
|
|
|
|
/** |
|
|
|
|
* Collect the {@link CachePutRequest} for all {@link CacheOperation} using |
|
|
|
|
* the specified result item. |
|
|
|
|
* @param contexts the contexts to handle |
|
|
|
|
* @param result the result item (never {@code null}) |
|
|
|
|
* @param putRequests the collection to update |
|
|
|
|
*/ |
|
|
|
|
private void collectPutRequests(Collection<CacheOperationContext> contexts, |
|
|
|
|
Object result, Collection<CachePutRequest> putRequests) { |
|
|
|
|
|
|
|
|
|
private Cache.ValueWrapper findInAnyCaches(Collection<CacheOperationContext> contexts, Object key) { |
|
|
|
|
for (CacheOperationContext context : contexts) { |
|
|
|
|
ValueWrapper wrapper = findInCaches(context, key); |
|
|
|
|
if (wrapper != null) { |
|
|
|
|
return wrapper; |
|
|
|
|
if (isConditionPassing(context, result)) { |
|
|
|
|
Object key = generateKey(context, result); |
|
|
|
|
putRequests.add(new CachePutRequest(context, key)); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return null; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private Cache.ValueWrapper findInCaches(CacheOperationContext context, Object key) { |
|
|
|
|
|