1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License
15  */
16 
17 package com.android.compatibility.common.util;
18 
19 import java.lang.reflect.InvocationTargetException;
20 import java.lang.reflect.Method;
21 import java.util.ArrayList;
22 import java.util.Arrays;
23 import java.util.List;
24 import java.util.regex.Matcher;
25 import java.util.regex.Pattern;
26 
27 import org.junit.AssumptionViolatedException;
28 
29 /**
30  * Resolves methods provided by the BusinessLogicService and invokes them
31  */
32 public abstract class BusinessLogicExecutor {
33 
34     protected static final String LOG_TAG = "BusinessLogicExecutor";
35 
36     /** String representations of the String class and String[] class */
37     protected static final String STRING_CLASS = "java.lang.String";
38     protected static final String STRING_ARRAY_CLASS = "[Ljava.lang.String;";
39 
40     private static final String REDACTED_PLACEHOLDER = "[redacted]";
41     /* List of regexes indicating a method arg should be redacted in the logs */
42     protected List<String> mRedactionRegexes = new ArrayList<>();
43 
44     /**
45      * Execute a business logic condition.
46      * @param method the name of the method to invoke. Must include fully qualified name of the
47      * enclosing class, followed by '.', followed by the name of the method
48      * @param args the string arguments to supply to the method
49      * @return the return value of the method invoked
50      * @throws RuntimeException when failing to resolve or invoke the condition method
51      */
executeCondition(String method, String... args)52     public boolean executeCondition(String method, String... args) {
53         logDebug("Executing condition: %s", formatExecutionString(method, args));
54         try {
55             return (Boolean) invokeMethod(method, args);
56         } catch (ClassNotFoundException | IllegalAccessException | InstantiationException |
57                 InvocationTargetException | NoSuchMethodException e) {
58             throw new RuntimeException(String.format(
59                     "BusinessLogic: Failed to invoke condition method %s with args: %s", method,
60                     Arrays.toString(args)), e);
61         }
62     }
63 
64     /**
65      * Execute a business logic action.
66      * @param method the name of the method to invoke. Must include fully qualified name of the
67      * enclosing class, followed by '.', followed by the name of the method
68      * @param args the string arguments to supply to the method
69      * @throws RuntimeException when failing to resolve or invoke the action method
70      */
executeAction(String method, String... args)71     public void executeAction(String method, String... args) {
72         logDebug("Executing action: %s", formatExecutionString(method, args));
73         try {
74             invokeMethod(method, args);
75         } catch (ClassNotFoundException | IllegalAccessException | InstantiationException |
76                 NoSuchMethodException e) {
77             throw new RuntimeException(String.format(
78                     "BusinessLogic: Failed to invoke action method %s with args: %s", method,
79                     Arrays.toString(args)), e);
80         } catch (InvocationTargetException e) {
81             // This action throws an exception, so throw the original exception (e.g.
82             // AssertionFailedError) for a more readable stacktrace.
83             Throwable t = e.getCause();
84             if (AssumptionViolatedException.class.isInstance(t)) {
85                 // This is an assumption failure (registered as a "pass") so don't wrap this
86                 // throwable in a RuntimeException
87                 throw (AssumptionViolatedException) t;
88             } else {
89                 RuntimeException re = new RuntimeException(t.getMessage(), t.getCause());
90                 re.setStackTrace(t.getStackTrace());
91                 throw re;
92             }
93         }
94     }
95 
96     /**
97      * Format invocation information as "method(args[0], args[1], ...)".
98      */
formatExecutionString(String method, String... args)99     protected abstract String formatExecutionString(String method, String... args);
100 
101     /** Substitute sensitive information with REDACTED_PLACEHOLDER if necessary. */
formatArgs(String[] args)102     protected String[] formatArgs(String[] args) {
103         List<String> formattedArgs = new ArrayList<>();
104         for (String arg : args) {
105             formattedArgs.add(formatArg(arg));
106         }
107         return formattedArgs.toArray(new String[0]);
108     }
109 
formatArg(String arg)110     private String formatArg(String arg) {
111         for (String regex : mRedactionRegexes) {
112             Pattern pattern = Pattern.compile(regex);
113             Matcher matcher = pattern.matcher(arg);
114             if (matcher.find()) {
115                 return REDACTED_PLACEHOLDER;
116             }
117         }
118         return arg;
119     }
120 
121     /**
122      * Execute a business logic method.
123      * @param method the name of the method to invoke. Must include fully qualified name of the
124      * enclosing class, followed by '.', followed by the name of the method
125      * @param args the string arguments to supply to the method
126      * @return the return value of the method invoked (type Boolean if method is a condition)
127      * @throws RuntimeException when failing to resolve or invoke the method
128      */
invokeMethod(String method, String... args)129     protected Object invokeMethod(String method, String... args) throws ClassNotFoundException,
130             IllegalAccessException, InstantiationException, InvocationTargetException,
131             NoSuchMethodException {
132         // Method names served by the BusinessLogic service should assume format
133         // classname.methodName, but also handle format classname#methodName since test names use
134         // this format
135         int index = (method.indexOf('#') == -1) ? method.lastIndexOf('.') : method.indexOf('#');
136         if (index == -1) {
137             throw new RuntimeException(String.format("BusinessLogic: invalid method name "
138                     + "\"%s\". Method string must include fully qualified class name. "
139                     + "For example, \"com.android.packagename.ClassName.methodName\".", method));
140         }
141         String className = method.substring(0, index);
142         Class cls = Class.forName(className);
143         Object obj = null;
144         if (getTestObject() != null && cls.isAssignableFrom(getTestObject().getClass())) {
145             // The given method is a member of the test class, use the known test class instance
146             obj = getTestObject();
147         } else {
148             // Only instantiate a new object if we don't already have one.
149             // Otherwise the class could have been an interface which isn't instantiatable.
150             obj = cls.getDeclaredConstructor().newInstance();
151         }
152         ResolvedMethod rm = getResolvedMethod(cls, method.substring(index + 1), args);
153         return rm.invoke(obj);
154     }
155 
156     /**
157      * Log information with whichever logging mechanism is available to the instance. This varies
158      * from host-side to device-side, so implementations are left to subclasses.
159      * See {@link String.format(String, Object...)} for parameter information.
160      */
logInfo(String format, Object... args)161     public abstract void logInfo(String format, Object... args);
162 
163     /**
164      * Log debugging information to the host or device logs (depending on implementation).
165      * See {@link String.format(String, Object...)} for parameter information.
166      */
logDebug(String format, Object... args)167     public abstract void logDebug(String format, Object... args);
168 
169     /**
170      * Get the test object. This method is left abstract, since non-abstract subclasses will set
171      * the test object in the constructor.
172      * @return the test case instance
173      */
getTestObject()174     protected abstract Object getTestObject();
175 
176     /**
177      * Get the method and list of arguments corresponding to the class, method name, and proposed
178      * argument values, in the form of a {@link ResolvedMethod} object. This object stores all
179      * information required to successfully invoke the method. getResolvedMethod is left abstract,
180      * since argument types differ between device-side (e.g. Context) and host-side
181      * (e.g. ITestDevice) implementations of this class.
182      * @param cls the Class to which the method belongs
183      * @param methodName the name of the method to invoke
184      * @param args the string arguments to use when invoking the method
185      * @return a {@link ResolvedMethod}
186      * @throws ClassNotFoundException
187      */
getResolvedMethod(Class cls, String methodName, String... args)188     protected abstract ResolvedMethod getResolvedMethod(Class cls, String methodName,
189             String... args) throws ClassNotFoundException;
190 
191     /**
192      * Retrieve all methods within a class that match a given name
193      * @param cls the class
194      * @param name the method name
195      * @return a list of method objects
196      */
getMethodsWithName(Class cls, String name)197     protected List<Method> getMethodsWithName(Class cls, String name) {
198         List<Method> methodList = new ArrayList<>();
199         for (Method m : cls.getMethods()) {
200             if (name.equals(m.getName())) {
201                 methodList.add(m);
202             }
203         }
204         return methodList;
205     }
206 
207     /**
208      * Helper class for storing a method object, and a list of arguments to use when invoking the
209      * method. The class is also equipped with an "invoke" method for convenience.
210      */
211     protected static class ResolvedMethod {
212         private Method mMethod;
213         List<Object> mArgs;
214 
ResolvedMethod(Method method)215         public ResolvedMethod(Method method) {
216             mMethod = method;
217             mArgs = new ArrayList<>();
218         }
219 
220         /** Add an argument to the argument list for this instance */
addArg(Object arg)221         public void addArg(Object arg) {
222             mArgs.add(arg);
223         }
224 
225         /** Invoke the stored method with the stored args on a given object */
invoke(Object instance)226         public Object invoke(Object instance) throws IllegalAccessException,
227                 InvocationTargetException {
228             return mMethod.invoke(instance, mArgs.toArray());
229         }
230     }
231 }
232