2025-12-19 00:29:42 +01:00
---
2025-12-20 02:08:04 +00:00
summary: "Nodes: pairing, capabilities, permissions, and CLI helpers for canvas/camera/screen/system"
2025-12-19 00:29:42 +01:00
read_when:
- Pairing iOS/Android nodes to a gateway
- Using node canvas/camera for agent context
- Adding new node commands or CLI helpers
---
# Nodes
2026-01-22 22:02:00 +00:00
A **node** is a companion device (macOS/iOS/Android/headless) that connects to the Gateway **WebSocket** (same port as operators) with `role: "node"` and exposes a command surface (e.g. `canvas.*` , `camera.*` , `system.*` ) via `node.invoke` . Protocol details: [Gateway protocol ](/gateway/protocol ).
2025-12-19 00:29:42 +01:00
2026-01-22 23:07:58 +00:00
Legacy transport: [Bridge protocol ](/gateway/bridge-protocol ) (TCP JSONL; deprecated/removed for current nodes).
2026-01-22 22:02:00 +00:00
macOS can also run in **node mode** : the menubar app connects to the Gateway’ s WS server and exposes its local canvas/camera commands as a node (so `clawdbot nodes …` works against this Mac).
2025-12-19 01:48:19 +01:00
2026-01-11 03:17:06 +01:00
Notes:
2026-01-21 17:45:12 +00:00
- Nodes are **peripherals** , not gateways. They don’ t run the gateway service.
2026-01-11 03:17:06 +01:00
- Telegram/WhatsApp/etc. messages land on the **gateway** , not on nodes.
2025-12-19 00:29:42 +01:00
## Pairing + status
2026-01-22 22:02:00 +00:00
**WS nodes use device pairing.** Nodes present a device identity during `connect` ; the Gateway
creates a device pairing request for `role: node` . Approve via the devices CLI (or UI).
2025-12-19 00:29:42 +01:00
Quick CLI:
```bash
2026-01-22 22:02:00 +00:00
clawdbot devices list
clawdbot devices approve < requestId >
clawdbot devices reject < requestId >
2026-01-04 14:32:47 +00:00
clawdbot nodes status
clawdbot nodes describe --node < idOrNameOrIp >
2025-12-19 00:29:42 +01:00
```
2025-12-27 01:36:24 +01:00
Notes:
2026-01-22 22:02:00 +00:00
- `nodes status` marks a node as **paired** when its device pairing role includes `node` .
- `node.pair.*` (CLI: `clawdbot nodes pending/approve/reject` ) is a separate gateway-owned
node pairing store; it does **not** gate the WS `connect` handshake.
2025-12-27 01:36:24 +01:00
2026-01-21 04:14:34 +00:00
## Remote node host (system.run)
Use a **node host** when your Gateway runs on one machine and you want commands
to execute on another. The model still talks to the **gateway** ; the gateway
forwards `exec` calls to the **node host** when `host=node` is selected.
### What runs where
- **Gateway host**: receives messages, runs the model, routes tool calls.
- **Node host**: executes `system.run` /`system.which` on the node machine.
- **Approvals**: enforced on the node host via `~/.clawdbot/exec-approvals.json` .
### Start a node host (foreground)
On the node machine:
```bash
2026-01-21 16:48:31 +00:00
clawdbot node run --host < gateway-host > --port 18789 --display-name "Build Node"
2026-01-21 04:14:34 +00:00
```
### Start a node host (service)
```bash
2026-01-21 16:48:31 +00:00
clawdbot node install --host < gateway-host > --port 18789 --display-name "Build Node"
2026-01-22 23:07:58 +00:00
clawdbot node restart
2026-01-21 04:14:34 +00:00
```
### Pair + name
On the gateway host:
```bash
clawdbot nodes pending
clawdbot nodes approve < requestId >
clawdbot nodes list
```
Naming options:
2026-01-21 16:48:31 +00:00
- `--display-name` on `clawdbot node run` / `clawdbot node install` (persists in `~/.clawdbot/node.json` on the node).
2026-01-21 04:14:34 +00:00
- `clawdbot nodes rename --node <id|name|ip> --name "Build Node"` (gateway override).
### Allowlist the commands
Exec approvals are **per node host** . Add allowlist entries from the gateway:
```bash
clawdbot approvals allowlist add --node < id | name | ip > "/usr/bin/uname"
clawdbot approvals allowlist add --node < id | name | ip > "/usr/bin/sw_vers"
```
Approvals live on the node host at `~/.clawdbot/exec-approvals.json` .
### Point exec at the node
Configure defaults (gateway config):
```bash
clawdbot config set tools.exec.host node
clawdbot config set tools.exec.security allowlist
clawdbot config set tools.exec.node "< id-or-name > "
```
Or per session:
```
/exec host=node security=allowlist node=< id-or-name >
```
Once set, any `exec` call with `host=node` runs on the node host (subject to the
node allowlist/approvals).
Related:
- [Node host CLI ](/cli/node )
- [Exec tool ](/tools/exec )
- [Exec approvals ](/tools/exec-approvals )
2025-12-19 00:29:42 +01:00
## Invoking commands
Low-level (raw RPC):
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes invoke --node < idOrNameOrIp > --command canvas.eval --params '{"javaScript":"location.href"}'
2025-12-19 00:29:42 +01:00
```
Higher-level helpers exist for the common “give the agent a MEDIA attachment” workflows.
## Screenshots (canvas snapshots)
If the node is showing the Canvas (WebView), `canvas.snapshot` returns `{ format, base64 }` .
CLI helper (writes to a temp file and prints `MEDIA:<path>` ):
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes canvas snapshot --node < idOrNameOrIp > --format png
clawdbot nodes canvas snapshot --node < idOrNameOrIp > --format jpg --max-width 1200 --quality 0.9
2025-12-19 00:29:42 +01:00
```
2026-01-11 02:44:32 +00:00
### Canvas controls
```bash
clawdbot nodes canvas present --node < idOrNameOrIp > --target https://example.com
clawdbot nodes canvas hide --node < idOrNameOrIp >
clawdbot nodes canvas navigate https://example.com --node < idOrNameOrIp >
clawdbot nodes canvas eval --node < idOrNameOrIp > --js "document.title"
```
Notes:
- `canvas present` accepts URLs or local file paths (`--target` ), plus optional `--x/--y/--width/--height` for positioning.
- `canvas eval` accepts inline JS (`--js` ) or a positional arg.
### A2UI (Canvas)
```bash
clawdbot nodes canvas a2ui push --node < idOrNameOrIp > --text "Hello"
clawdbot nodes canvas a2ui push --node < idOrNameOrIp > --jsonl ./payload.jsonl
clawdbot nodes canvas a2ui reset --node < idOrNameOrIp >
```
Notes:
- Only A2UI v0.8 JSONL is supported (v0.9/createSurface is rejected).
2025-12-19 00:29:42 +01:00
## Photos + videos (node camera)
Photos (`jpg` ):
```bash
2026-01-11 02:44:32 +00:00
clawdbot nodes camera list --node < idOrNameOrIp >
2026-01-04 14:32:47 +00:00
clawdbot nodes camera snap --node < idOrNameOrIp > # default: both facings (2 MEDIA lines)
clawdbot nodes camera snap --node < idOrNameOrIp > --facing front
2025-12-19 00:29:42 +01:00
```
Video clips (`mp4` ):
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes camera clip --node < idOrNameOrIp > --duration 10s
clawdbot nodes camera clip --node < idOrNameOrIp > --duration 3000 --no-audio
2025-12-19 00:29:42 +01:00
```
Notes:
- The node must be **foregrounded** for `canvas.*` and `camera.*` (background calls return `NODE_BACKGROUND_UNAVAILABLE` ).
- Clip duration is clamped (currently `<= 60s` ) to avoid oversized base64 payloads.
- Android will prompt for `CAMERA` /`RECORD_AUDIO` permissions when possible; denied permissions fail with `*_PERMISSION_REQUIRED` .
2025-12-19 02:56:48 +01:00
## Screen recordings (nodes)
2025-12-19 02:33:43 +01:00
2025-12-19 02:56:48 +01:00
Nodes expose `screen.record` (mp4). Example:
2025-12-19 02:33:43 +01:00
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes screen record --node < idOrNameOrIp > --duration 10s --fps 10
clawdbot nodes screen record --node < idOrNameOrIp > --duration 10s --fps 10 --no-audio
2025-12-19 02:33:43 +01:00
```
2025-12-19 02:56:48 +01:00
Notes:
- `screen.record` requires the node app to be foregrounded.
- Android will show the system screen-capture prompt before recording.
- Screen recordings are clamped to `<= 60s` .
2025-12-19 03:16:25 +01:00
- `--no-audio` disables microphone capture (supported on iOS/Android; macOS uses system capture audio).
2026-01-11 02:44:32 +00:00
- Use `--screen <index>` to select a display when multiple screens are available.
2025-12-19 02:56:48 +01:00
2026-01-04 00:54:44 +01:00
## Location (nodes)
Nodes expose `location.get` when Location is enabled in settings.
CLI helper:
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes location get --node < idOrNameOrIp >
clawdbot nodes location get --node < idOrNameOrIp > --accuracy precise --max-age 15000 --location-timeout 10000
2026-01-04 00:54:44 +01:00
```
Notes:
- Location is **off by default** .
- “Always” requires system permission; background fetch is best-effort.
- The response includes lat/lon, accuracy (meters), and timestamp.
2026-01-04 13:27:30 +01:00
## SMS (Android nodes)
Android nodes can expose `sms.send` when the user grants **SMS** permission and the device supports telephony.
Low-level invoke:
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes invoke --node < idOrNameOrIp > --command sms.send --params '{"to":"+15555550123","message":"Hello from Clawdbot"}'
2026-01-04 13:27:30 +01:00
```
Notes:
- The permission prompt must be accepted on the Android device before the capability is advertised.
- Wi-Fi-only devices without telephony will not advertise `sms.send` .
2026-01-18 07:44:28 +00:00
## System commands (node host / mac node)
2025-12-20 02:08:04 +00:00
2026-01-18 15:23:36 +00:00
The macOS node exposes `system.run` , `system.notify` , and `system.execApprovals.get/set` .
The headless node host exposes `system.run` , `system.which` , and `system.execApprovals.get/set` .
2025-12-20 02:08:04 +00:00
Examples:
```bash
2026-01-04 14:32:47 +00:00
clawdbot nodes run --node < idOrNameOrIp > -- echo "Hello from mac node"
clawdbot nodes notify --node < idOrNameOrIp > --title "Ping" --body "Gateway ready"
2025-12-20 02:08:04 +00:00
```
Notes:
- `system.run` returns stdout/stderr/exit code in the payload.
- `system.notify` respects notification permission state on the macOS app.
2026-01-11 02:44:32 +00:00
- `system.run` supports `--cwd` , `--env KEY=VAL` , `--command-timeout` , and `--needs-screen-recording` .
- `system.notify` supports `--priority <passive|active|timeSensitive>` and `--delivery <system|overlay|auto>` .
2026-01-22 23:07:58 +00:00
- macOS nodes drop `PATH` overrides; headless node hosts only accept `PATH` when it prepends the node host PATH.
2026-01-18 07:44:28 +00:00
- On macOS node mode, `system.run` is gated by exec approvals in the macOS app (Settings → Exec approvals).
Ask/allowlist/full behave the same as the headless node host; denied prompts return `SYSTEM_RUN_DENIED` .
- On headless node host, `system.run` is gated by exec approvals (`~/.clawdbot/exec-approvals.json` ).
2025-12-20 02:08:04 +00:00
2026-01-18 08:26:28 +00:00
## Exec node binding
When multiple nodes are available, you can bind exec to a specific node.
This sets the default node for `exec host=node` (and can be overridden per agent).
Global default:
```bash
clawdbot config set tools.exec.node "node-id-or-name"
```
Per-agent override:
```bash
clawdbot config get agents.list
clawdbot config set agents.list[0].tools.exec.node "node-id-or-name"
```
Unset to allow any node:
```bash
clawdbot config unset tools.exec.node
clawdbot config unset agents.list[0].tools.exec.node
```
2025-12-20 02:08:04 +00:00
## Permissions map
Nodes may include a `permissions` map in `node.list` / `node.describe` , keyed by permission name (e.g. `screenRecording` , `accessibility` ) with boolean values (`true` = granted).
2026-01-18 07:44:28 +00:00
## Headless node host (cross-platform)
Clawdbot can run a **headless node host** (no UI) that connects to the Gateway
2026-01-22 22:02:00 +00:00
WebSocket and exposes `system.run` / `system.which` . This is useful on Linux/Windows
2026-01-18 07:44:28 +00:00
or for running a minimal node alongside a server.
Start it:
```bash
2026-01-22 23:07:58 +00:00
clawdbot node run --host < gateway-host > --port 18789
2026-01-18 07:44:28 +00:00
```
Notes:
- Pairing is still required (the Gateway will show a node approval prompt).
2026-01-22 23:07:58 +00:00
- The node host stores its node id, token, display name, and gateway connection info in `~/.clawdbot/node.json` .
2026-01-18 07:44:28 +00:00
- Exec approvals are enforced locally via `~/.clawdbot/exec-approvals.json`
(see [Exec approvals ](/tools/exec-approvals )).
2026-01-21 04:46:47 +00:00
- On macOS, the headless node host prefers the companion app exec host when reachable and falls
back to local execution if the app is unavailable. Set `CLAWDBOT_NODE_EXEC_HOST=app` to require
the app, or `CLAWDBOT_NODE_EXEC_FALLBACK=0` to disable fallback.
2026-01-22 22:02:00 +00:00
- Add `--tls` / `--tls-fingerprint` when the Gateway WS uses TLS.
2026-01-18 07:44:28 +00:00
2025-12-19 02:33:43 +01:00
## Mac node mode
2026-01-22 22:02:00 +00:00
- The macOS menubar app connects to the Gateway WS server as a node (so `clawdbot nodes …` works against this Mac).
- In remote mode, the app opens an SSH tunnel for the Gateway port and connects to `localhost` .