mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-15 18:01:22 +00:00
169 lines
6.8 KiB
Markdown
169 lines
6.8 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) |
|
|
| `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 |
|
|
| `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) |
|
|
| `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 |
|
|
|
|
### 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 |
|
|
|
|
### External Dependencies
|
|
|
|
| Extension | Description |
|
|
|-----------|-------------|
|
|
| `chalk-logger.ts` | Uses chalk from parent node_modules (demonstrates jiti module resolution) |
|
|
| `with-deps/` | Extension with its own package.json and dependencies |
|
|
| `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
|
|
}
|
|
}
|
|
});
|
|
```
|