USB 接続で Android の操作は adb command を使いますが、Android アプリ自身からコマンドを送ることができます。
Runtime.getRuntime().exec あるいは ProcessBuilder を使います.
下はProcessBuilderを使ってアプリパッケージのリストを表示させた例です。
2021.2.1
ProcessBuilder の設定
以前は様々なコマンドが使えたのですが、セキュリティが厳しくなり使えるものは少なくなりました。それでも使えるものとしてLogcatの取得とパッケージ検索などが可能です
Logcatの取得:
エラーが発生したときなどadb shell から
1 |
$ logcat -v time |
のコマンドでLogcatが取得できます
パッケージ検索:
また、アプリケーション・パッケージを探すコマンドは
1 |
$ pm list packages |
PCから接続している場合は、ターミナルで adb shell からシェルに入り、pm list packages を打ち込むとリストが見られます
ProcessBuilderに処理させるには以下のようにします
アプリ内なので adb shell はいりません
1 |
ProcessBuilder processBuilder = new ProcessBuilder("logcat", "-v", "time"); |
Logcatの取得・保存
リアルタイムで監視しながら1秒おきにLogcatを取得して
アプリ専用のストレージ領域に保存します
MainActivity.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 |
//package your.package.name; import android.content.Context; import android.os.Bundle; import android.util.Log; import android.widget.Button; import androidx.appcompat.app.AppCompatActivity; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.io.OutputStream; import java.io.OutputStreamWriter; import java.io.PrintWriter; import java.nio.charset.StandardCharsets; public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button button = findViewById(R.id.button); button.setOnClickListener(v -> { Thread getLogCat = new GetLogCat(getApplicationContext()); getLogCat.start(); }); } // リアルタイムで監視するスレッドクラス public static class GetLogCat extends Thread { private final Context context; public GetLogCat(Context context) { this.context = context; } @Override public void run() { Process process = null; BufferedReader bReader = null; final String pId = Integer.toString(android.os.Process.myPid()); Log.i("debug", pId); try { process = new ProcessBuilder("logcat", "-v", "time").start(); bReader = new BufferedReader(new InputStreamReader(process.getInputStream()), 1024); String line; do { line = bReader.readLine(); // logが無い時は休む if (line.length() == 0) { try { sleep(1000); } catch (InterruptedException ignored) { } continue; } if (line.contains(pId)) { // try-with-resources // 端末内のアプリ専用内部ストレージに保存する try (OutputStream out = context.openFileOutput("logcat.text", Context.MODE_PRIVATE | Context.MODE_APPEND); PrintWriter writer = new PrintWriter(new OutputStreamWriter(out, StandardCharsets.UTF_8));) { writer.println(line); } catch (Exception e) { e.printStackTrace(); } } } while (true); } catch (IOException e) { e.printStackTrace(); } finally { if (bReader != null) { try { bReader.close(); } catch (IOException e) { e.printStackTrace(); } } } } } } |
開始させるためのButtonを置いただけです
activity_main.xml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <Button android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/button" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" app:layout_constraintVertical_bias="0.45" /> </androidx.constraintlayout.widget.ConstraintLayout> |
リソース
strings.xml
1 2 3 4 |
<resources> <string name="app_name">TestProcessBuilderLog</string> <string name="button">button</string> </resources> |
アプリを実行した後で、Teminalからアプリのストレージ領域を確認してみましょう
adb shell でアプリのパッケージ名で入りますが
セキュリティのためrun-asを使います
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
> adb shell sargo:/ $ ls ... $ cd data/data $ run-as com.example.testprocessbuilderlog $ ls cache code_cache files $ cd files/ $ ls logcat.text $ cat logcat.text 07-05 11:33:05.906 I/ocessbuilderlo(12116): Late-enabling -Xcheck:jni 07-05 11:33:05.957 D/ProcessState(12116): Binder ioctl to enable oneway spam detection failed: Invalid argument 07-05 11:33:06.001 D/CompatibilityChangeReporter(12116): Compat change id reported: 171979766; UID 10644; state: ENABLED ... ... 07-05 11:33:17.292 I/debug (12116): 12116 $ |
Logcatが保存されていました
package検索
端末にあるアプリのpackage名を検索します
MainActivity.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 |
//package your.package.name; import androidx.appcompat.app.AppCompatActivity; import android.os.Bundle; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import android.widget.TextView; public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); String[] commands = {"pm", "list","packages"}; String result = comandExe(commands); TextView textView = findViewById(R.id.text_view); textView.setText(result); } private String comandExe(String[] command) { StringBuilder suilder = new StringBuilder(); ProcessBuilder processBuilder = new ProcessBuilder(command); InputStream iStream = null; InputStreamReader isReader = null; try{ Process proc = processBuilder.start(); iStream = proc.getInputStream(); isReader = new InputStreamReader(iStream); BufferedReader bufferedReader = new BufferedReader(isReader); String line; while ((line = bufferedReader.readLine()) != null) { suilder.append(line); suilder.append("\n"); } } catch(Exception e){ e.printStackTrace(); } finally { try{ if(iStream != null){ iStream.close(); } if(isReader != null){ isReader.close(); } } catch(Exception e){ e.printStackTrace(); } } return suilder.toString(); } } |
レイアウトです
activity_main.xml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
<?xml version="1.0" encoding="utf-8"?> <ScrollView xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="wrap_content" android:padding="16dp" tools:context=".MainActivity" > <TextView android:id="@+id/text_view" android:textSize="14sp" android:layout_width="wrap_content" android:layout_height="wrap_content"/> </ScrollView> |
また、Runtime.getRuntime().exec を使って
Runtime.getRuntime().exec(new String[] {“pm”,”list”,”com.example.testapp”});
のようにすればできます