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.io.PrintWriter;
20 import java.io.StringWriter;
21 import java.util.ArrayList;
22 import java.util.Arrays;
23 import java.util.Date;
24 import java.util.HashMap;
25 import java.util.List;
26 import java.util.Map;
27 import java.util.Set;
28 
29 import org.junit.AssumptionViolatedException;
30 
31 /**
32  * Helper and constants accessible to host and device components that enable Business Logic
33  * configuration
34  */
35 public class BusinessLogic {
36 
37     // Device location to which business logic data is pushed
38     public static final String DEVICE_FILE = "/sdcard/bl";
39 
40     /* A map from testcase name to the business logic rules for the test case */
41     protected Map<String, List<BusinessLogicRulesList>> mRules;
42     /* Feature flag determining if device specific tests are executed. */
43     public boolean mConditionalTestsEnabled;
44     private AuthenticationStatusEnum mAuthenticationStatus = AuthenticationStatusEnum.UNKNOWN;
45 
46     // A Date denoting the time of request from the business logic service
47     protected Date mTimestamp;
48 
49     // A list of regexes triggering log redaction
50     protected List<String> mRedactionRegexes = new ArrayList<>();
51 
52     /**
53      * Determines whether business logic exists for a given test name
54      * @param testName the name of the test case, prefixed by fully qualified class name, then '#'.
55      * For example, "com.android.foo.FooTest#testFoo"
56      * @return whether business logic exists for this test for this suite
57      */
hasLogicFor(String testName)58     public boolean hasLogicFor(String testName) {
59         List<BusinessLogicRulesList> rulesLists = mRules.get(testName);
60         return rulesLists != null && !rulesLists.isEmpty();
61     }
62 
63     /**
64      * Return whether multiple rule lists exist in the BusinessLogic for this test name.
65      */
hasLogicsFor(String testName)66     private boolean hasLogicsFor(String testName) {
67         List<BusinessLogicRulesList> rulesLists = mRules.get(testName);
68         return rulesLists != null && rulesLists.size() > 1;
69     }
70 
71     /**
72      * Apply business logic for the given test.
73      * @param testName the name of the test case, prefixed by fully qualified class name, then '#'.
74      * For example, "com.android.foo.FooTest#testFoo"
75      * @param executor a {@link BusinessLogicExecutor}
76      */
applyLogicFor(String testName, BusinessLogicExecutor executor)77     public void applyLogicFor(String testName, BusinessLogicExecutor executor) {
78         if (!hasLogicFor(testName)) {
79             return;
80         }
81         if (hasLogicsFor(testName)) {
82             applyLogicsFor(testName, executor); // handle this special case separately
83             return;
84         }
85         // expecting exactly one rules list at this point
86         BusinessLogicRulesList rulesList = mRules.get(testName).get(0);
87         rulesList.invokeRules(executor);
88     }
89 
90     /**
91      * Handle special case in which multiple rule lists exist for the test name provided.
92      * Execute each rule list in a sandbox and store an exception for each rule list that
93      * triggers failure or skipping for the test.
94      * If all rule lists trigger skipping, rethrow AssumptionViolatedException to report a 'skip'
95      * for the test as a whole.
96      * If one or more rule lists trigger failure, rethrow RuntimeException with a list containing
97      * each failure.
98      */
applyLogicsFor(String testName, BusinessLogicExecutor executor)99     private void applyLogicsFor(String testName, BusinessLogicExecutor executor) {
100         Map<String, RuntimeException> failedMap = new HashMap<>();
101         Map<String, RuntimeException> skippedMap = new HashMap<>();
102         List<BusinessLogicRulesList> rulesLists = mRules.get(testName);
103         for (int index = 0; index < rulesLists.size(); index++) {
104             BusinessLogicRulesList rulesList = rulesLists.get(index);
105             String description = cleanDescription(rulesList.getDescription(), index);
106             try {
107                 rulesList.invokeRules(executor);
108             } catch (RuntimeException re) {
109                 if (AssumptionViolatedException.class.isInstance(re)) {
110                     skippedMap.put(description, re);
111                     executor.logInfo("Test %s (%s) skipped for reason: %s", testName, description,
112                             re.getMessage());
113                 } else {
114                     failedMap.put(description, re);
115                 }
116             }
117         }
118         if (skippedMap.size() == rulesLists.size()) {
119             throwAggregatedException(skippedMap, false);
120         } else if (failedMap.size() > 0) {
121             throwAggregatedException(failedMap, true);
122         } // else this test should be reported as a pure pass
123     }
124 
125     /**
126      * Helper to aggregate the messages of many {@link RuntimeException}s, and optionally their
127      * stack traces, before throwing an exception.
128      * @param exceptions a map from description strings to exceptions. The descriptive keySet is
129      * used to differentiate which BusinessLogicRulesList caused which exception
130      * @param failed whether to trigger failure. When false, throws assumption failure instead, and
131      * excludes stack traces from the exception message.
132      */
throwAggregatedException(Map<String, RuntimeException> exceptions, boolean failed)133     private static void throwAggregatedException(Map<String, RuntimeException> exceptions,
134             boolean failed) {
135         Set<String> keySet = exceptions.keySet();
136         String[] descriptions = keySet.toArray(new String[keySet.size()]);
137         StringBuilder msg = new StringBuilder("");
138         msg.append(String.format("Test %s for cases: ", (failed) ? "failed" : "skipped"));
139         msg.append(Arrays.toString(descriptions));
140         msg.append("\nReasons include:");
141         for (String description : descriptions) {
142             RuntimeException re = exceptions.get(description);
143             msg.append(String.format("\nMessage [%s]: %s", description, re.getMessage()));
144             if (failed) {
145                 StringWriter sw = new StringWriter();
146                 re.printStackTrace(new PrintWriter(sw));
147                 msg.append(String.format("\nStack Trace: %s", sw.toString()));
148             }
149         }
150         if (failed) {
151             throw new RuntimeException(msg.toString());
152         } else {
153             throw new AssumptionViolatedException(msg.toString());
154         }
155     }
156 
157     /**
158      * Helper method to generate a meaningful description in case the provided description is null
159      * or empty. In this case, returns a string representation of the index provided.
160      */
cleanDescription(String description, int index)161     private String cleanDescription(String description, int index) {
162         return (description == null || description.length() == 0)
163                 ? Integer.toString(index)
164                 : description;
165     }
166 
setAuthenticationStatus(String authenticationStatus)167     public void setAuthenticationStatus(String authenticationStatus) {
168         try {
169             mAuthenticationStatus = Enum.valueOf(AuthenticationStatusEnum.class,
170                     authenticationStatus);
171         } catch (IllegalArgumentException e) {
172             // Invalid value, set to unknown
173             mAuthenticationStatus = AuthenticationStatusEnum.UNKNOWN;
174         }
175     }
176 
isAuthorized()177     public boolean isAuthorized() {
178         return AuthenticationStatusEnum.AUTHORIZED.equals(mAuthenticationStatus);
179     }
180 
getTimestamp()181     public Date getTimestamp() {
182         return mTimestamp;
183     }
184 
getRedactionRegexes()185     public List<String> getRedactionRegexes() {
186         return new ArrayList<String>(mRedactionRegexes);
187     }
188 
189     /**
190      * Builds a user readable string tha explains the authentication status and the effect on tests
191      * which require authentication to execute.
192      */
getAuthenticationStatusMessage()193     public String getAuthenticationStatusMessage() {
194         switch (mAuthenticationStatus) {
195             case AUTHORIZED:
196                 return "Authorized";
197             case NOT_AUTHENTICATED:
198                 return "authorization failed, please ensure the service account key is "
199                         + "properly installed.";
200             case NOT_AUTHORIZED:
201                 return "service account is not authorized to access information for this device. "
202                         + "Please verify device properties are set correctly and account "
203                         + "permissions are configured to the Business Logic Api.";
204             case NO_DEVICE_INFO:
205                 return "unable to read device info files. Retry without --skip-device-info flag.";
206             default:
207                 return "something went wrong, please try again.";
208         }
209     }
210 
211     /**
212      * A list of BusinessLogicRules, wrapped with an optional description to differentiate rule
213      * lists that apply to the same test.
214      */
215     protected static class BusinessLogicRulesList {
216 
217         /* Stored description and rules */
218         protected List<BusinessLogicRule> mRulesList;
219         protected String mDescription;
220 
BusinessLogicRulesList(List<BusinessLogicRule> rulesList)221         public BusinessLogicRulesList(List<BusinessLogicRule> rulesList) {
222             mRulesList = rulesList;
223         }
224 
BusinessLogicRulesList(List<BusinessLogicRule> rulesList, String description)225         public BusinessLogicRulesList(List<BusinessLogicRule> rulesList, String description) {
226             mRulesList = rulesList;
227             mDescription = description;
228         }
229 
getDescription()230         public String getDescription() {
231             return mDescription;
232         }
233 
getRules()234         public List<BusinessLogicRule> getRules() {
235             return mRulesList;
236         }
237 
invokeRules(BusinessLogicExecutor executor)238         public void invokeRules(BusinessLogicExecutor executor) {
239             for (BusinessLogicRule rule : mRulesList) {
240                 // Check conditions
241                 if (rule.invokeConditions(executor)) {
242                     rule.invokeActions(executor);
243                 }
244             }
245         }
246     }
247 
248     /**
249      * Nested class representing an Business Logic Rule. Stores a collection of conditions
250      * and actions for later invocation.
251      */
252     protected static class BusinessLogicRule {
253 
254         /* Stored conditions and actions */
255         protected List<BusinessLogicRuleCondition> mConditions;
256         protected List<BusinessLogicRuleAction> mActions;
257 
BusinessLogicRule(List<BusinessLogicRuleCondition> conditions, List<BusinessLogicRuleAction> actions)258         public BusinessLogicRule(List<BusinessLogicRuleCondition> conditions,
259                 List<BusinessLogicRuleAction> actions) {
260             mConditions = conditions;
261             mActions = actions;
262         }
263 
264         /**
265          * Method that invokes all Business Logic conditions for this rule, and returns true
266          * if all conditions evaluate to true.
267          */
invokeConditions(BusinessLogicExecutor executor)268         public boolean invokeConditions(BusinessLogicExecutor executor) {
269             for (BusinessLogicRuleCondition condition : mConditions) {
270                 if (!condition.invoke(executor)) {
271                     return false;
272                 }
273             }
274             return true;
275         }
276 
277         /**
278          * Method that invokes all Business Logic actions for this rule
279          */
invokeActions(BusinessLogicExecutor executor)280         public void invokeActions(BusinessLogicExecutor executor) {
281             for (BusinessLogicRuleAction action : mActions) {
282                 action.invoke(executor);
283             }
284         }
285     }
286 
287     /**
288      * Nested class representing an Business Logic Rule Condition. Stores the name of a method
289      * to invoke, as well as String args to use during invocation.
290      */
291     protected static class BusinessLogicRuleCondition {
292 
293         /* Stored method name and String args */
294         protected String mMethodName;
295         protected List<String> mMethodArgs;
296         /* Whether or not the boolean result of this condition should be reversed */
297         protected boolean mNegated;
298 
299 
BusinessLogicRuleCondition(String methodName, List<String> methodArgs, boolean negated)300         public BusinessLogicRuleCondition(String methodName, List<String> methodArgs,
301                 boolean negated) {
302             mMethodName = methodName;
303             mMethodArgs = methodArgs;
304             mNegated = negated;
305         }
306 
307         /**
308          * Invoke this Business Logic condition with an executor.
309          */
invoke(BusinessLogicExecutor executor)310         public boolean invoke(BusinessLogicExecutor executor) {
311             // XOR the negated boolean with the return value of the method
312             return (mNegated != executor.executeCondition(mMethodName,
313                     mMethodArgs.toArray(new String[mMethodArgs.size()])));
314         }
315     }
316 
317     /**
318      * Nested class representing an Business Logic Rule Action. Stores the name of a method
319      * to invoke, as well as String args to use during invocation.
320      */
321     protected static class BusinessLogicRuleAction {
322 
323         /* Stored method name and String args */
324         protected String mMethodName;
325         protected List<String> mMethodArgs;
326 
BusinessLogicRuleAction(String methodName, List<String> methodArgs)327         public BusinessLogicRuleAction(String methodName, List<String> methodArgs) {
328             mMethodName = methodName;
329             mMethodArgs = methodArgs;
330         }
331 
332         /**
333          * Invoke this Business Logic action with an executor.
334          */
invoke(BusinessLogicExecutor executor)335         public void invoke(BusinessLogicExecutor executor) {
336             executor.executeAction(mMethodName,
337                     mMethodArgs.toArray(new String[mMethodArgs.size()]));
338         }
339     }
340 
341     /**
342      * Nested enum of the possible authentication statuses.
343      */
344     protected enum AuthenticationStatusEnum {
345         UNKNOWN,
346         NOT_AUTHENTICATED,
347         NOT_AUTHORIZED,
348         AUTHORIZED,
349         NO_DEVICE_INFO
350     }
351 
352 }
353