docs/guide/team-mode.md
Parallel multi-agent coordination for omo, modeled after Claude Code's experimental Agent Teams.
OFF by default. Enable via JSONC config.
Add to user config ~/.config/opencode/oh-my-openagent.jsonc or project config .opencode/oh-my-openagent.jsonc:
{
"team_mode": {
"enabled": true,
"max_parallel_members": 4,
"max_members": 8,
"tmux_visualization": false
}
}
After enabling, restart opencode. The 12 team_* tools become available.
All fields live under team_mode:
enabled (boolean, default false)tmux_visualization (boolean, default false)max_parallel_members (int, 1..8, default 4)max_members (int, 1..8, default 8)max_messages_per_run (int, >=1, default 10000)max_wall_clock_minutes (int, >=1, default 120)max_member_turns (int, >=1, default 500)base_dir (optional string; default resolves to ~/.omo)message_payload_max_bytes (int, >=1024, default 32768)recipient_unread_max_bytes (int, >=1024, default 262144)mailbox_poll_interval_ms (int, >=500, default 3000)Team specs live under ~/.omo/teams/{name}/config.json (user scope) or <project>/.omo/teams/{name}/config.json (project scope):
{
"name": "ccapi-explorers",
"description": "Explore the ccapi project structure.",
"lead": { "kind": "subagent_type", "subagent_type": "sisyphus" },
"members": [
{ "kind": "category", "name": "scout-1", "category": "deep", "prompt": "Scout the src/ dir for auth patterns." },
{ "kind": "category", "name": "scout-2", "category": "quick", "prompt": "Scout tests for auth coverage." }
]
}
When both scopes define the same team name, project scope wins.
version, createdAt, and leadAgentId are optional in config files. The loader fills them automatically. You can either write a top-level lead: {...} shorthand, mark one member with isLead: true, or omit both when the team has exactly one member.
kind: "subagent_type" — direct agent (atlas, sisyphus, sisyphus-junior, hephaestus). prompt optional.kind: "category" — routed through sisyphus-junior with the chosen category model. prompt REQUIRED.sisyphus, atlas, sisyphus-junior.hephaestus (needs teammate permission teammate: "allow"; otherwise use subagent_type: "sisyphus").oracle, librarian, explore, multimodal-looker, metis, momus, prometheus.Hard-reject agents fail TeamSpec parsing because they cannot write mailbox state. Use delegate-task for those agents.
team_create — spawns team and member sessions.team_send_message, team_task_create.team_task_update with status: "claimed"), report back via team_send_message.team_shutdown_request → member or lead acks via team_approve_shutdown / team_reject_shutdown.team_delete — removes runtime state, worktrees, optional tmux layout.| Tool | Purpose |
|---|---|
team_create | Spawn a team. |
team_delete | Tear down (lead only, no active members). |
team_shutdown_request | Lead asks a member to wrap up. |
team_approve_shutdown / team_reject_shutdown | Member or lead responds. |
team_send_message | Peer-to-peer mailbox; lead-only broadcast. |
team_task_create / _list / _update / _get | Shared task list. |
team_status | Aggregate runtime view. |
team_list | Declared + active teams. |
Add "worktreePath": "../wt-scout" to a member entry. Path is filesystem-relative or absolute; bare branch names are rejected. Requires git.
Set tmux_visualization: true. Requires running inside a tmux session and tmux on PATH. Failures are isolated - a missing tmux never blocks team creation.
When enabled, each member gets a dedicated tmux pane attached to that member's session via opencode attach. The pane runs the full interactive opencode TUI for the member so you can watch streaming output in real time. Panes start in each member worktree when configured, otherwise the repo root.
team_delete closes the panes and tears down the team layout. Per-member shutdown closes just that pane and rebalances the remaining layout.
team_create).team_send_message is fire-and-forget).delegate-task (budget defaults to 0).team_delete rejects active members.bunx oh-my-opencode doctor includes a team-mode check showing tmux/git availability, declared team count, and active runtime dirs.
~/.omo/
├── teams/{name}/config.json # declared specs
├── .highwatermark # parity marker for runtime state
└── runtime/{teamRunId}/
├── state.json # durable runtime state
├── inboxes/{member}/{uuid}.json # mailbox (atomic per-message files)
├── inboxes/{member}/.delivering-{uuid}.json # transient live-delivery reservation
├── inboxes/{member}/processed/ # acked messages
└── tasks/{id}.json # shared task list
.delivering-{uuid}.json files exist only while a message is being live-delivered via promptAsync. They are committed to processed/ on delivery success, released back to {uuid}.json on failure, or reclaimed on team resume if stranded by a crash (10 minute TTL). listUnreadMessages ignores dotfile entries so the fallback poll never double-injects a reserved message.
Full design: .sisyphus/plans/team-mode.md.