Bypass method traversal for annotation introspection if possible
The isCandidateClass mechanism is consistently used for a bypass check before method traversal attempts. While by default this is only bypassing standard java types, the same mechanism can be used with index metadata which indicates non-presence of certain annotations. See gh-22420
This commit is contained in:
		
							parent
							
								
									6266370a7a
								
							
						
					
					
						commit
						e3a9826e56
					
				| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2017 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -21,6 +21,7 @@ import java.lang.annotation.Annotation;
 | 
			
		|||
import org.springframework.aop.ClassFilter;
 | 
			
		||||
import org.springframework.aop.MethodMatcher;
 | 
			
		||||
import org.springframework.aop.Pointcut;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -96,7 +97,7 @@ public class AnnotationMatchingPointcut implements Pointcut {
 | 
			
		|||
			this.classFilter = new AnnotationClassFilter(classAnnotationType, checkInherited);
 | 
			
		||||
		}
 | 
			
		||||
		else {
 | 
			
		||||
			this.classFilter = ClassFilter.TRUE;
 | 
			
		||||
			this.classFilter = new AnnotationCandidateClassFilter(methodAnnotationType);
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if (methodAnnotationType != null) {
 | 
			
		||||
| 
						 | 
				
			
			@ -164,4 +165,23 @@ public class AnnotationMatchingPointcut implements Pointcut {
 | 
			
		|||
		return new AnnotationMatchingPointcut(null, annotationType);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * {@link ClassFilter} that delegates to {@link AnnotationUtils#isCandidateClass}
 | 
			
		||||
	 * for filtering classes whose methods are not worth searching to begin with.
 | 
			
		||||
	 */
 | 
			
		||||
	private static class AnnotationCandidateClassFilter implements ClassFilter {
 | 
			
		||||
 | 
			
		||||
		private final Class<? extends Annotation> annotationType;
 | 
			
		||||
 | 
			
		||||
		public AnnotationCandidateClassFilter(Class<? extends Annotation> annotationType) {
 | 
			
		||||
			this.annotationType = annotationType;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public boolean matches(Class<?> clazz) {
 | 
			
		||||
			return AnnotationUtils.isCandidateClass(clazz, this.annotationType);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -59,6 +59,7 @@ import org.springframework.core.Ordered;
 | 
			
		|||
import org.springframework.core.PriorityOrdered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationAttributes;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
import org.springframework.util.ClassUtils;
 | 
			
		||||
| 
						 | 
				
			
			@ -244,25 +245,33 @@ public class AutowiredAnnotationBeanPostProcessor extends InstantiationAwareBean
 | 
			
		|||
 | 
			
		||||
		// Let's check for lookup methods here..
 | 
			
		||||
		if (!this.lookupMethodsChecked.contains(beanName)) {
 | 
			
		||||
			try {
 | 
			
		||||
				ReflectionUtils.doWithMethods(beanClass, method -> {
 | 
			
		||||
					Lookup lookup = method.getAnnotation(Lookup.class);
 | 
			
		||||
					if (lookup != null) {
 | 
			
		||||
						Assert.state(this.beanFactory != null, "No BeanFactory available");
 | 
			
		||||
						LookupOverride override = new LookupOverride(method, lookup.value());
 | 
			
		||||
						try {
 | 
			
		||||
							RootBeanDefinition mbd = (RootBeanDefinition) this.beanFactory.getMergedBeanDefinition(beanName);
 | 
			
		||||
							mbd.getMethodOverrides().addOverride(override);
 | 
			
		||||
						}
 | 
			
		||||
						catch (NoSuchBeanDefinitionException ex) {
 | 
			
		||||
							throw new BeanCreationException(beanName,
 | 
			
		||||
									"Cannot apply @Lookup to beans without corresponding bean definition");
 | 
			
		||||
						}
 | 
			
		||||
			if (AnnotationUtils.isCandidateClass(beanClass, Lookup.class)) {
 | 
			
		||||
				try {
 | 
			
		||||
					Class<?> targetClass = beanClass;
 | 
			
		||||
					do {
 | 
			
		||||
						ReflectionUtils.doWithLocalMethods(targetClass, method -> {
 | 
			
		||||
							Lookup lookup = method.getAnnotation(Lookup.class);
 | 
			
		||||
							if (lookup != null) {
 | 
			
		||||
								Assert.state(this.beanFactory != null, "No BeanFactory available");
 | 
			
		||||
								LookupOverride override = new LookupOverride(method, lookup.value());
 | 
			
		||||
								try {
 | 
			
		||||
									RootBeanDefinition mbd = (RootBeanDefinition) this.beanFactory.getMergedBeanDefinition(beanName);
 | 
			
		||||
									mbd.getMethodOverrides().addOverride(override);
 | 
			
		||||
								}
 | 
			
		||||
								catch (NoSuchBeanDefinitionException ex) {
 | 
			
		||||
									throw new BeanCreationException(beanName,
 | 
			
		||||
											"Cannot apply @Lookup to beans without corresponding bean definition");
 | 
			
		||||
								}
 | 
			
		||||
							}
 | 
			
		||||
						});
 | 
			
		||||
						targetClass = targetClass.getSuperclass();
 | 
			
		||||
					}
 | 
			
		||||
				});
 | 
			
		||||
			}
 | 
			
		||||
			catch (IllegalStateException ex) {
 | 
			
		||||
				throw new BeanCreationException(beanName, "Lookup method resolution failed", ex);
 | 
			
		||||
					while (targetClass != null && targetClass != Object.class);
 | 
			
		||||
 | 
			
		||||
				}
 | 
			
		||||
				catch (IllegalStateException ex) {
 | 
			
		||||
					throw new BeanCreationException(beanName, "Lookup method resolution failed", ex);
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			this.lookupMethodsChecked.add(beanName);
 | 
			
		||||
		}
 | 
			
		||||
| 
						 | 
				
			
			@ -433,6 +442,10 @@ public class AutowiredAnnotationBeanPostProcessor extends InstantiationAwareBean
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	private InjectionMetadata buildAutowiringMetadata(final Class<?> clazz) {
 | 
			
		||||
		if (!AnnotationUtils.isCandidateClass(clazz, this.autowiredAnnotationTypes)) {
 | 
			
		||||
			return new InjectionMetadata(clazz, Collections.emptyList());
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		List<InjectionMetadata.InjectedElement> elements = new ArrayList<>();
 | 
			
		||||
		Class<?> targetClass = clazz;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -24,7 +24,9 @@ import java.lang.reflect.InvocationTargetException;
 | 
			
		|||
import java.lang.reflect.Method;
 | 
			
		||||
import java.lang.reflect.Modifier;
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.Collection;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.LinkedHashSet;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
import java.util.Map;
 | 
			
		||||
| 
						 | 
				
			
			@ -41,6 +43,7 @@ import org.springframework.beans.factory.support.MergedBeanDefinitionPostProcess
 | 
			
		|||
import org.springframework.beans.factory.support.RootBeanDefinition;
 | 
			
		||||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.PriorityOrdered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.ClassUtils;
 | 
			
		||||
import org.springframework.util.ReflectionUtils;
 | 
			
		||||
| 
						 | 
				
			
			@ -196,6 +199,10 @@ public class InitDestroyAnnotationBeanPostProcessor
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	private LifecycleMetadata buildLifecycleMetadata(final Class<?> clazz) {
 | 
			
		||||
		if (!AnnotationUtils.isCandidateClass(clazz, Arrays.asList(this.initAnnotationType, this.destroyAnnotationType))) {
 | 
			
		||||
			return new LifecycleMetadata(clazz, Collections.emptyList(), Collections.emptyList());
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		List<LifecycleElement> initMethods = new ArrayList<>();
 | 
			
		||||
		List<LifecycleElement> destroyMethods = new ArrayList<>();
 | 
			
		||||
		Class<?> targetClass = clazz;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -103,6 +103,16 @@ public class AnnotationCacheOperationSource extends AbstractFallbackCacheOperati
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		for (CacheAnnotationParser parser : this.annotationParsers) {
 | 
			
		||||
			if (parser.isCandidateClass(targetClass)) {
 | 
			
		||||
				return true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	protected Collection<CacheOperation> findCacheOperations(Class<?> clazz) {
 | 
			
		||||
| 
						 | 
				
			
			@ -127,8 +137,8 @@ public class AnnotationCacheOperationSource extends AbstractFallbackCacheOperati
 | 
			
		|||
	@Nullable
 | 
			
		||||
	protected Collection<CacheOperation> determineCacheOperations(CacheOperationProvider provider) {
 | 
			
		||||
		Collection<CacheOperation> ops = null;
 | 
			
		||||
		for (CacheAnnotationParser annotationParser : this.annotationParsers) {
 | 
			
		||||
			Collection<CacheOperation> annOps = provider.getCacheOperations(annotationParser);
 | 
			
		||||
		for (CacheAnnotationParser parser : this.annotationParsers) {
 | 
			
		||||
			Collection<CacheOperation> annOps = provider.getCacheOperations(parser);
 | 
			
		||||
			if (annOps != null) {
 | 
			
		||||
				if (ops == null) {
 | 
			
		||||
					ops = annOps;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -30,12 +30,31 @@ import org.springframework.lang.Nullable;
 | 
			
		|||
 *
 | 
			
		||||
 * @author Costin Leau
 | 
			
		||||
 * @author Stephane Nicoll
 | 
			
		||||
 * @author Juergen Hoeller
 | 
			
		||||
 * @since 3.1
 | 
			
		||||
 * @see AnnotationCacheOperationSource
 | 
			
		||||
 * @see SpringCacheAnnotationParser
 | 
			
		||||
 */
 | 
			
		||||
public interface CacheAnnotationParser {
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for cache operations
 | 
			
		||||
	 * in the annotation format of this {@code CacheAnnotationParser}.
 | 
			
		||||
	 * <p>If this method returns {@code false}, the methods on the given class
 | 
			
		||||
	 * will not get traversed for {@code #parseCacheAnnotations} introspection.
 | 
			
		||||
	 * Returning {@code false} is therefore an optimization for non-affected
 | 
			
		||||
	 * classes, whereas {@code true} simply means that the class needs to get
 | 
			
		||||
	 * fully introspected for each method on the given class individually.
 | 
			
		||||
	 * @param targetClass the class to introspect
 | 
			
		||||
	 * @return {@code false} if the class is known to have no cache operation
 | 
			
		||||
	 * annotations at class or method level; {@code true} otherwise. The default
 | 
			
		||||
	 * implementation returns {@code true}, leading to regular introspection.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 */
 | 
			
		||||
	default boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return true;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Parse the cache definition for the given class,
 | 
			
		||||
	 * based on an annotation type understood by this parser.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -30,6 +30,7 @@ import org.springframework.cache.interceptor.CacheOperation;
 | 
			
		|||
import org.springframework.cache.interceptor.CachePutOperation;
 | 
			
		||||
import org.springframework.cache.interceptor.CacheableOperation;
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.StringUtils;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -58,6 +59,11 @@ public class SpringCacheAnnotationParser implements CacheAnnotationParser, Seria
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return AnnotationUtils.isCandidateClass(targetClass, CACHE_OPERATION_ANNOTATIONS);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public Collection<CacheOperation> parseCacheAnnotations(Class<?> type) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2012 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -27,10 +27,29 @@ import org.springframework.lang.Nullable;
 | 
			
		|||
 * source level, or elsewhere.
 | 
			
		||||
 *
 | 
			
		||||
 * @author Costin Leau
 | 
			
		||||
 * @author Juergen Hoeller
 | 
			
		||||
 * @since 3.1
 | 
			
		||||
 */
 | 
			
		||||
public interface CacheOperationSource {
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for cache operations
 | 
			
		||||
	 * in the metadata format of this {@code CacheOperationSource}.
 | 
			
		||||
	 * <p>If this method returns {@code false}, the methods on the given class
 | 
			
		||||
	 * will not get traversed for {@link #getCacheOperations} introspection.
 | 
			
		||||
	 * Returning {@code false} is therefore an optimization for non-affected
 | 
			
		||||
	 * classes, whereas {@code true} simply means that the class needs to get
 | 
			
		||||
	 * fully introspected for each method on the given class individually.
 | 
			
		||||
	 * @param targetClass the class to introspect
 | 
			
		||||
	 * @return {@code false} if the class is known to have no cache operation
 | 
			
		||||
	 * metadata at class or method level; {@code true} otherwise. The default
 | 
			
		||||
	 * implementation returns {@code true}, leading to regular introspection.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 */
 | 
			
		||||
	default boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return true;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Return the collection of cache operations for this method, or {@code null}
 | 
			
		||||
	 * if the method contains no <em>cacheable</em> annotations.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -19,6 +19,7 @@ package org.springframework.cache.interceptor;
 | 
			
		|||
import java.io.Serializable;
 | 
			
		||||
import java.lang.reflect.Method;
 | 
			
		||||
 | 
			
		||||
import org.springframework.aop.ClassFilter;
 | 
			
		||||
import org.springframework.aop.support.StaticMethodMatcherPointcut;
 | 
			
		||||
import org.springframework.cache.CacheManager;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
| 
						 | 
				
			
			@ -36,11 +37,13 @@ import org.springframework.util.ObjectUtils;
 | 
			
		|||
@SuppressWarnings("serial")
 | 
			
		||||
abstract class CacheOperationSourcePointcut extends StaticMethodMatcherPointcut implements Serializable {
 | 
			
		||||
 | 
			
		||||
	protected CacheOperationSourcePointcut() {
 | 
			
		||||
		setClassFilter(new CacheOperationSourceClassFilter());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean matches(Method method, Class<?> targetClass) {
 | 
			
		||||
		if (CacheManager.class.isAssignableFrom(targetClass)) {
 | 
			
		||||
			return false;
 | 
			
		||||
		}
 | 
			
		||||
		CacheOperationSource cas = getCacheOperationSource();
 | 
			
		||||
		return (cas != null && !CollectionUtils.isEmpty(cas.getCacheOperations(method, targetClass)));
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -75,4 +78,21 @@ abstract class CacheOperationSourcePointcut extends StaticMethodMatcherPointcut
 | 
			
		|||
	@Nullable
 | 
			
		||||
	protected abstract CacheOperationSource getCacheOperationSource();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * {@link ClassFilter} that delegates to {@link CacheOperationSource#isCandidateClass}
 | 
			
		||||
	 * for filtering classes whose methods are not worth searching to begin with.
 | 
			
		||||
	 */
 | 
			
		||||
	private class CacheOperationSourceClassFilter implements ClassFilter {
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public boolean matches(Class<?> clazz) {
 | 
			
		||||
			if (CacheManager.class.isAssignableFrom(clazz)) {
 | 
			
		||||
				return false;
 | 
			
		||||
			}
 | 
			
		||||
			CacheOperationSource cas = getCacheOperationSource();
 | 
			
		||||
			return (cas == null || cas.isCandidateClass(clazz));
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2016 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -29,6 +29,7 @@ import org.springframework.util.Assert;
 | 
			
		|||
 * over a given array of {@code CacheOperationSource} instances.
 | 
			
		||||
 *
 | 
			
		||||
 * @author Costin Leau
 | 
			
		||||
 * @author Juergen Hoeller
 | 
			
		||||
 * @since 3.1
 | 
			
		||||
 */
 | 
			
		||||
@SuppressWarnings("serial")
 | 
			
		||||
| 
						 | 
				
			
			@ -42,7 +43,7 @@ public class CompositeCacheOperationSource implements CacheOperationSource, Seri
 | 
			
		|||
	 * @param cacheOperationSources the CacheOperationSource instances to combine
 | 
			
		||||
	 */
 | 
			
		||||
	public CompositeCacheOperationSource(CacheOperationSource... cacheOperationSources) {
 | 
			
		||||
		Assert.notEmpty(cacheOperationSources, "cacheOperationSources array must not be empty");
 | 
			
		||||
		Assert.notEmpty(cacheOperationSources, "CacheOperationSource array must not be empty");
 | 
			
		||||
		this.cacheOperationSources = cacheOperationSources;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -54,18 +55,27 @@ public class CompositeCacheOperationSource implements CacheOperationSource, Seri
 | 
			
		|||
		return this.cacheOperationSources;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		for (CacheOperationSource source : this.cacheOperationSources) {
 | 
			
		||||
			if (source.isCandidateClass(targetClass)) {
 | 
			
		||||
				return true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public Collection<CacheOperation> getCacheOperations(Method method, @Nullable Class<?> targetClass) {
 | 
			
		||||
		Collection<CacheOperation> ops = null;
 | 
			
		||||
 | 
			
		||||
		for (CacheOperationSource source : this.cacheOperationSources) {
 | 
			
		||||
			Collection<CacheOperation> cacheOperations = source.getCacheOperations(method, targetClass);
 | 
			
		||||
			if (cacheOperations != null) {
 | 
			
		||||
				if (ops == null) {
 | 
			
		||||
					ops = new ArrayList<>();
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				ops.addAll(cacheOperations);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -64,6 +64,7 @@ import org.springframework.beans.factory.support.RootBeanDefinition;
 | 
			
		|||
import org.springframework.core.BridgeMethodResolver;
 | 
			
		||||
import org.springframework.core.MethodParameter;
 | 
			
		||||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.jndi.support.SimpleJndiBeanFactory;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
| 
						 | 
				
			
			@ -149,6 +150,8 @@ public class CommonAnnotationBeanPostProcessor extends InitDestroyAnnotationBean
 | 
			
		|||
	@Nullable
 | 
			
		||||
	private static Class<? extends Annotation> ejbRefClass;
 | 
			
		||||
 | 
			
		||||
	private static Set<Class<? extends Annotation>> resourceAnnotationTypes = new LinkedHashSet<>(4);
 | 
			
		||||
 | 
			
		||||
	static {
 | 
			
		||||
		try {
 | 
			
		||||
			@SuppressWarnings("unchecked")
 | 
			
		||||
| 
						 | 
				
			
			@ -159,6 +162,7 @@ public class CommonAnnotationBeanPostProcessor extends InitDestroyAnnotationBean
 | 
			
		|||
		catch (ClassNotFoundException ex) {
 | 
			
		||||
			webServiceRefClass = null;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		try {
 | 
			
		||||
			@SuppressWarnings("unchecked")
 | 
			
		||||
			Class<? extends Annotation> clazz = (Class<? extends Annotation>)
 | 
			
		||||
| 
						 | 
				
			
			@ -168,6 +172,14 @@ public class CommonAnnotationBeanPostProcessor extends InitDestroyAnnotationBean
 | 
			
		|||
		catch (ClassNotFoundException ex) {
 | 
			
		||||
			ejbRefClass = null;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		resourceAnnotationTypes.add(Resource.class);
 | 
			
		||||
		if (webServiceRefClass != null) {
 | 
			
		||||
			resourceAnnotationTypes.add(webServiceRefClass);
 | 
			
		||||
		}
 | 
			
		||||
		if (ejbRefClass != null) {
 | 
			
		||||
			resourceAnnotationTypes.add(ejbRefClass);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -356,6 +368,10 @@ public class CommonAnnotationBeanPostProcessor extends InitDestroyAnnotationBean
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	private InjectionMetadata buildResourceMetadata(final Class<?> clazz) {
 | 
			
		||||
		if (!AnnotationUtils.isCandidateClass(clazz, resourceAnnotationTypes)) {
 | 
			
		||||
			return new InjectionMetadata(clazz, Collections.emptyList());
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		List<InjectionMetadata.InjectedElement> elements = new ArrayList<>();
 | 
			
		||||
		Class<?> targetClass = clazz;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -24,9 +24,13 @@ import java.util.Set;
 | 
			
		|||
import org.apache.commons.logging.Log;
 | 
			
		||||
import org.apache.commons.logging.LogFactory;
 | 
			
		||||
 | 
			
		||||
import org.springframework.aop.framework.AopInfrastructureBean;
 | 
			
		||||
import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
 | 
			
		||||
import org.springframework.beans.factory.config.BeanDefinition;
 | 
			
		||||
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
 | 
			
		||||
import org.springframework.beans.factory.config.BeanPostProcessor;
 | 
			
		||||
import org.springframework.beans.factory.support.AbstractBeanDefinition;
 | 
			
		||||
import org.springframework.context.event.EventListenerFactory;
 | 
			
		||||
import org.springframework.core.Conventions;
 | 
			
		||||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
| 
						 | 
				
			
			@ -96,6 +100,12 @@ abstract class ConfigurationClassUtils {
 | 
			
		|||
			// Check already loaded Class if present...
 | 
			
		||||
			// since we possibly can't even load the class file for this Class.
 | 
			
		||||
			Class<?> beanClass = ((AbstractBeanDefinition) beanDef).getBeanClass();
 | 
			
		||||
			if (BeanFactoryPostProcessor.class.isAssignableFrom(beanClass) ||
 | 
			
		||||
					BeanPostProcessor.class.isAssignableFrom(beanClass) ||
 | 
			
		||||
					AopInfrastructureBean.class.isAssignableFrom(beanClass) ||
 | 
			
		||||
					EventListenerFactory.class.isAssignableFrom(beanClass)) {
 | 
			
		||||
				return false;
 | 
			
		||||
			}
 | 
			
		||||
			metadata = new StandardAnnotationMetadata(beanClass, true);
 | 
			
		||||
		}
 | 
			
		||||
		else {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -42,6 +42,7 @@ import org.springframework.context.ConfigurableApplicationContext;
 | 
			
		|||
import org.springframework.core.MethodIntrospector;
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.stereotype.Component;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
| 
						 | 
				
			
			@ -142,7 +143,10 @@ public class EventListenerMethodProcessor
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	private void processBean(final String beanName, final Class<?> targetType) {
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetType) && !isSpringContainerClass(targetType)) {
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetType) &&
 | 
			
		||||
				AnnotationUtils.isCandidateClass(targetType, EventListener.class) &&
 | 
			
		||||
				!isSpringContainerClass(targetType)) {
 | 
			
		||||
 | 
			
		||||
			Map<Method, EventListener> annotatedMethods = null;
 | 
			
		||||
			try {
 | 
			
		||||
				annotatedMethods = MethodIntrospector.selectMethods(targetType,
 | 
			
		||||
| 
						 | 
				
			
			@ -155,6 +159,7 @@ public class EventListenerMethodProcessor
 | 
			
		|||
					logger.debug("Could not resolve methods for bean with name '" + beanName + "'", ex);
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			if (CollectionUtils.isEmpty(annotatedMethods)) {
 | 
			
		||||
				this.nonAnnotatedClasses.add(targetType);
 | 
			
		||||
				if (logger.isTraceEnabled()) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -19,6 +19,7 @@ package org.springframework.scheduling.annotation;
 | 
			
		|||
import java.lang.reflect.Method;
 | 
			
		||||
import java.time.Duration;
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.Collection;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.IdentityHashMap;
 | 
			
		||||
| 
						 | 
				
			
			@ -59,6 +60,7 @@ import org.springframework.core.MethodIntrospector;
 | 
			
		|||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.scheduling.TaskScheduler;
 | 
			
		||||
import org.springframework.scheduling.Trigger;
 | 
			
		||||
| 
						 | 
				
			
			@ -340,7 +342,8 @@ public class ScheduledAnnotationBeanPostProcessor
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		Class<?> targetClass = AopProxyUtils.ultimateTargetClass(bean);
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetClass)) {
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetClass) &&
 | 
			
		||||
				AnnotationUtils.isCandidateClass(targetClass, Arrays.asList(Scheduled.class, Schedules.class))) {
 | 
			
		||||
			Map<Method, Set<Scheduled>> annotatedMethods = MethodIntrospector.selectMethods(targetClass,
 | 
			
		||||
					(MethodIntrospector.MetadataLookup<Set<Scheduled>>) method -> {
 | 
			
		||||
						Set<Scheduled> scheduledMethods = AnnotatedElementUtils.getMergedRepeatableAnnotations(
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -28,6 +28,7 @@ import java.lang.reflect.Modifier;
 | 
			
		|||
import java.lang.reflect.Proxy;
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.Collection;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.HashSet;
 | 
			
		||||
import java.util.LinkedHashMap;
 | 
			
		||||
| 
						 | 
				
			
			@ -151,6 +152,55 @@ public abstract class AnnotationUtils {
 | 
			
		|||
	private static transient Log logger;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for carrying one of the specified
 | 
			
		||||
	 * annotations (at type, method or field level).
 | 
			
		||||
	 * @param clazz the class to introspect
 | 
			
		||||
	 * @param annotationTypes the searchable annotation types
 | 
			
		||||
	 * @return {@code false} if the class is known to have no such annotations at any level;
 | 
			
		||||
	 * {@code true} otherwise. Callers will usually perform full method/field introspection
 | 
			
		||||
	 * if {@code true} is being returned here.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 * @see #isCandidateClass(Class, Class)
 | 
			
		||||
	 */
 | 
			
		||||
	public static boolean isCandidateClass(Class<?> clazz, Collection<Class<? extends Annotation>> annotationTypes) {
 | 
			
		||||
		for (Class<? extends Annotation> annotationType : annotationTypes) {
 | 
			
		||||
			if (isCandidateClass(clazz, annotationType)) {
 | 
			
		||||
				return true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for carrying the specified annotation
 | 
			
		||||
	 * (at type, method or field level).
 | 
			
		||||
	 * @param clazz the class to introspect
 | 
			
		||||
	 * @param annotationType the searchable annotation type
 | 
			
		||||
	 * @return {@code false} if the class is known to have no such annotations at any level;
 | 
			
		||||
	 * {@code true} otherwise. Callers will usually perform full method/field introspection
 | 
			
		||||
	 * if {@code true} is being returned here.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 * @see #isCandidateClass(Class, String)
 | 
			
		||||
	 */
 | 
			
		||||
	public static boolean isCandidateClass(Class<?> clazz, Class<? extends Annotation> annotationType) {
 | 
			
		||||
		return isCandidateClass(clazz, annotationType.getName());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for carrying the specified annotation
 | 
			
		||||
	 * (at type, method or field level).
 | 
			
		||||
	 * @param clazz the class to introspect
 | 
			
		||||
	 * @param annotationName the fully-qualified name of the searchable annotation type
 | 
			
		||||
	 * @return {@code false} if the class is known to have no such annotations at any level;
 | 
			
		||||
	 * {@code true} otherwise. Callers will usually perform full method/field introspection
 | 
			
		||||
	 * if {@code true} is being returned here.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 */
 | 
			
		||||
	public static boolean isCandidateClass(Class<?> clazz, String annotationName) {
 | 
			
		||||
		return !clazz.getName().startsWith("java");
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Get a single {@link Annotation} of {@code annotationType} from the supplied
 | 
			
		||||
	 * annotation: either the given annotation itself or a direct meta-annotation
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2017 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -24,6 +24,7 @@ import java.util.Map;
 | 
			
		|||
import java.util.Set;
 | 
			
		||||
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.MultiValueMap;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -137,37 +138,41 @@ public class StandardAnnotationMetadata extends StandardClassMetadata implements
 | 
			
		|||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean hasAnnotatedMethods(String annotationName) {
 | 
			
		||||
		try {
 | 
			
		||||
			Method[] methods = getIntrospectedClass().getDeclaredMethods();
 | 
			
		||||
			for (Method method : methods) {
 | 
			
		||||
				if (!method.isBridge() && method.getAnnotations().length > 0 &&
 | 
			
		||||
						AnnotatedElementUtils.isAnnotated(method, annotationName)) {
 | 
			
		||||
					return true;
 | 
			
		||||
		if (AnnotationUtils.isCandidateClass(getIntrospectedClass(), annotationName)) {
 | 
			
		||||
			try {
 | 
			
		||||
				Method[] methods = getIntrospectedClass().getDeclaredMethods();
 | 
			
		||||
				for (Method method : methods) {
 | 
			
		||||
					if (!method.isBridge() && method.getAnnotations().length > 0 &&
 | 
			
		||||
							AnnotatedElementUtils.isAnnotated(method, annotationName)) {
 | 
			
		||||
						return true;
 | 
			
		||||
					}
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			return false;
 | 
			
		||||
		}
 | 
			
		||||
		catch (Throwable ex) {
 | 
			
		||||
			throw new IllegalStateException("Failed to introspect annotated methods on " + getIntrospectedClass(), ex);
 | 
			
		||||
			catch (Throwable ex) {
 | 
			
		||||
				throw new IllegalStateException("Failed to introspect annotated methods on " + getIntrospectedClass(), ex);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public Set<MethodMetadata> getAnnotatedMethods(String annotationName) {
 | 
			
		||||
		try {
 | 
			
		||||
			Method[] methods = getIntrospectedClass().getDeclaredMethods();
 | 
			
		||||
			Set<MethodMetadata> annotatedMethods = new LinkedHashSet<>(4);
 | 
			
		||||
			for (Method method : methods) {
 | 
			
		||||
				if (!method.isBridge() && method.getAnnotations().length > 0 &&
 | 
			
		||||
						AnnotatedElementUtils.isAnnotated(method, annotationName)) {
 | 
			
		||||
					annotatedMethods.add(new StandardMethodMetadata(method, this.nestedAnnotationsAsMap));
 | 
			
		||||
		Set<MethodMetadata> annotatedMethods = new LinkedHashSet<>(4);
 | 
			
		||||
		if (AnnotationUtils.isCandidateClass(getIntrospectedClass(), annotationName)) {
 | 
			
		||||
			try {
 | 
			
		||||
				Method[] methods = getIntrospectedClass().getDeclaredMethods();
 | 
			
		||||
				for (Method method : methods) {
 | 
			
		||||
					if (!method.isBridge() && method.getAnnotations().length > 0 &&
 | 
			
		||||
							AnnotatedElementUtils.isAnnotated(method, annotationName)) {
 | 
			
		||||
						annotatedMethods.add(new StandardMethodMetadata(method, this.nestedAnnotationsAsMap));
 | 
			
		||||
					}
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			return annotatedMethods;
 | 
			
		||||
		}
 | 
			
		||||
		catch (Throwable ex) {
 | 
			
		||||
			throw new IllegalStateException("Failed to introspect annotated methods on " + getIntrospectedClass(), ex);
 | 
			
		||||
			catch (Throwable ex) {
 | 
			
		||||
				throw new IllegalStateException("Failed to introspect annotated methods on " + getIntrospectedClass(), ex);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return annotatedMethods;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -46,6 +46,7 @@ import org.springframework.core.MethodIntrospector;
 | 
			
		|||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotatedElementUtils;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.jms.config.JmsListenerConfigUtils;
 | 
			
		||||
import org.springframework.jms.config.JmsListenerContainerFactory;
 | 
			
		||||
import org.springframework.jms.config.JmsListenerEndpointRegistrar;
 | 
			
		||||
| 
						 | 
				
			
			@ -228,7 +229,8 @@ public class JmsListenerAnnotationBeanPostProcessor
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		Class<?> targetClass = AopProxyUtils.ultimateTargetClass(bean);
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetClass)) {
 | 
			
		||||
		if (!this.nonAnnotatedClasses.contains(targetClass) &&
 | 
			
		||||
				AnnotationUtils.isCandidateClass(targetClass, JmsListener.class)) {
 | 
			
		||||
			Map<Method, Set<JmsListener>> annotatedMethods = MethodIntrospector.selectMethods(targetClass,
 | 
			
		||||
					(MethodIntrospector.MetadataLookup<Set<JmsListener>>) method -> {
 | 
			
		||||
						Set<JmsListener> listenerMethods = AnnotatedElementUtils.getMergedRepeatableAnnotations(
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -23,6 +23,8 @@ import java.lang.reflect.Member;
 | 
			
		|||
import java.lang.reflect.Method;
 | 
			
		||||
import java.lang.reflect.Modifier;
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.LinkedList;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
import java.util.Map;
 | 
			
		||||
| 
						 | 
				
			
			@ -54,6 +56,7 @@ import org.springframework.beans.factory.support.RootBeanDefinition;
 | 
			
		|||
import org.springframework.core.BridgeMethodResolver;
 | 
			
		||||
import org.springframework.core.Ordered;
 | 
			
		||||
import org.springframework.core.PriorityOrdered;
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.jndi.JndiLocatorDelegate;
 | 
			
		||||
import org.springframework.jndi.JndiTemplate;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
| 
						 | 
				
			
			@ -412,12 +415,15 @@ public class PersistenceAnnotationBeanPostProcessor
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	private InjectionMetadata buildPersistenceMetadata(final Class<?> clazz) {
 | 
			
		||||
		if (!AnnotationUtils.isCandidateClass(clazz, Arrays.asList(PersistenceContext.class, PersistenceUnit.class))) {
 | 
			
		||||
			return new InjectionMetadata(clazz, Collections.emptyList());
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		List<InjectionMetadata.InjectedElement> elements = new ArrayList<>();
 | 
			
		||||
		Class<?> targetClass = clazz;
 | 
			
		||||
 | 
			
		||||
		do {
 | 
			
		||||
			final LinkedList<InjectionMetadata.InjectedElement> currElements =
 | 
			
		||||
					new LinkedList<>();
 | 
			
		||||
			final LinkedList<InjectionMetadata.InjectedElement> currElements = new LinkedList<>();
 | 
			
		||||
 | 
			
		||||
			ReflectionUtils.doWithLocalFields(targetClass, field -> {
 | 
			
		||||
				if (field.isAnnotationPresent(PersistenceContext.class) ||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -137,6 +137,16 @@ public class AnnotationTransactionAttributeSource extends AbstractFallbackTransa
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		for (TransactionAnnotationParser parser : this.annotationParsers) {
 | 
			
		||||
			if (parser.isCandidateClass(targetClass)) {
 | 
			
		||||
				return true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	protected TransactionAttribute findTransactionAttribute(Class<?> clazz) {
 | 
			
		||||
| 
						 | 
				
			
			@ -161,8 +171,8 @@ public class AnnotationTransactionAttributeSource extends AbstractFallbackTransa
 | 
			
		|||
	 */
 | 
			
		||||
	@Nullable
 | 
			
		||||
	protected TransactionAttribute determineTransactionAttribute(AnnotatedElement element) {
 | 
			
		||||
		for (TransactionAnnotationParser annotationParser : this.annotationParsers) {
 | 
			
		||||
			TransactionAttribute attr = annotationParser.parseTransactionAnnotation(element);
 | 
			
		||||
		for (TransactionAnnotationParser parser : this.annotationParsers) {
 | 
			
		||||
			TransactionAttribute attr = parser.parseTransactionAnnotation(element);
 | 
			
		||||
			if (attr != null) {
 | 
			
		||||
				return attr;
 | 
			
		||||
			}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -21,6 +21,7 @@ import java.lang.reflect.AnnotatedElement;
 | 
			
		|||
import javax.ejb.ApplicationException;
 | 
			
		||||
import javax.ejb.TransactionAttributeType;
 | 
			
		||||
 | 
			
		||||
import org.springframework.core.annotation.AnnotationUtils;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.transaction.interceptor.DefaultTransactionAttribute;
 | 
			
		||||
import org.springframework.transaction.interceptor.TransactionAttribute;
 | 
			
		||||
| 
						 | 
				
			
			@ -35,6 +36,11 @@ import org.springframework.transaction.interceptor.TransactionAttribute;
 | 
			
		|||
@SuppressWarnings("serial")
 | 
			
		||||
public class Ejb3TransactionAnnotationParser implements TransactionAnnotationParser, Serializable {
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return AnnotationUtils.isCandidateClass(targetClass, javax.ejb.TransactionAttribute.class);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public TransactionAttribute parseTransactionAnnotation(AnnotatedElement element) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -39,6 +39,11 @@ import org.springframework.transaction.interceptor.TransactionAttribute;
 | 
			
		|||
@SuppressWarnings("serial")
 | 
			
		||||
public class JtaTransactionAnnotationParser implements TransactionAnnotationParser, Serializable {
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return AnnotationUtils.isCandidateClass(targetClass, javax.transaction.Transactional.class);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public TransactionAttribute parseTransactionAnnotation(AnnotatedElement element) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -39,6 +39,11 @@ import org.springframework.transaction.interceptor.TransactionAttribute;
 | 
			
		|||
@SuppressWarnings("serial")
 | 
			
		||||
public class SpringTransactionAnnotationParser implements TransactionAnnotationParser, Serializable {
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return AnnotationUtils.isCandidateClass(targetClass, Transactional.class);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public TransactionAttribute parseTransactionAnnotation(AnnotatedElement element) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -37,6 +37,24 @@ import org.springframework.transaction.interceptor.TransactionAttribute;
 | 
			
		|||
 */
 | 
			
		||||
public interface TransactionAnnotationParser {
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for transaction attributes
 | 
			
		||||
	 * in the annotation format of this {@code TransactionAnnotationParser}.
 | 
			
		||||
	 * <p>If this method returns {@code false}, the methods on the given class
 | 
			
		||||
	 * will not get traversed for {@code #parseTransactionAnnotation} introspection.
 | 
			
		||||
	 * Returning {@code false} is therefore an optimization for non-affected
 | 
			
		||||
	 * classes, whereas {@code true} simply means that the class needs to get
 | 
			
		||||
	 * fully introspected for each method on the given class individually.
 | 
			
		||||
	 * @param targetClass the class to introspect
 | 
			
		||||
	 * @return {@code false} if the class is known to have no transaction
 | 
			
		||||
	 * annotations at class or method level; {@code true} otherwise. The default
 | 
			
		||||
	 * implementation returns {@code true}, leading to regular introspection.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 */
 | 
			
		||||
	default boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return true;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Parse the transaction attribute for the given method or class,
 | 
			
		||||
	 * based on an annotation type understood by this parser.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2012 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -39,7 +39,7 @@ public class CompositeTransactionAttributeSource implements TransactionAttribute
 | 
			
		|||
	 * Create a new CompositeTransactionAttributeSource for the given sources.
 | 
			
		||||
	 * @param transactionAttributeSources the TransactionAttributeSource instances to combine
 | 
			
		||||
	 */
 | 
			
		||||
	public CompositeTransactionAttributeSource(TransactionAttributeSource[] transactionAttributeSources) {
 | 
			
		||||
	public CompositeTransactionAttributeSource(TransactionAttributeSource... transactionAttributeSources) {
 | 
			
		||||
		Assert.notNull(transactionAttributeSources, "TransactionAttributeSource array must not be null");
 | 
			
		||||
		this.transactionAttributeSources = transactionAttributeSources;
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -53,13 +53,23 @@ public class CompositeTransactionAttributeSource implements TransactionAttribute
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		for (TransactionAttributeSource source : this.transactionAttributeSources) {
 | 
			
		||||
			if (source.isCandidateClass(targetClass)) {
 | 
			
		||||
				return true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	@Nullable
 | 
			
		||||
	public TransactionAttribute getTransactionAttribute(Method method, @Nullable Class<?> targetClass) {
 | 
			
		||||
		for (TransactionAttributeSource tas : this.transactionAttributeSources) {
 | 
			
		||||
			TransactionAttribute ta = tas.getTransactionAttribute(method, targetClass);
 | 
			
		||||
			if (ta != null) {
 | 
			
		||||
				return ta;
 | 
			
		||||
		for (TransactionAttributeSource source : this.transactionAttributeSources) {
 | 
			
		||||
			TransactionAttribute attr = source.getTransactionAttribute(method, targetClass);
 | 
			
		||||
			if (attr != null) {
 | 
			
		||||
				return attr;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return null;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,14 +34,31 @@ import org.springframework.lang.Nullable;
 | 
			
		|||
 */
 | 
			
		||||
public interface TransactionAttributeSource {
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Determine whether the given class is a candidate for transaction attributes
 | 
			
		||||
	 * in the metadata format of this {@code TransactionAttributeSource}.
 | 
			
		||||
	 * <p>If this method returns {@code false}, the methods on the given class
 | 
			
		||||
	 * will not get traversed for {@link #getTransactionAttribute} introspection.
 | 
			
		||||
	 * Returning {@code false} is therefore an optimization for non-affected
 | 
			
		||||
	 * classes, whereas {@code true} simply means that the class needs to get
 | 
			
		||||
	 * fully introspected for each method on the given class individually.
 | 
			
		||||
	 * @param targetClass the class to introspect
 | 
			
		||||
	 * @return {@code false} if the class is known to have no transaction
 | 
			
		||||
	 * attributes at class or method level; {@code true} otherwise. The default
 | 
			
		||||
	 * implementation returns {@code true}, leading to regular introspection.
 | 
			
		||||
	 * @since 5.2
 | 
			
		||||
	 */
 | 
			
		||||
	default boolean isCandidateClass(Class<?> targetClass) {
 | 
			
		||||
		return true;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Return the transaction attribute for the given method,
 | 
			
		||||
	 * or {@code null} if the method is non-transactional.
 | 
			
		||||
	 * @param method the method to introspect
 | 
			
		||||
	 * @param targetClass the target class (may be {@code null},
 | 
			
		||||
	 * in which case the declaring class of the method must be used)
 | 
			
		||||
	 * @return the TransactionAttribute the matching transaction attribute,
 | 
			
		||||
	 * or {@code null} if none found
 | 
			
		||||
	 * @return the matching transaction attribute, or {@code null} if none found
 | 
			
		||||
	 */
 | 
			
		||||
	@Nullable
 | 
			
		||||
	TransactionAttribute getTransactionAttribute(Method method, @Nullable Class<?> targetClass);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2019 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -19,6 +19,7 @@ package org.springframework.transaction.interceptor;
 | 
			
		|||
import java.io.Serializable;
 | 
			
		||||
import java.lang.reflect.Method;
 | 
			
		||||
 | 
			
		||||
import org.springframework.aop.ClassFilter;
 | 
			
		||||
import org.springframework.aop.support.StaticMethodMatcherPointcut;
 | 
			
		||||
import org.springframework.dao.support.PersistenceExceptionTranslator;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
| 
						 | 
				
			
			@ -35,13 +36,13 @@ import org.springframework.util.ObjectUtils;
 | 
			
		|||
@SuppressWarnings("serial")
 | 
			
		||||
abstract class TransactionAttributeSourcePointcut extends StaticMethodMatcherPointcut implements Serializable {
 | 
			
		||||
 | 
			
		||||
	protected TransactionAttributeSourcePointcut() {
 | 
			
		||||
		setClassFilter(new TransactionAttributeSourceClassFilter());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public boolean matches(Method method, Class<?> targetClass) {
 | 
			
		||||
		if (TransactionalProxy.class.isAssignableFrom(targetClass) ||
 | 
			
		||||
				PlatformTransactionManager.class.isAssignableFrom(targetClass) ||
 | 
			
		||||
				PersistenceExceptionTranslator.class.isAssignableFrom(targetClass)) {
 | 
			
		||||
			return false;
 | 
			
		||||
		}
 | 
			
		||||
		TransactionAttributeSource tas = getTransactionAttributeSource();
 | 
			
		||||
		return (tas == null || tas.getTransactionAttribute(method, targetClass) != null);
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -76,4 +77,23 @@ abstract class TransactionAttributeSourcePointcut extends StaticMethodMatcherPoi
 | 
			
		|||
	@Nullable
 | 
			
		||||
	protected abstract TransactionAttributeSource getTransactionAttributeSource();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * {@link ClassFilter} that delegates to {@link TransactionAttributeSource#isCandidateClass}
 | 
			
		||||
	 * for filtering classes whose methods are not worth searching to begin with.
 | 
			
		||||
	 */
 | 
			
		||||
	private class TransactionAttributeSourceClassFilter implements ClassFilter {
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public boolean matches(Class<?> clazz) {
 | 
			
		||||
			if (TransactionalProxy.class.isAssignableFrom(clazz) ||
 | 
			
		||||
					PlatformTransactionManager.class.isAssignableFrom(clazz) ||
 | 
			
		||||
					PersistenceExceptionTranslator.class.isAssignableFrom(clazz)) {
 | 
			
		||||
				return false;
 | 
			
		||||
			}
 | 
			
		||||
			TransactionAttributeSource tas = getTransactionAttributeSource();
 | 
			
		||||
			return (tas == null || tas.isCandidateClass(clazz));
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue