org.xwiki.velocity.introspection.MethodArgumentsUberspector Maven / Gradle / Ivy
Show all versions of xwiki-commons-velocity Show documentation
/*
* See the NOTICE file distributed with this work for additional
* information regarding copyright ownership.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.xwiki.velocity.introspection;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.Iterator;
import org.apache.commons.lang3.ClassUtils;
import org.apache.commons.lang3.exception.ExceptionUtils;
import org.apache.commons.lang3.reflect.TypeUtils;
import org.apache.velocity.runtime.RuntimeServices;
import org.apache.velocity.util.RuntimeServicesAware;
import org.apache.velocity.util.introspection.AbstractChainableUberspector;
import org.apache.velocity.util.introspection.Info;
import org.apache.velocity.util.introspection.VelMethod;
import org.xwiki.component.manager.ComponentLookupException;
import org.xwiki.component.manager.ComponentManager;
import org.xwiki.properties.ConverterManager;
import org.xwiki.velocity.internal.inrospection.WrappingVelMethod;
/**
* Chainable Velocity Uberspector that tries to convert method arguments to formal parameter types when the passed
* arguments don't match the method signature. In other words, it looks for a method matching the passed arguments and
* if none is found then it tries the convert the arguments to match the available method signatures (the available
* methods with the same name and the same number of parameters but with different parameter types). E.g.:
*
* * {@code $obj.someMethod('VALUE')
* // will forward to
* obj.someMethod(SomeEnum.VALUE)
* // if obj has someMethod(SomeEnum) and not someMethod(String)}
*
*
* Also handle Optional return values and unwraps them.
*
* @since 4.1M2
* @version $Id: 5225e2b9ed46dfe0fcad7eac955bf849ddbae084 $
*/
// TODO: Implement TypeConversionHandler instead
public class MethodArgumentsUberspector extends AbstractChainableUberspector implements RuntimeServicesAware
{
/**
* The component used to convert method arguments to formal parameter types.
*/
private ConverterManager converterManager;
@Override
public void setRuntimeServices(RuntimeServices runtimeServices)
{
super.setRuntimeServices(runtimeServices);
ComponentManager componentManager =
(ComponentManager) runtimeServices.getApplicationAttribute(ComponentManager.class.getName());
try {
this.converterManager = componentManager.getInstance(ConverterManager.class);
} catch (ComponentLookupException e) {
this.log.warn(
"Failed to find an implementation for [{}]. Working in degraded mode without Velocity "
+ "parameter conversion. Root cause: [{}]",
ConverterManager.class.getName(), ExceptionUtils.getRootCauseMessage(e));
}
}
@Override
public VelMethod getMethod(Object obj, String methodName, Object[] args, Info i)
{
// Let Velocity find a matching method. However, Velocity finds the closest matching method.
// According to the JavaDoc of MethodMap:
// "Attempts to find the most specific applicable method using the algorithm described in the JLS section
// 15.12.2 (with the exception that it can't distinguish a primitive type argument from an object type
// argument, since in reflection primitive type arguments are represented by their object counterparts, so for
// an argument of type (say) java.lang.Integer, it will not be able to decide between a method that takes int
// and a method that takes java.lang.Integer as a parameter."
// Thus we need to apply the following logic:
// - if the returned VelMethod has a different number of parameters than the signature asked for, then go into
// our conversion code
// - if our conversion code doesn't find any match, then return the VelMethod found by Velocity.
VelMethod initialVelMethod = super.getMethod(obj, methodName, args, i);
VelMethod velMethod = initialVelMethod;
boolean shouldConvert = false;
if (this.converterManager != null) {
if (velMethod == null) {
shouldConvert = true;
} else {
Method method = velMethod.getMethod();
boolean sameParameterNumbers = method.getParameterTypes().length == args.length;
if (!sameParameterNumbers) {
shouldConvert = true;
}
}
}
if (shouldConvert) {
// Try to convert method arguments to formal parameter types.
Object[] convertedArguments = this.convertArguments(obj, methodName, args);
if (convertedArguments != null) {
velMethod = super.getMethod(obj, methodName, convertedArguments, i);
if (velMethod != null) {
velMethod = new ConvertingVelMethod(velMethod);
} else {
velMethod = initialVelMethod;
}
}
}
return velMethod;
}
/**
* Converts the given arguments to match a method with the specified name and the same number of formal parameters
* as the number of arguments.
*
* @param obj the object the method is invoked on, used to retrieve the list of available methods
* @param methodName the method we're looking for
* @param args the method arguments
* @return a new array of arguments where some values have been converted to match the formal method parameter
* types, {@code null} if no such method is found
*/
private Object[] convertArguments(Object obj, String methodName, Object[] args)
{
for (Iterator it =
Arrays.stream(obj.getClass().getMethods()).filter(m -> isSupported(m, methodName, args))
.sorted((m1, m2) -> compare(m1, m2, methodName, args)).iterator(); it.hasNext();) {
Method method = it.next();
try {
return convertArguments(args, method.getGenericParameterTypes(), method.isVarArgs());
} catch (Exception e) {
// Ignore and try the next method.
}
}
return null;
}
private boolean isSupported(Method method, String methodName, Object[] args)
{
return method.getName().equalsIgnoreCase(methodName)
&& (method.getGenericParameterTypes().length == args.length || method.isVarArgs());
}
private int compare(Method method1, Method method2, String methodName, Object[] args)
{
// Compare method names
int value = compareMethodNames(method1, method2, methodName);
if (value != 0) {
return value;
}
// Compare compatible number of parameters
value = compareMethodArguments(method1, method2, methodName, args);
if (value != 0) {
return value;
}
// Compare tostring to maintain the stable priority list across environments
return method1.toString().compareTo(method2.toString());
}
private int compareMethodNames(Method method1, Method method2, String methodName)
{
if (!method1.getName().equals(method2.getName())) {
if (methodName.equals(method1.getName())) {
// Priority to exact matching method name
return -1;
} else if (methodName.equals(method2.getName())) {
// Priority to exact matching method name
return 1;
}
}
return 0;
}
private int compareMethodArguments(Method method1, Method method2, String methodName, Object[] args)
{
double compatibleArtgumentsCount1 = countCompatibleParameters(method1, args);
double compatibleArtgumentsCount2 = countCompatibleParameters(method2, args);
if (compatibleArtgumentsCount1 > compatibleArtgumentsCount2) {
// Priority to higher number of compatible parameters
return -1;
} else if (compatibleArtgumentsCount1 < compatibleArtgumentsCount2) {
// Priority to higher number of compatible parameters
return 1;
} else if (!method1.isVarArgs() && method2.isVarArgs()) {
// Priority to non vararg method in case of same number of compatible parameters
return -1;
} else if (method1.isVarArgs() && !method2.isVarArgs()) {
// Priority to non vararg method in case of same number of compatible parameters
return 1;
}
return 0;
}
private double countCompatibleParameters(Method method, Object[] args)
{
double number = 0;
Type[] types = method.getGenericParameterTypes();
for (int i = 0; i < args.length && i < types.length; ++i) {
Type type = types[i];
Object arg = args[i];
if (arg != null) {
if (arg.getClass() == type) {
// Give high priority to exact same type
number += 10;
} else if (TypeUtils.isInstance(arg, type)) {
// Give normal priority to compatible type
number += 1;
} else if (type instanceof Class) {
Class typeClass = (Class) type;
if (isPrimitive(typeClass, arg)) {
// Class wrapper and corresponding primitive are interpreted as compatible type
number += 1;
} else if (isNumber(typeClass, arg)) {
// Give slight priority to number conversion since it's a very common use cases
number += 0.5;
}
}
}
}
return number;
}
private boolean isPrimitive(Class typeClass, Object arg)
{
return typeClass.isPrimitive() && ClassUtils.primitiveToWrapper(typeClass) == arg.getClass();
}
private boolean isNumber(Class typeClass, Object arg)
{
return ClassUtils.isAssignable(typeClass, Number.class) && TypeUtils.isInstance(arg, Number.class);
}
/**
* Tries to convert the given arguments to match the specified formal parameters types.
*
* Throws a runtime exception if the conversion fails.
*
* @param arguments the method actual arguments
* @param parameterTypes the method formal parameter types
* @param isVarArgs true if the method contains a varargs (ie the last parameter is a varargs)
* @return a new array of arguments where some values have been converted to match the formal method parameter types
*/
private Object[] convertArguments(Object[] arguments, Type[] parameterTypes, boolean isVarArgs)
{
Object[] convertedArguments = Arrays.copyOf(arguments, arguments.length);
for (int i = 0; i < arguments.length; i++) {
// Try to convert the argument if it's not null and if it doesn't match the parameter type.
// If the method is a varargs then extract the type from the vararg array
Type expectedType;
if (isVarArgs && i >= parameterTypes.length - 1) {
expectedType = ((Class) parameterTypes[parameterTypes.length - 1]).getComponentType();
} else {
expectedType = parameterTypes[i];
}
if (arguments[i] != null && !TypeUtils.isInstance(arguments[i], expectedType)) {
convertedArguments[i] = this.converterManager.convert(expectedType, arguments[i]);
}
}
return convertedArguments;
}
/**
* Wrapper for a real VelMethod that converts the passed arguments to the real arguments expected by the method.
*
* @version $Id: 5225e2b9ed46dfe0fcad7eac955bf849ddbae084 $
*/
private class ConvertingVelMethod extends WrappingVelMethod
{
ConvertingVelMethod(VelMethod realMethod)
{
super(realMethod);
}
@Override
public Object invoke(Object o, Object[] params) throws IllegalAccessException, InvocationTargetException
{
return getWrappedVelMethod().invoke(o, convertArguments(o, getWrappedVelMethod().getMethodName(), params));
}
}
}