package org.jdiameter.client.impl.annotation.internal; import java.lang.annotation.Annotation; import java.lang.reflect.Constructor; import java.lang.reflect.Method; import java.util.Collection; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; public class ClassInfo { private Storage storage; private Class<?> _class; private Map<Class<?>, Annotation> annotations; private Map<Method, MethodInfo> methods; private Map<Constructor, ConstructorInfo> constructors; private Collection<Annotation> classCache; private Collection<MethodInfo> methodCache; private Collection<ConstructorInfo> constructorCache; public ClassInfo (Storage storage, Class<?> _class){ this.storage = storage; this._class = _class; } public Class<?> getAttachedClass() { return _class; } public Collection<Annotation> getAnnotations() { if (classCache == null) { if (annotations == null) { annotations = new ConcurrentHashMap<Class<?>, Annotation>(); final Class<?> parent = getAttachedClass().getSuperclass(); if (parent != null) { addAnnotations(parent); } for (Class<?> i : getAttachedClass().getInterfaces()) { addAnnotations(i); } for (Annotation a : getAttachedClass().getDeclaredAnnotations()) { annotations.put(a.getClass().getInterfaces()[0], a); } } classCache = annotations.values(); } return classCache; } public <T> T getAnnotation(Class<?> annotation) { for (Annotation a : getAnnotations()) { if (a.annotationType() == annotation) { return (T) a; } } return null; } private void addAnnotations(Class<?> _class) { for (Annotation annotation : storage.getClassInfo(_class).getAnnotations()) { if (annotation != null) { for (Class<?> _interface : annotation.getClass().getInterfaces()) { annotations.put(_interface, annotation); } } } } public MethodInfo getMethodInfo(String methodName, Class<?>... args) { try { return getMethodInfo( getAttachedClass().getMethod(methodName, args) ); } catch (Exception e) { return null; } } public ConstructorInfo getConstructorInfo(Class<?>... args) { try { return getConstructorInfo( getAttachedClass().getConstructor(args) ); } catch (Exception e1) { // may be generic try { return getConstructorInfo( getAttachedClass().getConstructor(Object.class) ); } catch (Exception e2) {} return null; } } public MethodInfo getMethodInfo(Method method) { return getMethodMap().get(method); } public ConstructorInfo getConstructorInfo(Constructor constr) { return getConstructorMap().get(constr); } public Collection<MethodInfo> getMethodsInfo() { return methodCache == null ? (methodCache = getMethodMap().values()) : methodCache; } public Collection<ConstructorInfo> getConstructorsInfo() { return constructorCache == null ? (constructorCache = getConstructorMap().values()) : constructorCache; } private Map<Method, MethodInfo> getMethodMap(){ if (methods == null) { methods = new ConcurrentHashMap<Method, MethodInfo>(); for (Method method : getAttachedClass().getMethods()) { methods.put(method, new MethodInfo(storage, this, method)); } } return methods; } private Map<Constructor, ConstructorInfo> getConstructorMap(){ if (constructors == null) { constructors = new ConcurrentHashMap<Constructor, ConstructorInfo>(); for (Constructor constr : getAttachedClass().getConstructors()) { constructors.put(constr, new ConstructorInfo(storage, this, constr)); } } return constructors; } }