mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-16 07:04:25 +00:00
When the agent is running, we prefix an animated braille spinner to the title to show busy state. Its cleared when the agent is done
200 lines
8.2 KiB
Markdown
200 lines
8.2 KiB
Markdown
# Extension Examples
|
|
|
|
Example extensions for pi-coding-agent.
|
|
|
|
## Usage
|
|
|
|
```bash
|
|
# Load an extension with --extension flag
|
|
pi --extension examples/extensions/permission-gate.ts
|
|
|
|
# Or copy to extensions directory for auto-discovery
|
|
cp permission-gate.ts ~/.pi/agent/extensions/
|
|
```
|
|
|
|
## Examples
|
|
|
|
### Lifecycle & Safety
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `permission-gate.ts` | Prompts for confirmation before dangerous bash commands (rm -rf, sudo, etc.) |
|
|
| `protected-paths.ts` | Blocks writes to protected paths (.env, .git/, node_modules/) |
|
|
| `confirm-destructive.ts` | Confirms before destructive session actions (clear, switch, fork) |
|
|
| `dirty-repo-guard.ts` | Prevents session changes with uncommitted git changes |
|
|
| `sandbox/` | OS-level sandboxing using `@anthropic-ai/sandbox-runtime` with per-project config |
|
|
|
|
### Custom Tools
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `todo.ts` | Todo list tool + `/todos` command with custom rendering and state persistence |
|
|
| `hello.ts` | Minimal custom tool example |
|
|
| `question.ts` | Demonstrates `ctx.ui.select()` for asking the user questions with custom UI |
|
|
| `questionnaire.ts` | Multi-question input with tab bar navigation between questions |
|
|
| `tool-override.ts` | Override built-in tools (e.g., add logging/access control to `read`) |
|
|
| `truncated-tool.ts` | Wraps ripgrep with proper output truncation (50KB/2000 lines) |
|
|
| `antigravity-image-gen.ts` | Generate images via Google Antigravity with optional save-to-disk modes |
|
|
| `ssh.ts` | Delegate all tools to a remote machine via SSH using pluggable operations |
|
|
| `subagent/` | Delegate tasks to specialized subagents with isolated context windows |
|
|
|
|
### Commands & UI
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `preset.ts` | Named presets for model, thinking level, tools, and instructions via `--preset` flag and `/preset` command |
|
|
| `plan-mode/` | Claude Code-style plan mode for read-only exploration with `/plan` command and step tracking |
|
|
| `tools.ts` | Interactive `/tools` command to enable/disable tools with session persistence |
|
|
| `handoff.ts` | Transfer context to a new focused session via `/handoff <goal>` |
|
|
| `qna.ts` | Extracts questions from last response into editor via `ctx.ui.setEditorText()` |
|
|
| `status-line.ts` | Shows turn progress in footer via `ctx.ui.setStatus()` with themed colors |
|
|
| `widget-placement.ts` | Shows widgets above and below the editor via `ctx.ui.setWidget()` placement |
|
|
| `model-status.ts` | Shows model changes in status bar via `model_select` hook |
|
|
| `snake.ts` | Snake game with custom UI, keyboard handling, and session persistence |
|
|
| `send-user-message.ts` | Demonstrates `pi.sendUserMessage()` for sending user messages from extensions |
|
|
| `timed-confirm.ts` | Demonstrates AbortSignal for auto-dismissing `ctx.ui.confirm()` and `ctx.ui.select()` dialogs |
|
|
| `modal-editor.ts` | Custom vim-like modal editor via `ctx.ui.setEditorComponent()` |
|
|
| `rainbow-editor.ts` | Animated rainbow text effect via custom editor |
|
|
| `notify.ts` | Desktop notifications via OSC 777 when agent finishes (Ghostty, iTerm2, WezTerm) |
|
|
| `titlebar-spinner.ts` | Braille spinner animation in terminal title while the agent is working |
|
|
| `summarize.ts` | Summarize conversation with GPT-5.2 and show in transient UI |
|
|
| `custom-footer.ts` | Custom footer with git branch and token stats via `ctx.ui.setFooter()` |
|
|
| `custom-header.ts` | Custom header via `ctx.ui.setHeader()` |
|
|
| `overlay-test.ts` | Test overlay compositing with inline text inputs and edge cases |
|
|
| `overlay-qa-tests.ts` | Comprehensive overlay QA tests: anchors, margins, stacking, overflow, animation |
|
|
| `doom-overlay/` | DOOM game running as an overlay at 35 FPS (demonstrates real-time game rendering) |
|
|
| `shutdown-command.ts` | Adds `/quit` command demonstrating `ctx.shutdown()` |
|
|
| `interactive-shell.ts` | Run interactive commands (vim, htop) with full terminal via `user_bash` hook |
|
|
| `inline-bash.ts` | Expands `!{command}` patterns in prompts via `input` event transformation |
|
|
|
|
### Git Integration
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `git-checkpoint.ts` | Creates git stash checkpoints at each turn for code restoration on fork |
|
|
| `auto-commit-on-exit.ts` | Auto-commits on exit using last assistant message for commit message |
|
|
|
|
### System Prompt & Compaction
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `pirate.ts` | Demonstrates `systemPromptAppend` to dynamically modify system prompt |
|
|
| `claude-rules.ts` | Scans `.claude/rules/` folder and lists rules in system prompt |
|
|
| `custom-compaction.ts` | Custom compaction that summarizes entire conversation |
|
|
| `trigger-compact.ts` | Triggers compaction when context usage exceeds 100k tokens and adds `/trigger-compact` command |
|
|
|
|
### System Integration
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `mac-system-theme.ts` | Syncs pi theme with macOS dark/light mode |
|
|
|
|
### Resources
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `dynamic-resources/` | Loads skills, prompts, and themes using `resources_discover` |
|
|
|
|
### Messages & Communication
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `message-renderer.ts` | Custom message rendering with colors and expandable details via `registerMessageRenderer` |
|
|
| `event-bus.ts` | Inter-extension communication via `pi.events` |
|
|
|
|
### Session Metadata
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `session-name.ts` | Name sessions for the session selector via `setSessionName` |
|
|
| `bookmark.ts` | Bookmark entries with labels for `/tree` navigation via `setLabel` |
|
|
|
|
### Custom Providers
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `custom-provider-anthropic/` | Custom Anthropic provider with OAuth support and custom streaming implementation |
|
|
| `custom-provider-gitlab-duo/` | GitLab Duo provider using pi-ai's built-in Anthropic/OpenAI streaming via proxy |
|
|
| `custom-provider-qwen-cli/` | Qwen CLI provider with OAuth device flow and OpenAI-compatible models |
|
|
|
|
### External Dependencies
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `with-deps/` | Extension with its own package.json and dependencies (demonstrates jiti module resolution) |
|
|
| `file-trigger.ts` | Watches a trigger file and injects contents into conversation |
|
|
|
|
## Writing Extensions
|
|
|
|
See [docs/extensions.md](../../docs/extensions.md) for full documentation.
|
|
|
|
```typescript
|
|
import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
|
|
import { Type } from "@sinclair/typebox";
|
|
|
|
export default function (pi: ExtensionAPI) {
|
|
// Subscribe to lifecycle events
|
|
pi.on("tool_call", async (event, ctx) => {
|
|
if (event.toolName === "bash" && event.input.command?.includes("rm -rf")) {
|
|
const ok = await ctx.ui.confirm("Dangerous!", "Allow rm -rf?");
|
|
if (!ok) return { block: true, reason: "Blocked by user" };
|
|
}
|
|
});
|
|
|
|
// Register custom tools
|
|
pi.registerTool({
|
|
name: "greet",
|
|
label: "Greeting",
|
|
description: "Generate a greeting",
|
|
parameters: Type.Object({
|
|
name: Type.String({ description: "Name to greet" }),
|
|
}),
|
|
async execute(toolCallId, params, onUpdate, ctx, signal) {
|
|
return {
|
|
content: [{ type: "text", text: `Hello, ${params.name}!` }],
|
|
details: {},
|
|
};
|
|
},
|
|
});
|
|
|
|
// Register commands
|
|
pi.registerCommand("hello", {
|
|
description: "Say hello",
|
|
handler: async (args, ctx) => {
|
|
ctx.ui.notify("Hello!", "info");
|
|
},
|
|
});
|
|
}
|
|
```
|
|
|
|
## Key Patterns
|
|
|
|
**Use StringEnum for string parameters** (required for Google API compatibility):
|
|
```typescript
|
|
import { StringEnum } from "@mariozechner/pi-ai";
|
|
|
|
// Good
|
|
action: StringEnum(["list", "add"] as const)
|
|
|
|
// Bad - doesn't work with Google
|
|
action: Type.Union([Type.Literal("list"), Type.Literal("add")])
|
|
```
|
|
|
|
**State persistence via details:**
|
|
```typescript
|
|
// Store state in tool result details for proper forking support
|
|
return {
|
|
content: [{ type: "text", text: "Done" }],
|
|
details: { todos: [...todos], nextId }, // Persisted in session
|
|
};
|
|
|
|
// Reconstruct on session events
|
|
pi.on("session_start", async (_event, ctx) => {
|
|
for (const entry of ctx.sessionManager.getBranch()) {
|
|
if (entry.type === "message" && entry.message.toolName === "my_tool") {
|
|
const details = entry.message.details;
|
|
// Reconstruct state from details
|
|
}
|
|
}
|
|
});
|
|
```
|