Delegate tasks to specialized subagents with configurable context modes (spawn / fork).
There are many subagent extensions for pi, this one is mine.
Specialization — Use tailored agents for specific tasks like refactoring, documentation, or research.
Context Control — Choose spawn (fresh context) or fork (inherit current session context), depending on the task.
Parallel Execution — Run multiple agents at once.
A Simpler Fork — This extension intentionally trims features from other implementations (like chaining and scope selectors) to keep the surface area small and predictable. If you want the minimal, “just delegate” experience, this is it.
pi install npm:@mjakl/pi-subagentpi install git:github.com/mjakl/pi-subagentClone this repository to your Pi extensions directory:
cd ~/.pi/agent/extensions
git clone https://github.com/mjakl/pi-subagent.git
cd pi-subagent
npm installBy default, this extension enforces two runtime guards:
- Depth guard (
--subagent-max-depth, default3)- Main agent starts at depth
0 - Delegation is allowed while
currentDepth < maxDepth - With default depth
3: depth0,1, and2can delegate; depth3cannot
- Main agent starts at depth
- Cycle guard (
--subagent-prevent-cycles, defaulttrue)- Blocks delegating to any agent name already present in the current delegation stack
- Prevents self-recursion (
writer -> writer) and loops (planner -> reviewer -> planner)
You can configure depth with either:
- CLI flag:
--subagent-max-depth <n> - Environment variable:
PI_SUBAGENT_MAX_DEPTH=<n>
n must be a non-negative integer.
You can configure cycle prevention with either:
- CLI flag:
--subagent-prevent-cycles/--no-subagent-prevent-cycles - Environment variable:
PI_SUBAGENT_PREVENT_CYCLES=true|false
Internal env vars managed by the extension and propagated to child processes:
PI_SUBAGENT_DEPTHPI_SUBAGENT_MAX_DEPTHPI_SUBAGENT_STACK(JSON array of ancestor agent names, e.g.["scout","planner"])PI_SUBAGENT_PREVENT_CYCLES
Recommended extension-integration note:
If another extension needs to detect whether it is running inside a delegated subagent process, check PI_SUBAGENT_DEPTH. Treat PI_SUBAGENT_DEPTH > 0 as "this pi process is a subagent". This is the recommended way to suppress parent-only behavior such as bells, desktop notifications, or other attention-grabbing signals.
Examples:
# Default behavior: depth 3 + cycle prevention enabled
pi
# Restrict to one nested level (main -> child -> grandchild)
pi --subagent-max-depth 2
# Disable subagent delegation entirely
pi --subagent-max-depth 0
# Allow depth 3 but disable cycle prevention (not recommended)
pi --subagent-max-depth 3 --no-subagent-prevent-cyclessubagent supports a top-level mode switch:
spawn(default) — Child receives only the task string (Task: ...). Best for isolated, reproducible work; typically lower token/cost and less context leakage.fork— Child receives a forked snapshot of the current session context plus the task string. Best for follow-up work that depends on prior context; typically higher token/cost and may include sensitive context.
Quick rule of thumb:
- Start with
spawnfor one-off tasks. - Use
forkwhen the delegated task depends on the current session's prior discussion, reads, or decisions.
Examples:
{ "agent": "writer", "task": "Document the API", "mode": "spawn" }{ "agent": "review", "task": "Double-check this migration", "mode": "fork" }If omitted, mode defaults to spawn.
Subagents are defined as Markdown files with YAML frontmatter.
User Agents: ~/.pi/agent/agents/*.md by default, or $PI_CODING_AGENT_DIR/agents/*.md when PI_CODING_AGENT_DIR is set
Project Agents: .pi/agents/*.md
PI_CODING_AGENT_DIR follows Pi's config-dir override semantics: when it is set, the extension uses $PI_CODING_AGENT_DIR/agents as the user/global agent directory instead of ~/.pi/agent/agents. Project agents are still loaded in addition to the active user/global directory, and project agents win on name conflicts. When project agents are requested, Pi will prompt for confirmation before running them.
If no user or project subagents can be found, pi-subagent creates a starter user agent named explorer in the active user agents directory:
~/.pi/agent/agents/explorer.mdby default$PI_CODING_AGENT_DIR/agents/explorer.mdwhenPI_CODING_AGENT_DIRis set
The starter is read-only (read, grep, find, ls) and is meant for focused codebase exploration. Existing files are never overwritten. If you delete every subagent, the starter will be recreated the next time Pi starts with this extension.
Example agent (~/.pi/agent/agents/writer.md):
---
name: writer
description: Expert technical writer and editor
model: anthropic/claude-3-5-sonnet
tools: read, write
---
You are an expert technical writer. Your task is to improve the clarity and conciseness of the provided text.Note: this repository includes a sample agent in agents/oracle.md for reference.
| Field | Required | Default | Description |
|---|---|---|---|
name |
Yes | — | Agent identifier used in tool calls (must match exactly) |
description |
Yes | — | What the agent does (shown to the main agent) |
model |
No | Uses the default pi model | Overrides the model for this agent. You can include a provider prefix (e.g. anthropic/claude-3-5-sonnet or openrouter/claude-3.5-sonnet) to force a specific provider. |
thinking |
No | Uses Pi's default thinking level | Sets the thinking level (off, minimal, low, medium, high, xhigh). Equivalent to --thinking. |
tools |
No | read,bash,edit,write |
Comma-separated list of built-in tools to enable for this agent. If omitted, defaults apply. |
Notes:
modelacceptsprovider/modelsyntax — this is a Pi feature. Use it when multiple providers offer the same model ID.thinkinguses the same values as Pi's--thinkingflag; it's recommended to set it explicitly since thinking support varies by model.toolsonly controls built-in tools. Extension tools remain available unless extensions are disabled.- The Markdown body below the frontmatter becomes the agent's system prompt and is appended to Pi's default system prompt (it does not replace it).
- Description matters — the main agent uses the
descriptionto decide which subagent to call, so be specific about what the agent is good at. - Tool scope is optional but helpful — reducing tools can keep the agent focused, but you can leave defaults if unsure.
- Model + thinking is the power combo — selecting the right model and thinking level is often the biggest quality boost.
Available Tools (default: read, bash, edit, write):
read— Read file contentsbash— Execute bash commandsedit— Edit files with find/replacewrite— Write files (creates/overwrites)grep— Search file contents (read-only, off by default)find— Find files by glob pattern (read-only, off by default)ls— List directory contents (read-only, off by default)
Tip: for a read-only tool selection, use read,find,ls,grep. As soon as you include edit, write, or bash, the agent can practically go wild.
Each subagent always runs in a separate pi process:
- ❌ No shared memory/state with the parent process
- ❌ No visibility into sibling subagents
- ✅ Its own model/tool/runtime loop
- ✅ Started with
PI_OFFLINE=1to skip startup network operations and reduce spawn latency - ✅ Inherits relevant parent CLI configuration such as extensions, provider/theme/skill flags, resolves inherited relative resource paths against the parent cwd, and reuses parent
--model/--thinking/--toolsvalues when the agent file does not override them
What it can see depends on mode:
spawn(default)- ✅ Receives: subagent system prompt +
Task: ... - ❌ Does not receive parent session history
- ✅ Receives: subagent system prompt +
fork- ✅ Receives: forked snapshot of current parent session context +
Task: ...
- ✅ Receives: forked snapshot of current parent session context +
subagent({ agent: "writer", task: "Document the API" }) sends:
[System Prompt from ~/.pi/agent/agents/writer.md]
User: Task: Document the API
No parent conversation history is included. In spawn, include all required context in task.
subagent({ agent: "writer", task: "Document the API", mode: "fork" }) sends:
[Forked snapshot of current session context]
[System Prompt from ~/.pi/agent/agents/writer.md]
User: Task: Document the API
Note: fork copies session context, not transient runtime-only prompt mutations from the parent process.
| Data | Main Agent Sees | TUI Shows |
|---|---|---|
| Final text output | ✅ Yes — full, unbounded | ✅ Yes |
| Tool calls made by subagent | ❌ No | ✅ Yes (expanded view) |
| Token usage / cost | ❌ No | ✅ Yes |
| Reasoning/thinking steps | ❌ No | ❌ No |
| Error messages | ✅ Yes (on failure) | ✅ Yes |
Key point: The main agent receives only the final assistant text from each subagent. Not the tool calls, not the reasoning, not the intermediate steps. This prevents context pollution while still giving you the results.
When running multiple agents in parallel:
- All subagents start simultaneously (up to 4 concurrent)
- The top-level
modeapplies to all tasks in that call - Main agent receives a combined result after all finish:
Parallel: 3/3 succeeded
[writer] completed: Full output text here...
[tester] completed: Full output text here...
[reviewer] completed: Full output text here...
- Auto-Discovery — Agents are found at startup and their descriptions are injected into the main agent's system prompt.
- Context Mode Switch —
spawn(fresh context) andfork(session snapshot + task) per call. - Depth + Cycle Guards — Depth limiting and ancestry-cycle checks prevent runaway recursive delegation by default.
- Streaming Updates — Watch subagent progress in real-time as tool calls and outputs stream in.
- Rich TUI Rendering — Collapsed/expanded views with usage stats, tool call previews, and markdown output.
- Security Confirmation — Project-local agents require explicit user approval before execution.
index.ts — Extension entry point: lifecycle hooks, tool registration, mode orchestration
agents.ts — Agent discovery: reads and parses .md files from the active Pi config dir and project directories
runner-cli.js — Parent CLI inheritance: parses and normalizes flags forwarded to child processes
runner.ts — Process runner: starts `pi` subprocesses in spawn/fork context modes and streams JSON events
render.ts — TUI rendering: renderCall and renderResult for the subagent tool
types.ts — Shared types and pure helper functions
Inspired by implementations from vaayne/agent-kit and mariozechner/pi-mono.
MIT