@oh-my-pi/pi-coding-agent 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +1629 -0
- package/README.md +1041 -0
- package/docs/compaction.md +403 -0
- package/docs/config-usage.md +113 -0
- package/docs/custom-tools.md +541 -0
- package/docs/extension-loading.md +1004 -0
- package/docs/hooks.md +867 -0
- package/docs/rpc.md +1040 -0
- package/docs/sdk.md +994 -0
- package/docs/session-tree-plan.md +441 -0
- package/docs/session.md +240 -0
- package/docs/skills.md +290 -0
- package/docs/theme.md +670 -0
- package/docs/tree.md +197 -0
- package/docs/tui.md +341 -0
- package/examples/README.md +21 -0
- package/examples/custom-tools/README.md +124 -0
- package/examples/custom-tools/hello/index.ts +20 -0
- package/examples/custom-tools/question/index.ts +84 -0
- package/examples/custom-tools/subagent/README.md +172 -0
- package/examples/custom-tools/subagent/agents/planner.md +37 -0
- package/examples/custom-tools/subagent/agents/scout.md +50 -0
- package/examples/custom-tools/subagent/agents/worker.md +24 -0
- package/examples/custom-tools/subagent/agents.ts +156 -0
- package/examples/custom-tools/subagent/commands/implement-and-review.md +10 -0
- package/examples/custom-tools/subagent/commands/implement.md +10 -0
- package/examples/custom-tools/subagent/commands/scout-and-plan.md +9 -0
- package/examples/custom-tools/subagent/index.ts +1002 -0
- package/examples/custom-tools/todo/index.ts +212 -0
- package/examples/hooks/README.md +56 -0
- package/examples/hooks/auto-commit-on-exit.ts +49 -0
- package/examples/hooks/confirm-destructive.ts +59 -0
- package/examples/hooks/custom-compaction.ts +116 -0
- package/examples/hooks/dirty-repo-guard.ts +52 -0
- package/examples/hooks/file-trigger.ts +41 -0
- package/examples/hooks/git-checkpoint.ts +53 -0
- package/examples/hooks/handoff.ts +150 -0
- package/examples/hooks/permission-gate.ts +34 -0
- package/examples/hooks/protected-paths.ts +30 -0
- package/examples/hooks/qna.ts +119 -0
- package/examples/hooks/snake.ts +343 -0
- package/examples/hooks/status-line.ts +40 -0
- package/examples/sdk/01-minimal.ts +22 -0
- package/examples/sdk/02-custom-model.ts +49 -0
- package/examples/sdk/03-custom-prompt.ts +44 -0
- package/examples/sdk/04-skills.ts +44 -0
- package/examples/sdk/05-tools.ts +90 -0
- package/examples/sdk/06-hooks.ts +61 -0
- package/examples/sdk/07-context-files.ts +36 -0
- package/examples/sdk/08-slash-commands.ts +42 -0
- package/examples/sdk/09-api-keys-and-oauth.ts +55 -0
- package/examples/sdk/10-settings.ts +38 -0
- package/examples/sdk/11-sessions.ts +48 -0
- package/examples/sdk/12-full-control.ts +95 -0
- package/examples/sdk/README.md +154 -0
- package/package.json +89 -0
- package/src/bun-imports.d.ts +16 -0
- package/src/capability/context-file.ts +40 -0
- package/src/capability/extension.ts +48 -0
- package/src/capability/hook.ts +40 -0
- package/src/capability/index.ts +616 -0
- package/src/capability/instruction.ts +37 -0
- package/src/capability/mcp.ts +52 -0
- package/src/capability/prompt.ts +35 -0
- package/src/capability/rule.ts +56 -0
- package/src/capability/settings.ts +35 -0
- package/src/capability/skill.ts +49 -0
- package/src/capability/slash-command.ts +40 -0
- package/src/capability/system-prompt.ts +35 -0
- package/src/capability/tool.ts +38 -0
- package/src/capability/types.ts +166 -0
- package/src/cli/args.ts +259 -0
- package/src/cli/file-processor.ts +121 -0
- package/src/cli/list-models.ts +104 -0
- package/src/cli/plugin-cli.ts +661 -0
- package/src/cli/session-picker.ts +41 -0
- package/src/cli/update-cli.ts +274 -0
- package/src/cli.ts +10 -0
- package/src/config.ts +391 -0
- package/src/core/agent-session.ts +2178 -0
- package/src/core/auth-storage.ts +258 -0
- package/src/core/bash-executor.ts +197 -0
- package/src/core/compaction/branch-summarization.ts +315 -0
- package/src/core/compaction/compaction.ts +664 -0
- package/src/core/compaction/index.ts +7 -0
- package/src/core/compaction/utils.ts +153 -0
- package/src/core/custom-commands/bundled/review/index.ts +156 -0
- package/src/core/custom-commands/index.ts +15 -0
- package/src/core/custom-commands/loader.ts +226 -0
- package/src/core/custom-commands/types.ts +112 -0
- package/src/core/custom-tools/index.ts +22 -0
- package/src/core/custom-tools/loader.ts +248 -0
- package/src/core/custom-tools/types.ts +185 -0
- package/src/core/custom-tools/wrapper.ts +29 -0
- package/src/core/exec.ts +139 -0
- package/src/core/export-html/index.ts +159 -0
- package/src/core/export-html/template.css +774 -0
- package/src/core/export-html/template.generated.ts +2 -0
- package/src/core/export-html/template.html +45 -0
- package/src/core/export-html/template.js +1185 -0
- package/src/core/export-html/template.macro.ts +24 -0
- package/src/core/file-mentions.ts +54 -0
- package/src/core/hooks/index.ts +16 -0
- package/src/core/hooks/loader.ts +288 -0
- package/src/core/hooks/runner.ts +434 -0
- package/src/core/hooks/tool-wrapper.ts +98 -0
- package/src/core/hooks/types.ts +770 -0
- package/src/core/index.ts +53 -0
- package/src/core/logger.ts +112 -0
- package/src/core/mcp/client.ts +185 -0
- package/src/core/mcp/config.ts +248 -0
- package/src/core/mcp/index.ts +45 -0
- package/src/core/mcp/loader.ts +99 -0
- package/src/core/mcp/manager.ts +235 -0
- package/src/core/mcp/tool-bridge.ts +156 -0
- package/src/core/mcp/transports/http.ts +316 -0
- package/src/core/mcp/transports/index.ts +6 -0
- package/src/core/mcp/transports/stdio.ts +252 -0
- package/src/core/mcp/types.ts +228 -0
- package/src/core/messages.ts +211 -0
- package/src/core/model-registry.ts +334 -0
- package/src/core/model-resolver.ts +494 -0
- package/src/core/plugins/doctor.ts +67 -0
- package/src/core/plugins/index.ts +38 -0
- package/src/core/plugins/installer.ts +189 -0
- package/src/core/plugins/loader.ts +339 -0
- package/src/core/plugins/manager.ts +672 -0
- package/src/core/plugins/parser.ts +105 -0
- package/src/core/plugins/paths.ts +37 -0
- package/src/core/plugins/types.ts +190 -0
- package/src/core/sdk.ts +900 -0
- package/src/core/session-manager.ts +1837 -0
- package/src/core/settings-manager.ts +860 -0
- package/src/core/skills.ts +352 -0
- package/src/core/slash-commands.ts +132 -0
- package/src/core/system-prompt.ts +442 -0
- package/src/core/timings.ts +25 -0
- package/src/core/title-generator.ts +110 -0
- package/src/core/tools/ask.ts +193 -0
- package/src/core/tools/bash-interceptor.ts +120 -0
- package/src/core/tools/bash.ts +91 -0
- package/src/core/tools/context.ts +32 -0
- package/src/core/tools/edit-diff.ts +487 -0
- package/src/core/tools/edit.ts +140 -0
- package/src/core/tools/exa/company.ts +59 -0
- package/src/core/tools/exa/index.ts +63 -0
- package/src/core/tools/exa/linkedin.ts +59 -0
- package/src/core/tools/exa/mcp-client.ts +368 -0
- package/src/core/tools/exa/render.ts +200 -0
- package/src/core/tools/exa/researcher.ts +90 -0
- package/src/core/tools/exa/search.ts +338 -0
- package/src/core/tools/exa/types.ts +167 -0
- package/src/core/tools/exa/websets.ts +248 -0
- package/src/core/tools/find.ts +244 -0
- package/src/core/tools/grep.ts +584 -0
- package/src/core/tools/index.ts +283 -0
- package/src/core/tools/ls.ts +142 -0
- package/src/core/tools/lsp/client.ts +767 -0
- package/src/core/tools/lsp/clients/biome-client.ts +207 -0
- package/src/core/tools/lsp/clients/index.ts +49 -0
- package/src/core/tools/lsp/clients/lsp-linter-client.ts +98 -0
- package/src/core/tools/lsp/config.ts +845 -0
- package/src/core/tools/lsp/edits.ts +110 -0
- package/src/core/tools/lsp/index.ts +1364 -0
- package/src/core/tools/lsp/render.ts +560 -0
- package/src/core/tools/lsp/rust-analyzer.ts +145 -0
- package/src/core/tools/lsp/types.ts +495 -0
- package/src/core/tools/lsp/utils.ts +526 -0
- package/src/core/tools/notebook.ts +182 -0
- package/src/core/tools/output.ts +198 -0
- package/src/core/tools/path-utils.ts +61 -0
- package/src/core/tools/read.ts +507 -0
- package/src/core/tools/renderers.ts +820 -0
- package/src/core/tools/review.ts +275 -0
- package/src/core/tools/rulebook.ts +124 -0
- package/src/core/tools/task/agents.ts +158 -0
- package/src/core/tools/task/artifacts.ts +114 -0
- package/src/core/tools/task/commands.ts +157 -0
- package/src/core/tools/task/discovery.ts +217 -0
- package/src/core/tools/task/executor.ts +531 -0
- package/src/core/tools/task/index.ts +548 -0
- package/src/core/tools/task/model-resolver.ts +176 -0
- package/src/core/tools/task/parallel.ts +38 -0
- package/src/core/tools/task/render.ts +502 -0
- package/src/core/tools/task/subprocess-tool-registry.ts +89 -0
- package/src/core/tools/task/types.ts +142 -0
- package/src/core/tools/truncate.ts +265 -0
- package/src/core/tools/web-fetch.ts +2511 -0
- package/src/core/tools/web-search/auth.ts +199 -0
- package/src/core/tools/web-search/index.ts +583 -0
- package/src/core/tools/web-search/providers/anthropic.ts +198 -0
- package/src/core/tools/web-search/providers/exa.ts +196 -0
- package/src/core/tools/web-search/providers/perplexity.ts +195 -0
- package/src/core/tools/web-search/render.ts +372 -0
- package/src/core/tools/web-search/types.ts +180 -0
- package/src/core/tools/write.ts +63 -0
- package/src/core/ttsr.ts +211 -0
- package/src/core/utils.ts +187 -0
- package/src/discovery/agents-md.ts +75 -0
- package/src/discovery/builtin.ts +647 -0
- package/src/discovery/claude.ts +623 -0
- package/src/discovery/cline.ts +104 -0
- package/src/discovery/codex.ts +571 -0
- package/src/discovery/cursor.ts +266 -0
- package/src/discovery/gemini.ts +368 -0
- package/src/discovery/github.ts +120 -0
- package/src/discovery/helpers.test.ts +127 -0
- package/src/discovery/helpers.ts +249 -0
- package/src/discovery/index.ts +84 -0
- package/src/discovery/mcp-json.ts +127 -0
- package/src/discovery/vscode.ts +99 -0
- package/src/discovery/windsurf.ts +219 -0
- package/src/index.ts +192 -0
- package/src/main.ts +507 -0
- package/src/migrations.ts +156 -0
- package/src/modes/cleanup.ts +23 -0
- package/src/modes/index.ts +48 -0
- package/src/modes/interactive/components/armin.ts +382 -0
- package/src/modes/interactive/components/assistant-message.ts +86 -0
- package/src/modes/interactive/components/bash-execution.ts +199 -0
- package/src/modes/interactive/components/bordered-loader.ts +41 -0
- package/src/modes/interactive/components/branch-summary-message.ts +42 -0
- package/src/modes/interactive/components/compaction-summary-message.ts +45 -0
- package/src/modes/interactive/components/custom-editor.ts +122 -0
- package/src/modes/interactive/components/diff.ts +147 -0
- package/src/modes/interactive/components/dynamic-border.ts +25 -0
- package/src/modes/interactive/components/extensions/extension-dashboard.ts +296 -0
- package/src/modes/interactive/components/extensions/extension-list.ts +479 -0
- package/src/modes/interactive/components/extensions/index.ts +9 -0
- package/src/modes/interactive/components/extensions/inspector-panel.ts +313 -0
- package/src/modes/interactive/components/extensions/state-manager.ts +558 -0
- package/src/modes/interactive/components/extensions/types.ts +191 -0
- package/src/modes/interactive/components/hook-editor.ts +117 -0
- package/src/modes/interactive/components/hook-input.ts +64 -0
- package/src/modes/interactive/components/hook-message.ts +96 -0
- package/src/modes/interactive/components/hook-selector.ts +91 -0
- package/src/modes/interactive/components/model-selector.ts +560 -0
- package/src/modes/interactive/components/oauth-selector.ts +136 -0
- package/src/modes/interactive/components/plugin-settings.ts +481 -0
- package/src/modes/interactive/components/queue-mode-selector.ts +56 -0
- package/src/modes/interactive/components/session-selector.ts +220 -0
- package/src/modes/interactive/components/settings-defs.ts +597 -0
- package/src/modes/interactive/components/settings-selector.ts +545 -0
- package/src/modes/interactive/components/show-images-selector.ts +45 -0
- package/src/modes/interactive/components/status-line/index.ts +4 -0
- package/src/modes/interactive/components/status-line/presets.ts +94 -0
- package/src/modes/interactive/components/status-line/segments.ts +350 -0
- package/src/modes/interactive/components/status-line/separators.ts +55 -0
- package/src/modes/interactive/components/status-line/types.ts +81 -0
- package/src/modes/interactive/components/status-line-segment-editor.ts +357 -0
- package/src/modes/interactive/components/status-line.ts +384 -0
- package/src/modes/interactive/components/theme-selector.ts +62 -0
- package/src/modes/interactive/components/thinking-selector.ts +64 -0
- package/src/modes/interactive/components/tool-execution.ts +946 -0
- package/src/modes/interactive/components/tree-selector.ts +877 -0
- package/src/modes/interactive/components/ttsr-notification.ts +82 -0
- package/src/modes/interactive/components/user-message-selector.ts +159 -0
- package/src/modes/interactive/components/user-message.ts +18 -0
- package/src/modes/interactive/components/visual-truncate.ts +50 -0
- package/src/modes/interactive/components/welcome.ts +228 -0
- package/src/modes/interactive/interactive-mode.ts +2669 -0
- package/src/modes/interactive/theme/dark.json +102 -0
- package/src/modes/interactive/theme/defaults/dark-arctic.json +111 -0
- package/src/modes/interactive/theme/defaults/dark-catppuccin.json +106 -0
- package/src/modes/interactive/theme/defaults/dark-cyberpunk.json +109 -0
- package/src/modes/interactive/theme/defaults/dark-dracula.json +105 -0
- package/src/modes/interactive/theme/defaults/dark-forest.json +103 -0
- package/src/modes/interactive/theme/defaults/dark-github.json +112 -0
- package/src/modes/interactive/theme/defaults/dark-gruvbox.json +119 -0
- package/src/modes/interactive/theme/defaults/dark-monochrome.json +101 -0
- package/src/modes/interactive/theme/defaults/dark-monokai.json +105 -0
- package/src/modes/interactive/theme/defaults/dark-nord.json +104 -0
- package/src/modes/interactive/theme/defaults/dark-ocean.json +108 -0
- package/src/modes/interactive/theme/defaults/dark-one.json +107 -0
- package/src/modes/interactive/theme/defaults/dark-retro.json +99 -0
- package/src/modes/interactive/theme/defaults/dark-rose-pine.json +95 -0
- package/src/modes/interactive/theme/defaults/dark-solarized.json +96 -0
- package/src/modes/interactive/theme/defaults/dark-sunset.json +106 -0
- package/src/modes/interactive/theme/defaults/dark-synthwave.json +102 -0
- package/src/modes/interactive/theme/defaults/dark-tokyo-night.json +108 -0
- package/src/modes/interactive/theme/defaults/index.ts +67 -0
- package/src/modes/interactive/theme/defaults/light-arctic.json +106 -0
- package/src/modes/interactive/theme/defaults/light-catppuccin.json +105 -0
- package/src/modes/interactive/theme/defaults/light-cyberpunk.json +103 -0
- package/src/modes/interactive/theme/defaults/light-forest.json +107 -0
- package/src/modes/interactive/theme/defaults/light-github.json +114 -0
- package/src/modes/interactive/theme/defaults/light-gruvbox.json +115 -0
- package/src/modes/interactive/theme/defaults/light-monochrome.json +100 -0
- package/src/modes/interactive/theme/defaults/light-ocean.json +106 -0
- package/src/modes/interactive/theme/defaults/light-one.json +105 -0
- package/src/modes/interactive/theme/defaults/light-retro.json +105 -0
- package/src/modes/interactive/theme/defaults/light-solarized.json +101 -0
- package/src/modes/interactive/theme/defaults/light-sunset.json +106 -0
- package/src/modes/interactive/theme/defaults/light-synthwave.json +105 -0
- package/src/modes/interactive/theme/defaults/light-tokyo-night.json +118 -0
- package/src/modes/interactive/theme/light.json +99 -0
- package/src/modes/interactive/theme/theme-schema.json +424 -0
- package/src/modes/interactive/theme/theme.ts +2211 -0
- package/src/modes/print-mode.ts +163 -0
- package/src/modes/rpc/rpc-client.ts +527 -0
- package/src/modes/rpc/rpc-mode.ts +494 -0
- package/src/modes/rpc/rpc-types.ts +203 -0
- package/src/prompts/architect-plan.md +10 -0
- package/src/prompts/branch-summary-preamble.md +3 -0
- package/src/prompts/branch-summary.md +28 -0
- package/src/prompts/browser.md +71 -0
- package/src/prompts/compaction-summary.md +34 -0
- package/src/prompts/compaction-turn-prefix.md +16 -0
- package/src/prompts/compaction-update-summary.md +41 -0
- package/src/prompts/explore.md +82 -0
- package/src/prompts/implement-with-critic.md +11 -0
- package/src/prompts/implement.md +11 -0
- package/src/prompts/init.md +30 -0
- package/src/prompts/plan.md +54 -0
- package/src/prompts/reviewer.md +81 -0
- package/src/prompts/summarization-system.md +3 -0
- package/src/prompts/system-prompt.md +27 -0
- package/src/prompts/task.md +56 -0
- package/src/prompts/title-system.md +8 -0
- package/src/prompts/tools/ask.md +24 -0
- package/src/prompts/tools/bash.md +23 -0
- package/src/prompts/tools/edit.md +9 -0
- package/src/prompts/tools/find.md +6 -0
- package/src/prompts/tools/grep.md +12 -0
- package/src/prompts/tools/lsp.md +14 -0
- package/src/prompts/tools/output.md +23 -0
- package/src/prompts/tools/read.md +25 -0
- package/src/prompts/tools/web-fetch.md +8 -0
- package/src/prompts/tools/web-search.md +10 -0
- package/src/prompts/tools/write.md +10 -0
- package/src/utils/changelog.ts +99 -0
- package/src/utils/clipboard.ts +265 -0
- package/src/utils/fuzzy.ts +108 -0
- package/src/utils/mime.ts +30 -0
- package/src/utils/shell-snapshot.ts +218 -0
- package/src/utils/shell.ts +364 -0
- package/src/utils/tools-manager.ts +265 -0
|
@@ -0,0 +1,248 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Custom tool loader - loads TypeScript tool modules using native Bun import.
|
|
3
|
+
*
|
|
4
|
+
* Dependencies (@sinclair/typebox and pi-coding-agent) are injected via the CustomToolAPI
|
|
5
|
+
* to avoid import resolution issues with custom tools loaded from user directories.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import * as os from "node:os";
|
|
9
|
+
import * as path from "node:path";
|
|
10
|
+
import * as typebox from "@sinclair/typebox";
|
|
11
|
+
import { toolCapability } from "../../capability/tool";
|
|
12
|
+
import { type CustomTool, loadSync } from "../../discovery";
|
|
13
|
+
import * as piCodingAgent from "../../index";
|
|
14
|
+
import { theme } from "../../modes/interactive/theme/theme";
|
|
15
|
+
import type { ExecOptions } from "../exec";
|
|
16
|
+
import { execCommand } from "../exec";
|
|
17
|
+
import type { HookUIContext } from "../hooks/types";
|
|
18
|
+
import { logger } from "../logger";
|
|
19
|
+
import { getAllPluginToolPaths } from "../plugins/loader";
|
|
20
|
+
import type { CustomToolAPI, CustomToolFactory, CustomToolsLoadResult, LoadedCustomTool } from "./types";
|
|
21
|
+
|
|
22
|
+
const UNICODE_SPACES = /[\u00A0\u2000-\u200A\u202F\u205F\u3000]/g;
|
|
23
|
+
|
|
24
|
+
function normalizeUnicodeSpaces(str: string): string {
|
|
25
|
+
return str.replace(UNICODE_SPACES, " ");
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function expandPath(p: string): string {
|
|
29
|
+
const normalized = normalizeUnicodeSpaces(p);
|
|
30
|
+
if (normalized.startsWith("~/")) {
|
|
31
|
+
return path.join(os.homedir(), normalized.slice(2));
|
|
32
|
+
}
|
|
33
|
+
if (normalized.startsWith("~")) {
|
|
34
|
+
return path.join(os.homedir(), normalized.slice(1));
|
|
35
|
+
}
|
|
36
|
+
return normalized;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Resolve tool path.
|
|
41
|
+
* - Absolute paths used as-is
|
|
42
|
+
* - Paths starting with ~ expanded to home directory
|
|
43
|
+
* - Relative paths resolved from cwd
|
|
44
|
+
*/
|
|
45
|
+
function resolveToolPath(toolPath: string, cwd: string): string {
|
|
46
|
+
const expanded = expandPath(toolPath);
|
|
47
|
+
|
|
48
|
+
if (path.isAbsolute(expanded)) {
|
|
49
|
+
return expanded;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// Relative paths resolved from cwd
|
|
53
|
+
return path.resolve(cwd, expanded);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Create a no-op UI context for headless modes.
|
|
58
|
+
*/
|
|
59
|
+
function createNoOpUIContext(): HookUIContext {
|
|
60
|
+
return {
|
|
61
|
+
select: async () => undefined,
|
|
62
|
+
confirm: async () => false,
|
|
63
|
+
input: async () => undefined,
|
|
64
|
+
notify: () => {},
|
|
65
|
+
setStatus: () => {},
|
|
66
|
+
custom: async () => undefined as never,
|
|
67
|
+
setEditorText: () => {},
|
|
68
|
+
getEditorText: () => "",
|
|
69
|
+
editor: async () => undefined,
|
|
70
|
+
get theme() {
|
|
71
|
+
return theme;
|
|
72
|
+
},
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/** Error with source metadata */
|
|
77
|
+
interface ToolLoadError {
|
|
78
|
+
path: string;
|
|
79
|
+
error: string;
|
|
80
|
+
source?: { provider: string; providerName: string; level: "user" | "project" };
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Load a single tool module using native Bun import.
|
|
85
|
+
*/
|
|
86
|
+
async function loadTool(
|
|
87
|
+
toolPath: string,
|
|
88
|
+
cwd: string,
|
|
89
|
+
sharedApi: CustomToolAPI,
|
|
90
|
+
source?: { provider: string; providerName: string; level: "user" | "project" },
|
|
91
|
+
): Promise<{ tools: LoadedCustomTool[] | null; error: ToolLoadError | null }> {
|
|
92
|
+
const resolvedPath = resolveToolPath(toolPath, cwd);
|
|
93
|
+
|
|
94
|
+
// Skip declarative tool files (.md, .json) - these are metadata only, not executable modules
|
|
95
|
+
if (resolvedPath.endsWith(".md") || resolvedPath.endsWith(".json")) {
|
|
96
|
+
return {
|
|
97
|
+
tools: null,
|
|
98
|
+
error: {
|
|
99
|
+
path: toolPath,
|
|
100
|
+
error: "Declarative tool files (.md, .json) cannot be loaded as executable modules",
|
|
101
|
+
source,
|
|
102
|
+
},
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
try {
|
|
107
|
+
const module = await import(resolvedPath);
|
|
108
|
+
const factory = (module.default ?? module) as CustomToolFactory;
|
|
109
|
+
|
|
110
|
+
if (typeof factory !== "function") {
|
|
111
|
+
return { tools: null, error: { path: toolPath, error: "Tool must export a default function", source } };
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
const toolResult = await factory(sharedApi);
|
|
115
|
+
const toolsArray = Array.isArray(toolResult) ? toolResult : [toolResult];
|
|
116
|
+
|
|
117
|
+
const loadedTools: LoadedCustomTool[] = toolsArray.map((tool) => ({
|
|
118
|
+
path: toolPath,
|
|
119
|
+
resolvedPath,
|
|
120
|
+
tool,
|
|
121
|
+
source,
|
|
122
|
+
}));
|
|
123
|
+
|
|
124
|
+
return { tools: loadedTools, error: null };
|
|
125
|
+
} catch (err) {
|
|
126
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
127
|
+
return { tools: null, error: { path: toolPath, error: `Failed to load tool: ${message}`, source } };
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/** Tool path with optional source metadata */
|
|
132
|
+
interface ToolPathWithSource {
|
|
133
|
+
path: string;
|
|
134
|
+
source?: { provider: string; providerName: string; level: "user" | "project" };
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Load all tools from configuration.
|
|
139
|
+
* @param pathsWithSources - Array of tool paths with optional source metadata
|
|
140
|
+
* @param cwd - Current working directory for resolving relative paths
|
|
141
|
+
* @param builtInToolNames - Names of built-in tools to check for conflicts
|
|
142
|
+
*/
|
|
143
|
+
export async function loadCustomTools(
|
|
144
|
+
pathsWithSources: ToolPathWithSource[],
|
|
145
|
+
cwd: string,
|
|
146
|
+
builtInToolNames: string[],
|
|
147
|
+
): Promise<CustomToolsLoadResult> {
|
|
148
|
+
const tools: LoadedCustomTool[] = [];
|
|
149
|
+
const errors: ToolLoadError[] = [];
|
|
150
|
+
const seenNames = new Set<string>(builtInToolNames);
|
|
151
|
+
|
|
152
|
+
// Shared API object - all tools get the same instance
|
|
153
|
+
const sharedApi: CustomToolAPI = {
|
|
154
|
+
cwd,
|
|
155
|
+
exec: (command: string, args: string[], options?: ExecOptions) =>
|
|
156
|
+
execCommand(command, args, options?.cwd ?? cwd, options),
|
|
157
|
+
ui: createNoOpUIContext(),
|
|
158
|
+
hasUI: false,
|
|
159
|
+
logger,
|
|
160
|
+
typebox,
|
|
161
|
+
pi: piCodingAgent,
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
for (const { path: toolPath, source } of pathsWithSources) {
|
|
165
|
+
const { tools: loadedTools, error } = await loadTool(toolPath, cwd, sharedApi, source);
|
|
166
|
+
|
|
167
|
+
if (error) {
|
|
168
|
+
errors.push(error);
|
|
169
|
+
continue;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
if (loadedTools) {
|
|
173
|
+
for (const loadedTool of loadedTools) {
|
|
174
|
+
// Check for name conflicts
|
|
175
|
+
if (seenNames.has(loadedTool.tool.name)) {
|
|
176
|
+
errors.push({
|
|
177
|
+
path: toolPath,
|
|
178
|
+
error: `Tool name "${loadedTool.tool.name}" conflicts with existing tool`,
|
|
179
|
+
source,
|
|
180
|
+
});
|
|
181
|
+
continue;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
seenNames.add(loadedTool.tool.name);
|
|
185
|
+
tools.push(loadedTool);
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
return {
|
|
191
|
+
tools,
|
|
192
|
+
errors,
|
|
193
|
+
setUIContext(uiContext, hasUI) {
|
|
194
|
+
sharedApi.ui = uiContext;
|
|
195
|
+
sharedApi.hasUI = hasUI;
|
|
196
|
+
},
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
/**
|
|
201
|
+
* Discover and load tools from standard locations via capability system:
|
|
202
|
+
* 1. User and project tools discovered by capability providers
|
|
203
|
+
* 2. Installed plugins (~/.omp/plugins/node_modules/*)
|
|
204
|
+
* 3. Explicitly configured paths from settings or CLI
|
|
205
|
+
*
|
|
206
|
+
* @param configuredPaths - Explicit paths from settings.json and CLI --tool flags
|
|
207
|
+
* @param cwd - Current working directory
|
|
208
|
+
* @param builtInToolNames - Names of built-in tools to check for conflicts
|
|
209
|
+
*/
|
|
210
|
+
export async function discoverAndLoadCustomTools(
|
|
211
|
+
configuredPaths: string[],
|
|
212
|
+
cwd: string,
|
|
213
|
+
builtInToolNames: string[],
|
|
214
|
+
): Promise<CustomToolsLoadResult> {
|
|
215
|
+
const allPathsWithSources: ToolPathWithSource[] = [];
|
|
216
|
+
const seen = new Set<string>();
|
|
217
|
+
|
|
218
|
+
// Helper to add paths without duplicates
|
|
219
|
+
const addPath = (p: string, source?: { provider: string; providerName: string; level: "user" | "project" }) => {
|
|
220
|
+
const resolved = path.resolve(p);
|
|
221
|
+
if (!seen.has(resolved)) {
|
|
222
|
+
seen.add(resolved);
|
|
223
|
+
allPathsWithSources.push({ path: p, source });
|
|
224
|
+
}
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
// 1. Discover tools via capability system (user + project from all providers)
|
|
228
|
+
const discoveredTools = loadSync<CustomTool>(toolCapability.id, { cwd });
|
|
229
|
+
for (const tool of discoveredTools.items) {
|
|
230
|
+
addPath(tool.path, {
|
|
231
|
+
provider: tool._source.provider,
|
|
232
|
+
providerName: tool._source.providerName,
|
|
233
|
+
level: tool.level,
|
|
234
|
+
});
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
// 2. Plugin tools: ~/.omp/plugins/node_modules/*/
|
|
238
|
+
for (const pluginPath of getAllPluginToolPaths(cwd)) {
|
|
239
|
+
addPath(pluginPath, { provider: "plugin", providerName: "Plugin", level: "user" });
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
// 3. Explicitly configured paths (can override/add)
|
|
243
|
+
for (const configPath of configuredPaths) {
|
|
244
|
+
addPath(resolveToolPath(configPath, cwd), { provider: "config", providerName: "Config", level: "project" });
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
return loadCustomTools(allPathsWithSources, cwd, builtInToolNames);
|
|
248
|
+
}
|
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Custom tool types.
|
|
3
|
+
*
|
|
4
|
+
* Custom tools are TypeScript modules that define additional tools for the agent.
|
|
5
|
+
* They can provide custom rendering for tool calls and results in the TUI.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { AgentToolResult, AgentToolUpdateCallback } from "@oh-my-pi/pi-agent-core";
|
|
9
|
+
import type { Model } from "@oh-my-pi/pi-ai";
|
|
10
|
+
import type { Component } from "@oh-my-pi/pi-tui";
|
|
11
|
+
import type { Static, TSchema } from "@sinclair/typebox";
|
|
12
|
+
import type { Theme } from "../../modes/interactive/theme/theme";
|
|
13
|
+
import type { ExecOptions, ExecResult } from "../exec";
|
|
14
|
+
import type { HookUIContext } from "../hooks/types";
|
|
15
|
+
import type { Logger } from "../logger";
|
|
16
|
+
import type { ModelRegistry } from "../model-registry";
|
|
17
|
+
import type { ReadonlySessionManager } from "../session-manager";
|
|
18
|
+
|
|
19
|
+
/** Alias for clarity */
|
|
20
|
+
export type CustomToolUIContext = HookUIContext;
|
|
21
|
+
|
|
22
|
+
/** Re-export for custom tools to use in execute signature */
|
|
23
|
+
export type { AgentToolResult, AgentToolUpdateCallback };
|
|
24
|
+
|
|
25
|
+
// Re-export for backward compatibility
|
|
26
|
+
export type { ExecOptions, ExecResult } from "../exec";
|
|
27
|
+
|
|
28
|
+
/** API passed to custom tool factory (stable across session changes) */
|
|
29
|
+
export interface CustomToolAPI {
|
|
30
|
+
/** Current working directory */
|
|
31
|
+
cwd: string;
|
|
32
|
+
/** Execute a command */
|
|
33
|
+
exec(command: string, args: string[], options?: ExecOptions): Promise<ExecResult>;
|
|
34
|
+
/** UI methods for user interaction (select, confirm, input, notify, custom) */
|
|
35
|
+
ui: CustomToolUIContext;
|
|
36
|
+
/** Whether UI is available (false in print/RPC mode) */
|
|
37
|
+
hasUI: boolean;
|
|
38
|
+
/** File logger for error/warning/debug messages */
|
|
39
|
+
logger: Logger;
|
|
40
|
+
/** Injected @sinclair/typebox module */
|
|
41
|
+
typebox: typeof import("@sinclair/typebox");
|
|
42
|
+
/** Injected pi-coding-agent exports */
|
|
43
|
+
pi: typeof import("../../index.js");
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Context passed to tool execute and onSession callbacks.
|
|
48
|
+
* Provides access to session state and model information.
|
|
49
|
+
*/
|
|
50
|
+
export interface CustomToolContext {
|
|
51
|
+
/** Session manager (read-only) */
|
|
52
|
+
sessionManager: ReadonlySessionManager;
|
|
53
|
+
/** Model registry - use for API key resolution and model retrieval */
|
|
54
|
+
modelRegistry: ModelRegistry;
|
|
55
|
+
/** Current model (may be undefined if no model is selected yet) */
|
|
56
|
+
model: Model<any> | undefined;
|
|
57
|
+
/** Whether the agent is idle (not streaming) */
|
|
58
|
+
isIdle(): boolean;
|
|
59
|
+
/** Whether there are queued messages waiting to be processed */
|
|
60
|
+
hasQueuedMessages(): boolean;
|
|
61
|
+
/** Abort the current agent operation (fire-and-forget, does not wait) */
|
|
62
|
+
abort(): void;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/** Session event passed to onSession callback */
|
|
66
|
+
export interface CustomToolSessionEvent {
|
|
67
|
+
/** Reason for the session event */
|
|
68
|
+
reason: "start" | "switch" | "branch" | "tree" | "shutdown";
|
|
69
|
+
/** Previous session file path, or undefined for "start" and "shutdown" */
|
|
70
|
+
previousSessionFile: string | undefined;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/** Rendering options passed to renderResult */
|
|
74
|
+
export interface RenderResultOptions {
|
|
75
|
+
/** Whether the result view is expanded */
|
|
76
|
+
expanded: boolean;
|
|
77
|
+
/** Whether this is a partial/streaming result */
|
|
78
|
+
isPartial: boolean;
|
|
79
|
+
/** Current spinner frame index for animated elements (0-9, only provided during partial results) */
|
|
80
|
+
spinnerFrame?: number;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export type CustomToolResult<TDetails = any> = AgentToolResult<TDetails>;
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* Custom tool definition.
|
|
87
|
+
*
|
|
88
|
+
* Custom tools are standalone - they don't extend AgentTool directly.
|
|
89
|
+
* When loaded, they are wrapped in an AgentTool for the agent to use.
|
|
90
|
+
*
|
|
91
|
+
* The execute callback receives a ToolContext with access to session state,
|
|
92
|
+
* model registry, and current model.
|
|
93
|
+
*
|
|
94
|
+
* @example
|
|
95
|
+
* ```typescript
|
|
96
|
+
* const factory: CustomToolFactory = (pi) => ({
|
|
97
|
+
* name: "my_tool",
|
|
98
|
+
* label: "My Tool",
|
|
99
|
+
* description: "Does something useful",
|
|
100
|
+
* parameters: Type.Object({ input: Type.String() }),
|
|
101
|
+
*
|
|
102
|
+
* async execute(toolCallId, params, onUpdate, ctx, signal) {
|
|
103
|
+
* // Access session state via ctx.sessionManager
|
|
104
|
+
* // Access model registry via ctx.modelRegistry
|
|
105
|
+
* // Current model via ctx.model
|
|
106
|
+
* return { content: [{ type: "text", text: "Done" }] };
|
|
107
|
+
* },
|
|
108
|
+
*
|
|
109
|
+
* onSession(event, ctx) {
|
|
110
|
+
* if (event.reason === "shutdown") {
|
|
111
|
+
* // Cleanup
|
|
112
|
+
* }
|
|
113
|
+
* // Reconstruct state from ctx.sessionManager.getEntries()
|
|
114
|
+
* }
|
|
115
|
+
* });
|
|
116
|
+
* ```
|
|
117
|
+
*/
|
|
118
|
+
export interface CustomTool<TParams extends TSchema = TSchema, TDetails = any> {
|
|
119
|
+
/** Tool name (used in LLM tool calls) */
|
|
120
|
+
name: string;
|
|
121
|
+
/** Human-readable label for UI */
|
|
122
|
+
label: string;
|
|
123
|
+
/** Description for LLM */
|
|
124
|
+
description: string;
|
|
125
|
+
/** Parameter schema (TypeBox) */
|
|
126
|
+
parameters: TParams;
|
|
127
|
+
/** If true, tool is excluded unless explicitly listed in --tools or agent's tools field */
|
|
128
|
+
hidden?: boolean;
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Execute the tool.
|
|
132
|
+
* @param toolCallId - Unique ID for this tool call
|
|
133
|
+
* @param params - Parsed parameters matching the schema
|
|
134
|
+
* @param onUpdate - Callback for streaming partial results (for UI, not LLM)
|
|
135
|
+
* @param ctx - Context with session manager, model registry, and current model
|
|
136
|
+
* @param signal - Optional abort signal for cancellation
|
|
137
|
+
*/
|
|
138
|
+
execute(
|
|
139
|
+
toolCallId: string,
|
|
140
|
+
params: Static<TParams>,
|
|
141
|
+
onUpdate: AgentToolUpdateCallback<TDetails> | undefined,
|
|
142
|
+
ctx: CustomToolContext,
|
|
143
|
+
signal?: AbortSignal,
|
|
144
|
+
): Promise<AgentToolResult<TDetails>>;
|
|
145
|
+
|
|
146
|
+
/** Called on session lifecycle events - use to reconstruct state or cleanup resources */
|
|
147
|
+
onSession?: (event: CustomToolSessionEvent, ctx: CustomToolContext) => void | Promise<void>;
|
|
148
|
+
/** Custom rendering for tool call display - return a Component */
|
|
149
|
+
renderCall?: (args: Static<TParams>, theme: Theme) => Component;
|
|
150
|
+
|
|
151
|
+
/** Custom rendering for tool result display - return a Component */
|
|
152
|
+
renderResult?: (result: CustomToolResult<TDetails>, options: RenderResultOptions, theme: Theme) => Component;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/** Factory function that creates a custom tool or array of tools */
|
|
156
|
+
export type CustomToolFactory = (
|
|
157
|
+
pi: CustomToolAPI,
|
|
158
|
+
) => CustomTool<any, any> | CustomTool<any, any>[] | Promise<CustomTool<any, any> | CustomTool<any, any>[]>;
|
|
159
|
+
|
|
160
|
+
/** Loaded custom tool with metadata and wrapped AgentTool */
|
|
161
|
+
export interface LoadedCustomTool {
|
|
162
|
+
/** Original path (as specified) */
|
|
163
|
+
path: string;
|
|
164
|
+
/** Resolved absolute path */
|
|
165
|
+
resolvedPath: string;
|
|
166
|
+
/** The original custom tool instance */
|
|
167
|
+
tool: CustomTool;
|
|
168
|
+
/** Source metadata (provider and level) */
|
|
169
|
+
source?: { provider: string; providerName: string; level: "user" | "project" };
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
/** Error with source metadata */
|
|
173
|
+
export interface ToolLoadError {
|
|
174
|
+
path: string;
|
|
175
|
+
error: string;
|
|
176
|
+
source?: { provider: string; providerName: string; level: "user" | "project" };
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/** Result from loading custom tools */
|
|
180
|
+
export interface CustomToolsLoadResult {
|
|
181
|
+
tools: LoadedCustomTool[];
|
|
182
|
+
errors: ToolLoadError[];
|
|
183
|
+
/** Update the UI context for all loaded tools. Call when mode initializes. */
|
|
184
|
+
setUIContext(uiContext: CustomToolUIContext, hasUI: boolean): void;
|
|
185
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wraps CustomTool instances into AgentTool for use with the agent.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import type { AgentTool } from "@oh-my-pi/pi-agent-core";
|
|
6
|
+
import type { CustomTool, CustomToolContext, LoadedCustomTool } from "./types";
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Wrap a CustomTool into an AgentTool.
|
|
10
|
+
* The wrapper injects the ToolContext into execute calls.
|
|
11
|
+
*/
|
|
12
|
+
export function wrapCustomTool(tool: CustomTool, getContext: () => CustomToolContext): AgentTool {
|
|
13
|
+
return {
|
|
14
|
+
name: tool.name,
|
|
15
|
+
label: tool.label,
|
|
16
|
+
description: tool.description,
|
|
17
|
+
parameters: tool.parameters,
|
|
18
|
+
hidden: tool.hidden,
|
|
19
|
+
execute: (toolCallId, params, signal, onUpdate, context) =>
|
|
20
|
+
tool.execute(toolCallId, params, onUpdate, context ?? getContext(), signal),
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Wrap all loaded custom tools into AgentTools.
|
|
26
|
+
*/
|
|
27
|
+
export function wrapCustomTools(loadedTools: LoadedCustomTool[], getContext: () => CustomToolContext): AgentTool[] {
|
|
28
|
+
return loadedTools.map((lt) => wrapCustomTool(lt.tool, getContext));
|
|
29
|
+
}
|
package/src/core/exec.ts
ADDED
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared command execution utilities for hooks and custom tools.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import type { Subprocess } from "bun";
|
|
6
|
+
import { logger } from "./logger";
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Options for executing shell commands.
|
|
10
|
+
*/
|
|
11
|
+
export interface ExecOptions {
|
|
12
|
+
/** AbortSignal to cancel the command */
|
|
13
|
+
signal?: AbortSignal;
|
|
14
|
+
/** Timeout in milliseconds */
|
|
15
|
+
timeout?: number;
|
|
16
|
+
/** Working directory */
|
|
17
|
+
cwd?: string;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Result of executing a shell command.
|
|
22
|
+
*/
|
|
23
|
+
export interface ExecResult {
|
|
24
|
+
stdout: string;
|
|
25
|
+
stderr: string;
|
|
26
|
+
code: number;
|
|
27
|
+
killed: boolean;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Execute a shell command and return stdout/stderr/code.
|
|
32
|
+
* Supports timeout and abort signal.
|
|
33
|
+
*/
|
|
34
|
+
export async function execCommand(
|
|
35
|
+
command: string,
|
|
36
|
+
args: string[],
|
|
37
|
+
cwd: string,
|
|
38
|
+
options?: ExecOptions,
|
|
39
|
+
): Promise<ExecResult> {
|
|
40
|
+
return new Promise((resolve) => {
|
|
41
|
+
const proc: Subprocess = Bun.spawn([command, ...args], {
|
|
42
|
+
cwd,
|
|
43
|
+
stdin: "ignore",
|
|
44
|
+
stdout: "pipe",
|
|
45
|
+
stderr: "pipe",
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
let stdout = "";
|
|
49
|
+
let stderr = "";
|
|
50
|
+
let killed = false;
|
|
51
|
+
let timeoutId: Timer | undefined;
|
|
52
|
+
|
|
53
|
+
const killProcess = () => {
|
|
54
|
+
if (!killed) {
|
|
55
|
+
killed = true;
|
|
56
|
+
proc.kill();
|
|
57
|
+
// Force kill after 5 seconds if first kill doesn't work
|
|
58
|
+
setTimeout(() => {
|
|
59
|
+
try {
|
|
60
|
+
proc.kill(9);
|
|
61
|
+
} catch {
|
|
62
|
+
// Ignore if already dead
|
|
63
|
+
}
|
|
64
|
+
}, 5000);
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
// Handle abort signal
|
|
69
|
+
if (options?.signal) {
|
|
70
|
+
if (options.signal.aborted) {
|
|
71
|
+
killProcess();
|
|
72
|
+
} else {
|
|
73
|
+
options.signal.addEventListener("abort", killProcess, { once: true });
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Handle timeout
|
|
78
|
+
if (options?.timeout && options.timeout > 0) {
|
|
79
|
+
timeoutId = setTimeout(() => {
|
|
80
|
+
killProcess();
|
|
81
|
+
}, options.timeout);
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
// Read streams asynchronously
|
|
85
|
+
(async () => {
|
|
86
|
+
try {
|
|
87
|
+
const stdoutReader = (proc.stdout as ReadableStream<Uint8Array>).getReader();
|
|
88
|
+
const stderrReader = (proc.stderr as ReadableStream<Uint8Array>).getReader();
|
|
89
|
+
|
|
90
|
+
// Read both streams and wait for process exit
|
|
91
|
+
const [stdoutResult, stderrResult, exitCode] = await Promise.all([
|
|
92
|
+
(async () => {
|
|
93
|
+
const chunks: Uint8Array[] = [];
|
|
94
|
+
try {
|
|
95
|
+
while (true) {
|
|
96
|
+
const { done, value } = await stdoutReader.read();
|
|
97
|
+
if (done) break;
|
|
98
|
+
chunks.push(value);
|
|
99
|
+
}
|
|
100
|
+
} finally {
|
|
101
|
+
stdoutReader.releaseLock();
|
|
102
|
+
}
|
|
103
|
+
return Buffer.concat(chunks).toString();
|
|
104
|
+
})(),
|
|
105
|
+
(async () => {
|
|
106
|
+
const chunks: Uint8Array[] = [];
|
|
107
|
+
try {
|
|
108
|
+
while (true) {
|
|
109
|
+
const { done, value } = await stderrReader.read();
|
|
110
|
+
if (done) break;
|
|
111
|
+
chunks.push(value);
|
|
112
|
+
}
|
|
113
|
+
} finally {
|
|
114
|
+
stderrReader.releaseLock();
|
|
115
|
+
}
|
|
116
|
+
return Buffer.concat(chunks).toString();
|
|
117
|
+
})(),
|
|
118
|
+
proc.exited,
|
|
119
|
+
]);
|
|
120
|
+
|
|
121
|
+
stdout = stdoutResult;
|
|
122
|
+
stderr = stderrResult;
|
|
123
|
+
|
|
124
|
+
if (timeoutId) clearTimeout(timeoutId);
|
|
125
|
+
if (options?.signal) {
|
|
126
|
+
options.signal.removeEventListener("abort", killProcess);
|
|
127
|
+
}
|
|
128
|
+
resolve({ stdout, stderr, code: exitCode ?? 0, killed });
|
|
129
|
+
} catch (err) {
|
|
130
|
+
logger.debug("Process stream error", { error: String(err) });
|
|
131
|
+
if (timeoutId) clearTimeout(timeoutId);
|
|
132
|
+
if (options?.signal) {
|
|
133
|
+
options.signal.removeEventListener("abort", killProcess);
|
|
134
|
+
}
|
|
135
|
+
resolve({ stdout, stderr, code: 1, killed });
|
|
136
|
+
}
|
|
137
|
+
})();
|
|
138
|
+
});
|
|
139
|
+
}
|