march-cli 0.1.24 → 0.1.25
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/bin/march.mjs +13 -13
- package/package.json +49 -49
- package/src/agent/command-exec-tool.mjs +172 -172
- package/src/agent/context-stats-tool.mjs +57 -57
- package/src/agent/editing/diff-apply.mjs +28 -28
- package/src/agent/editing/diff-format.mjs +57 -57
- package/src/agent/editing/lsp-report.mjs +69 -69
- package/src/agent/file-edit-tool.mjs +250 -250
- package/src/agent/file-tools/read-file-tool.mjs +112 -112
- package/src/agent/file-tools/read-image-tool.mjs +76 -76
- package/src/agent/model-payload-dumper.mjs +208 -208
- package/src/agent/pi-session/pi-session-sidecar-failure.mjs +10 -10
- package/src/agent/provider/payload-messages.mjs +138 -138
- package/src/agent/runner/codex-large-context-guard.mjs +87 -87
- package/src/agent/runner/codex-transport-compression.mjs +180 -180
- package/src/agent/runner/codex-transport-debug.mjs +113 -113
- package/src/agent/runner/codex-websocket-event-debug.mjs +130 -130
- package/src/agent/runner/fast-model.mjs +36 -36
- package/src/agent/runner/runner-cleanup.mjs +12 -12
- package/src/agent/runner/runner-init.mjs +15 -15
- package/src/agent/runner/runner-session-state.mjs +40 -40
- package/src/agent/runner/runner-utils.mjs +24 -24
- package/src/agent/runner.mjs +299 -299
- package/src/agent/runtime/ipc/ipc-peer.mjs +99 -99
- package/src/agent/runtime/ipc/process-ipc-transport.mjs +16 -16
- package/src/agent/runtime/remote-runner-client.mjs +73 -73
- package/src/agent/runtime/remote-ui-client.mjs +20 -20
- package/src/agent/runtime/runner-ipc-target.mjs +125 -125
- package/src/agent/runtime/runner-process-client.mjs +47 -47
- package/src/agent/runtime/runner-process-entry.mjs +11 -11
- package/src/agent/runtime/runner-process-factory.mjs +111 -108
- package/src/agent/runtime/runner-runtime-host.mjs +79 -79
- package/src/agent/runtime/runtime-factory.mjs +42 -42
- package/src/agent/runtime/runtime-host.mjs +34 -34
- package/src/agent/runtime/ui-event-bridge.mjs +95 -95
- package/src/agent/screen-tools/list-windows-tool.mjs +39 -39
- package/src/agent/screen-tools/screen-tool.mjs +49 -49
- package/src/agent/screen-tools/windows-screen.mjs +133 -133
- package/src/agent/session/session-auto-name.mjs +41 -41
- package/src/agent/session/session-binding.mjs +12 -12
- package/src/agent/session/session-options.mjs +47 -47
- package/src/agent/tool-names.mjs +1 -1
- package/src/agent/tool-result.mjs +3 -3
- package/src/agent/tool-summary.mjs +112 -112
- package/src/agent/tools.mjs +58 -58
- package/src/agent/turn/turn-events.mjs +111 -111
- package/src/agent/turn/turn-logging.mjs +30 -30
- package/src/agent/turn/turn-runner.mjs +196 -196
- package/src/agent/vision-capability.mjs +14 -14
- package/src/auth/login-command.mjs +90 -90
- package/src/auth/storage.mjs +34 -34
- package/src/cli/args.mjs +96 -79
- package/src/cli/commands/copy-command.mjs +87 -87
- package/src/cli/commands/export-command.mjs +206 -206
- package/src/cli/commands/extensions-command.mjs +53 -53
- package/src/cli/commands/help-command.mjs +7 -7
- package/src/cli/commands/model-command.mjs +141 -141
- package/src/cli/commands/paste-image-command.mjs +43 -43
- package/src/cli/commands/provider-command.mjs +59 -59
- package/src/cli/commands/status-command.mjs +196 -194
- package/src/cli/commands/thinking-command.mjs +87 -87
- package/src/cli/fallback-ui.mjs +156 -156
- package/src/cli/input/attachment-tokens.mjs +20 -20
- package/src/cli/input/autocomplete.mjs +74 -74
- package/src/cli/input/external-editor.mjs +39 -39
- package/src/cli/input/file-search/index.mjs +160 -160
- package/src/cli/input/history-store.mjs +35 -35
- package/src/cli/input/image-clipboard.mjs +55 -55
- package/src/cli/input/keybinding-dispatch.mjs +76 -76
- package/src/cli/input/keybindings.mjs +96 -96
- package/src/cli/input/mode-state.mjs +43 -43
- package/src/cli/input/prompt-templates.mjs +84 -84
- package/src/cli/input/select-with-keyboard.mjs +86 -86
- package/src/cli/permissions.mjs +103 -103
- package/src/cli/repl-commands.mjs +86 -86
- package/src/cli/repl-loop.mjs +183 -183
- package/src/cli/selector-list.mjs +21 -21
- package/src/cli/session/pi-session-switch-command.mjs +41 -41
- package/src/cli/session/session-command.mjs +23 -23
- package/src/cli/session/session-list-command.mjs +68 -68
- package/src/cli/session/session-name-command.mjs +26 -26
- package/src/cli/session/session-source-command.mjs +89 -89
- package/src/cli/session/session-switch-command.mjs +1 -1
- package/src/cli/shell/shell-command.mjs +55 -55
- package/src/cli/shell/shell-drawer-controls.mjs +33 -33
- package/src/cli/shell/shell-drawer.mjs +192 -192
- package/src/cli/shell/shell-split-layout.mjs +70 -70
- package/src/cli/slash-commands.mjs +192 -192
- package/src/cli/startup/create-runtime-runner.mjs +61 -61
- package/src/cli/startup/runtime-close.mjs +23 -23
- package/src/cli/startup/startup-banner.mjs +71 -71
- package/src/cli/startup/startup-session.mjs +51 -51
- package/src/cli/status-line-updater.mjs +75 -75
- package/src/cli/tool-output.mjs +9 -9
- package/src/cli/tui/editor/external-editor-runner.mjs +24 -24
- package/src/cli/tui/input/mouse-selection-controller.mjs +91 -91
- package/src/cli/tui/input/mouse-tracking.mjs +20 -20
- package/src/cli/tui/layout/main-pane-layout.mjs +47 -47
- package/src/cli/tui/layout/safe-render-boundary.mjs +46 -46
- package/src/cli/tui/markdown-renderer.mjs +285 -285
- package/src/cli/tui/output/scroll-state.mjs +79 -79
- package/src/cli/tui/output/text-line-renderer.mjs +50 -50
- package/src/cli/tui/output/tool-card-renderer.mjs +59 -59
- package/src/cli/tui/output/visible-lines.mjs +8 -8
- package/src/cli/tui/output-buffer.mjs +293 -293
- package/src/cli/tui/permission-request-ui.mjs +18 -18
- package/src/cli/tui/recall-rendering.mjs +28 -25
- package/src/cli/tui/render/render-scheduler.mjs +26 -26
- package/src/cli/tui/render/stream-delta-buffer.mjs +46 -46
- package/src/cli/tui/select/editor-select-list.mjs +111 -111
- package/src/cli/tui/selection-screen.mjs +269 -269
- package/src/cli/tui/status/retry-status.mjs +72 -72
- package/src/cli/tui/status/spinner-status.mjs +42 -42
- package/src/cli/tui/status/status-bar.mjs +225 -225
- package/src/cli/tui/syntax/highlighting.mjs +260 -260
- package/src/cli/tui/syntax/languages.mjs +91 -91
- package/src/cli/tui/syntax/tree-sitter/bash.highlights.scm +261 -261
- package/src/cli/tui/syntax/tree-sitter/c.highlights.scm +341 -341
- package/src/cli/tui/syntax/tree-sitter/cpp.highlights.scm +268 -268
- package/src/cli/tui/syntax/tree-sitter/csharp.highlights.scm +577 -577
- package/src/cli/tui/syntax/tree-sitter/css.highlights.scm +109 -109
- package/src/cli/tui/syntax/tree-sitter/diff.highlights.scm +49 -49
- package/src/cli/tui/syntax/tree-sitter/go.highlights.scm +254 -254
- package/src/cli/tui/syntax/tree-sitter/html.highlights.scm +13 -13
- package/src/cli/tui/syntax/tree-sitter/java.highlights.scm +330 -330
- package/src/cli/tui/syntax/tree-sitter/json.highlights.scm +38 -38
- package/src/cli/tui/syntax/tree-sitter/php.highlights.scm +203 -203
- package/src/cli/tui/syntax/tree-sitter/python.highlights.scm +137 -137
- package/src/cli/tui/syntax/tree-sitter/ruby.highlights.scm +309 -309
- package/src/cli/tui/syntax/tree-sitter/rust.highlights.scm +531 -531
- package/src/cli/tui/syntax/tree-sitter/toml.highlights.scm +39 -39
- package/src/cli/tui/syntax/tree-sitter/tsx.highlights.scm +35 -35
- package/src/cli/tui/syntax/tree-sitter/typescript.highlights.scm +35 -35
- package/src/cli/tui/syntax/tree-sitter/yaml.highlights.scm +99 -99
- package/src/cli/tui/tool-rendering.mjs +87 -87
- package/src/cli/tui/tui-diff-rendering.mjs +157 -157
- package/src/cli/tui/tui-handlers.mjs +111 -111
- package/src/cli/tui/tui-input-controller.mjs +61 -61
- package/src/cli/tui/ui-theme.mjs +157 -157
- package/src/cli/ui.mjs +297 -297
- package/src/config/config-json.mjs +108 -84
- package/src/config/dotenv.mjs +20 -20
- package/src/config/features.mjs +75 -75
- package/src/config/loader.mjs +156 -143
- package/src/config/settings-command.mjs +97 -97
- package/src/context/engine.mjs +199 -198
- package/src/context/injections.mjs +26 -26
- package/src/context/profiles.mjs +39 -39
- package/src/context/project-context.mjs +20 -20
- package/src/context/session-status.mjs +25 -17
- package/src/context/shell-layers.mjs +23 -23
- package/src/context/system-core/base.md +50 -50
- package/src/context/system-core/prompts/deepseek-v4-pro.md +3 -3
- package/src/context/system-core/prompts/default.md +3 -3
- package/src/context/system-core.mjs +35 -35
- package/src/debug/logger.mjs +141 -141
- package/src/debug/model-context-dumper.mjs +52 -52
- package/src/extensions/discovery.mjs +40 -40
- package/src/extensions/lifecycle-adapter.mjs +210 -210
- package/src/extensions/lifecycle-manifest.mjs +69 -69
- package/src/image-gen/index.mjs +7 -7
- package/src/image-gen/provider.mjs +231 -231
- package/src/image-gen/tool.mjs +84 -84
- package/src/lsp/client.mjs +257 -257
- package/src/lsp/diagnostic-store.mjs +42 -42
- package/src/lsp/diagnostics-format.mjs +72 -72
- package/src/lsp/managed-node-server.mjs +99 -99
- package/src/lsp/path-match.mjs +10 -10
- package/src/lsp/server-definitions.mjs +188 -188
- package/src/lsp/servers.mjs +165 -165
- package/src/lsp/service.mjs +110 -110
- package/src/lsp/status-message.mjs +9 -9
- package/src/lsp/typescript-project-resolver.mjs +186 -186
- package/src/main.mjs +294 -299
- package/src/mcp/client.mjs +195 -195
- package/src/mcp/config.mjs +130 -130
- package/src/mcp/index.mjs +48 -48
- package/src/mcp/tools.mjs +98 -98
- package/src/memory/command.mjs +120 -0
- package/src/memory/markdown/markdown-delete.mjs +23 -23
- package/src/memory/markdown/markdown-format.mjs +128 -128
- package/src/memory/markdown/markdown-recall.mjs +28 -28
- package/src/memory/markdown/ripgrep.mjs +16 -16
- package/src/memory/markdown/sqlite-index.mjs +87 -87
- package/src/memory/markdown-store.mjs +272 -286
- package/src/memory/markdown-tools.mjs +174 -103
- package/src/memory/remote/client.mjs +68 -0
- package/src/memory/remote/config.mjs +52 -0
- package/src/memory/remote/server.mjs +99 -0
- package/src/memory/search.mjs +183 -0
- package/src/network/environment.mjs +131 -131
- package/src/notification/desktop-notifier.mjs +262 -262
- package/src/platform/open-file.mjs +28 -28
- package/src/platform/spawn-command.mjs +27 -27
- package/src/provider/accept-command.mjs +89 -89
- package/src/provider/command.mjs +21 -21
- package/src/provider/config-command.mjs +129 -129
- package/src/provider/custom-provider.mjs +113 -113
- package/src/provider/hosted-tools.mjs +111 -111
- package/src/provider/presets.mjs +72 -72
- package/src/provider/share-command.mjs +79 -79
- package/src/provider/share-payload.mjs +52 -52
- package/src/session/attachment-display.mjs +16 -16
- package/src/session/attachment-references.mjs +65 -65
- package/src/session/attachments.mjs +140 -140
- package/src/session/persist.mjs +1 -1
- package/src/session/pi-manager.mjs +34 -34
- package/src/session/session-utils.mjs +16 -16
- package/src/session/sidecar-sync.mjs +19 -19
- package/src/session/sidecar.mjs +69 -69
- package/src/session/transcript.mjs +83 -83
- package/src/session/tree.mjs +42 -42
- package/src/shell/cli-runtime.mjs +11 -11
- package/src/shell/hints.mjs +12 -12
- package/src/shell/node-pty-adapter.mjs +81 -81
- package/src/shell/runtime-state.mjs +126 -126
- package/src/shell/runtime.mjs +252 -252
- package/src/shell/screen-buffer.mjs +136 -136
- package/src/shell/tool-read.mjs +74 -74
- package/src/shell/tools.mjs +299 -299
- package/src/supergrok/actions/image-generate.mjs +60 -60
- package/src/supergrok/actions/search.mjs +78 -78
- package/src/supergrok/auth.mjs +36 -36
- package/src/supergrok/constants.mjs +18 -18
- package/src/supergrok/oauth-provider.mjs +278 -278
- package/src/supergrok/provider.mjs +35 -35
- package/src/supergrok/response.mjs +76 -76
- package/src/supergrok/tool.mjs +61 -61
- package/src/text/ansi.mjs +3 -3
- package/src/web/config-command.mjs +43 -43
- package/src/web/fetch.mjs +78 -78
- package/src/web/presets.mjs +16 -16
- package/src/web/search.mjs +83 -83
- package/src/web/tools.mjs +107 -107
|
@@ -1,136 +1,136 @@
|
|
|
1
|
-
import headless from "@xterm/headless";
|
|
2
|
-
|
|
3
|
-
const { Terminal } = headless;
|
|
4
|
-
|
|
5
|
-
export function createTerminalScreenBuffer({ cols = 80, rows = 24 } = {}) {
|
|
6
|
-
const terminal = new Terminal({
|
|
7
|
-
allowProposedApi: true,
|
|
8
|
-
cols: normalizePositiveInt(cols, 80),
|
|
9
|
-
rows: normalizePositiveInt(rows, 24),
|
|
10
|
-
scrollback: 0,
|
|
11
|
-
});
|
|
12
|
-
|
|
13
|
-
let pendingWrites = 0;
|
|
14
|
-
|
|
15
|
-
return {
|
|
16
|
-
write(data) {
|
|
17
|
-
pendingWrites += 1;
|
|
18
|
-
terminal.write(String(data ?? ""), () => {
|
|
19
|
-
pendingWrites = Math.max(0, pendingWrites - 1);
|
|
20
|
-
});
|
|
21
|
-
},
|
|
22
|
-
resize(nextCols, nextRows) {
|
|
23
|
-
terminal.resize(
|
|
24
|
-
normalizePositiveInt(nextCols, terminal.cols),
|
|
25
|
-
normalizePositiveInt(nextRows, terminal.rows),
|
|
26
|
-
);
|
|
27
|
-
},
|
|
28
|
-
snapshot() {
|
|
29
|
-
const lines = readViewportLines(terminal);
|
|
30
|
-
return {
|
|
31
|
-
cols: terminal.cols,
|
|
32
|
-
rows: terminal.rows,
|
|
33
|
-
pendingWrites,
|
|
34
|
-
plain: lines.map((line) => line.plain).join("\n"),
|
|
35
|
-
ansi: lines.map((line) => line.ansi).join("\n"),
|
|
36
|
-
};
|
|
37
|
-
},
|
|
38
|
-
dispose() {
|
|
39
|
-
terminal.dispose();
|
|
40
|
-
},
|
|
41
|
-
};
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function readViewportLines(terminal) {
|
|
45
|
-
const buffer = terminal.buffer.active;
|
|
46
|
-
const start = buffer.baseY;
|
|
47
|
-
const rows = [];
|
|
48
|
-
for (let y = start; y < start + terminal.rows; y++) {
|
|
49
|
-
const line = buffer.getLine(y);
|
|
50
|
-
rows.push(line ? lineToSnapshot(line) : { plain: "", ansi: "" });
|
|
51
|
-
}
|
|
52
|
-
return trimTrailingBlankRows(rows);
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
function lineToSnapshot(line) {
|
|
56
|
-
const end = findContentEnd(line);
|
|
57
|
-
if (end === 0) return { plain: "", ansi: "" };
|
|
58
|
-
|
|
59
|
-
const plain = line.translateToString(false, 0, end);
|
|
60
|
-
let ansi = "";
|
|
61
|
-
let activeStyle = "";
|
|
62
|
-
for (let x = 0; x < end; x++) {
|
|
63
|
-
const cell = line.getCell(x);
|
|
64
|
-
if (!cell || cell.getWidth() === 0) continue;
|
|
65
|
-
const chars = cell.getChars() || " ";
|
|
66
|
-
const style = cellToSgr(cell);
|
|
67
|
-
if (style !== activeStyle) {
|
|
68
|
-
if (activeStyle) ansi += "\x1b[0m";
|
|
69
|
-
if (style) ansi += style;
|
|
70
|
-
activeStyle = style;
|
|
71
|
-
}
|
|
72
|
-
ansi += chars;
|
|
73
|
-
}
|
|
74
|
-
if (activeStyle) ansi += "\x1b[0m";
|
|
75
|
-
return { plain, ansi };
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
function findContentEnd(line) {
|
|
79
|
-
for (let x = line.length - 1; x >= 0; x--) {
|
|
80
|
-
const cell = line.getCell(x);
|
|
81
|
-
if (cell?.getChars()) return x + cell.getWidth();
|
|
82
|
-
}
|
|
83
|
-
return 0;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
function cellToSgr(cell) {
|
|
87
|
-
if (cell.isAttributeDefault()) return "";
|
|
88
|
-
const codes = [];
|
|
89
|
-
if (cell.isBold()) codes.push(1);
|
|
90
|
-
if (cell.isDim()) codes.push(2);
|
|
91
|
-
if (cell.isItalic()) codes.push(3);
|
|
92
|
-
if (cell.isUnderline()) codes.push(4);
|
|
93
|
-
if (cell.isBlink()) codes.push(5);
|
|
94
|
-
if (cell.isInverse()) codes.push(7);
|
|
95
|
-
if (cell.isInvisible()) codes.push(8);
|
|
96
|
-
if (cell.isStrikethrough()) codes.push(9);
|
|
97
|
-
if (cell.isOverline()) codes.push(53);
|
|
98
|
-
pushColorCodes(codes, cell, "fg");
|
|
99
|
-
pushColorCodes(codes, cell, "bg");
|
|
100
|
-
return codes.length ? `\x1b[${codes.join(";")}m` : "";
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
function pushColorCodes(codes, cell, target) {
|
|
104
|
-
const isFg = target === "fg";
|
|
105
|
-
const isDefault = isFg ? cell.isFgDefault() : cell.isBgDefault();
|
|
106
|
-
if (isDefault) return;
|
|
107
|
-
|
|
108
|
-
const color = isFg ? cell.getFgColor() : cell.getBgColor();
|
|
109
|
-
const base = isFg ? 30 : 40;
|
|
110
|
-
const brightBase = isFg ? 90 : 100;
|
|
111
|
-
if (isFg ? cell.isFgRGB() : cell.isBgRGB()) {
|
|
112
|
-
codes.push(isFg ? 38 : 48, 2, (color >> 16) & 255, (color >> 8) & 255, color & 255);
|
|
113
|
-
return;
|
|
114
|
-
}
|
|
115
|
-
if (isFg ? cell.isFgPalette() : cell.isBgPalette()) {
|
|
116
|
-
if (color < 8) {
|
|
117
|
-
codes.push(base + color);
|
|
118
|
-
} else if (color < 16) {
|
|
119
|
-
codes.push(brightBase + color - 8);
|
|
120
|
-
} else {
|
|
121
|
-
codes.push(isFg ? 38 : 48, 5, color);
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
function trimTrailingBlankRows(rows) {
|
|
127
|
-
let end = rows.length;
|
|
128
|
-
while (end > 0 && rows[end - 1].plain === "") end -= 1;
|
|
129
|
-
return rows.slice(0, end);
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
function normalizePositiveInt(value, fallback) {
|
|
133
|
-
const number = Number(value);
|
|
134
|
-
if (!Number.isFinite(number) || number <= 0) return Math.max(1, Math.trunc(Number(fallback) || 1));
|
|
135
|
-
return Math.max(1, Math.trunc(number));
|
|
136
|
-
}
|
|
1
|
+
import headless from "@xterm/headless";
|
|
2
|
+
|
|
3
|
+
const { Terminal } = headless;
|
|
4
|
+
|
|
5
|
+
export function createTerminalScreenBuffer({ cols = 80, rows = 24 } = {}) {
|
|
6
|
+
const terminal = new Terminal({
|
|
7
|
+
allowProposedApi: true,
|
|
8
|
+
cols: normalizePositiveInt(cols, 80),
|
|
9
|
+
rows: normalizePositiveInt(rows, 24),
|
|
10
|
+
scrollback: 0,
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
let pendingWrites = 0;
|
|
14
|
+
|
|
15
|
+
return {
|
|
16
|
+
write(data) {
|
|
17
|
+
pendingWrites += 1;
|
|
18
|
+
terminal.write(String(data ?? ""), () => {
|
|
19
|
+
pendingWrites = Math.max(0, pendingWrites - 1);
|
|
20
|
+
});
|
|
21
|
+
},
|
|
22
|
+
resize(nextCols, nextRows) {
|
|
23
|
+
terminal.resize(
|
|
24
|
+
normalizePositiveInt(nextCols, terminal.cols),
|
|
25
|
+
normalizePositiveInt(nextRows, terminal.rows),
|
|
26
|
+
);
|
|
27
|
+
},
|
|
28
|
+
snapshot() {
|
|
29
|
+
const lines = readViewportLines(terminal);
|
|
30
|
+
return {
|
|
31
|
+
cols: terminal.cols,
|
|
32
|
+
rows: terminal.rows,
|
|
33
|
+
pendingWrites,
|
|
34
|
+
plain: lines.map((line) => line.plain).join("\n"),
|
|
35
|
+
ansi: lines.map((line) => line.ansi).join("\n"),
|
|
36
|
+
};
|
|
37
|
+
},
|
|
38
|
+
dispose() {
|
|
39
|
+
terminal.dispose();
|
|
40
|
+
},
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function readViewportLines(terminal) {
|
|
45
|
+
const buffer = terminal.buffer.active;
|
|
46
|
+
const start = buffer.baseY;
|
|
47
|
+
const rows = [];
|
|
48
|
+
for (let y = start; y < start + terminal.rows; y++) {
|
|
49
|
+
const line = buffer.getLine(y);
|
|
50
|
+
rows.push(line ? lineToSnapshot(line) : { plain: "", ansi: "" });
|
|
51
|
+
}
|
|
52
|
+
return trimTrailingBlankRows(rows);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function lineToSnapshot(line) {
|
|
56
|
+
const end = findContentEnd(line);
|
|
57
|
+
if (end === 0) return { plain: "", ansi: "" };
|
|
58
|
+
|
|
59
|
+
const plain = line.translateToString(false, 0, end);
|
|
60
|
+
let ansi = "";
|
|
61
|
+
let activeStyle = "";
|
|
62
|
+
for (let x = 0; x < end; x++) {
|
|
63
|
+
const cell = line.getCell(x);
|
|
64
|
+
if (!cell || cell.getWidth() === 0) continue;
|
|
65
|
+
const chars = cell.getChars() || " ";
|
|
66
|
+
const style = cellToSgr(cell);
|
|
67
|
+
if (style !== activeStyle) {
|
|
68
|
+
if (activeStyle) ansi += "\x1b[0m";
|
|
69
|
+
if (style) ansi += style;
|
|
70
|
+
activeStyle = style;
|
|
71
|
+
}
|
|
72
|
+
ansi += chars;
|
|
73
|
+
}
|
|
74
|
+
if (activeStyle) ansi += "\x1b[0m";
|
|
75
|
+
return { plain, ansi };
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function findContentEnd(line) {
|
|
79
|
+
for (let x = line.length - 1; x >= 0; x--) {
|
|
80
|
+
const cell = line.getCell(x);
|
|
81
|
+
if (cell?.getChars()) return x + cell.getWidth();
|
|
82
|
+
}
|
|
83
|
+
return 0;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function cellToSgr(cell) {
|
|
87
|
+
if (cell.isAttributeDefault()) return "";
|
|
88
|
+
const codes = [];
|
|
89
|
+
if (cell.isBold()) codes.push(1);
|
|
90
|
+
if (cell.isDim()) codes.push(2);
|
|
91
|
+
if (cell.isItalic()) codes.push(3);
|
|
92
|
+
if (cell.isUnderline()) codes.push(4);
|
|
93
|
+
if (cell.isBlink()) codes.push(5);
|
|
94
|
+
if (cell.isInverse()) codes.push(7);
|
|
95
|
+
if (cell.isInvisible()) codes.push(8);
|
|
96
|
+
if (cell.isStrikethrough()) codes.push(9);
|
|
97
|
+
if (cell.isOverline()) codes.push(53);
|
|
98
|
+
pushColorCodes(codes, cell, "fg");
|
|
99
|
+
pushColorCodes(codes, cell, "bg");
|
|
100
|
+
return codes.length ? `\x1b[${codes.join(";")}m` : "";
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function pushColorCodes(codes, cell, target) {
|
|
104
|
+
const isFg = target === "fg";
|
|
105
|
+
const isDefault = isFg ? cell.isFgDefault() : cell.isBgDefault();
|
|
106
|
+
if (isDefault) return;
|
|
107
|
+
|
|
108
|
+
const color = isFg ? cell.getFgColor() : cell.getBgColor();
|
|
109
|
+
const base = isFg ? 30 : 40;
|
|
110
|
+
const brightBase = isFg ? 90 : 100;
|
|
111
|
+
if (isFg ? cell.isFgRGB() : cell.isBgRGB()) {
|
|
112
|
+
codes.push(isFg ? 38 : 48, 2, (color >> 16) & 255, (color >> 8) & 255, color & 255);
|
|
113
|
+
return;
|
|
114
|
+
}
|
|
115
|
+
if (isFg ? cell.isFgPalette() : cell.isBgPalette()) {
|
|
116
|
+
if (color < 8) {
|
|
117
|
+
codes.push(base + color);
|
|
118
|
+
} else if (color < 16) {
|
|
119
|
+
codes.push(brightBase + color - 8);
|
|
120
|
+
} else {
|
|
121
|
+
codes.push(isFg ? 38 : 48, 5, color);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
function trimTrailingBlankRows(rows) {
|
|
127
|
+
let end = rows.length;
|
|
128
|
+
while (end > 0 && rows[end - 1].plain === "") end -= 1;
|
|
129
|
+
return rows.slice(0, end);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
function normalizePositiveInt(value, fallback) {
|
|
133
|
+
const number = Number(value);
|
|
134
|
+
if (!Number.isFinite(number) || number <= 0) return Math.max(1, Math.trunc(Number(fallback) || 1));
|
|
135
|
+
return Math.max(1, Math.trunc(number));
|
|
136
|
+
}
|
package/src/shell/tool-read.mjs
CHANGED
|
@@ -1,74 +1,74 @@
|
|
|
1
|
-
import { defineTool } from "@earendil-works/pi-coding-agent";
|
|
2
|
-
import { Type } from "typebox";
|
|
3
|
-
import { toolText } from "../agent/tool-result.mjs";
|
|
4
|
-
|
|
5
|
-
export function createTerminalReadTool(shellRuntime) {
|
|
6
|
-
return defineTool({
|
|
7
|
-
name: "terminal_read",
|
|
8
|
-
label: "Terminal Read",
|
|
9
|
-
description: "Read plain-text output from a running interactive terminal. Use this for normal shell inspection after shell hints, terminal_list, or terminal_send.",
|
|
10
|
-
parameters: Type.Object({
|
|
11
|
-
shell_id: Type.String({ description: "Shell id or name returned by terminal_spawn, terminal_list, or shell_hints" }),
|
|
12
|
-
source: Type.Optional(Type.String({ description: "screen (default), scrollback, or both" })),
|
|
13
|
-
lines: Type.Optional(Type.Number({ description: "Number of trailing lines to return. Default 80, max 300" })),
|
|
14
|
-
}),
|
|
15
|
-
execute: async (_toolCallId, params) => {
|
|
16
|
-
const resolved = resolveShellId(shellRuntime, params.shell_id);
|
|
17
|
-
if (!resolved.ok) return toolText(`Error: ${resolved.error}`, { error: true });
|
|
18
|
-
const snapshot = shellRuntime.snapshotShell(resolved.id);
|
|
19
|
-
const source = normalizeReadSource(params.source);
|
|
20
|
-
const lineLimit = normalizeLineLimit(params.lines);
|
|
21
|
-
const text = formatTerminalRead(snapshot, { source, lines: lineLimit });
|
|
22
|
-
return toolText(text, { shell: snapshot.shell, source, lines: lineLimit, snapshot });
|
|
23
|
-
},
|
|
24
|
-
});
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
function resolveShellId(shellRuntime, ref) {
|
|
28
|
-
const value = String(ref ?? "").trim();
|
|
29
|
-
const shells = shellRuntime.listShells();
|
|
30
|
-
const shell = shells.find((shell) => shell.id === value || shell.name === value);
|
|
31
|
-
if (shell) return { ok: true, id: shell.id, shell };
|
|
32
|
-
const available = shells.length
|
|
33
|
-
? ` Active shells: ${shells.map((shell) => `${shell.id} (${shell.name})`).join(", ")}.`
|
|
34
|
-
: " No active shells.";
|
|
35
|
-
return { ok: false, error: `shell not found: ${value}.${available}` };
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
function normalizeReadSource(value) {
|
|
39
|
-
const normalized = String(value ?? "screen").trim().toLowerCase();
|
|
40
|
-
if (normalized === "scrollback") return "scrollback";
|
|
41
|
-
if (normalized === "both") return "both";
|
|
42
|
-
return "screen";
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
function normalizeLineLimit(value) {
|
|
46
|
-
const number = Math.trunc(Number(value) || 80);
|
|
47
|
-
return Math.min(300, Math.max(1, number));
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function formatTerminalRead(snapshot, { source, lines }) {
|
|
51
|
-
const shell = snapshot.shell;
|
|
52
|
-
const header = [
|
|
53
|
-
`## ${shell.name} (${shell.id})`,
|
|
54
|
-
`status: ${shell.status}`,
|
|
55
|
-
`command: ${shell.command}${shell.args?.length ? ` ${shell.args.join(" ")}` : ""}`,
|
|
56
|
-
`cwd: ${shell.cwd}`,
|
|
57
|
-
`source: ${source}`,
|
|
58
|
-
`lines: ${lines}`,
|
|
59
|
-
"",
|
|
60
|
-
].join("\n");
|
|
61
|
-
if (source === "both") {
|
|
62
|
-
return `${header}${formatReadSection("screen", snapshot.screen?.plain, lines)}\n\n${formatReadSection("scrollback", snapshot.plain, lines)}`;
|
|
63
|
-
}
|
|
64
|
-
const content = source === "scrollback" ? snapshot.plain : snapshot.screen?.plain;
|
|
65
|
-
return `${header}${lastLines(content, lines) || "(empty shell output)"}`;
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
function formatReadSection(label, text, lines) {
|
|
69
|
-
return `-- ${label} --\n${lastLines(text, lines) || "(empty shell output)"}`;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
function lastLines(text, lines) {
|
|
73
|
-
return String(text ?? "").split(/\r?\n/).slice(-lines).join("\n").trimEnd();
|
|
74
|
-
}
|
|
1
|
+
import { defineTool } from "@earendil-works/pi-coding-agent";
|
|
2
|
+
import { Type } from "typebox";
|
|
3
|
+
import { toolText } from "../agent/tool-result.mjs";
|
|
4
|
+
|
|
5
|
+
export function createTerminalReadTool(shellRuntime) {
|
|
6
|
+
return defineTool({
|
|
7
|
+
name: "terminal_read",
|
|
8
|
+
label: "Terminal Read",
|
|
9
|
+
description: "Read plain-text output from a running interactive terminal. Use this for normal shell inspection after shell hints, terminal_list, or terminal_send.",
|
|
10
|
+
parameters: Type.Object({
|
|
11
|
+
shell_id: Type.String({ description: "Shell id or name returned by terminal_spawn, terminal_list, or shell_hints" }),
|
|
12
|
+
source: Type.Optional(Type.String({ description: "screen (default), scrollback, or both" })),
|
|
13
|
+
lines: Type.Optional(Type.Number({ description: "Number of trailing lines to return. Default 80, max 300" })),
|
|
14
|
+
}),
|
|
15
|
+
execute: async (_toolCallId, params) => {
|
|
16
|
+
const resolved = resolveShellId(shellRuntime, params.shell_id);
|
|
17
|
+
if (!resolved.ok) return toolText(`Error: ${resolved.error}`, { error: true });
|
|
18
|
+
const snapshot = shellRuntime.snapshotShell(resolved.id);
|
|
19
|
+
const source = normalizeReadSource(params.source);
|
|
20
|
+
const lineLimit = normalizeLineLimit(params.lines);
|
|
21
|
+
const text = formatTerminalRead(snapshot, { source, lines: lineLimit });
|
|
22
|
+
return toolText(text, { shell: snapshot.shell, source, lines: lineLimit, snapshot });
|
|
23
|
+
},
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function resolveShellId(shellRuntime, ref) {
|
|
28
|
+
const value = String(ref ?? "").trim();
|
|
29
|
+
const shells = shellRuntime.listShells();
|
|
30
|
+
const shell = shells.find((shell) => shell.id === value || shell.name === value);
|
|
31
|
+
if (shell) return { ok: true, id: shell.id, shell };
|
|
32
|
+
const available = shells.length
|
|
33
|
+
? ` Active shells: ${shells.map((shell) => `${shell.id} (${shell.name})`).join(", ")}.`
|
|
34
|
+
: " No active shells.";
|
|
35
|
+
return { ok: false, error: `shell not found: ${value}.${available}` };
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function normalizeReadSource(value) {
|
|
39
|
+
const normalized = String(value ?? "screen").trim().toLowerCase();
|
|
40
|
+
if (normalized === "scrollback") return "scrollback";
|
|
41
|
+
if (normalized === "both") return "both";
|
|
42
|
+
return "screen";
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function normalizeLineLimit(value) {
|
|
46
|
+
const number = Math.trunc(Number(value) || 80);
|
|
47
|
+
return Math.min(300, Math.max(1, number));
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function formatTerminalRead(snapshot, { source, lines }) {
|
|
51
|
+
const shell = snapshot.shell;
|
|
52
|
+
const header = [
|
|
53
|
+
`## ${shell.name} (${shell.id})`,
|
|
54
|
+
`status: ${shell.status}`,
|
|
55
|
+
`command: ${shell.command}${shell.args?.length ? ` ${shell.args.join(" ")}` : ""}`,
|
|
56
|
+
`cwd: ${shell.cwd}`,
|
|
57
|
+
`source: ${source}`,
|
|
58
|
+
`lines: ${lines}`,
|
|
59
|
+
"",
|
|
60
|
+
].join("\n");
|
|
61
|
+
if (source === "both") {
|
|
62
|
+
return `${header}${formatReadSection("screen", snapshot.screen?.plain, lines)}\n\n${formatReadSection("scrollback", snapshot.plain, lines)}`;
|
|
63
|
+
}
|
|
64
|
+
const content = source === "scrollback" ? snapshot.plain : snapshot.screen?.plain;
|
|
65
|
+
return `${header}${lastLines(content, lines) || "(empty shell output)"}`;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function formatReadSection(label, text, lines) {
|
|
69
|
+
return `-- ${label} --\n${lastLines(text, lines) || "(empty shell output)"}`;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
function lastLines(text, lines) {
|
|
73
|
+
return String(text ?? "").split(/\r?\n/).slice(-lines).join("\n").trimEnd();
|
|
74
|
+
}
|