| /* |
| * Copyright (c) 2007 Mockito contributors |
| * This program is made available under the terms of the MIT License. |
| */ |
| package org.mockito.internal.util.reflection; |
| |
| import org.junit.Test; |
| |
| import java.io.Serializable; |
| import java.lang.reflect.Method; |
| import java.lang.reflect.Type; |
| import java.lang.reflect.TypeVariable; |
| import java.util.*; |
| |
| import static org.assertj.core.api.Assertions.assertThat; |
| import static org.junit.Assert.fail; |
| import static org.mockito.internal.util.reflection.GenericMetadataSupport.inferFrom; |
| |
| public class GenericMetadataSupportTest { |
| |
| interface GenericsSelfReference<T extends GenericsSelfReference<T>> { |
| T self(); |
| } |
| interface UpperBoundedTypeWithClass<E extends Number & Comparable<E>> { |
| E get(); |
| } |
| interface UpperBoundedTypeWithInterfaces<E extends Comparable<E> & Cloneable> { |
| E get(); |
| } |
| interface ListOfNumbers extends List<Number> {} |
| interface AnotherListOfNumbers extends ListOfNumbers {} |
| |
| abstract class ListOfNumbersImpl implements ListOfNumbers {} |
| abstract class AnotherListOfNumbersImpl extends ListOfNumbersImpl {} |
| |
| interface ListOfAnyNumbers<N extends Number & Cloneable> extends List<N> {} |
| |
| interface GenericsNest<K extends Comparable<K> & Cloneable> extends Map<K, Set<Number>> { |
| Set<Number> remove(Object key); // override with fixed ParameterizedType |
| List<? super Integer> returning_wildcard_with_class_lower_bound(); |
| List<? super K> returning_wildcard_with_typeVar_lower_bound(); |
| List<? extends K> returning_wildcard_with_typeVar_upper_bound(); |
| K returningK(); |
| <O extends K> List<O> paramType_with_type_params(); |
| <S, T extends S> T two_type_params(); |
| <O extends K> O typeVar_with_type_params(); |
| } |
| |
| static class StringList extends ArrayList<String> { } |
| |
| @Test |
| public void typeVariable_of_self_type() { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsSelfReference.class).resolveGenericReturnType(firstNamedMethod("self", GenericsSelfReference.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(GenericsSelfReference.class); |
| } |
| |
| @Test |
| public void can_get_raw_type_from_Class() throws Exception { |
| assertThat(inferFrom(ListOfAnyNumbers.class).rawType()).isEqualTo(ListOfAnyNumbers.class); |
| assertThat(inferFrom(ListOfNumbers.class).rawType()).isEqualTo(ListOfNumbers.class); |
| assertThat(inferFrom(GenericsNest.class).rawType()).isEqualTo(GenericsNest.class); |
| assertThat(inferFrom(StringList.class).rawType()).isEqualTo(StringList.class); |
| } |
| |
| @Test |
| public void can_get_raw_type_from_ParameterizedType() throws Exception { |
| assertThat(inferFrom(ListOfAnyNumbers.class.getGenericInterfaces()[0]).rawType()).isEqualTo(List.class); |
| assertThat(inferFrom(ListOfNumbers.class.getGenericInterfaces()[0]).rawType()).isEqualTo(List.class); |
| assertThat(inferFrom(GenericsNest.class.getGenericInterfaces()[0]).rawType()).isEqualTo(Map.class); |
| assertThat(inferFrom(StringList.class.getGenericSuperclass()).rawType()).isEqualTo(ArrayList.class); |
| } |
| |
| @Test |
| public void can_get_type_variables_from_Class() throws Exception { |
| assertThat(inferFrom(GenericsNest.class).actualTypeArguments().keySet()).hasSize(1).extracting("name").contains("K"); |
| assertThat(inferFrom(ListOfNumbers.class).actualTypeArguments().keySet()).isEmpty(); |
| assertThat(inferFrom(ListOfAnyNumbers.class).actualTypeArguments().keySet()).hasSize(1).extracting("name").contains("N"); |
| assertThat(inferFrom(Map.class).actualTypeArguments().keySet()).hasSize(2).extracting("name").contains("K", "V"); |
| assertThat(inferFrom(Serializable.class).actualTypeArguments().keySet()).isEmpty(); |
| assertThat(inferFrom(StringList.class).actualTypeArguments().keySet()).isEmpty(); |
| } |
| |
| @Test |
| public void can_resolve_type_variables_from_ancestors() throws Exception { |
| Method listGet = List.class.getMethod("get", int.class); |
| assertThat(inferFrom(AnotherListOfNumbers.class).resolveGenericReturnType(listGet).rawType()).isEqualTo(Number.class); |
| assertThat(inferFrom(AnotherListOfNumbersImpl.class).resolveGenericReturnType(listGet).rawType()).isEqualTo(Number.class); |
| } |
| |
| @Test |
| public void can_get_type_variables_from_ParameterizedType() throws Exception { |
| assertThat(inferFrom(GenericsNest.class.getGenericInterfaces()[0]).actualTypeArguments().keySet()).hasSize(2).extracting("name").contains("K", "V"); |
| assertThat(inferFrom(ListOfAnyNumbers.class.getGenericInterfaces()[0]).actualTypeArguments().keySet()).hasSize(1).extracting("name").contains("E"); |
| assertThat(inferFrom(Integer.class.getGenericInterfaces()[0]).actualTypeArguments().keySet()).hasSize(1).extracting("name").contains("T"); |
| assertThat(inferFrom(StringBuilder.class.getGenericInterfaces()[0]).actualTypeArguments().keySet()).isEmpty(); |
| assertThat(inferFrom(StringList.class).actualTypeArguments().keySet()).isEmpty(); |
| } |
| |
| @Test |
| public void typeVariable_return_type_of____iterator____resolved_to_Iterator_and_type_argument_to_String() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(StringList.class).resolveGenericReturnType(firstNamedMethod("iterator", StringList.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Iterator.class); |
| assertThat(genericMetadata.actualTypeArguments().values()).contains(String.class); |
| } |
| |
| @Test |
| public void typeVariable_return_type_of____get____resolved_to_Set_and_type_argument_to_Number() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("get", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Set.class); |
| assertThat(genericMetadata.actualTypeArguments().values()).contains(Number.class); |
| } |
| |
| @Test |
| public void bounded_typeVariable_return_type_of____returningK____resolved_to_Comparable_and_with_BoundedType() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("returningK", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Comparable.class); |
| GenericMetadataSupport extraInterface_0 = inferFrom(genericMetadata.extraInterfaces().get(0)); |
| assertThat(extraInterface_0.rawType()).isEqualTo(Cloneable.class); |
| } |
| |
| @Test |
| public void fixed_ParamType_return_type_of____remove____resolved_to_Set_and_type_argument_to_Number() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("remove", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Set.class); |
| assertThat(genericMetadata.actualTypeArguments().values()).contains(Number.class); |
| } |
| |
| @Test |
| public void paramType_return_type_of____values____resolved_to_Collection_and_type_argument_to_Parameterized_Set() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("values", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Collection.class); |
| GenericMetadataSupport fromTypeVariableE = inferFrom(typeVariableValue(genericMetadata.actualTypeArguments(), "E")); |
| assertThat(fromTypeVariableE.rawType()).isEqualTo(Set.class); |
| assertThat(fromTypeVariableE.actualTypeArguments().values()).contains(Number.class); |
| } |
| |
| @Test |
| public void paramType_with_type_parameters_return_type_of____paramType_with_type_params____resolved_to_Collection_and_type_argument_to_Parameterized_Set() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("paramType_with_type_params", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(List.class); |
| Type firstBoundOfE = ((GenericMetadataSupport.TypeVarBoundedType) typeVariableValue(genericMetadata.actualTypeArguments(), "E")).firstBound(); |
| assertThat(inferFrom(firstBoundOfE).rawType()).isEqualTo(Comparable.class); |
| } |
| |
| @Test |
| public void typeVariable_with_type_parameters_return_type_of____typeVar_with_type_params____resolved_K_hence_to_Comparable_and_with_BoundedType() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("typeVar_with_type_params", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(Comparable.class); |
| GenericMetadataSupport extraInterface_0 = inferFrom(genericMetadata.extraInterfaces().get(0)); |
| assertThat(extraInterface_0.rawType()).isEqualTo(Cloneable.class); |
| } |
| |
| @Test |
| public void class_return_type_of____append____resolved_to_StringBuilder_and_type_arguments() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(StringBuilder.class).resolveGenericReturnType(firstNamedMethod("append", StringBuilder.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(StringBuilder.class); |
| assertThat(genericMetadata.actualTypeArguments()).isEmpty(); |
| } |
| |
| |
| |
| @Test |
| public void paramType_with_wildcard_return_type_of____returning_wildcard_with_class_lower_bound____resolved_to_List_and_type_argument_to_Integer() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("returning_wildcard_with_class_lower_bound", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(List.class); |
| GenericMetadataSupport.BoundedType boundedType = (GenericMetadataSupport.BoundedType) typeVariableValue(genericMetadata.actualTypeArguments(), "E"); |
| assertThat(boundedType.firstBound()).isEqualTo(Integer.class); |
| assertThat(boundedType.interfaceBounds()).isEmpty(); |
| } |
| |
| @Test |
| public void paramType_with_wildcard_return_type_of____returning_wildcard_with_typeVar_lower_bound____resolved_to_List_and_type_argument_to_Integer() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("returning_wildcard_with_typeVar_lower_bound", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(List.class); |
| GenericMetadataSupport.BoundedType boundedType = (GenericMetadataSupport.BoundedType) typeVariableValue(genericMetadata.actualTypeArguments(), "E"); |
| |
| assertThat(inferFrom(boundedType.firstBound()).rawType()).isEqualTo(Comparable.class); |
| assertThat(boundedType.interfaceBounds()).contains(Cloneable.class); } |
| |
| @Test |
| public void paramType_with_wildcard_return_type_of____returning_wildcard_with_typeVar_upper_bound____resolved_to_List_and_type_argument_to_Integer() throws Exception { |
| GenericMetadataSupport genericMetadata = inferFrom(GenericsNest.class).resolveGenericReturnType(firstNamedMethod("returning_wildcard_with_typeVar_upper_bound", GenericsNest.class)); |
| |
| assertThat(genericMetadata.rawType()).isEqualTo(List.class); |
| GenericMetadataSupport.BoundedType boundedType = (GenericMetadataSupport.BoundedType) typeVariableValue(genericMetadata.actualTypeArguments(), "E"); |
| |
| assertThat(inferFrom(boundedType.firstBound()).rawType()).isEqualTo(Comparable.class); |
| assertThat(boundedType.interfaceBounds()).contains(Cloneable.class); |
| } |
| |
| private Type typeVariableValue(Map<TypeVariable<?>, Type> typeVariables, String typeVariableName) { |
| for (Map.Entry<TypeVariable<?>, Type> typeVariableTypeEntry : typeVariables.entrySet()) { |
| if (typeVariableTypeEntry.getKey().getName().equals(typeVariableName)) { |
| return typeVariableTypeEntry.getValue(); |
| } |
| } |
| |
| fail("'" + typeVariableName + "' was not found in " + typeVariables); |
| return null; // unreachable |
| } |
| |
| private Method firstNamedMethod(String methodName, Class<?> clazz) { |
| for (Method method : clazz.getMethods()) { |
| boolean protect_against_different_jdk_ordering_avoiding_bridge_methods = !method.isBridge(); |
| if (method.getName().contains(methodName) && protect_against_different_jdk_ordering_avoiding_bridge_methods) { |
| return method; |
| } |
| } |
| throw new IllegalStateException("The method : '" + methodName + "' do not exist in '" + clazz.getSimpleName() + "'"); |
| } |
| } |