mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-17 05:00:16 +00:00
Merge branch 'fix-311'
This commit is contained in:
commit
3f0ffc6064
3 changed files with 83 additions and 4 deletions
|
|
@ -7,6 +7,10 @@
|
||||||
- **Automatic custom system prompt loading**: Pi now auto-loads `SYSTEM.md` files to replace the default system prompt. Project-local `.pi/SYSTEM.md` takes precedence over global `~/.pi/agent/SYSTEM.md`. CLI `--system-prompt` flag overrides both. ([#309](https://github.com/badlogic/pi-mono/issues/309))
|
- **Automatic custom system prompt loading**: Pi now auto-loads `SYSTEM.md` files to replace the default system prompt. Project-local `.pi/SYSTEM.md` takes precedence over global `~/.pi/agent/SYSTEM.md`. CLI `--system-prompt` flag overrides both. ([#309](https://github.com/badlogic/pi-mono/issues/309))
|
||||||
- **Unified `/settings` command**: New settings menu consolidating thinking level, theme, queue mode, auto-compact, show images, hide thinking, and collapse changelog. Replaces individual `/thinking`, `/queue`, `/theme`, `/autocompact`, and `/show-images` commands. ([#310](https://github.com/badlogic/pi-mono/issues/310))
|
- **Unified `/settings` command**: New settings menu consolidating thinking level, theme, queue mode, auto-compact, show images, hide thinking, and collapse changelog. Replaces individual `/thinking`, `/queue`, `/theme`, `/autocompact`, and `/show-images` commands. ([#310](https://github.com/badlogic/pi-mono/issues/310))
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- **Custom tools/hooks with typebox subpath imports**: Fixed jiti alias for `@sinclair/typebox` to point to package root instead of entry file, allowing imports like `@sinclair/typebox/compiler` to resolve correctly. ([#311](https://github.com/badlogic/pi-mono/issues/311) by [@kim0](https://github.com/kim0))
|
||||||
|
|
||||||
## [0.29.0] - 2025-12-25
|
## [0.29.0] - 2025-12-25
|
||||||
|
|
||||||
### Breaking Changes
|
### Breaking Changes
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,10 @@
|
||||||
/**
|
/**
|
||||||
* Custom tool loader - loads TypeScript tool modules using jiti.
|
* Custom tool loader - loads TypeScript tool modules using jiti.
|
||||||
|
*
|
||||||
|
* For Bun compiled binaries, custom tools that import from @mariozechner/* packages
|
||||||
|
* are not supported because Bun's plugin system doesn't intercept imports from
|
||||||
|
* external files loaded at runtime. Users should use the npm-installed version
|
||||||
|
* for custom tools that depend on pi packages.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { spawn } from "node:child_process";
|
import { spawn } from "node:child_process";
|
||||||
|
|
@ -9,7 +14,7 @@ import * as os from "node:os";
|
||||||
import * as path from "node:path";
|
import * as path from "node:path";
|
||||||
import { fileURLToPath } from "node:url";
|
import { fileURLToPath } from "node:url";
|
||||||
import { createJiti } from "jiti";
|
import { createJiti } from "jiti";
|
||||||
import { getAgentDir } from "../../config.js";
|
import { getAgentDir, isBunBinary } from "../../config.js";
|
||||||
import type { HookUIContext } from "../hooks/types.js";
|
import type { HookUIContext } from "../hooks/types.js";
|
||||||
import type {
|
import type {
|
||||||
CustomToolFactory,
|
CustomToolFactory,
|
||||||
|
|
@ -31,11 +36,19 @@ function getAliases(): Record<string, string> {
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
const packageIndex = path.resolve(__dirname, "../..", "index.js");
|
const packageIndex = path.resolve(__dirname, "../..", "index.js");
|
||||||
|
|
||||||
|
// For typebox, we need the package root directory (not the entry file)
|
||||||
|
// because jiti's alias is prefix-based: imports like "@sinclair/typebox/compiler"
|
||||||
|
// get the alias prepended. If we alias to the entry file (.../build/cjs/index.js),
|
||||||
|
// then "@sinclair/typebox/compiler" becomes ".../build/cjs/index.js/compiler" (invalid).
|
||||||
|
// By aliasing to the package root, it becomes ".../typebox/compiler" which resolves correctly.
|
||||||
|
const typeboxEntry = require.resolve("@sinclair/typebox");
|
||||||
|
const typeboxRoot = typeboxEntry.replace(/\/build\/cjs\/index\.js$/, "");
|
||||||
|
|
||||||
_aliases = {
|
_aliases = {
|
||||||
"@mariozechner/pi-coding-agent": packageIndex,
|
"@mariozechner/pi-coding-agent": packageIndex,
|
||||||
"@mariozechner/pi-tui": require.resolve("@mariozechner/pi-tui"),
|
"@mariozechner/pi-tui": require.resolve("@mariozechner/pi-tui"),
|
||||||
"@mariozechner/pi-ai": require.resolve("@mariozechner/pi-ai"),
|
"@mariozechner/pi-ai": require.resolve("@mariozechner/pi-ai"),
|
||||||
"@sinclair/typebox": require.resolve("@sinclair/typebox"),
|
"@sinclair/typebox": typeboxRoot,
|
||||||
};
|
};
|
||||||
return _aliases;
|
return _aliases;
|
||||||
}
|
}
|
||||||
|
|
@ -169,7 +182,56 @@ function createNoOpUIContext(): HookUIContext {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load a single tool module using jiti.
|
* Load a tool in Bun binary mode.
|
||||||
|
*
|
||||||
|
* Since Bun plugins don't work for dynamically loaded external files,
|
||||||
|
* custom tools that import from @mariozechner/* packages won't work.
|
||||||
|
* Tools that only use standard npm packages (installed in the tool's directory)
|
||||||
|
* may still work.
|
||||||
|
*/
|
||||||
|
async function loadToolWithBun(
|
||||||
|
resolvedPath: string,
|
||||||
|
sharedApi: ToolAPI,
|
||||||
|
): Promise<{ tools: LoadedCustomTool[] | null; error: string | null }> {
|
||||||
|
try {
|
||||||
|
// Try to import directly - will work for tools without @mariozechner/* imports
|
||||||
|
const module = await import(resolvedPath);
|
||||||
|
const factory = (module.default ?? module) as CustomToolFactory;
|
||||||
|
|
||||||
|
if (typeof factory !== "function") {
|
||||||
|
return { tools: null, error: "Tool must export a default function" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const toolResult = await factory(sharedApi);
|
||||||
|
const toolsArray = Array.isArray(toolResult) ? toolResult : [toolResult];
|
||||||
|
|
||||||
|
const loadedTools: LoadedCustomTool[] = toolsArray.map((tool) => ({
|
||||||
|
path: resolvedPath,
|
||||||
|
resolvedPath,
|
||||||
|
tool,
|
||||||
|
}));
|
||||||
|
|
||||||
|
return { tools: loadedTools, error: null };
|
||||||
|
} catch (err) {
|
||||||
|
const message = err instanceof Error ? err.message : String(err);
|
||||||
|
|
||||||
|
// Check if it's a module resolution error for our packages
|
||||||
|
if (message.includes("Cannot find module") && message.includes("@mariozechner/")) {
|
||||||
|
return {
|
||||||
|
tools: null,
|
||||||
|
error:
|
||||||
|
`${message}\n` +
|
||||||
|
"Note: Custom tools importing from @mariozechner/* packages are not supported in the standalone binary.\n" +
|
||||||
|
"Please install pi via npm: npm install -g @mariozechner/pi-coding-agent",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return { tools: null, error: `Failed to load tool: ${message}` };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load a single tool module using jiti (or Bun.build for compiled binaries).
|
||||||
*/
|
*/
|
||||||
async function loadTool(
|
async function loadTool(
|
||||||
toolPath: string,
|
toolPath: string,
|
||||||
|
|
@ -178,6 +240,11 @@ async function loadTool(
|
||||||
): Promise<{ tools: LoadedCustomTool[] | null; error: string | null }> {
|
): Promise<{ tools: LoadedCustomTool[] | null; error: string | null }> {
|
||||||
const resolvedPath = resolveToolPath(toolPath, cwd);
|
const resolvedPath = resolveToolPath(toolPath, cwd);
|
||||||
|
|
||||||
|
// Use Bun.build for compiled binaries since jiti can't resolve bundled modules
|
||||||
|
if (isBunBinary) {
|
||||||
|
return loadToolWithBun(resolvedPath, sharedApi);
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Create jiti instance for TypeScript/ESM loading
|
// Create jiti instance for TypeScript/ESM loading
|
||||||
// Use aliases to resolve package imports since tools are loaded from user directories
|
// Use aliases to resolve package imports since tools are loaded from user directories
|
||||||
|
|
|
||||||
|
|
@ -23,12 +23,20 @@ function getAliases(): Record<string, string> {
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
const packageIndex = path.resolve(__dirname, "../..", "index.js");
|
const packageIndex = path.resolve(__dirname, "../..", "index.js");
|
||||||
|
|
||||||
|
// For typebox, we need the package root directory (not the entry file)
|
||||||
|
// because jiti's alias is prefix-based: imports like "@sinclair/typebox/compiler"
|
||||||
|
// get the alias prepended. If we alias to the entry file (.../build/cjs/index.js),
|
||||||
|
// then "@sinclair/typebox/compiler" becomes ".../build/cjs/index.js/compiler" (invalid).
|
||||||
|
// By aliasing to the package root, it becomes ".../typebox/compiler" which resolves correctly.
|
||||||
|
const typeboxEntry = require.resolve("@sinclair/typebox");
|
||||||
|
const typeboxRoot = typeboxEntry.replace(/\/build\/cjs\/index\.js$/, "");
|
||||||
|
|
||||||
_aliases = {
|
_aliases = {
|
||||||
"@mariozechner/pi-coding-agent": packageIndex,
|
"@mariozechner/pi-coding-agent": packageIndex,
|
||||||
"@mariozechner/pi-coding-agent/hooks": path.resolve(__dirname, "index.js"),
|
"@mariozechner/pi-coding-agent/hooks": path.resolve(__dirname, "index.js"),
|
||||||
"@mariozechner/pi-tui": require.resolve("@mariozechner/pi-tui"),
|
"@mariozechner/pi-tui": require.resolve("@mariozechner/pi-tui"),
|
||||||
"@mariozechner/pi-ai": require.resolve("@mariozechner/pi-ai"),
|
"@mariozechner/pi-ai": require.resolve("@mariozechner/pi-ai"),
|
||||||
"@sinclair/typebox": require.resolve("@sinclair/typebox"),
|
"@sinclair/typebox": typeboxRoot,
|
||||||
};
|
};
|
||||||
return _aliases;
|
return _aliases;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue