001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.collections4.functors; 018 019import java.lang.reflect.InvocationTargetException; 020import java.lang.reflect.Method; 021import java.util.Objects; 022 023import org.apache.commons.collections4.FunctorException; 024import org.apache.commons.collections4.Transformer; 025 026/** 027 * Transformer implementation that creates a new object instance by reflection. 028 * <p> 029 * <strong>WARNING:</strong> from v4.1 onwards this class will <strong>not</strong> be serializable anymore 030 * in order to prevent potential remote code execution exploits. Please refer to 031 * <a href="https://issues.apache.org/jira/browse/COLLECTIONS-580">COLLECTIONS-580</a> 032 * for more details. 033 * </p> 034 * 035 * @param <T> the type of the input to the function. 036 * @param <R> the type of the result of the function. 037 * @since 3.0 038 */ 039public class InvokerTransformer<T, R> implements Transformer<T, R> { 040 041 /** 042 * Gets an instance of this transformer calling a specific method with no arguments. 043 * 044 * @param <I> the input type 045 * @param <O> the output type 046 * @param methodName the method name to call 047 * @return an invoker transformer 048 * @throws NullPointerException if methodName is null 049 * @since 3.1 050 */ 051 public static <I, O> Transformer<I, O> invokerTransformer(final String methodName) { 052 return new InvokerTransformer<>(Objects.requireNonNull(methodName, "methodName")); 053 } 054 /** 055 * Gets an instance of this transformer calling a specific method with specific values. 056 * 057 * @param <I> the input type 058 * @param <O> the output type 059 * @param methodName the method name to call 060 * @param paramTypes the parameter types of the method 061 * @param args the arguments to pass to the method 062 * @return an invoker transformer 063 * @throws NullPointerException if methodName is null 064 * @throws IllegalArgumentException if paramTypes does not match args 065 */ 066 public static <I, O> Transformer<I, O> invokerTransformer(final String methodName, final Class<?>[] paramTypes, 067 final Object[] args) { 068 Objects.requireNonNull(methodName, "methodName"); 069 if (paramTypes == null && args != null 070 || paramTypes != null && args == null 071 || paramTypes != null && args != null && paramTypes.length != args.length) { 072 throw new IllegalArgumentException("The parameter types must match the arguments"); 073 } 074 if (paramTypes == null || paramTypes.length == 0) { 075 return new InvokerTransformer<>(methodName); 076 } 077 return new InvokerTransformer<>(methodName, paramTypes, args); 078 } 079 /** The method name to call */ 080 private final String iMethodName; 081 082 /** The array of reflection parameter types */ 083 private final Class<?>[] iParamTypes; 084 085 /** The array of reflection arguments */ 086 private final Object[] iArgs; 087 088 /** 089 * Constructor for no arg instance. 090 * 091 * @param methodName the method to call 092 */ 093 private InvokerTransformer(final String methodName) { 094 iMethodName = methodName; 095 iParamTypes = null; 096 iArgs = null; 097 } 098 099 /** 100 * Constructor that performs no validation. 101 * Use {@code invokerTransformer} if you want that. 102 * <p> 103 * Note: from 4.0, the input parameters will be cloned 104 * 105 * @param methodName the method to call 106 * @param paramTypes the constructor parameter types 107 * @param args the constructor arguments 108 */ 109 public InvokerTransformer(final String methodName, final Class<?>[] paramTypes, final Object[] args) { 110 iMethodName = methodName; 111 iParamTypes = paramTypes != null ? paramTypes.clone() : null; 112 iArgs = args != null ? args.clone() : null; 113 } 114 115 /** 116 * Transforms the input to result by invoking a method on the input. 117 * 118 * @param input the input object to transform 119 * @return the transformed result, null if null input 120 */ 121 @Override 122 @SuppressWarnings("unchecked") 123 public R transform(final Object input) { 124 if (input == null) { 125 return null; 126 } 127 try { 128 final Class<?> cls = input.getClass(); 129 final Method method = cls.getMethod(iMethodName, iParamTypes); 130 return (R) method.invoke(input, iArgs); 131 } catch (final NoSuchMethodException ex) { 132 throw new FunctorException("InvokerTransformer: The method '" + iMethodName + "' on '" + 133 input.getClass() + "' does not exist"); 134 } catch (final IllegalAccessException ex) { 135 throw new FunctorException("InvokerTransformer: The method '" + iMethodName + "' on '" + 136 input.getClass() + "' cannot be accessed"); 137 } catch (final InvocationTargetException ex) { 138 throw new FunctorException("InvokerTransformer: The method '" + iMethodName + "' on '" + 139 input.getClass() + "' threw an exception", ex); 140 } 141 } 142 143}