org.apache.camel.maven.packaging.generics.GenericsUtil Maven / Gradle / Ivy
/*
* 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.camel.maven.packaging.generics;
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.GenericArrayType;
import java.lang.reflect.GenericDeclaration;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.lang.reflect.WildcardType;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;
/**
* Utility classes for generic type operations.
*/
public final class GenericsUtil {
/*
* Private constructor
*/
private GenericsUtil() {
}
public static boolean satisfiesDependency(
boolean isDelegateOrEvent, boolean isProducer, Type injectionPointType, Type beanType) {
if (beanType instanceof TypeVariable || beanType instanceof WildcardType || beanType instanceof GenericArrayType) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, beanType);
} else {
Type injectionPointRawType = injectionPointType instanceof ParameterizedType
? ((ParameterizedType) injectionPointType).getRawType() : injectionPointType;
Type beanRawType = beanType instanceof ParameterizedType ? ((ParameterizedType) beanType).getRawType() : beanType;
if (ClassUtil.isSame(injectionPointRawType, beanRawType)) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, beanType);
}
}
return false;
}
public static boolean satisfiesDependencyRaw(
boolean isDelegateOrEvent, boolean isProducer, Type injectionPointType, Type beanType) {
if (beanType instanceof TypeVariable || beanType instanceof WildcardType || beanType instanceof GenericArrayType) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, beanType);
} else {
Type injectionPointRawType = injectionPointType instanceof ParameterizedType
? ((ParameterizedType) injectionPointType).getRawType() : injectionPointType;
Type beanRawType = beanType instanceof ParameterizedType ? ((ParameterizedType) beanType).getRawType() : beanType;
if (ClassUtil.isSame(injectionPointRawType, beanRawType)) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointRawType, beanRawType);
} else {
Class bean = (Class) beanType;
if (bean.getSuperclass() != null && ClassUtil.isRawClassEquals(injectionPointType, bean.getSuperclass())) {
return true;
}
Class>[] interfaces = bean.getInterfaces();
if (interfaces == null || interfaces.length == 0) {
return false;
}
for (Class> clazz : interfaces) {
if (ClassUtil.isRawClassEquals(injectionPointType, clazz)) {
return true;
}
}
}
}
return false;
}
/**
* 5.2.3 and 5.2.4
*/
public static boolean isAssignableFrom(boolean isDelegateOrEvent, boolean isProducer, Type requiredType, Type beanType) {
if (requiredType instanceof Class) {
return isAssignableFrom(isDelegateOrEvent, (Class>) requiredType, beanType);
} else if (requiredType instanceof ParameterizedType) {
return isAssignableFrom(isDelegateOrEvent, isProducer, (ParameterizedType) requiredType, beanType);
} else if (requiredType instanceof TypeVariable) {
return isAssignableFrom(isDelegateOrEvent, (TypeVariable>) requiredType, beanType);
} else if (requiredType instanceof GenericArrayType) {
return Class.class.isInstance(beanType) && Class.class.cast(beanType).isArray()
&& isAssignableFrom(isDelegateOrEvent, (GenericArrayType) requiredType, beanType);
} else if (requiredType instanceof WildcardType) {
return isAssignableFrom(isDelegateOrEvent, (WildcardType) requiredType, beanType);
} else {
throw new IllegalArgumentException("Unsupported type " + requiredType.getClass());
}
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, Class> injectionPointType, Type beanType) {
if (beanType instanceof Class) {
return isAssignableFrom(injectionPointType, (Class>) beanType);
} else if (beanType instanceof TypeVariable) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (TypeVariable>) beanType);
} else if (beanType instanceof ParameterizedType) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (ParameterizedType) beanType);
} else if (beanType instanceof GenericArrayType) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (GenericArrayType) beanType);
} else if (beanType instanceof WildcardType) {
return isAssignableFrom(isDelegateOrEvent, (Type) injectionPointType, (WildcardType) beanType);
} else {
throw new IllegalArgumentException("Unsupported type " + injectionPointType.getClass());
}
}
private static boolean isAssignableFrom(Class> injectionPointType, Class> beanType) {
return ClassUtil.isClassAssignableFrom(injectionPointType, beanType);
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, Class> injectionPointType, TypeVariable> beanType) {
for (Type bounds : beanType.getBounds()) {
if (isAssignableFrom(isDelegateOrEvent, injectionPointType, bounds)) {
return true;
}
}
return false;
}
/**
* CDI Spec. 5.2.4: "A parameterized bean type is considered assignable to a raw required type if the raw generics
* are identical and all type parameters of the bean type are either unbounded type variables or java.lang.Object."
*/
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, Class> injectionPointType, ParameterizedType beanType) {
if (beanType.getRawType() != injectionPointType) {
return false; // raw generics don't match
}
if (isDelegateOrEvent) {
// for delegate and events we match 'in reverse' kind off
// @Observes ProcessInjectionPoint, Instance> does also match
// Instance
return isAssignableFrom(true, injectionPointType, beanType.getRawType());
}
for (Type typeArgument : beanType.getActualTypeArguments()) {
if (typeArgument == Object.class) {
continue;
}
if (!(typeArgument instanceof TypeVariable)) {
return false; // neither object nor type variable
}
TypeVariable> typeVariable = (TypeVariable>) typeArgument;
for (Type bounds : typeVariable.getBounds()) {
if (bounds != Object.class) {
return false; // bound type variable
}
}
}
return true;
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, Class> injectionPointType, GenericArrayType beanType) {
return injectionPointType.isArray() && isAssignableFrom(isDelegateOrEvent, injectionPointType.getComponentType(),
beanType.getGenericComponentType());
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, Type injectionPointType, WildcardType beanType) {
for (Type bounds : beanType.getLowerBounds()) {
if (!isAssignableFrom(isDelegateOrEvent, false, bounds, injectionPointType)) {
return false;
}
}
for (Type bounds : beanType.getUpperBounds()) {
if (isAssignableFrom(isDelegateOrEvent, false, injectionPointType, bounds)) {
return true;
}
}
return false;
}
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, boolean isProducer, ParameterizedType injectionPointType, Type beanType) {
if (beanType instanceof Class) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, (Class>) beanType);
} else if (beanType instanceof TypeVariable) {
return isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, (TypeVariable>) beanType);
} else if (beanType instanceof ParameterizedType) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (ParameterizedType) beanType);
} else if (beanType instanceof WildcardType) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (WildcardType) beanType);
} else if (beanType instanceof GenericArrayType) {
return false;
} else {
throw new IllegalArgumentException("Unsupported type " + beanType.getClass());
}
}
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, boolean isProducer, ParameterizedType injectionPointType, Class> beanType) {
Class> rawInjectionPointType = getRawType(injectionPointType);
if (rawInjectionPointType.equals(beanType)) {
if (isProducer) {
for (final Type t : injectionPointType.getActualTypeArguments()) {
if (!TypeVariable.class.isInstance(t) || !isNotBound(TypeVariable.class.cast(t).getBounds())) {
if (!Class.class.isInstance(t) || Object.class != t) {
return false;
}
}
}
}
return true;
}
if (!rawInjectionPointType.isAssignableFrom(beanType)) {
return false;
}
if (beanType.getSuperclass() != null
&& isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, beanType.getGenericSuperclass())) {
return true;
}
for (Type genericInterface : beanType.getGenericInterfaces()) {
if (isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, genericInterface)) {
return true;
}
}
return false;
}
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, boolean isProducer, ParameterizedType injectionPointType, TypeVariable> beanType) {
final Type[] types = beanType.getBounds();
if (isNotBound(types)) {
return true;
}
for (final Type bounds : types) {
if (isAssignableFrom(isDelegateOrEvent, isProducer, injectionPointType, bounds)) {
return true;
}
}
return false;
}
/**
* CDI Spec. 5.2.4
*/
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, ParameterizedType injectionPointType, ParameterizedType beanType) {
if (injectionPointType.getRawType() != beanType.getRawType()) {
return false;
}
boolean swapParams = !isDelegateOrEvent;
Type[] injectionPointTypeArguments = injectionPointType.getActualTypeArguments();
Type[] beanTypeArguments = beanType.getActualTypeArguments();
for (int i = 0; i < injectionPointTypeArguments.length; i++) {
Type injectionPointTypeArgument = injectionPointTypeArguments[i];
Type beanTypeArgument = beanTypeArguments[i];
// for this special case it's actually an 'assignable to', thus we
// swap the params, see CDI-389
// but this special rule does not apply to Delegate injection
// points...
if (swapParams
&& (injectionPointTypeArgument instanceof Class || injectionPointTypeArgument instanceof TypeVariable)
&& beanTypeArgument instanceof TypeVariable) {
final Type[] bounds = ((TypeVariable>) beanTypeArgument).getBounds();
final boolean isNotBound = isNotBound(bounds);
if (!isNotBound) {
for (final Type upperBound : bounds) {
if (!isAssignableFrom(true, false, upperBound, injectionPointTypeArgument)) {
return false;
}
}
}
} else if (swapParams && injectionPointTypeArgument instanceof TypeVariable) {
return false;
} else if (isDelegateOrEvent && injectionPointTypeArgument instanceof Class && beanTypeArgument instanceof Class) {
// if no wildcard type was given then we require a real exact
// match.
return injectionPointTypeArgument.equals(beanTypeArgument);
} else if (!isAssignableFrom(isDelegateOrEvent, false, injectionPointTypeArgument, beanTypeArgument)) {
return false;
}
}
return true;
}
private static boolean isNotBound(final Type... bounds) {
return bounds == null || bounds.length == 0 || bounds.length == 1 && Object.class == bounds[0];
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, TypeVariable> injectionPointType, Type beanType) {
for (Type bounds : injectionPointType.getBounds()) {
if (!isAssignableFrom(isDelegateOrEvent, false, bounds, beanType)) {
return false;
}
}
return true;
}
// rules are a bit different when in an array so we handle ParameterizedType
// manually (not reusing isAssignableFrom)
private static boolean isAssignableFrom(boolean isDelegateOrEvent, GenericArrayType injectionPointType, Type beanType) {
final Type genericComponentType = injectionPointType.getGenericComponentType();
final Class componentType = Class.class.cast(beanType).getComponentType();
if (Class.class.isInstance(genericComponentType)) {
return Class.class.cast(genericComponentType).isAssignableFrom(componentType);
}
if (ParameterizedType.class.isInstance(genericComponentType)) {
return isAssignableFrom(isDelegateOrEvent, false, ParameterizedType.class.cast(genericComponentType).getRawType(),
componentType);
}
return isAssignableFrom(isDelegateOrEvent, false, genericComponentType, componentType);
}
private static boolean isAssignableFrom(boolean isDelegateOrEvent, WildcardType injectionPointType, Type beanType) {
if (beanType instanceof TypeVariable) {
return isAssignableFrom(isDelegateOrEvent, injectionPointType, (TypeVariable>) beanType);
}
for (Type bounds : injectionPointType.getLowerBounds()) {
if (!isAssignableFrom(isDelegateOrEvent, false, beanType, bounds)) {
return false;
}
}
for (Type bounds : injectionPointType.getUpperBounds()) {
Set beanTypeClosure = getTypeClosure(beanType);
boolean isAssignable = false;
for (Type beanSupertype : beanTypeClosure) {
if (isAssignableFrom(isDelegateOrEvent, false, bounds, beanSupertype)
|| Class.class.isInstance(bounds) && ParameterizedType.class.isInstance(beanSupertype)
&& bounds == ParameterizedType.class.cast(beanSupertype).getRawType()) {
isAssignable = true;
break;
}
}
if (!isAssignable) {
return false;
}
}
return true;
}
/**
* CDI 1.1 Spec. 5.2.4, third bullet point
*/
private static boolean isAssignableFrom(
boolean isDelegateOrEvent, WildcardType injectionPointType, TypeVariable> beanType) {
for (Type upperBound : injectionPointType.getUpperBounds()) {
for (Type bound : beanType.getBounds()) {
if (!isAssignableFrom(isDelegateOrEvent, false, upperBound, bound)
&& !isAssignableFrom(isDelegateOrEvent, false, bound, upperBound)) {
return false;
}
}
}
for (Type lowerBound : injectionPointType.getLowerBounds()) {
for (Type bound : beanType.getBounds()) {
if (!isAssignableFrom(isDelegateOrEvent, false, bound, lowerBound)) {
return false;
}
}
}
return true;
}
/**
* @return true, if the specified type declaration contains an unresolved type variable.
*/
public static boolean containsTypeVariable(Type type) {
if (type instanceof Class) {
return false;
} else if (type instanceof TypeVariable) {
return true;
} else if (type instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) type;
return containTypeVariable(parameterizedType.getActualTypeArguments());
} else if (type instanceof WildcardType) {
WildcardType wildcardType = (WildcardType) type;
return containTypeVariable(wildcardType.getUpperBounds()) || containTypeVariable(wildcardType.getLowerBounds());
} else if (type instanceof GenericArrayType) {
GenericArrayType arrayType = (GenericArrayType) type;
return containsTypeVariable(arrayType.getGenericComponentType());
} else {
throw new IllegalArgumentException("Unsupported type " + type.getClass().getName());
}
}
public static boolean containTypeVariable(Collection extends Type> types) {
return containTypeVariable(types.toArray(new Type[types.size()]));
}
public static boolean containTypeVariable(Type[] types) {
for (Type type : types) {
if (containsTypeVariable(type)) {
return true;
}
}
return false;
}
/**
* @param type to check
* @return {@code true} if the given type contains a {@link WildcardType} {@code false} otherwise
*/
public static boolean containsWildcardType(Type type) {
if (!(type instanceof ParameterizedType)) {
return false;
}
for (Type typeArgument : getParameterizedType(type).getActualTypeArguments()) {
if (ClassUtil.isParameterizedType(typeArgument)) {
if (containsWildcardType(typeArgument)) {
return true;
}
} else {
if (ClassUtil.isWildCardType(typeArgument)) {
return true;
}
}
}
return false;
}
/**
* Resolves the actual type of the specified field for the type hierarchy specified by the given subclass
*/
public static Type resolveType(Class> subclass, Field field) {
return resolveType(field.getGenericType(), subclass, newSeenList());
}
/**
* Resolves the actual return type of the specified method for the type hierarchy specified by the given subclass
*/
public static Type resolveReturnType(Class> subclass, Method method) {
return resolveType(method.getGenericReturnType(), subclass, newSeenList());
}
/**
* Resolves the actual parameter generics of the specified constructor for the type hierarchy specified by the given
* subclass
*/
public static Type[] resolveParameterTypes(Class> subclass, Constructor> constructor) {
return resolveTypes(constructor.getGenericParameterTypes(), subclass);
}
/**
* Resolves the actual parameter generics of the specified method for the type hierarchy specified by the given
* subclass
*/
public static Type[] resolveParameterTypes(Class> subclass, Method method) {
return resolveTypes(method.getGenericParameterTypes(), subclass);
}
/**
* Resolves the actual type of the specified type for the type hierarchy specified by the given subclass
*/
public static Type resolveType(Type type, Class> subclass, Member member) {
return resolveType(type, subclass, newSeenList());
}
public static Type resolveType(Type type, Class> subclass, Member member, Collection> seen) {
return resolveType(type, subclass, seen);
}
public static Type resolveType(Type type, Type actualType, Collection> seen) {
if (type instanceof Class) {
return type;
} else if (type instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) type;
Type[] resolvedTypeArguments;
if (Enum.class.equals(parameterizedType.getRawType())) {
// Enums derive from themselves, which would create an infinite
// loop
// we directly escape the loop if we detect this.
resolvedTypeArguments = new Type[] { new OwbWildcardTypeImpl(new Type[] { Enum.class }, ClassUtil.NO_TYPES) };
} else {
resolvedTypeArguments = resolveTypes(parameterizedType.getActualTypeArguments(), actualType, seen);
}
return new OwbParametrizedTypeImpl(
parameterizedType.getOwnerType(), parameterizedType.getRawType(), resolvedTypeArguments);
} else if (type instanceof TypeVariable) {
TypeVariable> variable = (TypeVariable>) type;
return resolveTypeVariable(variable, actualType, seen);
} else if (type instanceof WildcardType) {
WildcardType wildcardType = (WildcardType) type;
Type[] upperBounds = resolveTypes(wildcardType.getUpperBounds(), actualType, seen);
Type[] lowerBounds = resolveTypes(wildcardType.getLowerBounds(), actualType, seen);
return new OwbWildcardTypeImpl(upperBounds, lowerBounds);
} else if (type instanceof GenericArrayType) {
GenericArrayType arrayType = (GenericArrayType) type;
return createArrayType(resolveType(arrayType.getGenericComponentType(), actualType, seen));
} else {
throw new IllegalArgumentException("Unsupported type " + type.getClass().getName());
}
}
public static Type[] resolveTypes(Type[] types, Type actualType, Collection> seen) {
Type[] resolvedTypeArguments = new Type[types.length];
for (int i = 0; i < types.length; i++) {
final Type type = resolveType(types[i], actualType, seen);
if (type != null) { // means a stackoverflow was avoided, just keep
// what we have
resolvedTypeArguments[i] = type;
}
}
return resolvedTypeArguments;
}
public static Type[] resolveTypes(Type[] types, Type actualType) {
Type[] resolvedTypeArguments = new Type[types.length];
for (int i = 0; i < types.length; i++) {
resolvedTypeArguments[i] = resolveType(types[i], actualType, newSeenList());
}
return resolvedTypeArguments;
}
public static Set getTypeClosure(Class> type) {
return getTypeClosure(type, type);
}
public static Set getTypeClosure(Type actualType) {
return getTypeClosure(actualType, actualType);
}
/**
* Returns the type closure for the specified parameters.
* Example 1:
*
* Take the following classes:
*
*
* public class Foo {
* private T t;
* }
* public class Bar extends Foo {
* }
*
*
* To get the type closure of T in the context of Bar (which is {Number.class, Object.class}), you have to call this
* method like
*
*
* GenericUtil.getTypeClosure(Foo.class.getDeclaredField("t").getType(), Bar.class, Foo.class);
*
* Example 2:
*
* Take the following classes:
*
*
* public class Foo {
* private T t;
* }
* public class Bar extends Foo {
* }
*
*
* To get the type closure of Bar in the context of Foo (which are besides Object.class the
* ParameterizedTypes Bar and Foo), you have to call this method like
*
*
* GenericUtil.getTypeClosure(Foo.class, new TypeLiteral>() {}.getType(), Bar.class);
*
*
* @param type the type to get the closure for
* @param actualType the context to bind type variables
* @return the type closure
*/
public static Set getTypeClosure(Type type, Type actualType) {
Class> rawType = getRawType(type);
Class> actualRawType = getRawType(actualType);
if (rawType.isAssignableFrom(actualRawType) && rawType != actualRawType) {
return getTypeClosure(actualType, type);
}
if (hasTypeParameters(type)) {
type = getParameterizedType(type);
}
return getDirectTypeClosure(type, actualType);
}
public static Set getDirectTypeClosure(final Type type, final Type actualType) {
Set typeClosure = new HashSet<>();
typeClosure.add(Object.class);
fillTypeHierarchy(typeClosure, type, actualType);
return typeClosure;
}
private static void fillTypeHierarchy(Set set, Type type, Type actualType) {
if (type == null) {
return;
}
Type resolvedType = GenericsUtil.resolveType(type, actualType, newSeenList());
set.add(resolvedType);
Class> resolvedClass = GenericsUtil.getRawType(resolvedType, actualType);
if (resolvedClass.getSuperclass() != null) {
fillTypeHierarchy(set, resolvedClass.getGenericSuperclass(), resolvedType);
}
for (Type interfaceType : resolvedClass.getGenericInterfaces()) {
fillTypeHierarchy(set, interfaceType, resolvedType);
}
}
private static Collection> newSeenList() {
return new ArrayList<>();
}
public static boolean hasTypeParameters(Type type) {
if (type instanceof Class) {
Class> classType = (Class>) type;
return classType.getTypeParameters().length > 0;
}
return false;
}
public static ParameterizedType getParameterizedType(Type type) {
if (type instanceof ParameterizedType) {
return (ParameterizedType) type;
} else if (type instanceof Class) {
Class> classType = (Class>) type;
return new OwbParametrizedTypeImpl(classType.getDeclaringClass(), classType, classType.getTypeParameters());
} else {
throw new IllegalArgumentException(type.getClass().getSimpleName() + " is not supported");
}
}
public static Class getRawType(Type type) {
return getRawType(type, null);
}
static Class getRawType(Type type, Type actualType) {
if (type instanceof Class) {
return (Class) type;
} else if (type instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) type;
return getRawType(parameterizedType.getRawType(), actualType);
} else if (type instanceof TypeVariable) {
TypeVariable> typeVariable = (TypeVariable>) type;
Type mostSpecificType
= getMostSpecificType(getRawTypes(typeVariable.getBounds(), actualType), typeVariable.getBounds());
return getRawType(mostSpecificType, actualType);
} else if (type instanceof WildcardType) {
WildcardType wildcardType = (WildcardType) type;
Type mostSpecificType = getMostSpecificType(getRawTypes(wildcardType.getUpperBounds(), actualType),
wildcardType.getUpperBounds());
return getRawType(mostSpecificType, actualType);
} else if (type instanceof GenericArrayType) {
GenericArrayType arrayType = (GenericArrayType) type;
return getRawType(createArrayType(getRawType(arrayType.getGenericComponentType(), actualType)), actualType);
} else {
throw new IllegalArgumentException("Unsupported type " + type.getClass().getName());
}
}
private static Class[] getRawTypes(Type[] types) {
return getRawTypes(types, null);
}
private static Class[] getRawTypes(Type[] types, Type actualType) {
Class[] rawTypes = new Class[types.length];
for (int i = 0; i < types.length; i++) {
rawTypes[i] = getRawType(types[i], actualType);
}
return rawTypes;
}
private static Type getMostSpecificType(Class>[] types, Type[] genericTypes) {
Class> mostSpecificType = types[0];
int mostSpecificIndex = 0;
for (int i = 0; i < types.length; i++) {
if (mostSpecificType.isAssignableFrom(types[i])) {
mostSpecificType = types[i];
mostSpecificIndex = i;
}
}
return genericTypes[mostSpecificIndex];
}
private static Type resolveTypeVariable(TypeVariable> variable, Type actualType, Collection> seen) {
if (actualType == null) {
return variable;
}
Class> declaringClass = getDeclaringClass(variable.getGenericDeclaration());
Class> actualClass = getRawType(actualType);
if (actualClass == declaringClass) {
return resolveTypeVariable(variable, variable.getGenericDeclaration(), getParameterizedType(actualType), seen);
} else if (actualClass.isAssignableFrom(declaringClass)) {
Class> directSubclass = getDirectSubclass(declaringClass, actualClass);
Type[] typeArguments = resolveTypeArguments(directSubclass, actualType);
Type directSubtype = new OwbParametrizedTypeImpl(directSubclass.getDeclaringClass(), directSubclass, typeArguments);
return resolveTypeVariable(variable, directSubtype, seen);
} else { // if (declaringClass.isAssignableFrom(actualClass))
Type genericSuperclass = getGenericSuperclass(actualClass, declaringClass);
if (genericSuperclass == null) {
return variable;
} else if (genericSuperclass instanceof Class) {
// special handling for type erasure
Class> superclass = (Class>) genericSuperclass;
genericSuperclass = new OwbParametrizedTypeImpl(
superclass.getDeclaringClass(), superclass, getRawTypes(superclass.getTypeParameters()));
} else {
ParameterizedType genericSupertype = getParameterizedType(genericSuperclass);
Type[] typeArguments = resolveTypeArguments(getParameterizedType(actualType), genericSupertype);
genericSuperclass = new OwbParametrizedTypeImpl(
genericSupertype.getOwnerType(), genericSupertype.getRawType(), typeArguments);
}
Type resolvedType = resolveTypeVariable(variable, genericSuperclass, seen);
if (resolvedType instanceof TypeVariable) {
TypeVariable> resolvedTypeVariable = (TypeVariable>) resolvedType;
TypeVariable>[] typeParameters = actualClass.getTypeParameters();
for (int i = 0; i < typeParameters.length; i++) {
if (typeParameters[i].getName().equals(resolvedTypeVariable.getName())) {
resolvedType = getParameterizedType(actualType).getActualTypeArguments()[i];
break;
}
}
}
return resolvedType;
}
}
private static Class> getDeclaringClass(GenericDeclaration declaration) {
if (declaration instanceof Class) {
return (Class>) declaration;
} else if (declaration instanceof Member) {
return ((Member) declaration).getDeclaringClass();
} else {
throw new IllegalArgumentException("Unsupported type " + declaration.getClass());
}
}
private static Type resolveTypeVariable(
TypeVariable> variable, GenericDeclaration declaration, ParameterizedType type,
Collection> seen) {
int index = getIndex(declaration, variable);
if (declaration instanceof Class) {
if (index >= 0) {
return type.getActualTypeArguments()[index];
} else {
index = getIndex(type, variable);
if (index >= 0) {
return declaration.getTypeParameters()[index];
}
}
} else {
if (seen.contains(variable)) {
return null;
}
seen.add(variable);
Type[] resolvedBounds = resolveTypes(declaration.getTypeParameters()[index].getBounds(), type, seen);
return OwbTypeVariableImpl.createTypeVariable(variable, resolvedBounds);
}
return variable;
}
private static int getIndex(GenericDeclaration declaration, TypeVariable> variable) {
Type[] typeParameters = declaration.getTypeParameters();
for (int i = 0; i < typeParameters.length; i++) {
if (typeParameters[i] instanceof TypeVariable) {
TypeVariable> variableArgument = (TypeVariable>) typeParameters[i];
if (variableArgument.getName().equals(variable.getName())) {
return i;
}
}
}
return -1;
}
private static int getIndex(ParameterizedType type, TypeVariable> variable) {
Type[] actualTypeArguments = type.getActualTypeArguments();
for (int i = 0; i < actualTypeArguments.length; i++) {
if (actualTypeArguments[i] instanceof TypeVariable) {
TypeVariable> variableArgument = (TypeVariable>) actualTypeArguments[i];
if (variableArgument.getName().equals(variable.getName())) {
return i;
}
}
}
return -1;
}
private static Class> getDirectSubclass(Class> declaringClass, Class> actualClass) {
if (actualClass.isInterface()) {
Class> subclass = declaringClass;
for (Class> iface : declaringClass.getInterfaces()) {
if (iface == actualClass) {
return subclass;
}
if (actualClass.isAssignableFrom(iface)) {
subclass = iface;
} else {
subclass = declaringClass.getSuperclass();
}
}
return getDirectSubclass(subclass, actualClass);
} else {
Class> directSubclass = declaringClass;
while (directSubclass.getSuperclass() != actualClass) {
directSubclass = directSubclass.getSuperclass();
}
return directSubclass;
}
}
private static Type getGenericSuperclass(Class> subclass, Class> superclass) {
if (!superclass.isInterface()) {
return subclass.getGenericSuperclass();
} else {
for (Type genericInterface : subclass.getGenericInterfaces()) {
if (getRawType(genericInterface) == superclass) {
return genericInterface;
}
}
}
return superclass;
}
private static Type[] resolveTypeArguments(Class> subclass, Type supertype) {
if (supertype instanceof ParameterizedType) {
ParameterizedType parameterizedSupertype = (ParameterizedType) supertype;
return resolveTypeArguments(subclass, parameterizedSupertype);
} else {
return subclass.getTypeParameters();
}
}
private static Type[] resolveTypeArguments(Class> subclass, ParameterizedType parameterizedSupertype) {
Type genericSuperclass = getGenericSuperclass(subclass, getRawType(parameterizedSupertype));
if (!(genericSuperclass instanceof ParameterizedType)) {
return subclass.getTypeParameters();
}
ParameterizedType parameterizedSuperclass = (ParameterizedType) genericSuperclass;
Type[] typeParameters = subclass.getTypeParameters();
Type[] actualTypeArguments = parameterizedSupertype.getActualTypeArguments();
return resolveTypeArguments(parameterizedSuperclass, typeParameters, actualTypeArguments);
}
private static Type[] resolveTypeArguments(ParameterizedType subtype, ParameterizedType parameterizedSupertype) {
return resolveTypeArguments(getParameterizedType(getRawType(subtype)), parameterizedSupertype.getActualTypeArguments(),
subtype.getActualTypeArguments());
}
private static Type[] resolveTypeArguments(
ParameterizedType parameterizedType, Type[] typeParameters, Type[] actualTypeArguments) {
Type[] resolvedTypeArguments = new Type[typeParameters.length];
for (int i = 0; i < typeParameters.length; i++) {
resolvedTypeArguments[i] = resolveTypeArgument(parameterizedType, typeParameters[i], actualTypeArguments);
}
return resolvedTypeArguments;
}
private static Type resolveTypeArgument(
ParameterizedType parameterizedType, Type typeParameter, Type[] actualTypeArguments) {
if (typeParameter instanceof TypeVariable) {
TypeVariable> variable = (TypeVariable>) typeParameter;
int index = getIndex(parameterizedType, variable);
if (index == -1) {
return typeParameter;
} else {
return actualTypeArguments[index];
}
} else if (typeParameter instanceof GenericArrayType) {
GenericArrayType array = (GenericArrayType) typeParameter;
return createArrayType(
resolveTypeArgument(parameterizedType, array.getGenericComponentType(), actualTypeArguments));
} else {
return typeParameter;
}
}
private static Type createArrayType(Type componentType) {
if (componentType instanceof Class) {
return Array.newInstance((Class>) componentType, 0).getClass();
} else {
return new OwbGenericArrayTypeImpl(componentType);
}
}
public static Type resolveType(ParameterizedType parameterizedType, Type metadataType) {
return resolveType(parameterizedType, metadataType, newSeenList());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy