1 /*
2  * Copyright (C) 2015 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.performanceapp.tests;
18 
19 import android.app.ActivityManager;
20 import android.app.ActivityManager.RunningAppProcessInfo;
21 import android.content.ComponentName;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.pm.ActivityInfo;
25 import android.content.pm.PackageInfo;
26 import android.content.pm.PackageManager.NameNotFoundException;
27 import android.net.Uri;
28 import android.os.Bundle;
29 import android.os.Environment;
30 import android.os.ParcelFileDescriptor;
31 import android.test.InstrumentationTestCase;
32 import android.util.Log;
33 
34 import androidx.test.InstrumentationRegistry;
35 import androidx.test.filters.MediumTest;
36 
37 import java.io.ByteArrayOutputStream;
38 import java.io.File;
39 import java.io.FileOutputStream;
40 import java.io.IOException;
41 import java.io.InputStream;
42 import java.io.OutputStream;
43 import java.util.ArrayList;
44 import java.util.List;
45 
46 /**
47  * To test the App launch performance on the given target package for the list of activities. It
48  * launches the activities present in the target package the number of times in launch count or it
49  * launches only the activities mentioned in custom activity list the launch count times and returns
50  * the path to the files where the atrace logs are stored corresponding to each activity launch
51  */
52 public class AppLaunchTests extends InstrumentationTestCase {
53 
54     private static final String TAG = "AppLaunchInstrumentation";
55     private static final String TARGETPACKAGE = "targetpackage";
56     private static final String SIMPLEPERF_BIN = "simpleperf_bin";
57     private static final String SIMPLEPERF_EVT = "simpleperf_event";
58     private static final String SIMPLEPERF_DATA = "simpleperf_data";
59     private static final String DISPATCHER = "dispatcher";
60     private static final String ACTIVITYLIST = "activitylist";
61     private static final String LAUNCHCOUNT = "launchcount";
62     private static final String RECORDTRACE = "recordtrace";
63     private static final String ATRACE_CATEGORIES = "tracecategory";
64     private static final String DEFAULT_CATEGORIES = "am,view,gfx";
65     private static final String ATRACE_START = "atrace --async_start -b 100000 %s";
66     private static final String ATRACE_STOP = "atrace --async_stop";
67     private static final String FORCE_STOP = "am force-stop ";
68     private static final String TARGET_URL = "instanturl";
69     private static final String URL_PREFIX = "http://";
70     private static final int BUFFER_SIZE = 8192;
71 
72     private Context mContext;
73     private Bundle mResult;
74     private String mTargetPackageName;
75     private String mSimpleperfBin;
76     private String mSimpleperfEvt;
77     private String mSimpleperfDir;
78     private String mAtraceCategory;
79     private String mDispatcher;
80     private int mLaunchCount;
81     private String mCustomActivityList;
82     private String mTargetUrl;
83     private PackageInfo mPackageInfo;
84     private boolean mRecordTrace = true;
85     private List<String> mActivityList = new ArrayList<>();
86 
87     /**
88      * {@inheritDoc}
89      */
90     @Override
setUp()91     public void setUp() throws Exception {
92         super.setUp();
93         mContext = getInstrumentation().getTargetContext();
94         assertNotNull("Failed to get context", mContext);
95         Bundle args = InstrumentationRegistry.getArguments();
96         assertNotNull("Unable to get the args", args);
97         mTargetPackageName = args.getString(TARGETPACKAGE);
98         assertNotNull("Target package name not set", mTargetPackageName);
99         mSimpleperfEvt = args.getString(SIMPLEPERF_EVT);
100         mDispatcher = args.getString(DISPATCHER);
101 
102         mAtraceCategory = args.getString(ATRACE_CATEGORIES);
103         if (mAtraceCategory == null || mAtraceCategory.isEmpty()) {
104             mAtraceCategory = DEFAULT_CATEGORIES;
105         }
106         mAtraceCategory = mAtraceCategory.replace(",", " ");
107 
108         if (mDispatcher != null && !mDispatcher.isEmpty()) {
109             mSimpleperfBin = args.getString(SIMPLEPERF_BIN);
110             mSimpleperfEvt = args.getString(SIMPLEPERF_EVT);
111             mSimpleperfDir = args.getString(SIMPLEPERF_DATA);
112         }
113         mCustomActivityList = args.getString(ACTIVITYLIST);
114         if (mCustomActivityList == null || mCustomActivityList.isEmpty()) {
115             // Look for instant app configs exist.
116             mTargetUrl = args.getString(TARGET_URL);
117             if (mTargetUrl != null && !mTargetUrl.isEmpty()) {
118                 mActivityList.add(args.getString(TARGET_URL));
119             } else {
120                 // Get full list of activities from the target package
121                 mActivityList = getActivityList("");
122             }
123         } else {
124             // Get only the user defined list of activities from the target package
125             mActivityList = getActivityList(mCustomActivityList);
126         }
127         assertTrue("Activity List is empty", (mActivityList.size() > 0));
128         mLaunchCount = Integer.parseInt(args.getString(LAUNCHCOUNT));
129         assertTrue("Invalid Launch Count", mLaunchCount > 0);
130         if (args.getString(RECORDTRACE) != null
131                 && args.getString(RECORDTRACE).equalsIgnoreCase("false")) {
132             mRecordTrace = false;
133         }
134         mResult = new Bundle();
135     }
136 
137     @MediumTest
testAppLaunchPerformance()138     public void testAppLaunchPerformance() throws Exception {
139         assertTrue("Cannot write in External File", isExternalStorageWritable());
140         File root = Environment.getExternalStorageDirectory();
141         assertNotNull("Unable to get the root of the external storage", root);
142         File logsDir = new File(root, "atrace_logs");
143         assertTrue("Unable to create the directory to store atrace logs", logsDir.mkdir());
144         if (mDispatcher != null && !mDispatcher.isEmpty()) {
145             if (mSimpleperfDir == null)
146                 mSimpleperfDir = "/sdcard/perf_simpleperf/";
147             File simpleperfDir = new File(mSimpleperfDir);
148             assertTrue("Unable to create the directory to store simpleperf data",
149                     simpleperfDir.mkdir());
150         }
151         for (int count = 0; count < mLaunchCount; count++) {
152             for (String activityName : mActivityList) {
153                 Intent intent = new Intent(Intent.ACTION_MAIN);
154                 if (activityName.startsWith(URL_PREFIX)) {
155                     intent = new Intent(Intent.ACTION_VIEW, Uri.parse(activityName));
156                     intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
157                 } else {
158                     ComponentName cn = new ComponentName(mTargetPackageName,
159                             mDispatcher != null ? mDispatcher : activityName);
160                     intent = new Intent(Intent.ACTION_MAIN);
161                     intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
162                     intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
163                     intent.setComponent(cn);
164 
165                     if (mDispatcher != null) {
166                         intent.putExtra("ACTIVITY_NAME", activityName);
167                         intent.putExtra("SIMPLEPERF_DIR", mSimpleperfDir);
168                         intent.putExtra("SIMPLEPERF_EVT", mSimpleperfEvt);
169                         intent.putExtra("SIMPLEPERF_BIN", mSimpleperfBin);
170                     }
171                 }
172 
173                 // Start the atrace
174                 if (mRecordTrace) {
175                     ByteArrayOutputStream startStream = new ByteArrayOutputStream();
176                     try {
177                         writeDataToByteStream(getInstrumentation().getUiAutomation()
178                                 .executeShellCommand(String.format(ATRACE_START, mAtraceCategory)),
179                                 startStream);
180                     } finally {
181                         startStream.close();
182                     }
183 
184                     // Sleep for 5 secs to make sure atrace command is started
185                     Thread.sleep(5 * 1000);
186                 }
187 
188                 // Launch the activity
189                 mContext.startActivity(intent);
190                 Thread.sleep(5 * 1000);
191 
192                 // Make sure we stops simpleperf
193                 if (mDispatcher != null) {
194                     try {
195                         Runtime.getRuntime().exec("pkill -l SIGINT simpleperf").waitFor();
196                     } catch (Exception e) {
197                         Log.v(TAG, "simpleperf throw exception");
198                         e.printStackTrace();
199                     }
200                 }
201 
202                 // Dump atrace info and write it to file
203                 if (mRecordTrace) {
204                     int processId = getProcessId(mTargetPackageName);
205                     assertTrue("Not able to retrive the process id for the package:"
206                             + mTargetPackageName, processId > 0);
207                     String fileName = new String();
208                     if (!activityName.startsWith(URL_PREFIX)) {
209                         fileName = String.format("%s-%d-%d", activityName, count, processId);
210                     } else {
211                         fileName = String.format("%s-%d-%d", Uri.parse(activityName).getHost(),
212                                 count, processId);
213                     }
214 
215                     ByteArrayOutputStream stopStream = new ByteArrayOutputStream();
216                     File file = new File(logsDir, fileName);
217                     OutputStream fileOutputStream = new FileOutputStream(file);
218                     try {
219                         writeDataToByteStream(
220                                 getInstrumentation().getUiAutomation()
221                                         .executeShellCommand(ATRACE_STOP),
222                                 stopStream);
223                         fileOutputStream.write(stopStream.toByteArray());
224                     } finally {
225                         stopStream.close();
226                         fileOutputStream.close();
227                     }
228 
229                     // To keep track of the activity name,list of atrace file name
230                     if (!activityName.startsWith(URL_PREFIX)) {
231                         registerTraceFileNames(activityName, fileName);
232                     } else {
233                         registerTraceFileNames(Uri.parse(activityName).getHost(), fileName);
234                     }
235                 }
236 
237                 ByteArrayOutputStream killStream = new ByteArrayOutputStream();
238                 try {
239                     writeDataToByteStream(getInstrumentation().getUiAutomation()
240                             .executeShellCommand(
241                                     FORCE_STOP + mTargetPackageName),
242                             killStream);
243                 } finally {
244                     killStream.close();
245                 }
246 
247                 Thread.sleep(5 * 1000);
248             }
249         }
250         getInstrumentation().sendStatus(0, mResult);
251     }
252 
253     /**
254      * Method to check if external storage is writable
255      * @return
256      */
isExternalStorageWritable()257     public boolean isExternalStorageWritable() {
258         return Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState());
259     }
260 
261     /**
262      * Method to get list of activities present in given target package If customActivityList is
263      * passed then include only those activities
264      * @return list of activity names
265      */
getActivityList(String customActivityList)266     private List<String> getActivityList(String customActivityList) {
267         mActivityList = new ArrayList<String>();
268         try {
269             mPackageInfo = mContext.getPackageManager().getPackageInfo(
270                     mTargetPackageName, 1);
271             assertNotNull("Unable to get  the target package info", mPackageInfo);
272         } catch (NameNotFoundException e) {
273             fail(String.format("Target application: %s not found", mTargetPackageName));
274         }
275         for (ActivityInfo activityInfo : mPackageInfo.activities) {
276             mActivityList.add(activityInfo.name);
277         }
278         if (!customActivityList.isEmpty()) {
279             List<String> finalActivityList = new
280                     ArrayList<String>();
281             String customList[] = customActivityList.split(",");
282             for (int count = 0; count < customList.length; count++) {
283                 if (mActivityList.contains(customList[count])) {
284                     finalActivityList.add(customList[count]);
285                 } else {
286                     fail(String.format("Activity: %s not present in the target package : %s ",
287                             customList[count], mTargetPackageName));
288                 }
289             }
290             mActivityList = finalActivityList;
291         }
292         return mActivityList;
293     }
294 
295     /**
296      * Method to retrieve process id from the activity manager
297      * @param processName
298      * @return
299      */
getProcessId(String processName)300     private int getProcessId(String processName) {
301         ActivityManager am = (ActivityManager) getInstrumentation()
302                 .getContext().getSystemService(Context.ACTIVITY_SERVICE);
303         List<RunningAppProcessInfo> appsInfo = am.getRunningAppProcesses();
304         assertNotNull("Unable to retrieve running apps info", appsInfo);
305         for (RunningAppProcessInfo appInfo : appsInfo) {
306             if (appInfo.processName.equals(processName)) {
307                 return appInfo.pid;
308             }
309         }
310         return -1;
311     }
312 
313     /**
314      * To add the process id to the result map
315      * @param activityNamereturn
316      * @return
317      * @throws IOException
318      */
registerTraceFileNames(String activityName, String absPath)319     private void registerTraceFileNames(String activityName, String absPath)
320             throws IOException {
321         if (mResult.containsKey(activityName)) {
322             String existingResult = (String) mResult.get(activityName);
323             mResult.putString(activityName, existingResult + "," + absPath);
324         } else {
325             mResult.putString(activityName, "" + absPath);
326         }
327     }
328 
329     /**
330      * Method to write data into byte array
331      * @param pfDescriptor Used to read the content returned by shell command
332      * @param outputStream Write the data to this output stream read from pfDescriptor
333      * @throws IOException
334      */
writeDataToByteStream( ParcelFileDescriptor pfDescriptor, ByteArrayOutputStream outputStream)335     private void writeDataToByteStream(
336             ParcelFileDescriptor pfDescriptor, ByteArrayOutputStream outputStream)
337             throws IOException {
338         InputStream inputStream = new ParcelFileDescriptor.AutoCloseInputStream(pfDescriptor);
339         try {
340             byte[] buffer = new byte[BUFFER_SIZE];
341             int length;
342             while ((length = inputStream.read(buffer)) >= 0) {
343                 outputStream.write(buffer, 0, length);
344             }
345         } finally {
346             inputStream.close();
347         }
348     }
349 
350 }
351