Opinionated toolchain enforcement for Pi. Transparently rewrites commands to use preferred tools.

Opinionated toolchain enforcement for pi. Transparently mutates commands to use preferred tools or blocks mismatched commands.
pi install npm:@aliou/pi-toolchain
Or from git:
pi install git:github.com/aliou/pi-toolchain
Each feature is independently set to one of three modes:
"disabled": no action taken"mutate": transparently mutate matching commands before shell execution"block": block matching commands via tool_call hookMutates or blocks npm/yarn/bun commands to use the selected package manager. Also handles npx -> pnpm dlx/bunx.
Mutates or blocks bare python commands to use python3.
Mutates or blocks python/python3 to uv run python and pip/pip3 to uv pip.
Injects GIT_EDITOR=true and GIT_SEQUENCE_EDITOR=: env vars for git rebase commands so they run non-interactively. Only supports "disabled" and "mutate" modes (block is not applicable).
Run /toolchain:settings to open an interactive settings UI with two tabs:
.pi/extensions/toolchain.json)~/.pi/agent/extensions/toolchain.json)Use Tab / Shift+Tab to switch tabs. Feature modes and the package manager can be changed directly. Configure mutation notification visibility in JSON for now.
Configuration is loaded from two optional JSON files, merged in order (project overrides global):
~/.pi/agent/extensions/toolchain.json.pi/extensions/toolchain.json{
"enabled": true,
"features": {
"nodePackageManager": "disabled",
"pythonToPython3": "disabled",
"pythonToUv": "disabled",
"nonInteractiveGitRebase": "mutate"
},
"nodePackageManager": {
"selected": "pnpm"
},
"ui": {
"showMutationNotifications": false
}
}
All fields are optional. Missing fields use the defaults shown above.
| Feature | Default | Description |
|---|---|---|
nodePackageManager | "disabled" | Opt-in. Mutates or blocks package-manager commands. |
pythonToPython3 | "disabled" | Opt-in. Mutates or blocks bare python commands to python3. |
pythonToUv | "disabled" | Opt-in. Mutates or blocks python/pip commands to uv equivalents. |
nonInteractiveGitRebase | "mutate" | On by default. Injects non-interactive env vars for git rebase. |
ui.showMutationNotifications | false | Show a Pi notification each time a mutation happens. |
Enforce pnpm, mutate python commands, and show mutation notifications:
{
"features": {
"nodePackageManager": "mutate",
"pythonToUv": "mutate"
},
"nodePackageManager": {
"selected": "pnpm"
},
"ui": {
"showMutationNotifications": true
}
}
Block package-manager mismatches instead of mutating them:
{
"features": {
"nodePackageManager": "block"
},
"nodePackageManager": {
"selected": "pnpm"
}
}
The extension uses pi’s tool_call event hook:
"block" are rejected with a reason. The agent sees the block reason and retries with the correct command."mutate" are transparently rewritten before shell execution. The agent sees the original command in the tool call UI but gets the output of the mutated command.ui.showMutationNotifications is enabled, a warning-level Pi notification is shown for each mutation.All rewriters use structural shell parsing via @aliou/sh to identify command names in the AST. This avoids false positives where tool names appear in URLs, file paths, or strings. If the parser fails, the command passes through unchanged — a missed mutation is safe, a false positive mutation corrupts the command.
If you were using preventBrew, preventPython, or enforcePackageManager in your guardrails config:
@aliou/pi-toolchain.pi/extensions/toolchain.json with the equivalent configThe guardrails extension will continue to honor these features with a deprecation warning until they are removed in a future version.