Slim async local device drivers for Android Portal and iOS Portal, extracted from mobilerun.
Project description
mobilerun-core-cli is the slim async local-driver core of mobilerun.
No CLI, no agent, no LLM providers โ just local Android/iOS drivers for higher-level tools such as mobilerun-core.
- ๐ฑ Drive local devices โ Android over ADB+Portal, Android Portal HTTP-only, or iOS Portal HTTP.
- โก TCP-with-content-provider fallback โ fast Android HTTP path over
adb forward, transparent fallback to content-provider RPC. - ๐ HTTP-only drivers โ connect to already-running Android/iOS portals without taking over setup.
- ๐ Portal lifecycle โ download, install, accessibility enablement, auto-upgrade.
- ๐ชถ Slim โ a small async driver package with four runtime deps (
async_adbutils,httpx,requests,rich). - ๐ Embeddable โ designed to be wrapped by sync facades (e.g.
mobilerun-core) or used directly. - ๐ค In sync with upstream โ verbatim slice of
droidrun/mobilerun; behaviour and API track upstream.
๐ฆ Installation
Note: Python
>=3.11,<3.14. The Android ADB driver requires ADB onPATHand a device with USB debugging enabled. HTTP-only drivers require an already-running portal URL.
uv pip install mobilerun-core-cli
๐ Quick usage
import asyncio
from async_adbutils import adb
from mobilerun_core_cli import AndroidDriver, ensure_portal_ready
async def main():
# 1. Make sure Portal is installed + accessibility is on.
device = await adb.device()
await ensure_portal_ready(device)
# 2. Drive the device.
driver = AndroidDriver(serial=device.serial, use_tcp=True)
await driver.connect()
await driver.tap(540, 1200)
await driver.swipe(540, 1600, 540, 400, duration_ms=300)
await driver.input_text("hello", clear=True)
png_bytes = await driver.screenshot()
tree = await driver.get_ui_tree()
asyncio.run(main())
HTTP-only Android:
import asyncio
from mobilerun_core_cli import AndroidPortalHttpDriver
async def main():
driver = AndroidPortalHttpDriver(
url="http://127.0.0.1:18080",
token="...",
)
await driver.connect()
await driver.tap(540, 1200)
asyncio.run(main())
iOS Portal:
import asyncio
from mobilerun_core_cli import IOSPortalDriver
async def main():
driver = IOSPortalDriver("http://127.0.0.1:6643")
await driver.connect()
await driver.start_app("com.apple.Preferences")
asyncio.run(main())
๐งฑ Layout
mobilerun_core_cli/
โโโ __init__.py Re-exports the public surface
โโโ portal.py Portal APK lifecycle + content-provider helpers
โโโ driver/
โ โโโ base.py DeviceDriver ABC, DeviceDisconnectedError
โ โโโ android.py AndroidDriver โ ADB-backed concrete driver
โ โโโ android_http.py AndroidPortalHttpDriver โ HTTP-only Android driver
โ โโโ ios.py IOSPortalDriver โ ios-portal HTTP driver
โโโ transport/
โโโ portal_client.py PortalClient โ TCP-with-content-provider fallback
๐ Public API
Re-exported from mobilerun_core_cli:
| Symbol | What it is |
|---|---|
AndroidDriver |
ADB+Portal device driver. Async methods: tap, swipe, input_text, press_button, start_app, install_app, screenshot, get_ui_tree, get_apps, list_packages, get_date. |
AndroidPortalHttpDriver |
HTTP-only Android Portal driver. Requires url and bearer token; does not use ADB at runtime. |
IOSPortalDriver / IOSDriver |
iOS Portal HTTP driver. Requires an already-running ios-portal URL. |
DeviceDriver |
Abstract base for drivers. supported: set[str] declares which verbs a subclass implements. |
DeviceDisconnectedError |
Raised when the device drops mid-call. |
PortalClient |
HTTP-or-content-provider client for the on-device Portal. Used internally by AndroidDriver; can be constructed directly for low-level access. |
setup_portal(device) |
Download + install + enable the Portal APK on a device. |
ensure_portal_ready(device) |
Idempotent: install/upgrade Portal and enable accessibility if needed. |
setup_keyboard(device) |
Switch the device to the Mobilerun IME. |
ping_portal(device) |
Verify Portal is installed and reachable. |
PORTAL_PACKAGE_NAME, A11Y_SERVICE_NAME |
Portal identifiers. |
portal_content_uri(pkg, path) |
Build content://<pkg>/<path> URIs. |
portal_a11y_service(pkg), portal_ime_id(pkg) |
Accessibility service / IME component names. |
AndroidDriver accepts:
serial: str | Noneโ ADB serial;Nonepicks the only connected device.use_tcp: bool = Falseโ whenTrue, the underlyingPortalClientport-forwards Portal's HTTP server (localhost:N โ device:8080) and uses it instead of the content provider. Faster but requires a working forward; falls back transparently.
๐ชต Logging
All output goes through the "mobilerun_core_cli" logger. Configure it yourself; the package attaches no handlers.
import logging
logging.basicConfig(level=logging.INFO)
logging.getLogger("mobilerun_core_cli").setLevel(logging.DEBUG)
๐ Relationship to upstream mobilerun
This package owns local execution drivers. The full framework imports these drivers for CLI/agent flows; mobilerun-core wraps them behind a sync, backend-neutral API.
Use mobilerun-core-cli when you need async local drivers directly.
Use mobilerun when you want the full LLM-agent experience, CLI/TUI, and multi-platform support out of the box.
๐ License
MIT โ see LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mobilerun_core_cli-0.2.0.tar.gz.
File metadata
- Download URL: mobilerun_core_cli-0.2.0.tar.gz
- Upload date:
- Size: 43.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
dc13016c7983476cca564dd013fc281047474dc502de54f8c0bd5647b48211e2
|
|
| MD5 |
28032c1aa272e08ad95fb2f5ddf725af
|
|
| BLAKE2b-256 |
d67818cd91fccb2fb348a5150707245d303b6052890a9aca2c35bd391d776208
|
Provenance
The following attestation bundles were made for mobilerun_core_cli-0.2.0.tar.gz:
Publisher:
publish.yml on droidrun/mobilerun-core-cli
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mobilerun_core_cli-0.2.0.tar.gz -
Subject digest:
dc13016c7983476cca564dd013fc281047474dc502de54f8c0bd5647b48211e2 - Sigstore transparency entry: 1753153560
- Sigstore integration time:
-
Permalink:
droidrun/mobilerun-core-cli@ea3b12400ebbf2cccbc371a1503e00d5dcc36970 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/droidrun
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@ea3b12400ebbf2cccbc371a1503e00d5dcc36970 -
Trigger Event:
push
-
Statement type:
File details
Details for the file mobilerun_core_cli-0.2.0-py3-none-any.whl.
File metadata
- Download URL: mobilerun_core_cli-0.2.0-py3-none-any.whl
- Upload date:
- Size: 30.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d94c5478d299fc10ec3f8f850dc469d65e2533db4805e0d932b15925c8e5e351
|
|
| MD5 |
f70b64918d4c0cd9ae7219d65493ce8b
|
|
| BLAKE2b-256 |
40542fe03bf7a98fb50b70580acef5c5bf3ea1c51ccf6ef867b5df81201934ea
|
Provenance
The following attestation bundles were made for mobilerun_core_cli-0.2.0-py3-none-any.whl:
Publisher:
publish.yml on droidrun/mobilerun-core-cli
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mobilerun_core_cli-0.2.0-py3-none-any.whl -
Subject digest:
d94c5478d299fc10ec3f8f850dc469d65e2533db4805e0d932b15925c8e5e351 - Sigstore transparency entry: 1753153644
- Sigstore integration time:
-
Permalink:
droidrun/mobilerun-core-cli@ea3b12400ebbf2cccbc371a1503e00d5dcc36970 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/droidrun
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@ea3b12400ebbf2cccbc371a1503e00d5dcc36970 -
Trigger Event:
push
-
Statement type: