All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.core.type.classreading.AnnotationReadingVisitorUtils Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2002-2015 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.core.type.classreading;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.springframework.asm.Type;
import org.springframework.core.annotation.AnnotationAttributes;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.ObjectUtils;

/**
 * Internal utility class used when reading annotations via ASM.
 *
 * @author Juergen Hoeller
 * @author Mark Fisher
 * @author Costin Leau
 * @author Phillip Webb
 * @author Sam Brannen
 * @since 4.0
 */
abstract class AnnotationReadingVisitorUtils {

	public static AnnotationAttributes convertClassValues(ClassLoader classLoader, AnnotationAttributes original,
			boolean classValuesAsString) {

		if (original == null) {
			return null;
		}

		AnnotationAttributes result = new AnnotationAttributes(original.size());
		for (Map.Entry entry : original.entrySet()) {
			try {
				Object value = entry.getValue();
				if (value instanceof AnnotationAttributes) {
					value = convertClassValues(classLoader, (AnnotationAttributes) value, classValuesAsString);
				}
				else if (value instanceof AnnotationAttributes[]) {
					AnnotationAttributes[] values = (AnnotationAttributes[]) value;
					for (int i = 0; i < values.length; i++) {
						values[i] = convertClassValues(classLoader, values[i], classValuesAsString);
					}
				}
				else if (value instanceof Type) {
					value = (classValuesAsString ? ((Type) value).getClassName() :
							classLoader.loadClass(((Type) value).getClassName()));
				}
				else if (value instanceof Type[]) {
					Type[] array = (Type[]) value;
					Object[] convArray = (classValuesAsString ? new String[array.length] : new Class[array.length]);
					for (int i = 0; i < array.length; i++) {
						convArray[i] = (classValuesAsString ? array[i].getClassName() :
								classLoader.loadClass(array[i].getClassName()));
					}
					value = convArray;
				}
				else if (classValuesAsString) {
					if (value instanceof Class) {
						value = ((Class) value).getName();
					}
					else if (value instanceof Class[]) {
						Class[] clazzArray = (Class[]) value;
						String[] newValue = new String[clazzArray.length];
						for (int i = 0; i < clazzArray.length; i++) {
							newValue[i] = clazzArray[i].getName();
						}
						value = newValue;
					}
				}
				result.put(entry.getKey(), value);
			}
			catch (Exception ex) {
				// Class not found - can't resolve class reference in annotation attribute.
				result.put(entry.getKey(), ex);
			}
		}
		return result;
	}

	/**
	 * Retrieve the merged attributes of the annotation of the given type,
	 * if any, from the supplied {@code attributesMap}.
	 * 

Annotation attribute values appearing lower in the annotation * hierarchy (i.e., closer to the declaring class) will override those * defined higher in the annotation hierarchy. * @param attributesMap the map of annotation attribute lists, keyed by * annotation type name * @param metaAnnotationMap the map of meta annotation relationships, * keyed by annotation type name * @param annotationName the fully qualified class name of the annotation * type to look for * @return the merged annotation attributes, or {@code null} if no * matching annotation is present in the {@code attributesMap} * @since 4.0.3 */ public static AnnotationAttributes getMergedAnnotationAttributes( LinkedMultiValueMap attributesMap, Map> metaAnnotationMap, String annotationName) { // Get the unmerged list of attributes for the target annotation. List attributesList = attributesMap.get(annotationName); if (attributesList == null || attributesList.isEmpty()) { return null; } // To start with, we populate the results with a copy of all attribute // values from the target annotation. A copy is necessary so that we do // not inadvertently mutate the state of the metadata passed to this // method. AnnotationAttributes results = new AnnotationAttributes(attributesList.get(0)); Set overridableAttributeNames = new HashSet(results.keySet()); overridableAttributeNames.remove(AnnotationUtils.VALUE); // Since the map is a LinkedMultiValueMap, we depend on the ordering of // elements in the map and reverse the order of the keys in order to traverse // "down" the annotation hierarchy. List annotationTypes = new ArrayList(attributesMap.keySet()); Collections.reverse(annotationTypes); // No need to revisit the target annotation type: annotationTypes.remove(annotationName); for (String currentAnnotationType : annotationTypes) { List currentAttributesList = attributesMap.get(currentAnnotationType); if (!ObjectUtils.isEmpty(currentAttributesList)) { Set metaAnns = metaAnnotationMap.get(currentAnnotationType); if (metaAnns != null && metaAnns.contains(annotationName)) { AnnotationAttributes currentAttributes = currentAttributesList.get(0); for (String overridableAttributeName : overridableAttributeNames) { Object value = currentAttributes.get(overridableAttributeName); if (value != null) { // Store the value, potentially overriding a value from an // attribute of the same name found higher in the annotation // hierarchy. results.put(overridableAttributeName, value); } } } } } return results; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy