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

ninja.params.ControllerMethodInvoker Maven / Gradle / Ivy

/**
 * Copyright (C) 2012-2017 the original author or authors.
 *
 * 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 ninja.params;

import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.util.ArrayList;
import java.util.List;

import ninja.Context;
import ninja.RoutingException;
import ninja.params.ParamParsers.ArrayParamParser;
import ninja.validation.Validator;
import ninja.validation.WithValidator;

import com.google.inject.AbstractModule;
import com.google.inject.Injector;
import java.lang.reflect.Type;
import java.util.Optional;
import ninja.exceptions.BadRequestException;
import ninja.utils.NinjaConstant;
import ninja.utils.NinjaProperties;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Invokes methods on the controller, extracting arguments out
 *
 */
public class ControllerMethodInvoker {
    
    private final static Logger logger = LoggerFactory.getLogger(ControllerMethodInvoker.class);
    
    private final Method method;
    private final ArgumentExtractor[] argumentExtractors;
    private final boolean useStrictArgumentExtractors;
    
    private static boolean nonStrictModeWarningLoggedAlready = false;

    ControllerMethodInvoker(
            Method method, 
            ArgumentExtractor[] argumentExtractors,
            boolean useStrictArgumentExtractors) {
        this.method = method;
        this.argumentExtractors = argumentExtractors;
        this.useStrictArgumentExtractors = useStrictArgumentExtractors;
    }

    public Object invoke(Object controller, Context context) {
        // Extract arguments
        Object[] arguments = new Object[argumentExtractors.length];
        for (int i = 0; i < argumentExtractors.length; i++) {
            arguments[i] = argumentExtractors[i].extract(context);
        }
        
        checkNullArgumentsAndThrowBadRequestExceptionIfConfigured(arguments);
        
        try {
            return method.invoke(controller, arguments);
        } catch (IllegalAccessException | IllegalArgumentException e) {
            throw new RuntimeException(e);
        } catch (InvocationTargetException e) {
            if (e.getCause() instanceof RuntimeException) {
                throw (RuntimeException) e.getCause();
            } else {
                throw new RuntimeException(e.getCause());
            }
        }
    }
    
    private void checkNullArgumentsAndThrowBadRequestExceptionIfConfigured(Object[] arguments) {
        if (!useStrictArgumentExtractors) {
            return;
        }
        for (Object object : arguments) {
            if (object == null) {
                throw new BadRequestException();
            }
        }
    }

    /**
     * Builds an invoker for a functional method.  Understands what parameters
     * to inject and extract based on type and annotations.
     * @param functionalMethod The method to be invoked
     * @param implementationMethod The method to use for determining what
     *      actual parameters and annotations to use for each argument.  Useful
     *      when type/lambda erasure makes the functional interface not reliable
     *      for reflecting.
     * @param injector The guice injector
     * @param ninjaProperties The NinjaProperties of this application
     * @return An invoker
     */
    public static ControllerMethodInvoker build(
            Method functionalMethod, 
            Method implementationMethod, 
            Injector injector,
            NinjaProperties ninjaProperties) {
        // get both the parameters...
        final Type[] genericParameterTypes = implementationMethod.getGenericParameterTypes();
        final MethodParameter[] methodParameters = MethodParameter.convertIntoMethodParameters(genericParameterTypes);
        // ... and all annotations for the parameters
        final Annotation[][] paramAnnotations = implementationMethod
                .getParameterAnnotations();

        ArgumentExtractor[] argumentExtractors = new ArgumentExtractor[methodParameters.length];

        // now we skip through the parameters and process the annotations
        for (int i = 0; i < methodParameters.length; i++) {
            try {
                argumentExtractors[i] = getArgumentExtractor(methodParameters[i], paramAnnotations[i],
                        injector);
            } catch (RoutingException e) {
                throw new RoutingException("Error building argument extractor for parameter " + i +
                        " in method " + implementationMethod.getDeclaringClass().getName() + "." + implementationMethod.getName() + "()", e);
            }
        }

        // Replace a null extractor with a bodyAs extractor, but make sure there's only one
        int bodyAsFound = -1;
        for (int i = 0; i < argumentExtractors.length; i++) {
            if (argumentExtractors[i] == null) {
                if (bodyAsFound > -1) {
                    throw new RoutingException("Only one parameter may be deserialised as the body "
                            + implementationMethod.getDeclaringClass().getName() + "." + implementationMethod.getName() + "()\n"
                            + "Extracted parameter is type: " + methodParameters[bodyAsFound].parameterClass.getName() + "\n"
                            + "Extra parmeter is type: " + methodParameters[i].parameterClass.getName());
                } else {
                    argumentExtractors[i] = new ArgumentExtractors.BodyAsExtractor(methodParameters[i].parameterClass);
                    bodyAsFound = i;
                }
            }
        }

        // Now that every parameter has an argument extractor we can run validation on the annotated
        // parameters
        for (int i = 0; i < argumentExtractors.length; i++) {
            argumentExtractors[i] =
                    validateArgumentWithExtractor(
                            methodParameters[i],
                            paramAnnotations[i], 
                            injector,
                            argumentExtractors[i]);
        }
        
        boolean useStrictArgumentExtractors = determineWhetherToUseStrictArgumentExtractorMode(ninjaProperties);

        return new ControllerMethodInvoker(functionalMethod, argumentExtractors, useStrictArgumentExtractors);
    }
    
    private static boolean determineWhetherToUseStrictArgumentExtractorMode(NinjaProperties ninjaProperties) {
        boolean useStrictArgumentExtractors = ninjaProperties.getBooleanWithDefault(NinjaConstant.NINJA_STRICT_ARGUMENT_EXTRACTORS, false);
  
        if (useStrictArgumentExtractors == false && nonStrictModeWarningLoggedAlready == false) {
            String message = "Using deprecated non-strict mode for injection of parameters into controller "
                    + "(" + NinjaConstant.NINJA_STRICT_ARGUMENT_EXTRACTORS + " = false). "
                    + "This mode will soon be removed from Ninja. Make sure you upgrade your application as soon as possible. "
                    + "More: http://www.ninjaframework.org/documentation/basic_concepts/controllers.html 'A note about null and Optional'.";
            logger.warn(message);
            nonStrictModeWarningLoggedAlready = true;
        }
        
        return useStrictArgumentExtractors;
    }

    private static ArgumentExtractor getArgumentExtractor(
            MethodParameter methodParameter,
            Annotation[] annotations, Injector injector) {
        // First check if we have a static extractor
        ArgumentExtractor extractor = ArgumentExtractors.getExtractorForType(methodParameter.parameterClass);

        if (extractor == null) {
            // See if we have a WithArgumentExtractors annotated annotation for specialized extractors
            for (Annotation annotation : annotations) {
                WithArgumentExtractors withArgumentExtractors = annotation.annotationType()
                        .getAnnotation(WithArgumentExtractors.class);
                if (withArgumentExtractors != null) {
                    for (Class> argumentExtractor : withArgumentExtractors.value()) {
                        Class extractedType = (Class) ((ParameterizedType)(argumentExtractor.getGenericInterfaces()[0])).getActualTypeArguments()[0];
                        if (methodParameter.parameterClass.isAssignableFrom(extractedType)) {
                            extractor = instantiateComponent(argumentExtractor, annotation,
                                    methodParameter.parameterClass, injector);
                            break;
                        }
                    }
                }
            }
        }
        
        if (extractor == null) {
            // See if we have a WithArgumentExtractor annotated annotation
            for (Annotation annotation : annotations) {
                WithArgumentExtractor withArgumentExtractor = annotation.annotationType()
                        .getAnnotation(WithArgumentExtractor.class);
                if (withArgumentExtractor != null) {
                    extractor = instantiateComponent(withArgumentExtractor.value(), annotation,
                            methodParameter.parameterClass, injector);
                    break;
                }
            }
        }

        return extractor;
    }

    private static ArgumentExtractor validateArgumentWithExtractor(
            MethodParameter methodParameter,
            Annotation[] annotations, 
            Injector injector, 
            ArgumentExtractor extractor) {
        // We have validators that get applied before parsing, and validators
        // that get applied after parsing.
        List> preParseValidators = new ArrayList<>();
        List> postParseValidators = new ArrayList<>();

        Class boxedParamType = methodParameter.parameterClass;
        if (methodParameter.parameterClass.isPrimitive()) {
            boxedParamType = box(methodParameter.parameterClass);
        }

        // Now we have an extractor, lets apply validators that are able to validate
        for (Annotation annotation : annotations) {
            WithValidator withValidator = annotation.annotationType()
                    .getAnnotation(WithValidator.class);
            if (withValidator != null) {
                Validator validator = instantiateComponent(withValidator.value(), annotation,
                        methodParameter.parameterClass, injector);
                // If the validator can validate the extractors type, then it's a pre parse validator
                if (validator.getValidatedType().isAssignableFrom(extractor.getExtractedType())) {
                    preParseValidators.add(validator);
                    // If it can validate the parameter type, it's a post parse validator
                } else if (validator.getValidatedType().isAssignableFrom(boxedParamType)) {
                    postParseValidators.add(validator);
                    // Otherwise, we can't validate with this validator
                } else {
                    throw new RoutingException("Validator for field " + extractor.getFieldName() +
                            " validates type " + validator.getValidatedType() +
                            ", which doesn't match extracted type " + extractor.getExtractedType() +
                            " or parameter type " + methodParameter.parameterClass);
                }
            }
        }

        // If we have pre parse validators, wrap our extractor in them
        if (!preParseValidators.isEmpty()) {
            extractor = new ValidatingArgumentExtractor(extractor, preParseValidators);
        }

        // Either the extractor extracts a type that matches the param type, or it's a
        // String, and we can lookup a parser to parse it into the param type
        if (!boxedParamType.isAssignableFrom(extractor.getExtractedType())) {
            if (extractor.getFieldName() != null) {
                if (String.class.isAssignableFrom(extractor.getExtractedType())) {
                    // Look up a parser for a single-valued parameter
                    ParamParser parser = injector.getInstance(ParamParsers.class).getParamParser(methodParameter.parameterClass);
                    if (parser == null) {
                        throw new RoutingException("Can't find parameter parser for type "
                                + extractor.getExtractedType() + " on field "
                                + extractor.getFieldName());
                    } else {
                        extractor =
                                new ParsingArgumentExtractor(extractor, parser);
                        
                    }
                } else if (String[].class.isAssignableFrom(extractor.getExtractedType())) {
                    // Look up a parser for a multi-valued parameter
                    ArrayParamParser parser = injector.getInstance(ParamParsers.class).getArrayParser(methodParameter.parameterClass);
                    if (parser == null) {
                        throw new RoutingException("Can't find parameter array parser for type "
                                + extractor.getExtractedType() + " on field "
                                + extractor.getFieldName());
                    } else {
                        extractor =
                                new ParsingArrayExtractor(extractor, parser);
                    }

                } else {
                    throw new RoutingException("Extracted type " + extractor.getExtractedType()
                            + " for field " + extractor.getFieldName()
                            + " doesn't match parameter type " + methodParameter.parameterClass);
                }
            }
        }

        // If we have any post parse validators, wrap our extractor in them
        if (!postParseValidators.isEmpty()) {
            extractor = new ValidatingArgumentExtractor(extractor, postParseValidators);
        }
        
        if (methodParameter.isOptional) {
            extractor = new OptionalArgumentExtractor(extractor);
        }

        return extractor;
    }

    private static  T instantiateComponent(Class argumentExtractor,
            final Annotation annotation, final Class paramType,
            Injector injector) {
        // Noarg constructor
        Constructor noarg = getNoArgConstructor(argumentExtractor);
        if (noarg != null) {
            try {
                return (T) noarg.newInstance();
            } catch (Exception e) {
                throw new RoutingException(e);
            }
        }
        // Simple case, just takes the annotation
        Constructor simple = getSingleArgConstructor(argumentExtractor, annotation.annotationType());
        if (simple != null) {
            try {
                return (T) simple.newInstance(annotation);
            } catch (Exception e) {
                throw new RoutingException(e);
            }
        }
        // Simple case, just takes the parsed class
        Constructor simpleClass = getSingleArgConstructor(argumentExtractor, Class.class);
        if (simpleClass != null) {
            try {
                return (T) simpleClass.newInstance(paramType);
            } catch (Exception e) {
                throw new RoutingException(e);
            }
        }
        // Complex case, use Guice.  Create a child injector with the annotation in it.
        return injector.createChildInjector(new AbstractModule() {
            @Override
            protected void configure() {
                bind((Class) annotation.annotationType()).toInstance(annotation);
                bind(ArgumentClassHolder.class).toInstance(new ArgumentClassHolder(paramType));
            }
        }).getInstance(argumentExtractor);
    }

    private static Constructor getNoArgConstructor(Class clazz) {
        for (Constructor constructor : clazz.getConstructors()) {
            if (constructor.getParameterTypes().length == 0) {
                return constructor;
            }
        }
        return null;
    }

    private static Constructor getSingleArgConstructor(Class clazz, Class arg) {
        for (Constructor constructor : clazz.getConstructors()) {
            if (constructor.getParameterTypes().length == 1) {
                if (constructor.getParameterTypes()[0].isAssignableFrom(arg)) {
                    return constructor;
                }
            }
        }
        return null;
    }

    private static Class box(Class typeToBox) {
        if (typeToBox == int.class) {
            return Integer.class;
        } else if (typeToBox == boolean.class) {
            return Boolean.class;
        } else if (typeToBox == long.class) {
            return Long.class;
        } else if (typeToBox == float.class) {
            return Float.class;
        } else if (typeToBox == double.class) {
            return Double.class;
        } else if (typeToBox == byte.class) {
            return Byte.class;
        } else if (typeToBox == short.class) {
            return Short.class;
        } else if (typeToBox == char.class) {
            return Character.class;
        }
        throw new IllegalArgumentException("Don't know how to box type of " + typeToBox);
    }
    
    /**
     * Just a little helper that makes it possible to handle things like
     *     myControllerMethod(@Param("param1") Optional myValue)
     * 
     * It investigates the type parameter and allows to remember whether a type
     * was wrapped in an Optional or not. It stores the "real" type of the parameter
     * that the extractor should extract (String and not Optional in example above).
     */
    private static class MethodParameter {
        public boolean isOptional;
        public Class parameterClass;

        private MethodParameter(Type genericType) {
            try {
                // a ParameterizedType is something like Optional or List...
                if (genericType instanceof ParameterizedType) {
                    ParameterizedType parameterizedType = (ParameterizedType) genericType;
                    Class maybeOptional = getClass(parameterizedType.getRawType());

                    // The method expects an Optional, so we extract the first 
                    // generic which should determine the extractor that will be used to 
                    // extract the value;
                    if (maybeOptional.isAssignableFrom(Optional.class)) {
                        isOptional = true;
                        parameterClass = getClass(parameterizedType.getActualTypeArguments()[0]);
                    }
                }
                
                if (parameterClass == null) {
                    isOptional = false;
                    parameterClass = getClass(genericType);
                }
            } catch (Exception e) {
                throw new RuntimeException("Oops. Something went wrong while investigating method parameters for controller class invocation", e);
            }

        }
        
        public static MethodParameter[] convertIntoMethodParameters(Type[] genericParameterTypes) {
            MethodParameter[] methodParameters = new MethodParameter[genericParameterTypes.length];
            for (int i = 0; i < genericParameterTypes.length; i++) {
                methodParameters[i] = new MethodParameter(genericParameterTypes[i]);
            }
            return methodParameters;
        }
        
        private Class getClass(Type type) {
            if (type instanceof Class) {
                return (Class) type;
            } else {
                throw new RuntimeException(
                        "Oops. That's a strange internal Ninja error.\n"
                        + "Seems someone tried to convert a type into a class that is not a real class. ( " + type.getTypeName() + ")");
            }
        }
    
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy