co-mono/packages/coding-agent/examples/extensions
Mario Zechner 1668a59bff fix(pi-dosbox): force image re-render to prevent ghosting during streaming
The TUI only re-renders changed lines. During agent streaming, the
conversation lines above DOSBox change but DOSBox lines stay the same.
This caused the image to stay at its old position while text scrolled.

Fix: Add invisible timestamp to image line to force TUI to see it as
changed every render, ensuring the image is always re-rendered at the
correct position.
2026-01-22 05:30:57 +01:00
..
doom-overlay Don't force rerender in doom 2026-01-13 22:42:00 +01:00
pi-dosbox fix(pi-dosbox): force image re-render to prevent ghosting during streaming 2026-01-22 05:30:57 +01:00
plan-mode fix(plan-mode): change shortcut from Shift+P to Ctrl+Alt+P (#746) 2026-01-15 13:09:16 +01:00
sandbox Add MiniMax provider support (#656 by @dannote) 2026-01-13 02:27:09 +01:00
subagent Centralize frontmatter parsing + parse frontmatter with yaml library (#728) 2026-01-16 00:31:53 +01:00
with-deps Release v0.49.3 2026-01-22 01:42:43 +01:00
antigravity-image-gen.ts feat(extensions): add antigravity-image-gen example for image generation (#893) 2026-01-22 01:34:22 +01:00
auto-commit-on-exit.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
chalk-logger.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
claude-rules.ts Allow extensions to modify system prompt in before_agent_start 2026-01-08 19:54:34 +01:00
confirm-destructive.ts Rename /branch command to /fork 2026-01-11 23:12:31 +01:00
custom-compaction.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
custom-footer.ts feat(coding-agent): add FooterDataProvider for git branch and extension statuses 2026-01-09 07:39:30 -08:00
custom-header.ts feat(coding-agent): export VERSION and update custom-header example (#798) 2026-01-17 10:54:00 +01:00
dirty-repo-guard.ts Rename /branch command to /fork 2026-01-11 23:12:31 +01:00
file-trigger.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
git-checkpoint.ts Rename /branch command to /fork 2026-01-11 23:12:31 +01:00
handoff.ts Add changelog, update docs, fix handoff example for PR #642 2026-01-12 00:10:14 +01:00
hello.ts Flatten hello/ and question/ extension examples to single files 2026-01-05 01:45:10 +01:00
inline-bash.ts feat(extensions): add inline-bash example for expanding !{command} in prompts (#881) 2026-01-21 14:08:07 +01:00
input-transform.ts feat(coding-agent): add input event for extension input interception (#761) 2026-01-16 02:41:56 +01:00
interactive-shell.ts feat(coding-agent): add interactive-shell.ts extension example 2026-01-09 00:08:52 +01:00
mac-system-theme.ts Make mac-system-theme extension async to avoid blocking keyboard input 2026-01-09 00:49:31 +01:00
modal-editor.ts fix(tui): add vertical scrolling to Editor when content exceeds terminal height 2026-01-16 04:12:21 +01:00
model-status.ts feat(coding-agent): add model_select extension hook 2026-01-11 18:12:46 +01:00
notify.ts example: add desktop notification extension (OSC 777) (#658) 2026-01-12 16:33:03 +01:00
overlay-qa-tests.ts docs: move PR #667 overlay entries to unreleased section 2026-01-13 22:47:38 +01:00
overlay-test.ts feat(tui): hardware cursor positioning for IME support 2026-01-16 04:30:07 +01:00
permission-gate.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
pirate.ts Allow extensions to modify system prompt in before_agent_start 2026-01-08 19:54:34 +01:00
preset.ts Change getAllTools() to return ToolInfo[] instead of string[] 2026-01-12 17:18:43 +01:00
protected-paths.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
qna.ts feat(coding-agent): add ctx.ui.setEditorComponent() extension API 2026-01-07 16:11:49 +01:00
question.ts fix question option schema (#846) 2026-01-19 14:57:23 +01:00
questionnaire.ts fix(tui): add vertical scrolling to Editor when content exceeds terminal height 2026-01-16 04:12:21 +01:00
rainbow-editor.ts fix(tui): add vertical scrolling to Editor when content exceeds terminal height 2026-01-16 04:12:21 +01:00
README.md feat(extensions): add antigravity-image-gen example for image generation (#893) 2026-01-22 01:34:22 +01:00
send-user-message.ts Extensions: add pi.sendUserMessage() for sending user messages 2026-01-06 13:40:24 +01:00
shutdown-command.ts fix(coding-agent): add PR attribution to ctx.shutdown() changelog, fix example comments 2026-01-08 03:31:55 +01:00
snake.ts Revert "Remove Anthropic OAuth support" 2026-01-09 06:00:20 +01:00
space-invaders.ts feat(coding-agent): add Space Invaders example extension 2026-01-22 01:39:51 +01:00
ssh.ts feat(coding-agent): add user_bash event and theme API extensions 2026-01-08 21:50:56 +01:00
status-line.ts Merge hooks and custom-tools into unified extensions system (#454) 2026-01-05 01:43:35 +01:00
summarize.ts coding-agent: example extension to summarize conversation (#684) 2026-01-13 13:01:09 +01:00
timed-confirm.ts feat(coding-agent): add timeout option to extension dialogs with live countdown 2026-01-06 21:49:27 -08:00
todo.ts Rename /branch command to /fork 2026-01-11 23:12:31 +01:00
tool-override.ts docs(coding-agent): simplify tool-override example to use built-in renderer 2026-01-08 18:52:33 +01:00
tools.ts Change getAllTools() to return ToolInfo[] instead of string[] 2026-01-12 17:18:43 +01:00
trigger-compact.ts fix(coding-agent): show compaction UI for extensions 2026-01-17 11:48:54 +01:00
truncated-tool.ts Export truncation utilities for custom tools, add truncated-tool example 2026-01-06 22:13:08 +01:00
widget-placement.ts feat(coding-agent): Add widget placement option (#850) 2026-01-19 15:54:24 +01:00

Extension Examples

Example extensions for pi-coding-agent.

Usage

# 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)
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

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 for full documentation.

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):

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:

// 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
    }
  }
});