@mjakl/pi-subagent
Subagent extension for Pi coding agent. Delegate tasks to specialized agents.
Package details
Install @mjakl/pi-subagent from npm and Pi will load the resources declared by the package manifest.
$ pi install npm:@mjakl/pi-subagent- Package
@mjakl/pi-subagent- Version
1.4.1- Published
- Mar 24, 2026
- Downloads
- 232/mo · 38/wk
- Author
- mjakl
- License
- MIT
- Types
- extension
- Size
- 88.6 KB
- Dependencies
- 0 dependencies · 5 peers
Pi manifest JSON
{
"extensions": [
"./index.ts"
]
}Security note
Pi packages can execute code and influence agent behavior. Review the source before installing third-party packages.
README
Pi Subagent
Delegate tasks to specialized subagents with configurable context modes (spawn / fork).
There are many subagent extensions for pi, this one is mine.
Why Pi Subagent
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.
Install
Option 1: Install from npm (recommended)
pi install npm:@mjakl/pi-subagent
Option 2: Install via git
pi install git:github.com/mjakl/pi-subagent
Option 3: Manual Installation
Clone this repository to your Pi extensions directory:
cd ~/.pi/agent/extensions
git clone https://github.com/mjakl/pi-subagent.git
cd pi-subagent
npm install
Configuration
Delegation Guards (Depth + Cycle Prevention)
By 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
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-cycles
Context Mode (spawn vs fork)
subagent 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.
Subagent Definitions
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.
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.
Frontmatter Fields
| 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).
Writing a Good Agent File
- 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 Built-in Tools
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.
How Communication Works
The Isolation Model
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 +
What Gets Sent to Subagents
spawn mode (default)
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.
fork mode
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.
What Comes Back to the Main Agent
| 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.
Parallel Mode Behavior
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...
Features
- 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.
Project Structure
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
Attribution
Inspired by implementations from vaayne/agent-kit and mariozechner/pi-mono.
License
MIT