io.permazen.Util Maven / Gradle / Ivy
Show all versions of permazen-main Show documentation
/*
* Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
*/
package io.permazen;
import com.google.common.base.Preconditions;
import com.google.common.base.Throwables;
import com.google.common.collect.Iterables;
import com.google.common.reflect.TypeToken;
import io.permazen.annotation.OnValidate;
import java.lang.annotation.Annotation;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.lang.reflect.WildcardType;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.validation.Constraint;
import javax.validation.groups.Default;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Various utility routines.
*/
public final class Util {
private static final String ANNOTATION_ELEMENT_UTILS_CLASS_NAME = "org.springframework.core.annotation.AnnotatedElementUtils";
private static final String ANNOTATION_ELEMENT_UTILS_GET_MERGED_ANNOTATION_METHOD_NAME = "getMergedAnnotation";
private static final WildcardType QUESTION_MARK = new WildcardType() {
@Override
public Type[] getUpperBounds() {
return new Type[] { Object.class };
}
@Override
public Type[] getLowerBounds() {
return new Type[0];
}
@Override
public String toString() {
return "?";
}
};
private static Method newParameterizedTypeMethod;
private static BiFunction, Annotation> annotationRetriever;
private Util() {
}
/**
* Find the annotation on the given element.
*
*
* If {@code spring-core} is available on the classpath, the implementation in {@link AnnotationScanner} utilizes Spring's
* {@link org.springframework.core.annotation.AnnotatedElementUtils#getMergedAnnotation(AnnotatedElement, Class)} method
* to find annotations that are either present or meta-present on the element, and includes support for
* annotation attribute overrides; otherwise, it just invokes {@link AnnotatedElement#getAnnotation(Class)}.
*
* @param element element with annotation or meta-present annotation
* @param annotationType type of the annotation to find
* @param annotation type
* @return the annotation found, or null if not found
*/
public static A getAnnotation(AnnotatedElement element, Class annotationType) {
Preconditions.checkArgument(element != null, "null element");
Preconditions.checkArgument(annotationType != null, "null annotationType");
synchronized (Util.class) {
if (Util.annotationRetriever == null) {
final Logger log = LoggerFactory.getLogger(Util.class);
try {
final Class> cl = Class.forName(ANNOTATION_ELEMENT_UTILS_CLASS_NAME,
true, Thread.currentThread().getContextClassLoader());
final Method method = cl.getMethod(ANNOTATION_ELEMENT_UTILS_GET_MERGED_ANNOTATION_METHOD_NAME,
AnnotatedElement.class, Class.class);
Util.annotationRetriever = (elem, atype) -> {
try {
return atype.cast(method.invoke(null, elem, atype));
} catch (Exception e) {
throw new RuntimeException("internal error", e);
}
};
if (log.isDebugEnabled())
log.debug("using Spring's " + cl.getSimpleName() + "." + method.getName() + "() for annotation retrieval");
} catch (ClassNotFoundException e) {
if (log.isDebugEnabled())
log.debug("using JDK AnnotatedElement.getAnnotation() for annotation retrieval");
Util.annotationRetriever = (elem, atype) -> elem.getAnnotation(atype);
} catch (Exception e) {
log.warn("using JDK AnnotatedElement.getAnnotation() for annotation retrieval", e);
Util.annotationRetriever = (elem, atype) -> elem.getAnnotation(atype);
}
}
assert Util.annotationRetriever != null;
}
return annotationType.cast(Util.annotationRetriever.apply(element, annotationType));
}
/**
* Determine if any JSR 303 validation annotations are present on the given type itself
* or any of its methods (public methods only).
*
* @param type object type
* @return a non-null object with JSR 303 validation requirements, or null if none found
* @throws IllegalArgumentException if {@code type} is null
*/
public static AnnotatedElement hasValidation(Class> type) {
// Sanity check
Preconditions.checkArgument(type != null, "null type");
// Check for annotations on the class itself
if (Util.hasValidationAnnotation(type))
return type;
// Check methods
for (Method method : type.getDeclaredMethods()) {
// Check for JSR 303 annotation
if ((method.getModifiers() & Modifier.PUBLIC) != 0 && Util.hasValidationAnnotation(method))
return method;
}
// Recurse on supertypes
for (TypeToken> typeToken : TypeToken.of(type).getTypes()) {
final Class> superType = typeToken.getRawType();
if (superType == type)
continue;
final AnnotatedElement annotatedElement = Util.hasValidation(superType);
if (annotatedElement != null)
return annotatedElement;
}
// None found
return null;
}
/**
* Determine if instances of the given type require any validation under the default validation group.
*
*
* This will be true if {@code type} or any of its declared methods has a JSR 303 (public methods only)
* or {@link OnValidate @OnValidate} annotation, or if any of its super-types requires validation.
*
* @param type object type
* @return true if {@code type} has any validation requirements
* @throws IllegalArgumentException if {@code type} is null
* @see ValidationMode
*/
public static boolean requiresDefaultValidation(Class> type) {
// Sanity check
Preconditions.checkArgument(type != null, "null type");
// Check for annotations on the class itself
if (Util.hasDefaultValidationAnnotation(type))
return true;
// Check methods
for (Method method : type.getDeclaredMethods()) {
// Check for @OnValidate annotation
if (method.isAnnotationPresent(OnValidate.class))
return true;
// Check for JSR 303 annotation
if ((method.getModifiers() & Modifier.PUBLIC) != 0 && Util.requiresDefaultValidation(method))
return true;
}
// Recurse on superclasses
for (TypeToken> typeToken : TypeToken.of(type).getTypes()) {
final Class> superType = typeToken.getRawType();
if (superType != type && Util.requiresDefaultValidation(superType))
return true;
}
// Done
return false;
}
/**
* Determine if the given getter method, or any method it overrides, has a JSR 303 validation constraint
* applicable under the default validation group.
*
* @param method annotated method
* @return true if {@code obj} has one or more JSR 303 annotations
* @throws IllegalArgumentException if {@code method} is null
*/
public static boolean requiresDefaultValidation(Method method) {
Preconditions.checkArgument(method != null, "null method");
final String methodName = method.getName();
final Class>[] paramTypes = method.getParameterTypes();
for (TypeToken> typeToken : TypeToken.of(method.getDeclaringClass()).getTypes()) {
final Class> superType = typeToken.getRawType();
try {
method = superType.getMethod(methodName, paramTypes);
} catch (NoSuchMethodException e) {
continue;
}
if (Util.hasDefaultValidationAnnotation(method))
return true;
}
return false;
}
/**
* Determine whether the given object has any JSR 303 annotation(s) defining validation constraints in the default group.
*
* @param obj annotated element
* @return true if {@code obj} has one or more JSR 303 default validation constraint annotations
* @throws IllegalArgumentException if {@code obj} is null
*/
public static boolean hasDefaultValidationAnnotation(AnnotatedElement obj) {
return Util.hasValidationAnnotation(obj, new Class>[] { Default.class });
}
/**
* Determine whether the given object has any JSR 303 annotation(s).
*
* @param obj annotated element
* @return true if {@code obj} has one or more JSR 303 validation constraint annotations
* @throws IllegalArgumentException if {@code obj} is null
*/
public static boolean hasValidationAnnotation(AnnotatedElement obj) {
return Util.hasValidationAnnotation(obj, null);
}
private static boolean hasValidationAnnotation(AnnotatedElement obj, Class>[] validationGroups) {
Preconditions.checkArgument(obj != null, "null obj");
for (Annotation annotation : obj.getAnnotations()) {
final Class> annotationType = annotation.annotationType();
if (!annotationType.isAnnotationPresent(Constraint.class))
continue;
final Class>[] groups;
try {
groups = (Class>[])annotation.annotationType().getMethod("groups").invoke(annotation);
} catch (NoSuchMethodException e) {
return true;
} catch (RuntimeException e) {
throw e;
} catch (Exception e) {
throw new RuntimeException(e);
}
if (groups == null || groups.length == 0)
return true;
return validationGroups == null || Util.isAnyGroupBeingValidated(groups, validationGroups);
}
return false;
}
/**
* Determine if a constraint whose {@code groups()} contain the given constraint group should be applied
* when validating with the given validation groups.
*
* @param constraintGroup validation group associated with a validation constraint
* @param validationGroups groups for which validation is being performed
* @return whether to apply the validation constraint
* @throws IllegalArgumentException if any null values are encountered
*/
public static boolean isGroupBeingValidated(Class> constraintGroup, Class>[] validationGroups) {
Preconditions.checkArgument(constraintGroup != null, "null constraintGroup");
Preconditions.checkArgument(validationGroups != null, "null validationGroups");
for (Class> validationGroup : validationGroups) {
Preconditions.checkArgument(validationGroup != null, "null validationGroup");
if (constraintGroup.isAssignableFrom(validationGroup))
return true;
}
return false;
}
/**
* Determine if a constraint whose {@code groups()} contain the given constraint groups should be applied
* when validating with the given validation groups.
*
* @param constraintGroups validation groups associated with a validation constraint
* @param validationGroups groups for which validation is being performed
* @return whether to apply the validation constraint
* @throws IllegalArgumentException if any null values are encountered
*/
public static boolean isAnyGroupBeingValidated(Class>[] constraintGroups, Class>[] validationGroups) {
Preconditions.checkArgument(constraintGroups != null, "null constraintGroups");
for (Class> constraintGroup : constraintGroups) {
if (Util.isGroupBeingValidated(constraintGroup, validationGroups))
return true;
}
return false;
}
/**
* Find the getter method we should override corresponding to the nominal getter method.
*
*
* This deals with generic sub-type bridge methods.
*
* @param type Java type (possibly a sub-type of the type in which {@code getter} is declared)
* @param getter supertype Java bean property getter method
* @return corresponding Java bean property getter method in {@code type}, possibly {@code getter}
*/
static Method findJFieldGetterMethod(Class type, Method getter) {
Preconditions.checkArgument(type != null);
Preconditions.checkArgument(getter != null);
Preconditions.checkArgument(getter.getParameterTypes().length == 0);
Preconditions.checkArgument(getter.getReturnType() != void.class);
final TypeToken typeType = TypeToken.of(type);
final TypeToken> propertyType = typeType.resolveType(getter.getGenericReturnType());
for (TypeToken> superType : TypeToken.of(type).getTypes()) {
for (Method method : superType.getRawType().getDeclaredMethods()) {
if (!method.getName().equals(getter.getName()))
continue;
if (method.getParameterTypes().length != 0)
continue;
if (!typeType.resolveType(method.getGenericReturnType()).equals(propertyType))
continue;
if ((method.getModifiers() & (Modifier.PROTECTED | Modifier.PUBLIC)) == 0
|| (method.getModifiers() & Modifier.PRIVATE) != 0) {
throw new IllegalArgumentException("invalid getter method "
+ getter.getName() + "(): method must be public or protected");
}
return method;
}
}
return getter;
}
/**
* Find the setter method corresponding to a getter method. It must be either public or protected.
*
* @param type Java type (possibly a sub-type of the type in which {@code getter} is declared)
* @param getter Java bean property getter method
* @return Java bean property setter method
* @throws IllegalArgumentException if no corresponding setter method exists
*/
static Method findJFieldSetterMethod(Class type, Method getter) {
final Matcher matcher = Pattern.compile("(is|get)(.+)").matcher(getter.getName());
if (!matcher.matches()) {
throw new IllegalArgumentException("can't infer setter method name from getter method "
+ getter.getName() + "() because name does not follow Java bean naming conventions");
}
final String setterName = "set" + matcher.group(2);
final TypeToken typeType = TypeToken.of(type);
final TypeToken> propertyType = typeType.resolveType(getter.getGenericReturnType());
for (TypeToken> superType : TypeToken.of(type).getTypes()) {
for (Method setter : superType.getRawType().getDeclaredMethods()) {
if (!setter.getName().equals(setterName) || setter.getReturnType() != void.class)
continue;
final Type[] ptypes = setter.getGenericParameterTypes();
if (ptypes.length != 1)
continue;
if (!typeType.resolveType(ptypes[0]).equals(propertyType))
continue;
if ((setter.getModifiers() & (Modifier.PROTECTED | Modifier.PUBLIC)) == 0
|| (setter.getModifiers() & Modifier.PRIVATE) != 0) {
throw new IllegalArgumentException("invalid setter method " + setterName
+ "() corresponding to getter method " + getter.getName() + "(): method must be public or protected");
}
return setter;
}
}
throw new IllegalArgumentException("can't find any setter method " + setterName
+ "() corresponding to getter method " + getter.getName() + "() taking " + getter.getReturnType()
+ " and returning void");
}
/**
* Find unimplemented abstract methods in the given class.
*/
static Map findAbstractMethods(Class> type) {
final HashMap map = new HashMap<>();
// First find all methods, but don't include overridden supertype methods
for (TypeToken> superType : TypeToken.of(type).getTypes()) {
for (Method method : superType.getRawType().getDeclaredMethods()) {
final MethodKey key = new MethodKey(method);
if (!map.containsKey(key))
map.put(key, method);
}
}
// Now discard all the non-abstract methods
for (Iterator> i = map.entrySet().iterator(); i.hasNext(); ) {
final Map.Entry entry = i.next();
if ((entry.getValue().getModifiers() & Modifier.ABSTRACT) == 0)
i.remove();
}
// Done
return map;
}
/**
* Find the narrowest type that is a supertype of all of the given types.
*
*
* This method delegates to {@link #findLowestCommonAncestor findLowestCommonAncestor()}
* after converting the {@link Class} instances to {@link TypeToken}s.
*
* @param types sub-types
* @return narrowest common super-type
* @throws IllegalArgumentException if any type in {@code types} is null
* @see #findLowestCommonAncestor findLowestCommonAncestor()
*/
public static TypeToken> findLowestCommonAncestorOfClasses(Iterable> types) {
types.forEach(type -> {
if (type == null)
throw new IllegalArgumentException("null type");
});
return Util.findLowestCommonAncestor(Iterables.transform(types, TypeToken::of));
}
/**
* Find the narrowest type(s) each of which is a supertype of all of the given types.
*
*
* This method delegates to {@link #findLowestCommonAncestors findLowestCommonAncestors()}
* after converting the {@link Class} instances to {@link TypeToken}s.
*
* @param types sub-types
* @return maximally narrow common supertype(s)
* @throws IllegalArgumentException if any type in {@code types} is null
* @see #findLowestCommonAncestors findLowestCommonAncestors()
*/
public static Set> findLowestCommonAncestorsOfClasses(Iterable> types) {
types.forEach(type -> {
if (type == null)
throw new IllegalArgumentException("null type");
});
return Util.findLowestCommonAncestors(Iterables.transform(types, TypeToken::of));
}
/**
* Find the narrowest type that is a supertype of all of the given types.
*
*
* Note that there may be more than one such type. The returned type will always be as narrow
* as possible, but it's possible there for there to be multiple such types for which none
* is a sub-type of any other.
*
*
* When there is more than one choice, heuristics are used. For example, we prefer
* non-interface types, and {@link JObject} over other interface types.
*
* @param types sub-types
* @return narrowest common super-type
*/
public static TypeToken> findLowestCommonAncestor(Iterable> types) {
// Gather candidates
final Set> supertypes = Util.findLowestCommonAncestors(types);
// Pick the best candidate that's not Object, if possible
final TypeToken