Porting From pi-mono: A Practical Merge Guide
This guide is a repeatable checklist for porting changes from pi-mono into this repo. Use it for any merge: single file, feature branch, or full release sync.
Last Sync Point
Commit: b21b42d032919de2f2e6920a76fa9a37c3920c0aDate: 2026-03-22
Update this section after each sync; do not reuse the previous range.
When starting a new sync, generate patches from this commit forward:
git format-patch b21b42d032919de2f2e6920a76fa9a37c3920c0a..HEAD --stdout > changes.patch0) Define the scope
- Identify the upstream reference (commit, tag, or PR).
- List the packages or folders you plan to touch.
- Decide which features are in-scope and which are intentionally skipped.
1) Bring code over safely
- Prefer a clean, focused diff rather than a wholesale copy.
- Avoid copying built artifacts or generated files.
- If upstream added new files, add them explicitly and review contents.
2) Match import extension conventions
Most runtime TypeScript sources omit .js in internal imports, but some test/bench entrypoints keep .js for ESM runtime compatibility. Follow the local package’s existing style; do not blanket-strip extensions.
- In
packages/coding-agentruntime sources, keep internal imports extensionless unless importing non-TS assets. - In
packages/tui/testandpackages/natives/bench, keep.jswhere surrounding files already use it. - Keep real file extensions when required by tooling (e.g.,
.json,.css,.mdtext embeds). - Example:
import { x } from "./foo.js";→import { x } from "./foo";(only when the package convention is extensionless).
3) Replace import scopes
Upstream uses different package scopes. Replace them consistently.
- Replace old scopes with the local scope used here.
- Examples (adjust to match the actual packages you are porting):
@mariozechner/pi-coding-agent→@oh-my-pi/pi-coding-agent@mariozechner/pi-agent-core→@oh-my-pi/pi-agent-core@mariozechner/pi-tui→@oh-my-pi/pi-tui@mariozechner/pi-ai→@oh-my-pi/pi-ai
4) Use Bun APIs where they improve on Node
We run on Bun. Replace Node APIs only when Bun provides a better alternative.
DO replace:
- Process spawning:
child_process.spawn→ Bun Shell$for simple commands,Bun.spawn/Bun.spawnSyncfor streaming or long-running work - File I/O:
fs.readFileSync→Bun.file().text()/Bun.write() - HTTP clients:
node-fetch,axios→ nativefetch - Crypto hashing:
node:crypto→ Web Crypto orBun.hash - SQLite:
better-sqlite3→bun:sqlite - Env loading:
dotenv→ Bun loads.envautomatically
DO NOT replace (these work fine in Bun):
os.homedir()— do NOT replace withBun.env.HOME,Bun.env.HOME, or literal"~"os.tmpdir()— do NOT replace withBun.env.TMPDIR || "/tmp"or hardcoded pathsfs.mkdtempSync()— do NOT replace with manual path constructionpath.join(),path.resolve(), etc. — these are fine
Import style: Use the node: prefix with namespace imports only (no named imports from node:fs or node:path).
Additional Bun conventions:
- Prefer Bun Shell
$for short, non-streaming commands; useBun.spawnonly when you need streaming I/O or process control. - Use
Bun.file()/Bun.write()for files andnode:fs/promisesfor directories. - Avoid
Bun.file().exists()checks; useisEnoenthandling in try/catch. - Prefer
Bun.sleep(ms)oversetTimeoutwrappers.
Wrong:
// BROKEN: env vars may be undefined, "~" is not expanded
const home = Bun.env.HOME || "~";
const tmp = Bun.env.TMPDIR || "/tmp";Correct:
import * as os from "node:os";
import * as fs from "node:fs";
import * as path from "node:path";
const configDir = path.join(os.homedir(), ".config", "myapp");
const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), "myapp-"));5) Prefer Bun embeds (no copying)
Do not copy runtime assets or vendor files at build time.
- If upstream copies assets into a dist folder, replace with Bun-friendly embeds.
- Prompts are static
.mdfiles; use Bun text imports (with { type: "text" }) and Handlebars instead of inline prompt strings. - Use
import.meta.dir+Bun.fileto load adjacent non-text resources. - Keep assets in-repo and let the bundler include them.
- Eliminate copy scripts unless the user explicitly requests them.
- If upstream reads a bundled fallback file at runtime, replace filesystem reads with a Bun text embed import.
- Example (Codex instructions fallback):
const FALLBACK_PROMPT_PATH = join(import.meta.dir, "codex-instructions.md");-> removedimport FALLBACK_INSTRUCTIONS from "./codex-instructions.md" with { type: "text" };- Use
return FALLBACK_INSTRUCTIONS;instead ofreadFileSync(FALLBACK_PROMPT_PATH, "utf8")
- Example (Codex instructions fallback):
6) Port package.json carefully
Treat package.json as a contract. Merge intentionally.
- Keep existing
name,version,type,exports, andbinunless the port requires changes. - Replace npm/node scripts with Bun equivalents (e.g.,
bun check,bun test). - Ensure dependencies use the correct scope.
- Do not downgrade dependencies to fix type errors; upgrade instead.
- Validate workspace package links and
peerDependencies.
7) Align code style and tooling
- Keep existing formatting conventions.
- Do not introduce
anyunless required. - Avoid dynamic imports and inline type imports; use top-level imports only.
- Never build prompts in code; prompts are static
.mdfiles rendered with Handlebars. - In coding-agent, never use
console.log/console.warn/console.error; useloggerfrom@oh-my-pi/pi-utils. - Use
Promise.withResolvers()instead ofnew Promise((resolve, reject) => ...). - No
private/protected/publickeywords on class fields or methods. Use ES#private fields for encapsulation; leave accessible members bare (no keyword). The only exception is constructor parameter properties (constructor(private readonly x: T)), where the keyword is required by TypeScript. When porting upstream code that usesprivate fooorprotected bar, convert to#foo(private) or barebar(accessible). - Prefer existing helpers and utilities over new ad-hoc code.
- Preserve Bun-first infrastructure changes already made in this repo:
- Runtime is Bun (no Node entry points).
- Package manager is Bun (no npm lockfiles).
- Heavy Node APIs (
child_process,readline) are replaced with Bun equivalents. - Lightweight Node APIs (
os.homedir,os.tmpdir,fs.mkdtempSync,path.*) are kept. - CLI shebangs use
bun(notnode, nottsx). - Packages use source files directly (no TypeScript build step).
- CI workflows run Bun for install/check/test.
8) Remove old compatibility layers
Unless requested, remove upstream compatibility shims.
- Delete old APIs that were replaced.
- Update all call sites to the new API directly.
- Do not keep
*_v2or parallel versions.
9) Update docs and references
- Replace pi-mono repo links where appropriate.
- Update examples to use Bun and correct package scopes.
- Ensure README instructions still match the current repo behavior.
10) Validate the port
Run the standard checks after changes:
bun check
If the repo already has failing checks unrelated to your changes, call that out. Tests use Bun's runner (not Vitest), but only run bun test when explicitly requested.
11) Protect improved features (regression trap list)
If you already improved behavior locally, treat those as non‑negotiable. Before porting, write down the improvements and add explicit checks so they don’t get lost in the merge.
- Freeze the expected behavior: add a short “before/after” note for each improvement (inputs, outputs, defaults, edge cases). This prevents silent rollback.
- Map old → new APIs: if upstream renamed concepts (hooks → extensions, custom tools → tools, etc.), ensure every old entry point still wires through. One missed flag or export equals lost functionality.
- Verify exports: check
package.jsonexports, public types, and barrel files. Upstream ports often forget to re-export local additions. - Cover non‑happy paths: if you fixed error handling, timeouts, or fallback logic, add a test or at least a manual checklist that exercises those paths.
- Check defaults and config merge order: improvements often live in defaults. Confirm new defaults didn’t revert (e.g., new config precedence, disabled features, tool lists).
- Audit env/shell behavior: if you fixed execution or sandboxing, verify the new path still uses your sanitized env and does not reintroduce alias/function overrides.
- Re-run targeted samples: keep a minimal set of "known good" examples and run them after the port (CLI flags, extension registration, tool execution).
12) Detect and handle reworked code
Before porting a file, check if upstream significantly refactored it:
# Compare the file you're about to port against what you have locally
git diff HEAD upstream/main -- path/to/file.tsIf the diff shows the file was reworked (not just patched):
- New abstractions, renamed concepts, merged modules, changed data flow
Then you must read the new implementation thoroughly before porting. Blind merging of reworked code loses functionality because:
Note: interactive mode was recently split into controllers/utils/types. When backporting related changes, port updates into the individual files we created and ensure interactive-mode.ts wiring stays in sync.
Defaults change silently - A new variable
defaultFoo = [a, b]may replace an oldgetAllFoo()that returned[a, b, c, d, e].API options get dropped - When systems merge (e.g.,
hooks+customTools→extensions), old options may not wire through to the new implementation.Code paths go stale - A renamed concept (e.g.,
hookMessage→custom) needs updates in every switch statement, type guard, and handler—not just the definition.Context/capabilities shrink - Old APIs may have exposed
{ logger, typebox, pi }that new APIs forgot to include.
Semantic porting process
When upstream reworked a module:
Read the old implementation - Understand what it did, what options it accepted, what it exposed.
Read the new implementation - Understand the new abstractions and how they map to old behavior.
Verify feature parity - For each capability in the old code, confirm the new code preserves it or explicitly removes it.
Grep for stragglers - Search for old names/concepts that may have been missed in switch statements, handlers, UI components.
Test the boundaries - CLI flags, SDK options, event handlers, default values—these are where regressions hide.
Quick checks
# Find all uses of an old concept that may need updating
rg "oldConceptName" --type ts
# Compare default values between versions
git show upstream/main:path/to/file.ts | rg "default|DEFAULT"
# Check if all enum/union values have handlers
rg "case \"" path/to/file.ts13) Quick audit checklist
Use this as a final pass before you finish:
- [ ] Import extensions follow the local package convention (no blanket
.jsstripping) - [ ] No Node-only APIs in new/ported code
- [ ] All package scopes updated
- [ ]
package.jsonscripts use Bun - [ ] Prompts are
.mdtext imports (no inline prompt strings) - [ ] No
console.*in coding-agent (uselogger) - [ ] Assets load via Bun embed patterns (no copy scripts)
- [ ] Tests or checks run (or explicitly noted as blocked)
- [ ] No functionality regressions (see sections 11-12)
14) Commit message format
When committing a backport, follow the repo format <type>(scope): <past-tense description> and keep the commit range in the title.
fix(coding-agent): backported pi-mono changes (<from>..<to>)
packages/<package>:
- <type>: <description>
- <type>: <description> (#<issue> by @<contributor>)
packages/<other-package>:
- <type>: <description>Example:
fix(coding-agent): backported pi-mono changes (9f3eef65f..52532c7c0)
packages/ai:
- fix: handle "sensitive" stop reason from Anthropic API
- fix: normalize tool call IDs with special characters for Responses API
- fix: add overflow detection for Bedrock, MiniMax, Kimi providers
- fix: 429 status is rate limiting, not context overflow
packages/tui:
- fix: refactored autocomplete state tracking
- fix: file autocomplete should not trigger on empty text
- fix: configurable autocomplete max visible items
- fix: improved table column width calculation with word-aware wrapping
packages/coding-agent:
- fix: preserve external config.yml edits on save (#1046 by @nicobailonMD)
- fix: resolve macOS NFD and curly quote variants in file pathsRules:
- Group changes by package
- Use conventional commit types (
fix,feat,refactor,perf,docs) - Include upstream issue/PR numbers and contributor attribution for external contributions
- The commit range in the title helps track sync points
15) Intentional Divergences
Our fork has architectural decisions that differ from upstream. Do not port these upstream patterns:
UI Architecture
| Upstream | Our Fork | Reason |
|---|---|---|
FooterDataProvider class | StatusLineComponent | Simpler, integrated status line |
ctx.ui.setHeader() / ctx.ui.setFooter() | Stub in non-TUI modes | Implemented in TUI, no-op elsewhere |
ctx.ui.setEditorComponent() | Stub in non-TUI modes | Implemented in TUI, no-op elsewhere |
InteractiveModeOptions options object | Positional constructor args (options type still exported) | Keep constructor signature; update the type when upstream adds fields |
Component Naming
| Upstream | Our Fork |
|---|---|
extension-input.ts | hook-input.ts |
extension-selector.ts | hook-selector.ts |
ExtensionInputComponent | HookInputComponent |
ExtensionSelectorComponent | HookSelectorComponent |
API Naming
| Upstream | Our Fork | Notes |
|---|---|---|
sessionManager.appendSessionInfo(name) | sessionManager.setSessionName(name) | We use sessionName throughout |
sessionManager.getSessionName() | sessionManager.getSessionName() | Same (we unified to match upstream's RPC) |
agent.sessionName / setSessionName() | agent.sessionName / setSessionName() | Same |
File Consolidation
| Upstream | Our Fork | Reason |
|---|---|---|
clipboard.ts + clipboard-image.ts (tool files) | @oh-my-pi/pi-natives clipboard module | Merged into N-API native implementation |
Test Framework
| Upstream | Our Fork |
|---|---|
vitest with vi.mock() | bun:test with vi from bun |
node:test assertions | expect() matchers |
Tool Architecture
| Upstream | Our Fork | Notes |
|---|---|---|
createTool(cwd: string, options?) | createTools(session: ToolSession) via BUILTIN_TOOLS registry | Tool factories accept ToolSession and can return null |
Per-tool *Operations interfaces | Per-tool interfaces remain (FindOperations, GrepOperations) | Used for SSH/remote overrides |
Node.js fs/promises everywhere | Bun.file()/Bun.write() for files; node:fs/promises for dirs | Prefer Bun APIs when they simplify |
Auth Storage
| Upstream | Our Fork | Notes |
|---|---|---|
proper-lockfile + auth.json | agent.db (bun:sqlite) | Credentials stored exclusively in agent.db |
| Single credential per provider | Multi-credential with round-robin selection | Session affinity and backoff logic preserved |
Extensions
| Upstream | Our Fork |
|---|---|
jiti for TypeScript loading | Native Bun import() |
pkg.pi manifest field | pkg.omp ?? pkg.pi (prefer our namespace) |
Skip These Upstream Features
When porting, skip these files/features entirely:
footer-data-provider.ts— we use StatusLineComponentclipboard-image.ts— clipboard is in@oh-my-pi/pi-nativesN-API module- GitHub workflow files — we have our own CI
models.generated.ts— auto-generated, regenerate locally (as models.json instead)
Features We Added (Preserve These)
These exist in our fork but not upstream. Never overwrite:
StatusLineComponentin interactive mode- Multi-credential auth with session affinity
- Capability-based discovery system (
defineCapability,registerProvider,loadCapability,skillCapability, etc.) - MCP/Exa/SSH integrations
- LSP writethrough for format-on-save
- Bash interception (
checkBashInterception) - Fuzzy path suggestions in read tool