blob: 97a9769c5d45d514a9b21b59bf1a36442ad05664 [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// Licensed 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 com.google.devtools.common.options;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.primitives.Primitives;
import com.google.common.reflect.TypeToken;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
/**
* A helper class for {@link OptionsParserImpl} to help checking the return type
* of a {@link Converter} against the type of a field or the element type of a
* list.
*
* <p>This class has to go through considerable contortion to get the correct result
* from the Java reflection system, unfortunately. If the generic reflection part
* had been better designed, some of this would not be necessary.
*/
class GenericTypeHelper {
/**
* Returns the raw type of t, if t is either a raw or parameterized type.
* Otherwise, this method throws an {@link AssertionError}.
*/
@VisibleForTesting
static Class<?> getRawType(Type t) {
if (t instanceof Class<?>) {
return (Class<?>) t;
} else if (t instanceof ParameterizedType) {
return (Class<?>) ((ParameterizedType) t).getRawType();
} else {
throw new AssertionError("A known concrete type is not concrete");
}
}
/**
* If type is a parameterized type, searches the given type variable in the list
* of declared type variables, and then returns the corresponding actual type.
* Returns null if the type variable is not defined by type.
*/
private static Type matchTypeVariable(Type type, TypeVariable<?> variable) {
if (type instanceof ParameterizedType) {
Class<?> rawInterfaceType = getRawType(type);
TypeVariable<?>[] typeParameters = rawInterfaceType.getTypeParameters();
for (int i = 0; i < typeParameters.length; i++) {
if (variable.equals(typeParameters[i])) {
return ((ParameterizedType) type).getActualTypeArguments()[i];
}
}
}
return null;
}
/**
* Resolves the return type of a method, in particular if the generic return
* type ({@link Method#getGenericReturnType()}) is a type variable
* ({@link TypeVariable}), by checking all super-classes and directly
* implemented interfaces.
*
* <p>The method m must be defined by the given type or by its raw class type.
*
* @throws AssertionError if the generic return type could not be resolved
*/
// TODO(bazel-team): also check enclosing classes and indirectly implemented
// interfaces, which can also contribute type variables. This doesn't happen
// in the existing use cases.
public static Type getActualReturnType(Type type, Method method) {
Type returnType = method.getGenericReturnType();
if (returnType instanceof Class<?>) {
return returnType;
} else if (returnType instanceof ParameterizedType) {
return returnType;
} else if (returnType instanceof TypeVariable<?>) {
TypeVariable<?> variable = (TypeVariable<?>) returnType;
while (type != null) {
Type candidate = matchTypeVariable(type, variable);
if (candidate != null) {
return candidate;
}
Class<?> rawType = getRawType(type);
for (Type interfaceType : rawType.getGenericInterfaces()) {
candidate = matchTypeVariable(interfaceType, variable);
if (candidate != null) {
return candidate;
}
}
type = rawType.getGenericSuperclass();
}
}
throw new AssertionError("The type " + returnType
+ " is not a Class, ParameterizedType, or TypeVariable");
}
/**
* Determines if a value of a particular type (from) is assignable to a field of
* a particular type (to). Also allows assigning wrapper types to primitive
* types.
*
* <p>The checks done here should be identical to the checks done by
* {@link java.lang.reflect.Field#set}. I.e., if this method returns true, a
* subsequent call to {@link java.lang.reflect.Field#set} should succeed.
*/
public static boolean isAssignableFrom(Type to, Type from) {
if (to instanceof Class<?>) {
Class<?> toClass = (Class<?>) to;
if (toClass.isPrimitive()) {
return Primitives.wrap(toClass).equals(from);
}
}
return TypeToken.of(to).isSupertypeOf(from);
}
private GenericTypeHelper() {
// Prevents Java from creating a public constructor.
}
}