All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.tarantool.driver.mappers.MapperReflectionUtils Maven / Gradle / Ivy

package io.tarantool.driver.mappers;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.WildcardType;
import java.util.stream.Stream;

/**
 * Contains helper methods for converter classes lookup and determining its parameters at runtime
 */
public final class MapperReflectionUtils {

    private MapperReflectionUtils() {
    }

    /**
     * Get class for the runtime target type parameter of a converter
     *
     * @param converter a converter, must have at least one generic interface with 2 type parameters
     * @param        the target converter type
     * @return the converter class
     * @throws InterfaceParameterClassNotFoundException if the class of the parameter type on the specified position
     *                                                  cannot be determined or is not found
     */
    @SuppressWarnings("unchecked")
    public static  Class getConverterTargetType(Object converter)
        throws InterfaceParameterClassNotFoundException {
        Type[] genericInterfaces = getGenericInterfaces(converter);
        if (genericInterfaces.length < 1) {
            throw new RuntimeException(
                String.format("The passed converter object of type %s does not extend any generic interface",
                    converter.getClass()));
        }
        if (genericInterfaces.length > 1) {
            throw new RuntimeException(
                String.format("The passed converter object of type %s has more than one generic interfaces, " +
                    "unable to determine the target", converter.getClass()));
        }
        if (!(genericInterfaces[0] instanceof ParameterizedType)) {
            throw new RuntimeException(
                String.format("The passed converter object of type %s interface type is not a parameterized type",
                    converter.getClass()));
        }
        try {
            return (Class) Class.forName(
                ((ParameterizedType) genericInterfaces[0]).getActualTypeArguments()[1].getTypeName());
        } catch (ClassNotFoundException e) {
            throw new InterfaceParameterClassNotFoundException(e);
        }
    }

    /**
     * Call {@link #getInterfaceParameterType(Object, Class, int)} and get a class for the returned type
     *
     * @param converter             a converter, must have at least one generic interface with 2 type parameters
     * @param interfaceClass        the target converter generic interface
     * @param parameterTypePosition the position of the generic type parameter in the interface definition
     * @param                    the target generic interface parameter type
     * @return the generic interface parameter class
     * @throws InterfaceParameterTypeNotFoundException  if the parameter type on the specified position cannot be
     *                                                  determined
     * @throws InterfaceParameterClassNotFoundException if the class of the parameter type on the specified position
     *                                                  cannot be determined or is not found
     */
    @SuppressWarnings("unchecked")
    static  Class getInterfaceParameterClass(
        Object converter,
        Class interfaceClass,
        int parameterTypePosition)
        throws InterfaceParameterTypeNotFoundException, InterfaceParameterClassNotFoundException {
        try {
            return (Class) Class.forName(
                getInterfaceParameterType(converter, interfaceClass, parameterTypePosition).getTypeName());
        } catch (ClassNotFoundException e) {
            throw new InterfaceParameterClassNotFoundException(e);
        }
    }

    /**
     * 
{@code
     * SomeClass implements ValueConverter, ObjectConverter
     * <- interfaceClass: ValueConverter, parameterTypePosition: 0
     * -> V
     * }
* * @param converter a converter, must have at least one generic interface with 2 type parameters * @param interfaceClass the target converter generic interface * @param parameterTypePosition the position of the generic type parameter in the interface definition * @return the generic interface parameter type * @throws InterfaceParameterTypeNotFoundException if the parameter type on the specified position cannot be * determined * @throws InterfaceParameterClassNotFoundException if the class of the parameter type on the specified position * cannot be determined or is not found */ private static Type getInterfaceParameterType( Object converter, Class interfaceClass, int parameterTypePosition) throws InterfaceParameterTypeNotFoundException, InterfaceParameterClassNotFoundException { Type[] genericInterfaces = getGenericInterfaces(converter); try { for (Type iface : genericInterfaces) { if (iface instanceof ParameterizedType) { ParameterizedType parameterizedType = (ParameterizedType) iface; if (Class.forName(parameterizedType.getRawType().getTypeName()).isAssignableFrom(interfaceClass)) { return getParameterType(parameterizedType, parameterTypePosition); } } } } catch (ClassNotFoundException e) { throw new InterfaceParameterClassNotFoundException(e); } throw new InterfaceParameterTypeNotFoundException( "Unable to determine the generic parameter type on position %d for %s. " + "Either the class does not implement any generic interfaces or the parametrized types " + "cannot be determined due to type erasure", parameterTypePosition, converter.getClass()); } private static Type[] getGenericInterfaces(Object converter) { Type[] genericInterfaces = converter.getClass().getGenericInterfaces(); if (genericInterfaces == null) { throw new RuntimeException( String.format("Unable to determine the generic interfaces for %s", converter.getClass())); } return genericInterfaces; } /** *
{@code
     * ValueConverter
     * <- parameterTypePosition: 0
     * -> V
     * }
* * @param parameterizedType a type of the generic interface parameter type * @param parameterTypePosition the position of the generic type parameter in the interface definition * @return the generic interface parameter type */ private static Type getParameterType(ParameterizedType parameterizedType, int parameterTypePosition) { Type[] typeParams = parameterizedType.getActualTypeArguments(); if (typeParams == null || typeParams.length != 2) { throw new RuntimeException(String.format("Type %s does not have 2 type parameters", parameterizedType)); } if (typeParams[parameterTypePosition] instanceof ParameterizedType && Stream.of(((ParameterizedType) typeParams[parameterTypePosition]).getActualTypeArguments()) .allMatch(t -> t instanceof WildcardType)) { return ((ParameterizedType) typeParams[parameterTypePosition]).getRawType(); } else { return typeParams[parameterTypePosition]; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy