# ADB Driver `jumpstarter-driver-adb` tunnels Android Debug Bridge (ADB) connections over Jumpstarter, enabling remote Android device access via standard ADB tools such as Android Studio. ## Installation ```shell pip3 install --extra-index-url https://pkg.jumpstarter.dev/simple/ jumpstarter-driver-adb ``` For the optional Python ADB API: ```shell pip3 install --extra-index-url https://pkg.jumpstarter.dev/simple/ "jumpstarter-driver-adb[python-api]" ``` ## Configuration Example exporter configuration: ```yaml export: adb: type: jumpstarter_driver_adb.driver.AdbServer config: host: "127.0.0.1" port: 15037 ``` ### Configuration Parameters | Parameter | Description | Type | Required | Default | | --------- | ---------------------------------------------- | ---- | -------- | -------------------------- | | adb_path | Path to the ADB executable on the exporter | str | no | "adb" (resolved from PATH) | | host | Host address of the ADB server on the exporter | str | no | "127.0.0.1" | | port | Port of the ADB server on the exporter | int | no | 15037 | ### Port Assignment The exporter runs its own ADB server on a non-standard port (default: 15037) to avoid conflicting with the standard ADB server on port 5037 (if Jumpstarter is running in local mode). This is important because tools like Android Studio automatically start and maintain an ADB server on port 5037 and will restart it if killed. On the client side, the `tunnel` command binds to an auto-assigned port by default. Use `-P` to specify a port (such as 5037) if needed. ## Usage ### Run ADB commands All standard adb commands are passed through to the remote ADB server: ```bash # List devices j adb devices # Interactive shell j adb shell # Run a command on the device j adb shell getprop ro.product.model # Install an app j adb install app.apk # View device logs j adb logcat # Push/pull files j adb push local_file.txt /sdcard/ j adb pull /sdcard/remote_file.txt . ``` ### Persistent tunnel The `tunnel` command is the only Jumpstarter-specific command. All other commands (including `start-server`, `kill-server`, `connect`, `disconnect`, `reconnect`, `pair`) are passed through to the remote ADB server. ```bash # Create a persistent ADB tunnel (auto-assigned port) j adb tunnel # Create a tunnel on a specific port j adb tunnel -P 5038 # Background the tunnel for continued shell use j adb tunnel & ``` When a persistent tunnel is running, subsequent `j adb` commands will automatically reuse it instead of creating a new ephemeral tunnel. This makes commands faster and ensures a consistent connection. For native `adb` or external tools, export the env vars printed by the `tunnel` command in another terminal. ### Unsupported commands The `nodaemon` command is not supported as it would start a local ADB server process, ignoring the tunnel entirely. ## Integration with Android Ecosystem Tools ### Forward ADB for external tools The `tunnel` command creates a persistent tunnel that other `j adb` commands reuse automatically. For external tools, export the env vars printed by the command: ```bash # In the jmp shell: j adb tunnel ``` ```bash # In another terminal, using the port printed by the tunnel command: export ANDROID_ADB_SERVER_ADDRESS=127.0.0.1 export ANDROID_ADB_SERVER_PORT= adb devices ``` ### Android Studio Android Studio automatically starts and maintains its own ADB server on port 5037. Because of this, the `tunnel` command uses an auto-assigned port by default to avoid conflicts. To use the tunnel with Android Studio: 1. Note the port printed by `j adb tunnel` 2. Configure Android Studio to use a custom ADB server port, or: 3. Kill Android Studio's ADB server, bind the tunnel to port 5037, and restart Android Studio: ```bash adb kill-server j adb tunnel -P 5037 # Note: Android Studio may restart the ADB server on 5037 when opened, # causing a conflict. If this happens, use the auto-assigned port instead. ``` ### Trade Federation (tradefed) tradefed discovers devices through the ADB server via the `ANDROID_ADB_SERVER_PORT` environment variable: ```bash # Terminal 1: Start the tunnel j adb tunnel # Note the port, e.g. 54321 # Terminal 2: Run tradefed with the tunnel port export ANDROID_ADB_SERVER_PORT=54321 tradefed.sh # > list devices <-- shows remote devices ``` ### Python API You can also perform interactions via ADB using the [`adbutils`](https://github.com/openatx/adbutils) Python package. ```python # Requires: pip install jumpstarter-driver-adb[python-api] import adbutils with client.adb.forward_adb(port=0) as (host, port): adb = adbutils.AdbClient(host=host, port=port) for device in adb.device_list(): print(device.serial, device.prop.model) ``` ## CLI Reference ### Standard ADB commands (passed through) | Usage | Description | | ----------------------------- | ------------------------------------------------- | | `j adb [args...]` | Run any adb command against the remote ADB server | | `j adb devices` | List connected devices | | `j adb shell [command]` | Open a shell or run a command on the device | | `j adb install ` | Install an APK | | `j adb push ` | Push a file to the device | | `j adb pull ` | Pull a file from the device | | `j adb logcat` | View device logs | ### Jumpstarter-specific commands | Usage | Description | | ------------------------ | ----------------------------------------------------------------------- | | `j adb tunnel [-P PORT]` | Create a persistent ADB tunnel (auto-assigned port, or specify with -P) | ### Options | Option | Description | Default | | ------------ | ------------------------------------ | --------- | | `-H HOST` | Local address to tunnel ADB to | 127.0.0.1 | | `-P PORT` | Local port to tunnel ADB to (0=auto) | 0 | | `--adb PATH` | Path to local adb executable | adb | ## API Reference ### Driver ```{eval-rst} .. autoclass:: jumpstarter_driver_adb.driver.AdbServer() :members: start_server, kill_server, list_devices ``` ### Client ```{eval-rst} .. autoclass:: jumpstarter_driver_adb.client.AdbClient() :members: forward_adb, start_server, kill_server, list_devices ```