io.quarkus.cache.runtime.CacheInterceptor Maven / Gradle / Ivy
package io.quarkus.cache.runtime;
import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.CompletionStage;
import java.util.function.Supplier;
import jakarta.enterprise.inject.Instance;
import jakarta.inject.Inject;
import jakarta.interceptor.Interceptor.Priority;
import jakarta.interceptor.InvocationContext;
import org.jboss.logging.Logger;
import io.quarkus.arc.AbstractAnnotationLiteral;
import io.quarkus.arc.runtime.InterceptorBindings;
import io.quarkus.cache.Cache;
import io.quarkus.cache.CacheException;
import io.quarkus.cache.CacheKey;
import io.quarkus.cache.CacheKeyGenerator;
import io.quarkus.cache.CacheManager;
import io.quarkus.cache.CompositeCacheKey;
import io.smallrye.mutiny.Uni;
public abstract class CacheInterceptor {
public static final int BASE_PRIORITY = Priority.PLATFORM_BEFORE;
private static final Logger LOGGER = Logger.getLogger(CacheInterceptor.class);
private static final String PERFORMANCE_WARN_MSG = "Cache key resolution based on reflection calls. Please create a GitHub issue in the Quarkus repository, the maintainers might be able to improve your application performance.";
protected static final String UNHANDLED_ASYNC_RETURN_TYPE_MSG = "Unhandled async return type";
@Inject
CacheManager cacheManager;
@Inject
Instance keyGenerator;
/*
* The interception is almost always managed by Arc in a Quarkus application. In such a case, we want to retrieve the
* interceptor bindings stored by Arc in the invocation context data (very good performance-wise). But sometimes the
* interception is managed by another CDI interceptors implementation. It can happen for example while using caching
* annotations on a MicroProfile REST Client method. In that case, we have no other choice but to rely on reflection (with
* underlying synchronized blocks which are bad for performances) to retrieve the interceptor bindings.
*
* IMPORTANT: Normally would be , but that leads to type pollution
*/
protected CacheInterceptionContext getInterceptionContext(InvocationContext invocationContext,
Class interceptorBindingClass, boolean supportsCacheKey) {
return getArcCacheInterceptionContext(invocationContext, interceptorBindingClass)
.orElseGet(new Supplier>() {
@Override
public CacheInterceptionContext get() {
return getNonArcCacheInterceptionContext(invocationContext, interceptorBindingClass, supportsCacheKey);
}
});
}
@SuppressWarnings("unchecked")
private Optional> getArcCacheInterceptionContext(
InvocationContext invocationContext, Class interceptorBindingClass) {
Set bindings = InterceptorBindings.getInterceptorBindingLiterals(invocationContext);
if (bindings == null) {
LOGGER.trace("Interceptor bindings not found in ArC");
// This should only happen when the interception is not managed by Arc.
return Optional.empty();
}
List interceptorBindings = new ArrayList<>();
List cacheKeyParameterPositions = new ArrayList<>();
for (AbstractAnnotationLiteral binding : bindings) {
if (binding.annotationType().isAssignableFrom(CacheKeyParameterPositions.class)) {
for (short position : ((CacheKeyParameterPositions) binding).value()) {
cacheKeyParameterPositions.add(position);
}
} else if (binding.annotationType().isAssignableFrom((interceptorBindingClass))) {
interceptorBindings.add((T) binding);
}
}
return Optional.of(new CacheInterceptionContext<>(interceptorBindings, cacheKeyParameterPositions));
}
@SuppressWarnings("unchecked")
private CacheInterceptionContext getNonArcCacheInterceptionContext(
InvocationContext invocationContext, Class interceptorBindingClass, boolean supportsCacheKey) {
LOGGER.trace("Retrieving interceptor bindings using reflection");
List interceptorBindings = new ArrayList<>();
List cacheKeyParameterPositions = new ArrayList<>();
boolean cacheKeyParameterPositionsFound = false;
for (Annotation annotation : invocationContext.getMethod().getAnnotations()) {
if (annotation instanceof CacheKeyParameterPositions) {
cacheKeyParameterPositionsFound = true;
for (short position : ((CacheKeyParameterPositions) annotation).value()) {
cacheKeyParameterPositions.add(position);
}
} else if (interceptorBindingClass.isInstance(annotation)) {
interceptorBindings.add((T) annotation);
}
}
if (supportsCacheKey && !cacheKeyParameterPositionsFound) {
/*
* This block is a fallback that should ideally never be executed because of the poor performance of reflection
* calls. If the following warn message is displayed, then it means that we should update the build time bytecode
* generation to cover the missing case. See RestClientMethodEnhancer for more details.
*/
LOGGER.warn(PERFORMANCE_WARN_MSG);
Parameter[] parameters = invocationContext.getMethod().getParameters();
for (short i = 0; i < parameters.length; i++) {
if (parameters[i].isAnnotationPresent(CacheKey.class)) {
cacheKeyParameterPositions.add(i);
}
}
}
return new CacheInterceptionContext<>(interceptorBindings, cacheKeyParameterPositions);
}
@SuppressWarnings("unchecked")
private T cast(Annotation annotation, Class interceptorBindingClass) {
return (T) annotation;
}
protected Object getCacheKey(Cache cache, Class extends CacheKeyGenerator> keyGeneratorClass,
List cacheKeyParameterPositions, Method method, Object[] methodParameterValues) {
if (keyGeneratorClass != UndefinedCacheKeyGenerator.class) {
return generateKey(keyGeneratorClass, method, methodParameterValues);
} else if (methodParameterValues == null || methodParameterValues.length == 0) {
// If the intercepted method doesn't have any parameter, then the default cache key will be used.
return cache.getDefaultKey();
} else if (cacheKeyParameterPositions.size() == 1) {
// If exactly one @CacheKey-annotated parameter was identified for the intercepted method at build time, then this
// parameter will be used as the cache key.
return methodParameterValues[cacheKeyParameterPositions.get(0)];
} else if (cacheKeyParameterPositions.size() >= 2) {
// If two or more @CacheKey-annotated parameters were identified for the intercepted method at build time, then a
// composite cache key built from all these parameters will be used.
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy