Team Mode
June 29, 2026 · View on GitHub
Parallel multi-agent coordination for omo, modeled after Claude Code's experimental Agent Teams.
Status
OFF by default. Enable via JSONC config.
When to use
- Parallel exploration with bounded coordination.
- Long-running multi-step refactors split across specialised agents.
- Research + implementation pipelines that need shared task lists.
Enable
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.
Bug-fix note: v4.2.1 adds a fresh-install regression test for this minimal config and logs the resolved
team_modestate plus team tool count during startup. If the tools still do not appear after restart, inspectoh-my-opencode.logfor the loaded config path and[tool-registry] Built tool registryentry.
Config schema (11 fields)
All fields live under team_mode:
enabled(boolean, defaultfalse)tmux_visualization(boolean, defaultfalse)max_parallel_members(int,1..8, default4)max_members(int,1..8, default8)max_messages_per_run(int,>=1, default10000)max_wall_clock_minutes(int,>=1, default120)max_member_turns(int,>=1, default500)base_dir(optional string; default resolves to~/.omo)message_payload_max_bytes(int,>=1024, default32768)recipient_unread_max_bytes(int,>=1024, default262144)mailbox_poll_interval_ms(int,>=500, default3000)
Define a team
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 source directory 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.
Member kinds
kind: "subagent_type"— direct agent (atlas, sisyphus, sisyphus-junior, hephaestus).promptoptional.kind: "category"— routed throughsisyphus-juniorwith the chosen category model.promptREQUIRED.
Eligible agents
- Eligible:
sisyphus,atlas,sisyphus-junior. - Conditional:
hephaestus(needs teammate permissionteammate: "allow"; otherwise usesubagent_type: "sisyphus"). - Hard-reject:
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.
Lifecycle
team_create— spawns team and member sessions.- Lead delegates work via
team_send_message,team_task_create. - Members claim tasks (
team_task_updatewithstatus: "claimed"), report back viateam_send_message. team_shutdown_request→ member or lead acks viateam_approve_shutdown/team_reject_shutdown.team_delete— removes runtime state, worktrees, optional tmux layout.
12 tools
| 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. |
Bounds (defaults)
- 8 members max, 4 in flight.
- 32 KB per message body, 256 KB per recipient unread.
- 10 000 messages per run, 120 minutes wall clock, 500 turns per member.
Worktrees (optional per member)
Add "worktreePath": "../wt-scout" to a member entry. Path is filesystem-relative or absolute; bare branch names are rejected. Requires git.
tmux visualization (optional)
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.
What team mode does NOT do
- No nested teams (members cannot call
team_create). - No synchronous reply waits (
team_send_messageis fire-and-forget). - No member-driven
delegate-task(budget defaults to 0). - No shutdown bypass —
team_deleterejects active members.
Diagnostics
bunx oh-my-openagent doctor includes a team-mode check showing tmux/git availability, declared team count, and active runtime dirs.
Storage layout
~/.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.
Reference
Full design: .omo/plans/team-mode.md.