2026-01-06 18:25:52 +00:00
---
2026-01-13 07:15:57 +00:00
summary: "Multi-agent routing: isolated agents, channel accounts, and bindings"
2026-01-06 18:25:52 +00:00
title: Multi-Agent Routing
read_when: "You want multiple isolated agents (workspaces + auth) in one gateway process."
status: active
---
# Multi-Agent Routing
2026-01-13 07:15:57 +00:00
Goal: multiple *isolated* agents (separate workspace + `agentDir` + sessions), plus multiple channel accounts (e.g. two WhatsApps) in one running Gateway. Inbound is routed to an agent via bindings.
2026-01-06 18:25:52 +00:00
2026-01-06 22:44:07 +01:00
## What is “one agent”?
An **agent** is a fully scoped brain with its own:
- **Workspace** (files, AGENTS.md/SOUL.md/USER.md, local notes, persona rules).
- **State directory** (`agentDir` ) for auth profiles, model registry, and per-agent config.
- **Session store** (chat history + routing state) under `~/.clawdbot/agents/<agentId>/sessions` .
2026-01-15 04:41:38 +00:00
Auth profiles are **per-agent** . Each agent reads from its own:
```
~/.clawdbot/agents/< agentId > /agent/auth-profiles.json
```
Main agent credentials are **not** shared automatically. Never reuse `agentDir`
across agents (it causes auth/session collisions). If you want to share creds,
copy `auth-profiles.json` into the other agent's `agentDir` .
2026-01-09 01:37:19 +01:00
Skills are per-agent via each workspace’ s `skills/` folder, with shared skills
available from `~/.clawdbot/skills` . See [Skills: per-agent vs shared ](/tools/skills#per-agent-vs-shared-skills ).
2026-01-06 22:44:07 +01:00
The Gateway can host **one agent** (default) or **many agents** side-by-side.
2026-01-09 01:54:28 +01:00
**Workspace note:** each agent’ s workspace is the **default cwd** , not a hard
sandbox. Relative paths resolve inside the workspace, but absolute paths can
reach other host locations unless sandboxing is enabled. See
[Sandboxing ](/gateway/sandboxing ).
2026-01-08 07:49:07 +01:00
## Paths (quick map)
- Config: `~/.clawdbot/clawdbot.json` (or `CLAWDBOT_CONFIG_PATH` )
- State dir: `~/.clawdbot` (or `CLAWDBOT_STATE_DIR` )
- Workspace: `~/clawd` (or `~/clawd-<agentId>` )
2026-01-09 12:44:23 +00:00
- Agent dir: `~/.clawdbot/agents/<agentId>/agent` (or `agents.list[].agentDir` )
2026-01-08 07:49:07 +01:00
- Sessions: `~/.clawdbot/agents/<agentId>/sessions`
2026-01-06 22:44:07 +01:00
### Single-agent mode (default)
If you do nothing, Clawdbot runs a single agent:
- `agentId` defaults to ** `main` **.
- Sessions are keyed as `agent:main:<mainKey>` .
- Workspace defaults to `~/clawd` (or `~/clawd-<profile>` when `CLAWDBOT_PROFILE` is set).
- State defaults to `~/.clawdbot/agents/main/agent` .
2026-01-07 09:58:54 +01:00
## Agent helper
Use the agent wizard to add a new isolated agent:
```bash
clawdbot agents add work
```
2026-01-09 12:44:23 +00:00
Then add `bindings` (or let the wizard do it) to route inbound messages.
2026-01-07 09:58:54 +01:00
2026-01-08 07:49:07 +01:00
Verify with:
```bash
clawdbot agents list --bindings
```
2026-01-06 22:44:07 +01:00
## Multiple agents = multiple people, multiple personalities
With **multiple agents** , each `agentId` becomes a **fully isolated persona** :
2026-01-13 07:15:57 +00:00
- **Different phone numbers/accounts** (per channel `accountId` ).
2026-01-06 22:44:07 +01:00
- **Different personalities** (per-agent workspace files like `AGENTS.md` and `SOUL.md` ).
- **Separate auth + sessions** (no cross-talk unless explicitly enabled).
This lets **multiple people** share one Gateway server while keeping their AI “brains” and data isolated.
2026-01-12 01:25:53 +00:00
## One WhatsApp number, multiple people (DM split)
2026-01-12 01:28:52 +00:00
You can route **different WhatsApp DMs** to different agents while staying on **one WhatsApp account** . Match on sender E.164 (like `+15551234567` ) with `peer.kind: "dm"` . Replies still come from the same WhatsApp number (no per‑ agent sender identity).
2026-01-12 01:25:53 +00:00
Important detail: direct chats collapse to the agent’ s **main session key** , so true isolation requires **one agent per person** .
Example:
```json5
{
agents: {
list: [
{ id: "alex", workspace: "~/clawd-alex" },
{ id: "mia", workspace: "~/clawd-mia" }
]
},
bindings: [
2026-01-13 07:15:57 +00:00
{ agentId: "alex", match: { channel: "whatsapp", peer: { kind: "dm", id: "+15551230001" } } },
{ agentId: "mia", match: { channel: "whatsapp", peer: { kind: "dm", id: "+15551230002" } } }
2026-01-12 01:25:53 +00:00
],
2026-01-13 06:16:43 +00:00
channels: {
whatsapp: {
dmPolicy: "allowlist",
allowFrom: ["+15551230001", "+15551230002"]
}
2026-01-12 01:25:53 +00:00
}
}
```
Notes:
- DM access control is **global per WhatsApp account** (pairing/allowlist), not per agent.
- For shared groups, bind the group to one agent or use [Broadcast groups ](/broadcast-groups ).
2026-01-06 22:44:07 +01:00
## Routing rules (how messages pick an agent)
Bindings are **deterministic** and **most-specific wins** :
1. `peer` match (exact DM/group/channel id)
2. `guildId` (Discord)
3. `teamId` (Slack)
2026-01-13 07:15:57 +00:00
4. `accountId` match for a channel
5. channel-level match (`accountId: "*"` )
2026-01-09 12:44:23 +00:00
6. fallback to default agent (`agents.list[].default` , else first list entry, default: `main` )
2026-01-06 22:44:07 +01:00
## Multiple accounts / phone numbers
2026-01-13 07:15:57 +00:00
Channels that support **multiple accounts** (e.g. WhatsApp) use `accountId` to identify
2026-01-06 22:44:07 +01:00
each login. Each `accountId` can be routed to a different agent, so one server can host
multiple phone numbers without mixing sessions.
2026-01-06 18:25:52 +00:00
## Concepts
- `agentId` : one “brain” (workspace, per-agent auth, per-agent session store).
2026-01-13 07:15:57 +00:00
- `accountId` : one channel account instance (e.g. WhatsApp account `"personal"` vs `"biz"` ).
- `binding` : routes inbound messages to an `agentId` by `(channel, accountId, peer)` and optionally guild/team ids.
2026-01-06 18:25:52 +00:00
- Direct chats collapse to `agent:<agentId>:<mainKey>` (per-agent “main”; `session.mainKey` ).
## Example: two WhatsApps → two agents
`~/.clawdbot/clawdbot.json` (JSON5):
```js
{
2026-01-09 12:44:23 +00:00
agents: {
list: [
{
id: "home",
default: true,
2026-01-07 09:58:54 +01:00
name: "Home",
2026-01-06 18:25:52 +00:00
workspace: "~/clawd-home",
agentDir: "~/.clawdbot/agents/home/agent",
},
2026-01-09 12:44:23 +00:00
{
id: "work",
2026-01-07 09:58:54 +01:00
name: "Work",
2026-01-06 18:25:52 +00:00
workspace: "~/clawd-work",
agentDir: "~/.clawdbot/agents/work/agent",
},
2026-01-09 12:44:23 +00:00
],
},
2026-01-06 18:25:52 +00:00
2026-01-09 12:44:23 +00:00
// Deterministic routing: first match wins (most-specific first).
bindings: [
2026-01-13 07:15:57 +00:00
{ agentId: "home", match: { channel: "whatsapp", accountId: "personal" } },
{ agentId: "work", match: { channel: "whatsapp", accountId: "biz" } },
2026-01-09 12:44:23 +00:00
// Optional per-peer override (example: send a specific group to work agent).
{
agentId: "work",
match: {
2026-01-13 07:15:57 +00:00
channel: "whatsapp",
2026-01-09 12:44:23 +00:00
accountId: "personal",
peer: { kind: "group", id: "1203630...@g .us" },
2026-01-06 18:25:52 +00:00
},
2026-01-09 12:44:23 +00:00
},
],
2026-01-06 18:25:52 +00:00
2026-01-09 12:44:23 +00:00
// Off by default: agent-to-agent messaging must be explicitly enabled + allowlisted.
tools: {
2026-01-06 18:25:52 +00:00
agentToAgent: {
enabled: false,
allow: ["home", "work"],
},
},
2026-01-13 06:16:43 +00:00
channels: {
whatsapp: {
accounts: {
personal: {
// Optional override. Default: ~/.clawdbot/credentials/whatsapp/personal
// authDir: "~/.clawdbot/credentials/whatsapp/personal",
},
biz: {
// Optional override. Default: ~/.clawdbot/credentials/whatsapp/biz
// authDir: "~/.clawdbot/credentials/whatsapp/biz",
},
2026-01-06 18:25:52 +00:00
},
},
},
}
```
2026-01-07 11:59:04 +01:00
2026-01-12 05:59:32 +00:00
## Example: WhatsApp daily chat + Telegram deep work
2026-01-13 07:15:57 +00:00
Split by channel: route WhatsApp to a fast everyday agent and Telegram to an Opus agent.
2026-01-12 05:59:32 +00:00
```json5
{
agents: {
list: [
{
id: "chat",
name: "Everyday",
workspace: "~/clawd-chat",
model: "anthropic/claude-sonnet-4-5"
},
{
id: "opus",
name: "Deep Work",
workspace: "~/clawd-opus",
model: "anthropic/claude-opus-4-5"
}
]
},
bindings: [
2026-01-13 07:15:57 +00:00
{ agentId: "chat", match: { channel: "whatsapp" } },
{ agentId: "opus", match: { channel: "telegram" } }
2026-01-12 05:59:32 +00:00
]
}
```
Notes:
2026-01-13 07:15:57 +00:00
- If you have multiple accounts for a channel, add `accountId` to the binding (for example `{ channel: "whatsapp", accountId: "personal" }` ).
- To route a single DM/group to Opus while keeping the rest on chat, add a `match.peer` binding for that peer; peer matches always win over channel-wide rules.
2026-01-12 05:59:32 +00:00
2026-01-13 07:15:57 +00:00
## Example: same channel, one peer to Opus
2026-01-12 06:00:00 +00:00
Keep WhatsApp on the fast agent, but route one DM to Opus:
```json5
{
agents: {
list: [
{ id: "chat", name: "Everyday", workspace: "~/clawd-chat", model: "anthropic/claude-sonnet-4-5" },
{ id: "opus", name: "Deep Work", workspace: "~/clawd-opus", model: "anthropic/claude-opus-4-5" }
]
},
bindings: [
2026-01-13 07:15:57 +00:00
{ agentId: "opus", match: { channel: "whatsapp", peer: { kind: "dm", id: "+15551234567" } } },
{ agentId: "chat", match: { channel: "whatsapp" } }
2026-01-12 06:00:00 +00:00
]
}
```
2026-01-13 07:15:57 +00:00
Peer bindings always win, so keep them above the channel-wide rule.
2026-01-12 06:00:00 +00:00
2026-01-07 11:59:04 +01:00
## Per-Agent Sandbox and Tool Configuration
Starting with v2026.1.6, each agent can have its own sandbox and tool restrictions:
```js
{
2026-01-09 12:44:23 +00:00
agents: {
list: [
{
id: "personal",
2026-01-07 11:59:04 +01:00
workspace: "~/clawd-personal",
sandbox: {
mode: "off", // No sandbox for personal agent
},
// No tool restrictions - all tools available
},
2026-01-09 12:44:23 +00:00
{
id: "family",
2026-01-07 11:59:04 +01:00
workspace: "~/clawd-family",
sandbox: {
mode: "all", // Always sandboxed
scope: "agent", // One container per agent
2026-01-08 00:52:15 +01:00
docker: {
// Optional one-time setup after container creation
setupCommand: "apt-get update & & apt-get install -y git curl",
},
2026-01-07 11:59:04 +01:00
},
tools: {
allow: ["read"], // Only read tool
2026-01-12 03:42:49 +00:00
deny: ["exec", "write", "edit", "apply_patch"], // Deny others
2026-01-07 11:59:04 +01:00
},
},
2026-01-09 12:44:23 +00:00
],
2026-01-07 11:59:04 +01:00
},
}
```
**Benefits:**
- **Security isolation**: Restrict tools for untrusted agents
- **Resource control**: Sandbox specific agents while keeping others on host
- **Flexible policies**: Different permissions per agent
2026-01-09 12:44:23 +00:00
Note: `tools.elevated` is **global** and sender-based; it is not configurable per agent.
2026-01-12 02:49:55 +00:00
If you need per-agent boundaries, use `agents.list[].tools` to deny `exec` .
2026-01-09 12:44:23 +00:00
For group targeting, use `agents.list[].groupChat.mentionPatterns` so @mentions map cleanly to the intended agent.
2026-01-08 22:57:08 +01:00
2026-01-07 12:24:12 +01:00
See [Multi-Agent Sandbox & Tools ](/multi-agent-sandbox-tools ) for detailed examples.