blob: 1e52235e6c7a4c981115ef40e0c166ac0dc1bdae [file] [log] [blame]
/**
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.xbean.recipe;
import java.lang.annotation.Annotation;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.EnumSet;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import static org.apache.xbean.recipe.RecipeHelper.isAssignableFrom;
import org.apache.xbean.propertyeditor.PropertyEditorRegistry;
public final class ReflectionUtil {
private static final class ParameterLoader {
private static ParameterNameLoader PARAMETER_NAME_LOADER;
static {
if (isClassAvailable("org.apache.xbean.asm7.ClassReader")) {
PARAMETER_NAME_LOADER = new XbeanAsmParameterNameLoader();
} else if (isClassAvailable("org.objectweb.asm.ClassReader")) {
PARAMETER_NAME_LOADER = new AsmParameterNameLoader();
} else if (isClassAvailable("org.apache.xbean.asm.ClassReader")
|| isClassAvailable("org.apache.xbean.asm4.ClassReader")
|| isClassAvailable("org.apache.xbean.asm5.ClassReader")
|| isClassAvailable("org.apache.xbean.asm6.ClassReader")) {
throw new RuntimeException("Your xbean-asm-shade is too old, please upgrade to xbean-asm7-shade");
}
}
}
private ReflectionUtil() {
}
private static boolean isClassAvailable(String className) {
try {
ReflectionUtil.class.getClassLoader().loadClass(className);
return true;
} catch (Throwable ignored) {
return false;
}
}
public static Field findField(Class typeClass, String propertyName, Object propertyValue, Set<Option> options,
PropertyEditorRegistry registry) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (propertyName == null) throw new NullPointerException("name is null");
if (propertyName.length() == 0) throw new IllegalArgumentException("name is an empty string");
if (options == null) options = EnumSet.noneOf(Option.class);
int matchLevel = 0;
MissingAccessorException missException = null;
if (propertyName.contains("/")){
String[] strings = propertyName.split("/");
if (strings == null || strings.length != 2) throw new IllegalArgumentException("badly formed <class>/<attribute> property name: " + propertyName);
String className = strings[0];
propertyName = strings[1];
boolean found = false;
while(!typeClass.equals(Object.class) && !found){
if (typeClass.getName().equals(className)){
found = true;
break;
} else {
typeClass = typeClass.getSuperclass();
}
}
if (!found) throw new MissingAccessorException("Type not assignable to class: " + className, -1);
}
List<Field> fields = new ArrayList<Field>(Arrays.asList(typeClass.getDeclaredFields()));
Class parent = typeClass.getSuperclass();
while (parent != null){
fields.addAll(Arrays.asList(parent.getDeclaredFields()));
parent = parent.getSuperclass();
}
boolean allowPrivate = options.contains(Option.PRIVATE_PROPERTIES);
boolean allowStatic = options.contains(Option.STATIC_PROPERTIES);
boolean caseInsesnitive = options.contains(Option.CASE_INSENSITIVE_PROPERTIES);
for (Field field : fields) {
if (field.getName().equals(propertyName) || (caseInsesnitive && field.getName().equalsIgnoreCase(propertyName))) {
if (!allowPrivate && !Modifier.isPublic(field.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Field is not public: " + field, matchLevel);
}
continue;
}
if (!allowStatic && Modifier.isStatic(field.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Field is static: " + field, matchLevel);
}
continue;
}
Class fieldType = field.getType();
if (fieldType.isPrimitive() && propertyValue == null) {
if (matchLevel < 6) {
matchLevel = 6;
missException = new MissingAccessorException("Null can not be assigned to " +
fieldType.getName() + ": " + field, matchLevel);
}
continue;
}
if (!RecipeHelper.isInstance(fieldType, propertyValue) && !RecipeHelper.isConvertable(fieldType, propertyValue, registry)) {
if (matchLevel < 5) {
matchLevel = 5;
missException = new MissingAccessorException((propertyValue == null ? "null" : propertyValue.getClass().getName()) + " can not be assigned or converted to " +
fieldType.getName() + ": " + field, matchLevel);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(field.getModifiers())) {
setAccessible(field);
}
return field;
}
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid field: ");
buffer.append("public ").append(" ").append(propertyValue == null ? "null" : propertyValue.getClass().getName());
buffer.append(" ").append(propertyName).append(";");
throw new MissingAccessorException(buffer.toString(), -1);
}
}
public static Method findGetter(Class typeClass, String propertyName, Set<Option> options) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (propertyName == null) throw new NullPointerException("name is null");
if (propertyName.length() == 0) throw new IllegalArgumentException("name is an empty string");
if (options == null) options = EnumSet.noneOf(Option.class);
if (propertyName.contains("/")){
String[] strings = propertyName.split("/");
if (strings == null || strings.length != 2) throw new IllegalArgumentException("badly formed <class>/<attribute> property name: " + propertyName);
String className = strings[0];
propertyName = strings[1];
boolean found = false;
while(!typeClass.equals(Object.class) && !found){
if (typeClass.getName().equals(className)){
found = true;
break;
} else {
typeClass = typeClass.getSuperclass();
}
}
if (!found) throw new MissingAccessorException("Type not assignable to class: " + className, -1);
}
String getterName = "get" + Character.toUpperCase(propertyName.charAt(0));
if (propertyName.length() > 0) {
getterName += propertyName.substring(1);
}
boolean allowPrivate = options.contains(Option.PRIVATE_PROPERTIES);
boolean allowStatic = options.contains(Option.STATIC_PROPERTIES);
boolean caseInsesnitive = options.contains(Option.CASE_INSENSITIVE_PROPERTIES);
List<Method> methods = new ArrayList<Method>(Arrays.asList(typeClass.getMethods()));
methods.addAll(Arrays.asList(typeClass.getDeclaredMethods()));
for (Method method : methods) {
if (method.getName().equals(getterName) || (caseInsesnitive && method.getName().equalsIgnoreCase(getterName))) {
if (method.getParameterTypes().length > 0) {
continue;
}
if (method.getReturnType() == Void.TYPE) {
continue;
}
if (Modifier.isAbstract(method.getModifiers())) {
continue;
}
if (!allowPrivate && !Modifier.isPublic(method.getModifiers())) {
continue;
}
if (!allowStatic && Modifier.isStatic(method.getModifiers())) {
continue;
}
if (allowPrivate && !Modifier.isPublic(method.getModifiers())) {
setAccessible(method);
}
return method;
}
}
return null;
}
public static Method findSetter(Class typeClass, String propertyName, Object propertyValue, Set<Option> options,
PropertyEditorRegistry registry) {
List<Method> setters = findAllSetters(typeClass, propertyName, propertyValue, options, registry);
return setters.get(0);
}
/**
* Finds all valid setters for the property. Due to automatic type conversion there may be more than one possible
* setter that could be used to set the property. The setters that do not require type converstion will be a the
* head of the returned list of setters.
* @param typeClass the class to search for setters
* @param propertyName the name of the property
* @param propertyValue the value that must be settable either directly or after conversion
* @param options controls which setters are considered valid
* @return the valid setters; never null or empty
*/
public static List<Method> findAllSetters(Class typeClass, String propertyName, Object propertyValue, Set<Option> options,
PropertyEditorRegistry registry) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (propertyName == null) throw new NullPointerException("name is null");
if (propertyName.length() == 0) throw new IllegalArgumentException("name is an empty string");
if (options == null) options = EnumSet.noneOf(Option.class);
if (propertyName.contains("/")){
String[] strings = propertyName.split("/");
if (strings == null || strings.length != 2) throw new IllegalArgumentException("badly formed <class>/<attribute> property name: " + propertyName);
String className = strings[0];
propertyName = strings[1];
boolean found = false;
while(!typeClass.equals(Object.class) && !found){
if (typeClass.getName().equals(className)){
found = true;
break;
} else {
typeClass = typeClass.getSuperclass();
}
}
if (!found) throw new MissingAccessorException("Type not assignable to class: " + className, -1);
}
String setterName = "set" + Character.toUpperCase(propertyName.charAt(0));
if (propertyName.length() > 0) {
setterName += propertyName.substring(1);
}
int matchLevel = 0;
MissingAccessorException missException = null;
boolean allowPrivate = options.contains(Option.PRIVATE_PROPERTIES);
boolean allowStatic = options.contains(Option.STATIC_PROPERTIES);
boolean caseInsesnitive = options.contains(Option.CASE_INSENSITIVE_PROPERTIES);
LinkedList<Method> validSetters = new LinkedList<Method>();
List<Method> methods = new ArrayList<Method>(Arrays.asList(typeClass.getMethods()));
methods.addAll(Arrays.asList(typeClass.getDeclaredMethods()));
for (Method method : methods) {
if (method.getName().equals(setterName) || (caseInsesnitive && method.getName().equalsIgnoreCase(setterName))) {
if (method.getParameterTypes().length == 0) {
if (matchLevel < 1) {
matchLevel = 1;
missException = new MissingAccessorException("Setter takes no parameters: " + method, matchLevel);
}
continue;
}
if (method.getParameterTypes().length > 1) {
if (matchLevel < 1) {
matchLevel = 1;
missException = new MissingAccessorException("Setter takes more then one parameter: " + method, matchLevel);
}
continue;
}
if (method.getReturnType() != Void.TYPE) {
if (matchLevel < 2) {
matchLevel = 2;
missException = new MissingAccessorException("Setter returns a value: " + method, matchLevel);
}
continue;
}
if (Modifier.isAbstract(method.getModifiers())) {
if (matchLevel < 3) {
matchLevel = 3;
missException = new MissingAccessorException("Setter is abstract: " + method, matchLevel);
}
continue;
}
if (!allowPrivate && !Modifier.isPublic(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Setter is not public: " + method, matchLevel);
}
continue;
}
if (!allowStatic && Modifier.isStatic(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Setter is static: " + method, matchLevel);
}
continue;
}
Class methodParameterType = method.getParameterTypes()[0];
if (methodParameterType.isPrimitive() && propertyValue == null) {
if (matchLevel < 6) {
matchLevel = 6;
missException = new MissingAccessorException("Null can not be assigned to " +
methodParameterType.getName() + ": " + method, matchLevel);
}
continue;
}
if (!RecipeHelper.isInstance(methodParameterType, propertyValue) && !RecipeHelper.isConvertable(methodParameterType, propertyValue, registry)) {
if (matchLevel < 5) {
matchLevel = 5;
missException = new MissingAccessorException((propertyValue == null ? "null" : propertyValue.getClass().getName()) + " can not be assigned or converted to " +
methodParameterType.getName() + ": " + method, matchLevel);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(method.getModifiers())) {
setAccessible(method);
}
if (RecipeHelper.isInstance(methodParameterType, propertyValue)) {
// This setter requires no conversion, which means there can not be a conversion error.
// Therefore this setter is perferred and put a the head of the list
validSetters.addFirst(method);
} else {
validSetters.add(method);
}
}
}
if (!validSetters.isEmpty()) {
// remove duplicate methods (can happen with inheritance)
return new ArrayList<Method>(new LinkedHashSet<Method>(validSetters));
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid setter method: ");
buffer.append("public void ").append(typeClass.getName()).append(".");
buffer.append(setterName).append("(");
if (propertyValue == null) {
buffer.append("null");
} else if (propertyValue instanceof String || propertyValue instanceof Recipe) {
buffer.append("...");
} else {
buffer.append(propertyValue.getClass().getName());
}
buffer.append(")");
throw new MissingAccessorException(buffer.toString(), -1);
}
}
public static List<Field> findAllFieldsByType(Class typeClass, Object propertyValue, Set<Option> options,
PropertyEditorRegistry registry) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (options == null) options = EnumSet.noneOf(Option.class);
int matchLevel = 0;
MissingAccessorException missException = null;
List<Field> fields = new ArrayList<Field>(Arrays.asList(typeClass.getDeclaredFields()));
Class parent = typeClass.getSuperclass();
while (parent != null){
fields.addAll(Arrays.asList(parent.getDeclaredFields()));
parent = parent.getSuperclass();
}
boolean allowPrivate = options.contains(Option.PRIVATE_PROPERTIES);
boolean allowStatic = options.contains(Option.STATIC_PROPERTIES);
LinkedList<Field> validFields = new LinkedList<Field>();
for (Field field : fields) {
Class fieldType = field.getType();
if (RecipeHelper.isInstance(fieldType, propertyValue) || RecipeHelper.isConvertable(fieldType, propertyValue, registry)) {
if (!allowPrivate && !Modifier.isPublic(field.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Field is not public: " + field, matchLevel);
}
continue;
}
if (!allowStatic && Modifier.isStatic(field.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Field is static: " + field, matchLevel);
}
continue;
}
if (fieldType.isPrimitive() && propertyValue == null) {
if (matchLevel < 6) {
matchLevel = 6;
missException = new MissingAccessorException("Null can not be assigned to " +
fieldType.getName() + ": " + field, matchLevel);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(field.getModifiers())) {
setAccessible(field);
}
if (RecipeHelper.isInstance(fieldType, propertyValue)) {
// This field requires no conversion, which means there can not be a conversion error.
// Therefore this setter is perferred and put a the head of the list
validFields.addFirst(field);
} else {
validFields.add(field);
}
}
}
if (!validFields.isEmpty()) {
// remove duplicate methods (can happen with inheritance)
return new ArrayList<Field>(new LinkedHashSet<Field>(validFields));
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid field ");
if (propertyValue instanceof Recipe) {
buffer.append("for ").append(propertyValue == null ? "null" : propertyValue);
} else {
buffer.append("of type ").append(propertyValue == null ? "null" : propertyValue.getClass().getName());
}
buffer.append(" in class ").append(typeClass.getName());
throw new MissingAccessorException(buffer.toString(), -1);
}
}
public static List<Method> findAllSettersByType(Class typeClass, Object propertyValue, Set<Option> options,
PropertyEditorRegistry registry) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (options == null) options = EnumSet.noneOf(Option.class);
int matchLevel = 0;
MissingAccessorException missException = null;
boolean allowPrivate = options.contains(Option.PRIVATE_PROPERTIES);
boolean allowStatic = options.contains(Option.STATIC_PROPERTIES);
LinkedList<Method> validSetters = new LinkedList<Method>();
List<Method> methods = new ArrayList<Method>(Arrays.asList(typeClass.getMethods()));
methods.addAll(Arrays.asList(typeClass.getDeclaredMethods()));
for (Method method : methods) {
if (method.getName().startsWith("set") && method.getParameterTypes().length == 1 && (RecipeHelper.isInstance(method.getParameterTypes()[0], propertyValue) || RecipeHelper.isConvertable(method.getParameterTypes()[0], propertyValue, registry))) {
if (method.getReturnType() != Void.TYPE) {
if (matchLevel < 2) {
matchLevel = 2;
missException = new MissingAccessorException("Setter returns a value: " + method, matchLevel);
}
continue;
}
if (Modifier.isAbstract(method.getModifiers())) {
if (matchLevel < 3) {
matchLevel = 3;
missException = new MissingAccessorException("Setter is abstract: " + method, matchLevel);
}
continue;
}
if (!allowPrivate && !Modifier.isPublic(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Setter is not public: " + method, matchLevel);
}
continue;
}
Class methodParameterType = method.getParameterTypes()[0];
if (methodParameterType.isPrimitive() && propertyValue == null) {
if (matchLevel < 6) {
matchLevel = 6;
missException = new MissingAccessorException("Null can not be assigned to " +
methodParameterType.getName() + ": " + method, matchLevel);
}
continue;
}
if (!allowStatic && Modifier.isStatic(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingAccessorException("Setter is static: " + method, matchLevel);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(method.getModifiers())) {
setAccessible(method);
}
if (RecipeHelper.isInstance(methodParameterType, propertyValue)) {
// This setter requires no conversion, which means there can not be a conversion error.
// Therefore this setter is perferred and put a the head of the list
validSetters.addFirst(method);
} else {
validSetters.add(method);
}
}
}
if (!validSetters.isEmpty()) {
// remove duplicate methods (can happen with inheritance)
return new ArrayList<Method>(new LinkedHashSet<Method>(validSetters));
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid setter ");
if (propertyValue instanceof Recipe) {
buffer.append("for ").append(propertyValue == null ? "null" : propertyValue);
} else {
buffer.append("of type ").append(propertyValue == null ? "null" : propertyValue.getClass().getName());
}
buffer.append(" in class ").append(typeClass.getName());
throw new MissingAccessorException(buffer.toString(), -1);
}
}
public static ConstructorFactory findConstructor(Class typeClass, List<? extends Class<?>> parameterTypes, Set<Option> options) {
return findConstructor(typeClass, null, parameterTypes, null, options);
}
public static ConstructorFactory findConstructor(Class typeClass, List<String> parameterNames, List<? extends Class<?>> parameterTypes, Set<String> availableProperties, Set<Option> options) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (availableProperties == null) availableProperties = Collections.emptySet();
if (options == null) options = EnumSet.noneOf(Option.class);
//
// verify that it is a class we can construct
if (!Modifier.isPublic(typeClass.getModifiers())) {
throw new ConstructionException("Class is not public: " + typeClass.getName());
}
if (Modifier.isInterface(typeClass.getModifiers())) {
throw new ConstructionException("Class is an interface: " + typeClass.getName());
}
if (Modifier.isAbstract(typeClass.getModifiers())) {
throw new ConstructionException("Class is abstract: " + typeClass.getName());
}
// verify parameter names and types are the same length
if (parameterNames != null) {
if (parameterTypes == null) parameterTypes = Collections.nCopies(parameterNames.size(), null);
if (parameterNames.size() != parameterTypes.size()) {
throw new ConstructionException("Invalid ObjectRecipe: recipe has " + parameterNames.size() +
" parameter names and " + parameterTypes.size() + " parameter types");
}
} else if (!options.contains(Option.NAMED_PARAMETERS)) {
// Named parameters are not supported and no explicit parameters were given,
// so we will only use the no-arg constructor
parameterNames = Collections.emptyList();
parameterTypes = Collections.emptyList();
}
// get all methods sorted so that the methods with the most constructor args are first
List<Constructor> constructors = new ArrayList<Constructor>(Arrays.asList(typeClass.getConstructors()));
constructors.addAll(Arrays.asList(typeClass.getDeclaredConstructors()));
Collections.sort(constructors, new Comparator<Constructor>() {
public int compare(Constructor constructor1, Constructor constructor2) {
return constructor2.getParameterTypes().length - constructor1.getParameterTypes().length;
}
});
// as we check each constructor, we remember the closest invalid match so we can throw a nice exception to the user
int matchLevel = 0;
MissingFactoryMethodException missException = null;
boolean allowPrivate = options.contains(Option.PRIVATE_CONSTRUCTOR);
for (Constructor constructor : constructors) {
// if an explicit constructor is specified (via parameter types), look a constructor that matches
if (parameterTypes != null) {
if (constructor.getParameterTypes().length != parameterTypes.size()) {
if (matchLevel < 1) {
matchLevel = 1;
missException = new MissingFactoryMethodException("Constructor has " + constructor.getParameterTypes().length + " arugments " +
"but expected " + parameterTypes.size() + " arguments: " + constructor);
}
continue;
}
if (!isAssignableFrom(parameterTypes, Arrays.<Class<?>>asList(constructor.getParameterTypes()))) {
if (matchLevel < 2) {
matchLevel = 2;
missException = new MissingFactoryMethodException("Constructor has signature " +
"public static " + typeClass.getName() + toParameterList(constructor.getParameterTypes()) +
" but expected signature " +
"public static " + typeClass.getName() + toParameterList(parameterTypes));
}
continue;
}
} else {
// Implicit constructor selection based on named constructor args
//
// Only consider methods where we can supply a value for all of the parameters
parameterNames = getParameterNames(constructor);
if (parameterNames == null || !availableProperties.containsAll(parameterNames)) {
continue;
}
}
if (Modifier.isAbstract(constructor.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingFactoryMethodException("Constructor is abstract: " + constructor);
}
continue;
}
if (!allowPrivate && !Modifier.isPublic(constructor.getModifiers())) {
if (matchLevel < 5) {
matchLevel = 5;
missException = new MissingFactoryMethodException("Constructor is not public: " + constructor);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(constructor.getModifiers())) {
setAccessible(constructor);
}
return new ConstructorFactory(constructor, parameterNames);
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid constructor: ");
buffer.append("public void ").append(typeClass.getName()).append(toParameterList(parameterTypes));
throw new ConstructionException(buffer.toString());
}
}
public static StaticFactory findStaticFactory(Class typeClass, String factoryMethod, List<? extends Class<?>> parameterTypes, Set<Option> options) {
return findStaticFactory(typeClass, factoryMethod, null, parameterTypes, null, options);
}
public static StaticFactory findStaticFactory(Class typeClass, String factoryMethod, List<String> parameterNames, List<? extends Class<?>> parameterTypes, Set<String> allProperties, Set<Option> options) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (factoryMethod == null) throw new NullPointerException("name is null");
if (factoryMethod.length() == 0) throw new IllegalArgumentException("name is an empty string");
if (allProperties == null) allProperties = Collections.emptySet();
if (options == null) options = EnumSet.noneOf(Option.class);
//
// verify that it is a class we can construct
if (!Modifier.isPublic(typeClass.getModifiers())) {
throw new ConstructionException("Class is not public: " + typeClass.getName());
}
if (Modifier.isInterface(typeClass.getModifiers())) {
throw new ConstructionException("Class is an interface: " + typeClass.getName());
}
// verify parameter names and types are the same length
if (parameterNames != null) {
if (parameterTypes == null) parameterTypes = Collections.nCopies(parameterNames.size(), null);
if (parameterNames.size() != parameterTypes.size()) {
throw new ConstructionException("Invalid ObjectRecipe: recipe has " + parameterNames.size() +
" parameter names and " + parameterTypes.size() + " parameter types");
}
} else if (!options.contains(Option.NAMED_PARAMETERS)) {
// Named parameters are not supported and no explicit parameters were given,
// so we will only use the no-arg constructor
parameterNames = Collections.emptyList();
parameterTypes = Collections.emptyList();
}
// get all methods sorted so that the methods with the most constructor args are first
List<Method> methods = new ArrayList<Method>(Arrays.asList(typeClass.getMethods()));
methods.addAll(Arrays.asList(typeClass.getDeclaredMethods()));
Collections.sort(methods, new Comparator<Method>() {
public int compare(Method method2, Method method1) {
return method1.getParameterTypes().length - method2.getParameterTypes().length;
}
});
// as we check each constructor, we remember the closest invalid match so we can throw a nice exception to the user
int matchLevel = 0;
MissingFactoryMethodException missException = null;
boolean allowPrivate = options.contains(Option.PRIVATE_FACTORY);
boolean caseInsesnitive = options.contains(Option.CASE_INSENSITIVE_FACTORY);
for (Method method : methods) {
// Only consider methods where the name matches
if (!method.getName().equals(factoryMethod) && (!caseInsesnitive || !method.getName().equalsIgnoreCase(method.getName()))) {
continue;
}
// if an explicit constructor is specified (via parameter types), look a constructor that matches
if (parameterTypes != null) {
if (method.getParameterTypes().length != parameterTypes.size()) {
if (matchLevel < 1) {
matchLevel = 1;
missException = new MissingFactoryMethodException("Static factory method has " + method.getParameterTypes().length + " arugments " +
"but expected " + parameterTypes.size() + " arguments: " + method);
}
continue;
}
if (!isAssignableFrom(parameterTypes, Arrays.asList(method.getParameterTypes()))) {
if (matchLevel < 2) {
matchLevel = 2;
missException = new MissingFactoryMethodException("Static factory method has signature " +
"public static " + typeClass.getName() + "." + factoryMethod + toParameterList(method.getParameterTypes()) +
" but expected signature " +
"public static " + typeClass.getName() + "." + factoryMethod + toParameterList(parameterTypes));
}
continue;
}
} else {
// Implicit constructor selection based on named constructor args
//
// Only consider methods where we can supply a value for all of the parameters
parameterNames = getParameterNames(method);
if (parameterNames == null || !allProperties.containsAll(parameterNames)) {
continue;
}
}
if (method.getReturnType() == Void.TYPE) {
if (matchLevel < 3) {
matchLevel = 3;
missException = new MissingFactoryMethodException("Static factory method does not return a value: " + method);
}
continue;
}
if (Modifier.isAbstract(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingFactoryMethodException("Static factory method is abstract: " + method);
}
continue;
}
if (!allowPrivate && !Modifier.isPublic(method.getModifiers())) {
if (matchLevel < 5) {
matchLevel = 5;
missException = new MissingFactoryMethodException("Static factory method is not public: " + method);
}
continue;
}
if (!Modifier.isStatic(method.getModifiers())) {
if (matchLevel < 6) {
matchLevel = 6;
missException = new MissingFactoryMethodException("Static factory method is not static: " + method);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(method.getModifiers())) {
setAccessible(method);
}
return new StaticFactory(method, parameterNames);
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid factory method: ");
buffer.append("public void ").append(typeClass.getName()).append(".");
buffer.append(factoryMethod).append(toParameterList(parameterTypes));
throw new MissingFactoryMethodException(buffer.toString());
}
}
public static Method findInstanceFactory(Class typeClass, String factoryMethod, Set<Option> options) {
if (typeClass == null) throw new NullPointerException("typeClass is null");
if (factoryMethod == null) throw new NullPointerException("name is null");
if (factoryMethod.length() == 0) throw new IllegalArgumentException("name is an empty string");
if (options == null) options = EnumSet.noneOf(Option.class);
int matchLevel = 0;
MissingFactoryMethodException missException = null;
boolean allowPrivate = options.contains(Option.PRIVATE_FACTORY);
boolean caseInsesnitive = options.contains(Option.CASE_INSENSITIVE_FACTORY);
List<Method> methods = new ArrayList<Method>(Arrays.asList(typeClass.getMethods()));
methods.addAll(Arrays.asList(typeClass.getDeclaredMethods()));
for (Method method : methods) {
if (method.getName().equals(factoryMethod) || (caseInsesnitive && method.getName().equalsIgnoreCase(method.getName()))) {
if (Modifier.isStatic(method.getModifiers())) {
if (matchLevel < 1) {
matchLevel = 1;
missException = new MissingFactoryMethodException("Instance factory method is static: " + method);
}
continue;
}
if (method.getParameterTypes().length != 0) {
if (matchLevel < 2) {
matchLevel = 2;
missException = new MissingFactoryMethodException("Instance factory method has signature " +
"public " + typeClass.getName() + "." + factoryMethod + toParameterList(method.getParameterTypes()) +
" but expected signature " +
"public " + typeClass.getName() + "." + factoryMethod + "()");
}
continue;
}
if (method.getReturnType() == Void.TYPE) {
if (matchLevel < 3) {
matchLevel = 3;
missException = new MissingFactoryMethodException("Instance factory method does not return a value: " + method);
}
continue;
}
if (Modifier.isAbstract(method.getModifiers())) {
if (matchLevel < 4) {
matchLevel = 4;
missException = new MissingFactoryMethodException("Instance factory method is abstract: " + method);
}
continue;
}
if (!allowPrivate && !Modifier.isPublic(method.getModifiers())) {
if (matchLevel < 5) {
matchLevel = 5;
missException = new MissingFactoryMethodException("Instance factory method is not public: " + method);
}
continue;
}
if (allowPrivate && !Modifier.isPublic(method.getModifiers())) {
setAccessible(method);
}
return method;
}
}
if (missException != null) {
throw missException;
} else {
StringBuffer buffer = new StringBuffer("Unable to find a valid factory method: ");
buffer.append("public void ").append(typeClass.getName()).append(".");
buffer.append(factoryMethod).append("()");
throw new MissingFactoryMethodException(buffer.toString());
}
}
public static List<String> getParameterNames(Constructor<?> constructor) {
// use reflection to get Java6 ConstructorParameter annotation value
try {
Class<? extends Annotation> constructorPropertiesClass = ClassLoader.getSystemClassLoader().loadClass("java.beans.ConstructorProperties").asSubclass(Annotation.class);
Annotation constructorProperties = constructor.getAnnotation(constructorPropertiesClass);
if (constructorProperties != null) {
String[] parameterNames = (String[]) constructorPropertiesClass.getMethod("value").invoke(constructorProperties);
if (parameterNames != null) {
return Arrays.asList(parameterNames);
}
}
} catch (Throwable e) {
}
ParameterNames parameterNames = constructor.getAnnotation(ParameterNames.class);
if (parameterNames != null && parameterNames.value() != null) {
return Arrays.asList(parameterNames.value());
}
if (ParameterLoader.PARAMETER_NAME_LOADER != null) {
return ParameterLoader.PARAMETER_NAME_LOADER.get(constructor);
}
return null;
}
public static List<String> getParameterNames(Method method) {
ParameterNames parameterNames = method.getAnnotation(ParameterNames.class);
if (parameterNames != null && parameterNames.value() != null) {
return Arrays.asList(parameterNames.value());
}
if (ParameterLoader.PARAMETER_NAME_LOADER != null) {
return ParameterLoader.PARAMETER_NAME_LOADER.get(method);
}
return null;
}
public static interface Factory {
List<String> getParameterNames();
List<Type> getParameterTypes();
Object create(Object... parameters) throws ConstructionException;
}
public static class ConstructorFactory implements Factory {
private Constructor constructor;
private List<String> parameterNames;
public ConstructorFactory(Constructor constructor, List<String> parameterNames) {
if (constructor == null) throw new NullPointerException("constructor is null");
if (parameterNames == null) throw new NullPointerException("parameterNames is null");
this.constructor = constructor;
this.parameterNames = parameterNames;
}
public List<String> getParameterNames() {
return parameterNames;
}
public List<Type> getParameterTypes() {
return new ArrayList<Type>(Arrays.asList(constructor.getGenericParameterTypes()));
}
public Object create(Object... parameters) throws ConstructionException {
// create the instance
try {
Object instance = constructor.newInstance(parameters);
return instance;
} catch (Exception e) {
Throwable t = e;
if (e instanceof InvocationTargetException) {
InvocationTargetException invocationTargetException = (InvocationTargetException) e;
if (invocationTargetException.getCause() != null) {
t = invocationTargetException.getCause();
}
}
throw new ConstructionException("Error invoking constructor: " + constructor, t);
}
}
}
public static class StaticFactory implements Factory {
private Method staticFactory;
private List<String> parameterNames;
public StaticFactory(Method staticFactory, List<String> parameterNames) {
this.staticFactory = staticFactory;
this.parameterNames = parameterNames;
}
public List<String> getParameterNames() {
if (parameterNames == null) {
throw new ConstructionException("InstanceFactory has not been initialized");
}
return parameterNames;
}
public List<Type> getParameterTypes() {
return new ArrayList<Type>(Arrays.asList(staticFactory.getGenericParameterTypes()));
}
public Object create(Object... parameters) throws ConstructionException {
try {
Object instance = staticFactory.invoke(null, parameters);
return instance;
} catch (Exception e) {
Throwable t = e;
if (e instanceof InvocationTargetException) {
InvocationTargetException invocationTargetException = (InvocationTargetException) e;
if (invocationTargetException.getCause() != null) {
t = invocationTargetException.getCause();
}
}
throw new ConstructionException("Error invoking factory method: " + staticFactory, t);
}
}
}
private static void setAccessible(final AccessibleObject accessibleObject) {
accessibleObject.setAccessible(true);
}
private static String toParameterList(Class<?>[] parameterTypes) {
return toParameterList(parameterTypes != null ? Arrays.asList(parameterTypes) : null);
}
private static String toParameterList(List<? extends Class<?>> parameterTypes) {
StringBuffer buffer = new StringBuffer();
buffer.append("(");
if (parameterTypes != null) {
for (int i = 0; i < parameterTypes.size(); i++) {
Class type = parameterTypes.get(i);
if (i > 0) buffer.append(", ");
buffer.append(type.getName());
}
} else {
buffer.append("...");
}
buffer.append(")");
return buffer.toString();
}
}