1 /* 2 * Copyright (C) 2016 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 android.support.test.aupt; 18 19 import android.app.ActivityManager; 20 import android.app.Instrumentation; 21 import android.os.Environment; 22 import android.os.ParcelFileDescriptor; 23 import android.os.RemoteException; 24 import android.util.Log; 25 26 import java.io.ByteArrayOutputStream; 27 import java.io.File; 28 import java.io.FileNotFoundException; 29 import java.io.FileOutputStream; 30 import java.io.IOException; 31 import java.io.InputStream; 32 import java.io.OutputStream; 33 import java.text.SimpleDateFormat; 34 import java.util.Date; 35 import java.util.Locale; 36 37 public class FilesystemUtil { 38 private static final String TAG = FilesystemUtil.class.getSimpleName(); 39 40 /** Save the output of a process to a file */ saveProcessOutput(Instrumentation instr, String command, File file)41 public static void saveProcessOutput(Instrumentation instr, String command, File file) 42 throws IOException { 43 Log.d(TAG, String.format("Saving command \"%s\" output into file %s", 44 command, file.getAbsolutePath())); 45 46 OutputStream out = new FileOutputStream(file); 47 saveProcessOutput(instr, command, out); 48 out.close(); 49 } 50 51 /** Send the output of a process to an OutputStream. */ saveProcessOutput(Instrumentation instr, String command, OutputStream out)52 public static void saveProcessOutput(Instrumentation instr, String command, OutputStream out) 53 throws IOException { 54 try { 55 // First, try to execute via our UiAutomation 56 ParcelFileDescriptor pfd = instr.getUiAutomation().executeShellCommand(command); 57 pipe(new ParcelFileDescriptor.AutoCloseInputStream(pfd), out); 58 } catch (IllegalStateException ise) { 59 // If we don't have a UiAutomation, we'll get an IllegalStatException; 60 // so try to do it via an exec() 61 Process process = Runtime.getRuntime().exec(command); 62 pipe(process.getInputStream(), out); 63 64 // Wait for our process to finish 65 try { 66 process.waitFor(); 67 } catch (InterruptedException ie) { 68 throw new IOException("Thread interrupted waiting for command: " + command); 69 } 70 71 // Make sure it succeeded. 72 if (process.exitValue() != 0) { 73 throw new IOException("Failed to save output of command: " + command); 74 } 75 } 76 } 77 78 /** Save a bugreport to the given file */ saveBugreport(Instrumentation instr, String filename)79 public static void saveBugreport(Instrumentation instr, String filename) 80 throws IOException, InterruptedException { 81 ByteArrayOutputStream baos = new ByteArrayOutputStream(); 82 String cmdline = String.format("/system/bin/sh -c /system/bin/bugreport>%s", 83 templateToFilename(filename)); 84 saveProcessOutput(instr, cmdline, baos); 85 baos.close(); 86 } 87 88 /** Save a bugreport to the given file */ saveBugreportz(Instrumentation instr)89 public static void saveBugreportz(Instrumentation instr) throws IOException { 90 try { 91 ActivityManager.getService().requestFullBugReport(); 92 } catch (RemoteException e) { 93 throw new IOException("Could not capture bugreportz", e); 94 } 95 } 96 97 /** Save annotated Meminfo to our default logging directory */ dumpMeminfo(Instrumentation instr, String notes)98 public static void dumpMeminfo(Instrumentation instr, String notes) { 99 long epochSeconds = System.currentTimeMillis() / 1000; 100 File outputDir = new File(Environment.getExternalStorageDirectory(), "meminfo"); 101 Log.i(TAG, outputDir.toString()); 102 if (!outputDir.exists()) { 103 boolean yes = outputDir.mkdirs(); 104 Log.i(TAG, yes ? "created" : "not created"); 105 } 106 File outputFile = new File(outputDir, String.format("%d.txt", epochSeconds)); 107 Log.i(TAG, outputFile.toString()); 108 FileOutputStream fos = null; 109 110 try { 111 fos = new FileOutputStream(outputFile); 112 fos.write(String.format("notes: %s\n\n", notes).getBytes()); 113 114 saveProcessOutput(instr, "dumpsys meminfo -c", fos); 115 fos.close(); 116 } catch (FileNotFoundException e) { 117 Log.e(TAG, "exception while dumping meminfo", e); 118 } catch (IOException e) { 119 Log.e(TAG, "exception while dumping meminfo", e); 120 } 121 } 122 123 /** Splice the date into the "%s" in a file name */ templateToFilename(String filenameTemplate)124 public static String templateToFilename(String filenameTemplate) { 125 SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss", Locale.US); 126 return String.format(filenameTemplate, sdf.format(new Date())); 127 } 128 129 /** Pipe an inputstream to an outputstream. This matches Apache's IOUtils::copy */ pipe(InputStream in, OutputStream out)130 private static void pipe(InputStream in, OutputStream out) throws IOException { 131 byte[] buffer = new byte[4096]; 132 int bytesRead = 0; 133 134 try { 135 while (bytesRead >= 0) { 136 out.write(buffer, 0, bytesRead); 137 bytesRead = in.read(buffer); 138 } 139 } finally { 140 in.close(); 141 out.flush(); 142 } 143 } 144 } 145