@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,142 @@
|
|
|
1
|
+
import type { Usage } from "@oh-my-pi/pi-ai";
|
|
2
|
+
import { type Static, Type } from "@sinclair/typebox";
|
|
3
|
+
|
|
4
|
+
/** Source of an agent definition */
|
|
5
|
+
export type AgentSource = "bundled" | "user" | "project";
|
|
6
|
+
|
|
7
|
+
/** Maximum tasks per call */
|
|
8
|
+
export const MAX_PARALLEL_TASKS = 32;
|
|
9
|
+
|
|
10
|
+
/** Maximum concurrent workers */
|
|
11
|
+
export const MAX_CONCURRENCY = 16;
|
|
12
|
+
|
|
13
|
+
/** Maximum output bytes per agent */
|
|
14
|
+
export const MAX_OUTPUT_BYTES = 500_000;
|
|
15
|
+
|
|
16
|
+
/** Maximum output lines per agent */
|
|
17
|
+
export const MAX_OUTPUT_LINES = 5000;
|
|
18
|
+
|
|
19
|
+
/** Maximum agents to show in description */
|
|
20
|
+
export const MAX_AGENTS_IN_DESCRIPTION = 10;
|
|
21
|
+
|
|
22
|
+
/** Environment variable to inhibit subagent spawning (legacy, still checked for backwards compat) */
|
|
23
|
+
export const OMP_NO_SUBAGENTS_ENV = "OMP_NO_SUBAGENTS";
|
|
24
|
+
|
|
25
|
+
/** Environment variable containing blocked agent name (self-recursion prevention) */
|
|
26
|
+
export const OMP_BLOCKED_AGENT_ENV = "OMP_BLOCKED_AGENT";
|
|
27
|
+
|
|
28
|
+
/** Environment variable containing allowed spawn list (propagated to subprocesses) */
|
|
29
|
+
export const OMP_SPAWNS_ENV = "OMP_SPAWNS";
|
|
30
|
+
|
|
31
|
+
/** Single task item for parallel execution */
|
|
32
|
+
export const taskItemSchema = Type.Object({
|
|
33
|
+
agent: Type.String({ description: "Agent name" }),
|
|
34
|
+
task: Type.String({ description: "Task description for the agent" }),
|
|
35
|
+
model: Type.Optional(Type.String({ description: "Model override for this task" })),
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
export type TaskItem = Static<typeof taskItemSchema>;
|
|
39
|
+
|
|
40
|
+
/** Task tool parameters */
|
|
41
|
+
export const taskSchema = Type.Object({
|
|
42
|
+
context: Type.Optional(Type.String({ description: "Shared context prepended to all task prompts" })),
|
|
43
|
+
tasks: Type.Array(taskItemSchema, {
|
|
44
|
+
description: "Tasks to run in parallel",
|
|
45
|
+
maxItems: MAX_PARALLEL_TASKS,
|
|
46
|
+
}),
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
export type TaskParams = Static<typeof taskSchema>;
|
|
50
|
+
|
|
51
|
+
/** A code review finding reported by the reviewer agent */
|
|
52
|
+
export interface ReviewFinding {
|
|
53
|
+
title: string;
|
|
54
|
+
body: string;
|
|
55
|
+
priority: number;
|
|
56
|
+
confidence: number;
|
|
57
|
+
file_path: string;
|
|
58
|
+
line_start: number;
|
|
59
|
+
line_end: number;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/** Review summary submitted by the reviewer agent */
|
|
63
|
+
export interface ReviewSummary {
|
|
64
|
+
overall_correctness: "correct" | "incorrect";
|
|
65
|
+
explanation: string;
|
|
66
|
+
confidence: number;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/** Structured review data extracted from reviewer agent */
|
|
70
|
+
export interface ReviewData {
|
|
71
|
+
findings: ReviewFinding[];
|
|
72
|
+
summary?: ReviewSummary;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
/** Agent definition (bundled or discovered) */
|
|
76
|
+
export interface AgentDefinition {
|
|
77
|
+
name: string;
|
|
78
|
+
description: string;
|
|
79
|
+
systemPrompt: string;
|
|
80
|
+
tools?: string[];
|
|
81
|
+
spawns?: string[] | "*";
|
|
82
|
+
model?: string;
|
|
83
|
+
recursive?: boolean;
|
|
84
|
+
source: AgentSource;
|
|
85
|
+
filePath?: string;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/** Progress tracking for a single agent */
|
|
89
|
+
export interface AgentProgress {
|
|
90
|
+
index: number;
|
|
91
|
+
agent: string;
|
|
92
|
+
agentSource: AgentSource;
|
|
93
|
+
status: "pending" | "running" | "completed" | "failed" | "aborted";
|
|
94
|
+
task: string;
|
|
95
|
+
currentTool?: string;
|
|
96
|
+
currentToolArgs?: string;
|
|
97
|
+
currentToolStartMs?: number;
|
|
98
|
+
recentTools: Array<{ tool: string; args: string; endMs: number }>;
|
|
99
|
+
recentOutput: string[];
|
|
100
|
+
toolCount: number;
|
|
101
|
+
tokens: number;
|
|
102
|
+
durationMs: number;
|
|
103
|
+
modelOverride?: string;
|
|
104
|
+
/** Data extracted by registered subprocess tool handlers (keyed by tool name) */
|
|
105
|
+
extractedToolData?: Record<string, unknown[]>;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/** Result from a single agent execution */
|
|
109
|
+
export interface SingleResult {
|
|
110
|
+
index: number;
|
|
111
|
+
agent: string;
|
|
112
|
+
agentSource: AgentSource;
|
|
113
|
+
task: string;
|
|
114
|
+
exitCode: number;
|
|
115
|
+
output: string;
|
|
116
|
+
stderr: string;
|
|
117
|
+
truncated: boolean;
|
|
118
|
+
durationMs: number;
|
|
119
|
+
tokens: number;
|
|
120
|
+
modelOverride?: string;
|
|
121
|
+
error?: string;
|
|
122
|
+
aborted?: boolean;
|
|
123
|
+
jsonlEvents?: string[];
|
|
124
|
+
artifactPaths?: { inputPath: string; outputPath: string; jsonlPath?: string };
|
|
125
|
+
/** Aggregated usage from the subprocess, if available. */
|
|
126
|
+
usage?: Usage;
|
|
127
|
+
/** Data extracted by registered subprocess tool handlers (keyed by tool name) */
|
|
128
|
+
extractedToolData?: Record<string, unknown[]>;
|
|
129
|
+
/** Output metadata for Output tool integration */
|
|
130
|
+
outputMeta?: { lineCount: number; charCount: number };
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/** Tool details for TUI rendering */
|
|
134
|
+
export interface TaskToolDetails {
|
|
135
|
+
projectAgentsDir: string | null;
|
|
136
|
+
results: SingleResult[];
|
|
137
|
+
totalDurationMs: number;
|
|
138
|
+
/** Aggregated usage across all subagents. */
|
|
139
|
+
usage?: Usage;
|
|
140
|
+
outputPaths?: string[];
|
|
141
|
+
progress?: AgentProgress[];
|
|
142
|
+
}
|
|
@@ -0,0 +1,265 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared truncation utilities for tool outputs.
|
|
3
|
+
*
|
|
4
|
+
* Truncation is based on two independent limits - whichever is hit first wins:
|
|
5
|
+
* - Line limit (default: 2000 lines)
|
|
6
|
+
* - Byte limit (default: 50KB)
|
|
7
|
+
*
|
|
8
|
+
* Never returns partial lines (except bash tail truncation edge case).
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
export const DEFAULT_MAX_LINES = 2000;
|
|
12
|
+
export const DEFAULT_MAX_BYTES = 50 * 1024; // 50KB
|
|
13
|
+
export const GREP_MAX_LINE_LENGTH = 500; // Max chars per grep match line
|
|
14
|
+
|
|
15
|
+
export interface TruncationResult {
|
|
16
|
+
/** The truncated content */
|
|
17
|
+
content: string;
|
|
18
|
+
/** Whether truncation occurred */
|
|
19
|
+
truncated: boolean;
|
|
20
|
+
/** Which limit was hit: "lines", "bytes", or null if not truncated */
|
|
21
|
+
truncatedBy: "lines" | "bytes" | null;
|
|
22
|
+
/** Total number of lines in the original content */
|
|
23
|
+
totalLines: number;
|
|
24
|
+
/** Total number of bytes in the original content */
|
|
25
|
+
totalBytes: number;
|
|
26
|
+
/** Number of complete lines in the truncated output */
|
|
27
|
+
outputLines: number;
|
|
28
|
+
/** Number of bytes in the truncated output */
|
|
29
|
+
outputBytes: number;
|
|
30
|
+
/** Whether the last line was partially truncated (only for tail truncation edge case) */
|
|
31
|
+
lastLinePartial: boolean;
|
|
32
|
+
/** Whether the first line exceeded the byte limit (for head truncation) */
|
|
33
|
+
firstLineExceedsLimit: boolean;
|
|
34
|
+
/** The max lines limit that was applied */
|
|
35
|
+
maxLines: number;
|
|
36
|
+
/** The max bytes limit that was applied */
|
|
37
|
+
maxBytes: number;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export interface TruncationOptions {
|
|
41
|
+
/** Maximum number of lines (default: 2000) */
|
|
42
|
+
maxLines?: number;
|
|
43
|
+
/** Maximum number of bytes (default: 50KB) */
|
|
44
|
+
maxBytes?: number;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Format bytes as human-readable size.
|
|
49
|
+
*/
|
|
50
|
+
export function formatSize(bytes: number): string {
|
|
51
|
+
if (bytes < 1024) {
|
|
52
|
+
return `${bytes}B`;
|
|
53
|
+
} else if (bytes < 1024 * 1024) {
|
|
54
|
+
return `${(bytes / 1024).toFixed(1)}KB`;
|
|
55
|
+
} else {
|
|
56
|
+
return `${(bytes / (1024 * 1024)).toFixed(1)}MB`;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Truncate content from the head (keep first N lines/bytes).
|
|
62
|
+
* Suitable for file reads where you want to see the beginning.
|
|
63
|
+
*
|
|
64
|
+
* Never returns partial lines. If first line exceeds byte limit,
|
|
65
|
+
* returns empty content with firstLineExceedsLimit=true.
|
|
66
|
+
*/
|
|
67
|
+
export function truncateHead(content: string, options: TruncationOptions = {}): TruncationResult {
|
|
68
|
+
const maxLines = options.maxLines ?? DEFAULT_MAX_LINES;
|
|
69
|
+
const maxBytes = options.maxBytes ?? DEFAULT_MAX_BYTES;
|
|
70
|
+
|
|
71
|
+
const totalBytes = Buffer.byteLength(content, "utf-8");
|
|
72
|
+
const lines = content.split("\n");
|
|
73
|
+
const totalLines = lines.length;
|
|
74
|
+
|
|
75
|
+
// Check if no truncation needed
|
|
76
|
+
if (totalLines <= maxLines && totalBytes <= maxBytes) {
|
|
77
|
+
return {
|
|
78
|
+
content,
|
|
79
|
+
truncated: false,
|
|
80
|
+
truncatedBy: null,
|
|
81
|
+
totalLines,
|
|
82
|
+
totalBytes,
|
|
83
|
+
outputLines: totalLines,
|
|
84
|
+
outputBytes: totalBytes,
|
|
85
|
+
lastLinePartial: false,
|
|
86
|
+
firstLineExceedsLimit: false,
|
|
87
|
+
maxLines,
|
|
88
|
+
maxBytes,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Check if first line alone exceeds byte limit
|
|
93
|
+
const firstLineBytes = Buffer.byteLength(lines[0], "utf-8");
|
|
94
|
+
if (firstLineBytes > maxBytes) {
|
|
95
|
+
return {
|
|
96
|
+
content: "",
|
|
97
|
+
truncated: true,
|
|
98
|
+
truncatedBy: "bytes",
|
|
99
|
+
totalLines,
|
|
100
|
+
totalBytes,
|
|
101
|
+
outputLines: 0,
|
|
102
|
+
outputBytes: 0,
|
|
103
|
+
lastLinePartial: false,
|
|
104
|
+
firstLineExceedsLimit: true,
|
|
105
|
+
maxLines,
|
|
106
|
+
maxBytes,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// Collect complete lines that fit
|
|
111
|
+
const outputLinesArr: string[] = [];
|
|
112
|
+
let outputBytesCount = 0;
|
|
113
|
+
let truncatedBy: "lines" | "bytes" = "lines";
|
|
114
|
+
|
|
115
|
+
for (let i = 0; i < lines.length && i < maxLines; i++) {
|
|
116
|
+
const line = lines[i];
|
|
117
|
+
const lineBytes = Buffer.byteLength(line, "utf-8") + (i > 0 ? 1 : 0); // +1 for newline
|
|
118
|
+
|
|
119
|
+
if (outputBytesCount + lineBytes > maxBytes) {
|
|
120
|
+
truncatedBy = "bytes";
|
|
121
|
+
break;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
outputLinesArr.push(line);
|
|
125
|
+
outputBytesCount += lineBytes;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// If we exited due to line limit
|
|
129
|
+
if (outputLinesArr.length >= maxLines && outputBytesCount <= maxBytes) {
|
|
130
|
+
truncatedBy = "lines";
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
const outputContent = outputLinesArr.join("\n");
|
|
134
|
+
const finalOutputBytes = Buffer.byteLength(outputContent, "utf-8");
|
|
135
|
+
|
|
136
|
+
return {
|
|
137
|
+
content: outputContent,
|
|
138
|
+
truncated: true,
|
|
139
|
+
truncatedBy,
|
|
140
|
+
totalLines,
|
|
141
|
+
totalBytes,
|
|
142
|
+
outputLines: outputLinesArr.length,
|
|
143
|
+
outputBytes: finalOutputBytes,
|
|
144
|
+
lastLinePartial: false,
|
|
145
|
+
firstLineExceedsLimit: false,
|
|
146
|
+
maxLines,
|
|
147
|
+
maxBytes,
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Truncate content from the tail (keep last N lines/bytes).
|
|
153
|
+
* Suitable for bash output where you want to see the end (errors, final results).
|
|
154
|
+
*
|
|
155
|
+
* May return partial first line if the last line of original content exceeds byte limit.
|
|
156
|
+
*/
|
|
157
|
+
export function truncateTail(content: string, options: TruncationOptions = {}): TruncationResult {
|
|
158
|
+
const maxLines = options.maxLines ?? DEFAULT_MAX_LINES;
|
|
159
|
+
const maxBytes = options.maxBytes ?? DEFAULT_MAX_BYTES;
|
|
160
|
+
|
|
161
|
+
const totalBytes = Buffer.byteLength(content, "utf-8");
|
|
162
|
+
const lines = content.split("\n");
|
|
163
|
+
const totalLines = lines.length;
|
|
164
|
+
|
|
165
|
+
// Check if no truncation needed
|
|
166
|
+
if (totalLines <= maxLines && totalBytes <= maxBytes) {
|
|
167
|
+
return {
|
|
168
|
+
content,
|
|
169
|
+
truncated: false,
|
|
170
|
+
truncatedBy: null,
|
|
171
|
+
totalLines,
|
|
172
|
+
totalBytes,
|
|
173
|
+
outputLines: totalLines,
|
|
174
|
+
outputBytes: totalBytes,
|
|
175
|
+
lastLinePartial: false,
|
|
176
|
+
firstLineExceedsLimit: false,
|
|
177
|
+
maxLines,
|
|
178
|
+
maxBytes,
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
// Work backwards from the end
|
|
183
|
+
const outputLinesArr: string[] = [];
|
|
184
|
+
let outputBytesCount = 0;
|
|
185
|
+
let truncatedBy: "lines" | "bytes" = "lines";
|
|
186
|
+
let lastLinePartial = false;
|
|
187
|
+
|
|
188
|
+
for (let i = lines.length - 1; i >= 0 && outputLinesArr.length < maxLines; i--) {
|
|
189
|
+
const line = lines[i];
|
|
190
|
+
const lineBytes = Buffer.byteLength(line, "utf-8") + (outputLinesArr.length > 0 ? 1 : 0); // +1 for newline
|
|
191
|
+
|
|
192
|
+
if (outputBytesCount + lineBytes > maxBytes) {
|
|
193
|
+
truncatedBy = "bytes";
|
|
194
|
+
// Edge case: if we haven't added ANY lines yet and this line exceeds maxBytes,
|
|
195
|
+
// take the end of the line (partial)
|
|
196
|
+
if (outputLinesArr.length === 0) {
|
|
197
|
+
const truncatedLine = truncateStringToBytesFromEnd(line, maxBytes);
|
|
198
|
+
outputLinesArr.unshift(truncatedLine);
|
|
199
|
+
outputBytesCount = Buffer.byteLength(truncatedLine, "utf-8");
|
|
200
|
+
lastLinePartial = true;
|
|
201
|
+
}
|
|
202
|
+
break;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
outputLinesArr.unshift(line);
|
|
206
|
+
outputBytesCount += lineBytes;
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
// If we exited due to line limit
|
|
210
|
+
if (outputLinesArr.length >= maxLines && outputBytesCount <= maxBytes) {
|
|
211
|
+
truncatedBy = "lines";
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
const outputContent = outputLinesArr.join("\n");
|
|
215
|
+
const finalOutputBytes = Buffer.byteLength(outputContent, "utf-8");
|
|
216
|
+
|
|
217
|
+
return {
|
|
218
|
+
content: outputContent,
|
|
219
|
+
truncated: true,
|
|
220
|
+
truncatedBy,
|
|
221
|
+
totalLines,
|
|
222
|
+
totalBytes,
|
|
223
|
+
outputLines: outputLinesArr.length,
|
|
224
|
+
outputBytes: finalOutputBytes,
|
|
225
|
+
lastLinePartial,
|
|
226
|
+
firstLineExceedsLimit: false,
|
|
227
|
+
maxLines,
|
|
228
|
+
maxBytes,
|
|
229
|
+
};
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
/**
|
|
233
|
+
* Truncate a string to fit within a byte limit (from the end).
|
|
234
|
+
* Handles multi-byte UTF-8 characters correctly.
|
|
235
|
+
*/
|
|
236
|
+
function truncateStringToBytesFromEnd(str: string, maxBytes: number): string {
|
|
237
|
+
const buf = Buffer.from(str, "utf-8");
|
|
238
|
+
if (buf.length <= maxBytes) {
|
|
239
|
+
return str;
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
// Start from the end, skip maxBytes back
|
|
243
|
+
let start = buf.length - maxBytes;
|
|
244
|
+
|
|
245
|
+
// Find a valid UTF-8 boundary (start of a character)
|
|
246
|
+
while (start < buf.length && (buf[start] & 0xc0) === 0x80) {
|
|
247
|
+
start++;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
return buf.slice(start).toString("utf-8");
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
/**
|
|
254
|
+
* Truncate a single line to max characters, adding [truncated] suffix.
|
|
255
|
+
* Used for grep match lines.
|
|
256
|
+
*/
|
|
257
|
+
export function truncateLine(
|
|
258
|
+
line: string,
|
|
259
|
+
maxChars: number = GREP_MAX_LINE_LENGTH,
|
|
260
|
+
): { text: string; wasTruncated: boolean } {
|
|
261
|
+
if (line.length <= maxChars) {
|
|
262
|
+
return { text: line, wasTruncated: false };
|
|
263
|
+
}
|
|
264
|
+
return { text: `${line.slice(0, maxChars)}... [truncated]`, wasTruncated: true };
|
|
265
|
+
}
|