File |
Line |
org/apache/xbean/recipe/AsmParameterNameLoader.java |
162 |
org/apache/xbean/recipe/XbeanAsmParameterNameLoader.java |
162 |
XbeanAsmParameterNameLoader.AllParameterNamesDiscoveringVisitor visitor = new XbeanAsmParameterNameLoader.AllParameterNamesDiscoveringVisitor(clazz, methodName);
reader.accept(visitor, 0);
Map exceptions = visitor.getExceptions();
if (exceptions.size() == 1) {
throw new RuntimeException((Exception)exceptions.values().iterator().next());
}
if (!exceptions.isEmpty()) {
throw new RuntimeException(exceptions.toString());
}
methodParameters = visitor.getMethodParameters();
} catch (IOException ex) {
}
// Cache the names
for (Method method : methods) {
methodCache.put(method, methodParameters.get(method));
}
return methodParameters;
}
private Method[] getMethods(Class clazz, String methodName) {
List<Method> methods = new ArrayList<Method>(Arrays.asList(clazz.getMethods()));
methods.addAll(Arrays.asList(clazz.getDeclaredMethods()));
List<Method> matchingMethod = new ArrayList<Method>(methods.size());
for (Method method : methods) {
if (method.getName().equals(methodName)) {
matchingMethod.add(method);
}
}
return matchingMethod.toArray(new Method[matchingMethod.size()]);
}
private static ClassReader createClassReader(Class declaringClass) throws IOException {
InputStream in = null;
try {
ClassLoader classLoader = declaringClass.getClassLoader();
in = classLoader.getResourceAsStream(declaringClass.getName().replace('.', '/') + ".class");
ClassReader reader = new ClassReader(in);
return reader;
} finally {
if (in != null) {
try {
in.close();
} catch (IOException ignored) {
}
}
}
}
private static class AllParameterNamesDiscoveringVisitor extends EmptyVisitor {
private final Map<Constructor,List<String>> constructorParameters = new HashMap<Constructor,List<String>>();
private final Map<Method,List<String>> methodParameters = new HashMap<Method,List<String>>();
private final Map<String,Exception> exceptions = new HashMap<String,Exception>();
private final String methodName;
private final Map<String,Method> methodMap = new HashMap<String,Method>();
private final Map<String,Constructor> constructorMap = new HashMap<String,Constructor>();
public AllParameterNamesDiscoveringVisitor(Class type, String methodName) {
this.methodName = methodName;
List<Method> methods = new ArrayList<Method>(Arrays.asList(type.getMethods()));
methods.addAll(Arrays.asList(type.getDeclaredMethods()));
for (Method method : methods) {
if (method.getName().equals(methodName)) {
methodMap.put(Type.getMethodDescriptor(method), method);
}
}
}
public AllParameterNamesDiscoveringVisitor(Class type) {
this.methodName = "<init>";
List<Constructor> constructors = new ArrayList<Constructor>(Arrays.asList(type.getConstructors()));
constructors.addAll(Arrays.asList(type.getDeclaredConstructors()));
for (Constructor constructor : constructors) {
Type[] types = new Type[constructor.getParameterTypes().length];
for (int j = 0; j < types.length; j++) {
types[j] = Type.getType(constructor.getParameterTypes()[j]);
}
constructorMap.put(Type.getMethodDescriptor(Type.VOID_TYPE, types), constructor);
}
}
public Map<Constructor, List<String>> getConstructorParameters() {
return constructorParameters;
}
public Map<Method, List<String>> getMethodParameters() {
return methodParameters;
}
public Map<String,Exception> getExceptions() {
return exceptions;
}
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
if (!name.equals(this.methodName)) {
return null;
}
try {
final List<String> parameterNames;
final boolean isStaticMethod;
if (methodName.equals("<init>")) {
Constructor constructor = constructorMap.get(desc);
if (constructor == null) {
return null;
}
parameterNames = new ArrayList<String>(constructor.getParameterTypes().length);
parameterNames.addAll(Collections.<String>nCopies(constructor.getParameterTypes().length, null));
constructorParameters.put(constructor, parameterNames);
isStaticMethod = false;
} else {
Method method = methodMap.get(desc);
if (method == null) {
return null;
}
parameterNames = new ArrayList<String>(method.getParameterTypes().length);
parameterNames.addAll(Collections.<String>nCopies(method.getParameterTypes().length, null));
methodParameters.put(method, parameterNames);
isStaticMethod = Modifier.isStatic(method.getModifiers());
}
return new EmptyVisitor() {
// assume static method until we get a first parameter name
public void visitLocalVariable(String name, String description, String signature, Label start, Label end, int index) {
if (isStaticMethod) {
parameterNames.set(index, name);
} else if (index > 0) {
// for non-static the 0th arg is "this" so we need to offset by -1
parameterNames.set(index - 1, name);
}
}
};
} catch (Exception e) {
this.exceptions.put(signature, e);
}
return null;
}
}
}
|
File |
Line |
org/apache/xbean/recipe/AsmParameterNameLoader.java |
45 |
org/apache/xbean/recipe/XbeanAsmParameterNameLoader.java |
45 |
public class XbeanAsmParameterNameLoader implements ParameterNameLoader {
/**
* Weak map from Constructor to List<String>.
*/
private final WeakHashMap<Constructor,List<String>> constructorCache = new WeakHashMap<Constructor,List<String>>();
/**
* Weak map from Method to List<String>.
*/
private final WeakHashMap<Method,List<String>> methodCache = new WeakHashMap<Method,List<String>>();
/**
* Gets the parameter names of the specified method or null if the class was compiled without debug symbols on.
* @param method the method for which the parameter names should be retrieved
* @return the parameter names or null if the class was compilesd without debug symbols on
*/
public List<String> get(Method method) {
// check the cache
if (methodCache.containsKey(method)) {
return methodCache.get(method);
}
Map<Method,List<String>> allMethodParameters = getAllMethodParameters(method.getDeclaringClass(), method.getName());
return allMethodParameters.get(method);
}
/**
* Gets the parameter names of the specified constructor or null if the class was compiled without debug symbols on.
* @param constructor the constructor for which the parameters should be retrieved
* @return the parameter names or null if the class was compiled without debug symbols on
*/
public List<String> get(Constructor constructor) {
// check the cache
if (constructorCache.containsKey(constructor)) {
return constructorCache.get(constructor);
}
Map<Constructor,List<String>> allConstructorParameters = getAllConstructorParameters(constructor.getDeclaringClass());
return allConstructorParameters.get(constructor);
}
/**
* Gets the parameter names of all constructor or null if the class was compiled without debug symbols on.
* @param clazz the class for which the constructor parameter names should be retrieved
* @return a map from Constructor object to the parameter names or null if the class was compiled without debug symbols on
*/
public Map<Constructor,List<String>> getAllConstructorParameters(Class clazz) {
// Determine the constructors?
List<Constructor> constructors = new ArrayList<Constructor>(Arrays.asList(clazz.getConstructors()));
constructors.addAll(Arrays.asList(clazz.getDeclaredConstructors()));
if (constructors.isEmpty()) {
return Collections.emptyMap();
}
// Check the cache
if (constructorCache.containsKey(constructors.get(0))) {
Map<Constructor,List<String>> constructorParameters = new HashMap<Constructor,List<String>>();
for (Constructor constructor : constructors) {
constructorParameters.put(constructor, constructorCache.get(constructor));
}
return constructorParameters;
}
// Load the parameter names using ASM
Map<Constructor,List<String>> constructorParameters = new HashMap<Constructor,List<String>> ();
try {
ClassReader reader = XbeanAsmParameterNameLoader.createClassReader(clazz);
|
File |
Line |
org/apache/xbean/recipe/AsmParameterNameLoader.java |
113 |
org/apache/xbean/recipe/XbeanAsmParameterNameLoader.java |
113 |
XbeanAsmParameterNameLoader.AllParameterNamesDiscoveringVisitor visitor = new XbeanAsmParameterNameLoader.AllParameterNamesDiscoveringVisitor(clazz);
reader.accept(visitor, 0);
Map exceptions = visitor.getExceptions();
if (exceptions.size() == 1) {
throw new RuntimeException((Exception)exceptions.values().iterator().next());
}
if (!exceptions.isEmpty()) {
throw new RuntimeException(exceptions.toString());
}
constructorParameters = visitor.getConstructorParameters();
} catch (IOException ex) {
}
// Cache the names
for (Constructor constructor : constructors) {
constructorCache.put(constructor, constructorParameters.get(constructor));
}
return constructorParameters;
}
/**
* Gets the parameter names of all methods with the specified name or null if the class was compiled without debug symbols on.
* @param clazz the class for which the method parameter names should be retrieved
* @param methodName the of the method for which the parameters should be retrieved
* @return a map from Method object to the parameter names or null if the class was compiled without debug symbols on
*/
public Map<Method,List<String>> getAllMethodParameters(Class clazz, String methodName) {
// Determine the constructors?
Method[] methods = getMethods(clazz, methodName);
if (methods.length == 0) {
return Collections.emptyMap();
}
// Check the cache
if (methodCache.containsKey(methods[0])) {
Map<Method,List<String>> methodParameters = new HashMap<Method,List<String>>();
for (Method method : methods) {
methodParameters.put(method, methodCache.get(method));
}
return methodParameters;
}
// Load the parameter names using ASM
Map<Method,List<String>> methodParameters = new HashMap<Method,List<String>>();
try {
ClassReader reader = XbeanAsmParameterNameLoader.createClassReader(clazz);
|
File |
Line |
org/apache/xbean/recipe/ReflectionUtil.java |
157 |
org/apache/xbean/recipe/ReflectionUtil.java |
238 |
public static List<Method> findAllSetters(Class typeClass, String propertyName, Object propertyValue, 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 setterName = "set" + Character.toUpperCase(propertyName.charAt(0));
|
File |
Line |
org/apache/xbean/recipe/ArrayRecipe.java |
52 |
org/apache/xbean/recipe/CollectionRecipe.java |
75 |
public CollectionRecipe(CollectionRecipe collectionRecipe) {
if (collectionRecipe == null) throw new NullPointerException("setRecipe is null");
this.typeName = collectionRecipe.typeName;
this.typeClass = collectionRecipe.typeClass;
list = new ArrayList<Object>(collectionRecipe.list);
}
public void allow(Option option) {
options.add(option);
}
public void disallow(Option option) {
options.remove(option);
}
public List<Recipe> getNestedRecipes() {
List<Recipe> nestedRecipes = new ArrayList<Recipe>(list.size());
for (Object o : list) {
if (o instanceof Recipe) {
Recipe recipe = (Recipe) o;
nestedRecipes.add(recipe);
}
}
return nestedRecipes;
}
public List<Recipe> getConstructorRecipes() {
if (!options.contains(Option.LAZY_ASSIGNMENT)) {
return getNestedRecipes();
}
return Collections.emptyList();
}
public boolean canCreate(Type expectedType) {
Class myType = getType(expectedType);
|
File |
Line |
org/apache/xbean/recipe/ReflectionUtil.java |
68 |
org/apache/xbean/recipe/ReflectionUtil.java |
244 |
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);
}
|
File |
Line |
org/apache/xbean/recipe/ReflectionUtil.java |
300 |
org/apache/xbean/recipe/ReflectionUtil.java |
484 |
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;
}
|
File |
Line |
org/apache/xbean/recipe/ReflectionUtil.java |
578 |
org/apache/xbean/recipe/ReflectionUtil.java |
691 |
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()));
|
File |
Line |
org/apache/xbean/recipe/ReflectionUtil.java |
347 |
org/apache/xbean/recipe/ReflectionUtil.java |
521 |
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 ");
|