https://github.com/nativescript/simdeck
SimDeck is a developer tool built for streamlining mobile app development for coding agents. Drive iOS Simulators and Android emulators from the CLI using agents, browser, and automated tests on macOS.
https://github.com/nativescript/simdeck
android flutter ios jetpack-compose nativescript react-native swiftui typescript
Last synced: 10 days ago
JSON representation
SimDeck is a developer tool built for streamlining mobile app development for coding agents. Drive iOS Simulators and Android emulators from the CLI using agents, browser, and automated tests on macOS.
- Host: GitHub
- URL: https://github.com/nativescript/simdeck
- Owner: NativeScript
- License: mit
- Created: 2026-04-22T06:31:41.000Z (about 2 months ago)
- Default Branch: main
- Last Pushed: 2026-05-26T02:31:33.000Z (14 days ago)
- Last Synced: 2026-05-26T02:31:40.885Z (14 days ago)
- Topics: android, flutter, ios, jetpack-compose, nativescript, react-native, swiftui, typescript
- Language: Rust
- Homepage: https://simdeck.sh
- Size: 103 MB
- Stars: 33
- Watchers: 1
- Forks: 1
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Agents: AGENTS.md
Awesome Lists containing this project
README

SimDeck
SimDeck is a developer tool built for streamlining mobile app development using agents.
Drive iOS Simulators and Android emulators from your favorite IDE & CLI.

## Try it out
```sh
npx simdeck
```
Open the URL in your IDE of choice, for example in-app browser in Codex.
Install the CLI globally for agentic-use:
```sh
npm i -g simdeck@latest
```
After installing the CLI, install the Codex skill so agents know the stable
SimDeck workflow:
```sh
npx skills add NativeScript/SimDeck --skill simdeck -g
```
For VS Code, install the [`nativescript.simdeck-vscode`](https://marketplace.visualstudio.com/items?itemName=NativeScript.simdeck-vscode) extension to open the simulator
view inside the editor.
## Features
- Supports streaming both iOS simulators and Android emulators
- Full simulator control & inspection using private iOS accessibility APIs and Android UIAutomator - available using `simdeck` CLI
- Real-time screen `describe` command using accessibility view tree - available in token-efficient format for agents
- Profiling built-in: CPU, memory, disk writes, network throughput, hang signals, and stack sampling
- CoreSimulator chrome asset rendering for device bezels
- NativeScript, React Native, Flutter, UIKit and SwiftUI runtime inspector plugins to debug app's view hierarchy live
- `simdeck/test` for fast JS-based app tests that can query accessibility state and drive simulator controls
## Documentation
Full documentation lives at [simdeck.sh](https://simdeck.sh/), with guides, the CLI reference, the REST API, the video pipeline, and the inspector protocols.
For hosted pull request simulator sessions, use the GitHub Actions integration
documented in the [GitHub Actions guide](https://simdeck.sh/guide/github-actions).
## Quick start
```sh
simdeck
```
To focus a specific simulator by name or UDID, pass it as the only argument:
```sh
simdeck "iPhone 17 Pro Max"
```
Use `simdeck --open` to open the browser automatically, `simdeck -p 4311` to
use a non-default port, and `simdeck -a` to register the service for login
autostart.
The served loopback browser UI receives the generated API access token automatically.
LAN clients should pair with the printed code before receiving the API cookie.
For pairing with SimDeck iOS app:
```sh
simdeck pair
```
This starts or refreshes the LaunchAgent-backed SimDeck service, prints
local, LAN, and Tailscale URLs when available, and shows a QR code with a
`simdeck://pair` link. The QR contains the pairing code plus all detected
non-loopback addresses, so pairing once can save both the LAN and Tailscale
routes with the same service token.
Normal service restarts preserve that token so paired clients stay connected.
Use `simdeck service reset` only when you want to rotate the service token and
restart the LaunchAgent.
The service uses port 4310 unless you pass `-p` or `--port`.
Use `simdeck service kill` when you want to stop every SimDeck service process,
including services started from another checkout or installed binary.
CLI commands automatically use the same warm service:
```sh
simdeck list
simdeck use
simdeck tap 0.5 0.5 --normalized
simdeck tap "Continue"
simdeck describe --format agent --max-depth 2 --interactive
simdeck press @e3
simdeck snapshot --format agent --max-depth 2 -i
simdeck --device describe --format agent --max-depth 2
```
## CLI
```sh
simdeck list
simdeck use
simdeck boot
simdeck shutdown
simdeck erase
simdeck install /path/to/App.app
simdeck install /path/to/App.ipa
simdeck install android: /path/to/app.apk
simdeck uninstall com.example.App
simdeck open-url https://example.com
simdeck launch com.apple.Preferences
simdeck toggle-appearance
simdeck pasteboard set "hello"
simdeck pasteboard get
simdeck screenshot --output screen.png
simdeck screenshot --with-bezel --output screen-bezel.png
simdeck record --seconds 5 --output screen-recording.mp4
simdeck stream --frames 120 > stream.h264
simdeck describe
simdeck describe --format agent --max-depth 4
simdeck describe --format agent --max-depth 4 --interactive
simdeck snapshot --format agent --max-depth 4 -i
simdeck describe --point 120,240
simdeck wait-for --label "Welcome" --timeout-ms 5000
simdeck wait --label "Welcome" --timeout-ms 5000
simdeck assert --id login.button --source auto --max-depth 8
simdeck tap 120 240
simdeck tap --label "Continue" --wait-timeout-ms 5000
simdeck tap --id com.apple.settings.screenTime --expect-id BackButton
simdeck tap "Continue"
simdeck press @e3
simdeck back
simdeck swipe 200 700 200 200
simdeck gesture scroll-down
simdeck pinch --start-distance 160 --end-distance 80
simdeck rotate-gesture --radius 100 --degrees 90
simdeck touch 0.5 0.5 --phase began --normalized
simdeck touch 120 240 --down --up --delay-ms 800
simdeck key enter
simdeck key-sequence --keycodes h,e,l,l,o
simdeck key-combo --modifiers cmd --key a
simdeck type "hello"
simdeck type --file message.txt
simdeck button lock --duration-ms 1000
simdeck button volume-up
simdeck button action --duration-ms 1000
simdeck button digital-crown
simdeck crown --delta 50
simdeck button left-side-button
simdeck batch --step "tap --label Continue --expect-label Done" --step "type 'hello'" --step "back"
simdeck dismiss-keyboard
simdeck button software-keyboard
simdeck home
simdeck app-switcher
simdeck rotate-left
simdeck rotate-right
simdeck chrome-profile
simdeck logs --seconds 30 --limit 200
simdeck processes
simdeck stats --watch
simdeck sample --seconds 3
```
`simdeck list` defaults to compact JSON for agent-friendly device selection.
Use `simdeck list --format json` for the full inventory with paths and display
metadata.
`simdeck use ` stores a default simulator for the current project
directory. Most device commands accept `[]`; when it is omitted, SimDeck
uses `--device`, `SIMDECK_DEVICE`, `SIMDECK_UDID`, the saved project default,
or the only booted simulator, in that order.
## JS/TS Tests
```ts
import { connect } from "simdeck/test";
const sim = await connect({ udid: "" });
try {
await sim.tap(0.5, 0.5);
await sim.waitFor({ label: "Continue" });
await sim.screenshot();
await sim.screenshot({ withBezel: true });
await sim.record({ seconds: 5 });
} finally {
sim.close();
}
```
`connect()` starts the SimDeck service when needed and reuses it when it is
already healthy. Pass `udid` to `connect()`
to make it the default for session methods; each method still accepts an
explicit UDID as the first argument when needed. Query helpers such as
`tree()`, `query()`, `waitFor()`, `assert()`, and selector `tapElement()`
default to `source: "native-ax"` for fast agent control; pass
`source: "auto"` when a test intentionally wants richer framework inspector
trees first.
## NativeScript Inspector
NativeScript apps can connect directly to the running server from JS and expose
their NativeScript logical hierarchy plus raw UIKit backing views without
linking the Swift inspector framework:
```ts
import { startSimDeckInspector } from "@nativescript/simdeck-inspector";
if (__DEV__) {
startSimDeckInspector({ port: 4310 });
}
```
The runtime connects to `GET /api/inspector/connect` as a WebSocket. The Rust
server prefers connected NativeScript inspectors for hierarchy requests and
falls back to the Swift TCP inspector or the built-in native accessibility
bridge when no matching app inspector is available.
## React Native Inspector
React Native apps can expose their component tree and Metro dev-mode source
locations with the React Native inspector package:
```ts
import "react-native-simdeck/auto";
import "expo-router/entry";
```
Import it before `expo-router/entry` or `AppRegistry.registerComponent(...)`
so the package can capture React Fiber commits. The auto entrypoint no-ops
outside development, reads `EXPO_PUBLIC_SIMDECK_PORT` when present, and
otherwise scans common SimDeck service ports.
## Flutter Inspector
Flutter apps can expose their widget tree, render frames, semantics metadata,
and debug widget creation locations with the Flutter inspector package:
```dart
import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
import 'package:simdeck_flutter_inspector/simdeck_flutter_inspector.dart';
void main() {
WidgetsFlutterBinding.ensureInitialized();
if (kDebugMode) {
startSimDeckFlutterInspector(port: 4310);
}
runApp(const App());
}
```
## Contributing
Contributors should read [CONTRIBUTING.md](CONTRIBUTING.md) for local build
instructions, the dev workflow, and architecture notes.
## Copyright notice
Copyright [OpenJS Foundation](https://openjsf.org) and `NativeScript` contributors. All rights reserved. The [OpenJS Foundation](https://openjsf.org) has registered trademarks and uses trademarks. For a list of trademarks of the [OpenJS Foundation](https://openjsf.org), please see our [Trademark Policy](https://trademark-policy.openjsf.org/) and [Trademark List](https://trademark-list.openjsf.org/). Trademarks and logos not indicated on the [list of OpenJS Foundation trademarks](https://trademark-list.openjsf.org) are trademarks™ or registered® trademarks of their respective holders. Use of them does not imply any affiliation with or endorsement by them.
[The OpenJS Foundation](https://openjsf.org/) | [Terms of Use](https://terms-of-use.openjsf.org/) | [Privacy Policy](https://privacy-policy.openjsf.org/) | [OpenJS Foundation Bylaws](https://bylaws.openjsf.org/) | [Trademark Policy](https://trademark-policy.openjsf.org/) | [Trademark List](https://trademark-list.openjsf.org/) | [Cookie Policy](https://www.linuxfoundation.org/cookies/)
Made with ❤️