@aexol/spectral 0.7.1 → 0.7.5
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 +5 -0
- package/dist/agent/agents.js +1 -1
- package/dist/agent/index.js +199 -184
- package/dist/commands/serve.js +0 -3
- package/dist/designer/data/systems/renault/DESIGN.md +1 -1
- package/dist/designer/philosophies.js +668 -0
- package/dist/mcp/sampling-handler.js +1 -1
- package/dist/memory/commands/status.js +1 -1
- package/dist/memory/compaction.js +2 -2
- package/dist/memory/config.js +1 -1
- package/dist/memory/debug-log.js +1 -1
- package/dist/memory/hooks/compaction-hook.js +29 -0
- package/dist/memory/index.js +2 -0
- package/dist/memory/observer.js +2 -2
- package/dist/memory/project-observations-store.js +14 -0
- package/dist/memory/tokens.js +1 -1
- package/dist/memory/tools/read-project-observations.js +82 -0
- package/dist/memory/tools/recall-observation.js +2 -2
- package/dist/pi/agent-core/agent-loop.js +501 -0
- package/dist/pi/agent-core/agent.js +401 -0
- package/dist/pi/agent-core/harness/agent-harness.js +899 -0
- package/dist/pi/agent-core/harness/compaction/branch-summarization.js +173 -0
- package/dist/pi/agent-core/harness/compaction/compaction.js +532 -0
- package/dist/pi/agent-core/harness/compaction/utils.js +130 -0
- package/dist/pi/agent-core/harness/env/nodejs.js +485 -0
- package/dist/pi/agent-core/harness/messages.js +101 -0
- package/dist/pi/agent-core/harness/prompt-templates.js +229 -0
- package/dist/pi/agent-core/harness/session/jsonl-repo.js +100 -0
- package/dist/pi/agent-core/harness/session/jsonl-storage.js +230 -0
- package/dist/pi/agent-core/harness/session/memory-repo.js +41 -0
- package/dist/pi/agent-core/harness/session/memory-storage.js +113 -0
- package/dist/pi/agent-core/harness/session/repo-utils.js +38 -0
- package/dist/pi/agent-core/harness/session/session.js +196 -0
- package/dist/pi/agent-core/harness/session/uuid.js +49 -0
- package/dist/pi/agent-core/harness/skills.js +310 -0
- package/dist/pi/agent-core/harness/system-prompt.js +29 -0
- package/dist/pi/agent-core/harness/types.js +93 -0
- package/dist/pi/agent-core/harness/utils/shell-output.js +125 -0
- package/dist/pi/agent-core/harness/utils/truncate.js +289 -0
- package/dist/pi/agent-core/index.js +24 -0
- package/dist/pi/agent-core/node.js +2 -0
- package/dist/pi/agent-core/proxy.js +277 -0
- package/dist/pi/agent-core/types.js +1 -0
- package/dist/pi/ai/api-registry.js +43 -0
- package/dist/pi/ai/cli.js +120 -0
- package/dist/pi/ai/env-api-keys.js +169 -0
- package/dist/pi/ai/image-models.generated.js +441 -0
- package/dist/pi/ai/image-models.js +22 -0
- package/dist/pi/ai/images-api-registry.js +21 -0
- package/dist/pi/ai/images.js +13 -0
- package/dist/pi/ai/index.js +18 -0
- package/dist/pi/ai/models.generated.js +16220 -0
- package/dist/pi/ai/models.js +70 -0
- package/dist/pi/ai/oauth.js +1 -0
- package/dist/pi/ai/providers/anthropic.js +945 -0
- package/dist/pi/ai/providers/faux.js +367 -0
- package/dist/pi/ai/providers/github-copilot-headers.js +28 -0
- package/dist/pi/ai/providers/openai-completions.js +945 -0
- package/dist/pi/ai/providers/openai-prompt-cache.js +9 -0
- package/dist/pi/ai/providers/register-builtins.js +97 -0
- package/dist/pi/ai/providers/simple-options.js +40 -0
- package/dist/pi/ai/providers/transform-messages.js +183 -0
- package/dist/pi/ai/session-resources.js +21 -0
- package/dist/pi/ai/stream.js +26 -0
- package/dist/pi/ai/types.js +1 -0
- package/dist/pi/ai/utils/diagnostics.js +24 -0
- package/dist/pi/ai/utils/event-stream.js +80 -0
- package/dist/pi/ai/utils/hash.js +13 -0
- package/dist/pi/ai/utils/headers.js +7 -0
- package/dist/pi/ai/utils/json-parse.js +112 -0
- package/dist/pi/ai/utils/node-http-proxy.js +96 -0
- package/dist/pi/ai/utils/oauth/anthropic.js +334 -0
- package/dist/pi/ai/utils/oauth/device-code.js +54 -0
- package/dist/pi/ai/utils/oauth/github-copilot.js +270 -0
- package/dist/pi/ai/utils/oauth/index.js +121 -0
- package/dist/pi/ai/utils/oauth/oauth-page.js +104 -0
- package/dist/pi/ai/utils/oauth/openai-codex.js +384 -0
- package/dist/pi/ai/utils/oauth/pkce.js +30 -0
- package/dist/pi/ai/utils/oauth/types.js +1 -0
- package/dist/pi/ai/utils/overflow.js +150 -0
- package/dist/pi/ai/utils/sanitize-unicode.js +25 -0
- package/dist/pi/ai/utils/typebox-helpers.js +20 -0
- package/dist/pi/ai/utils/validation.js +280 -0
- package/dist/pi/coding-agent/bun/cli.js +7 -0
- package/dist/pi/coding-agent/bun/restore-sandbox-env.js +31 -0
- package/dist/pi/coding-agent/cli/args.js +340 -0
- package/dist/pi/coding-agent/cli/file-processor.js +82 -0
- package/dist/pi/coding-agent/cli/initial-message.js +21 -0
- package/dist/pi/coding-agent/cli.js +17 -0
- package/dist/pi/coding-agent/config.js +414 -0
- package/dist/pi/coding-agent/core/agent-session-runtime.js +299 -0
- package/dist/pi/coding-agent/core/agent-session-services.js +117 -0
- package/dist/pi/coding-agent/core/agent-session.js +2498 -0
- package/dist/pi/coding-agent/core/auth-guidance.js +20 -0
- package/dist/pi/coding-agent/core/auth-storage.js +441 -0
- package/dist/pi/coding-agent/core/bash-executor.js +110 -0
- package/dist/pi/coding-agent/core/compaction/branch-summarization.js +242 -0
- package/dist/pi/coding-agent/core/compaction/compaction.js +624 -0
- package/dist/pi/coding-agent/core/compaction/index.js +6 -0
- package/dist/pi/coding-agent/core/compaction/utils.js +152 -0
- package/dist/pi/coding-agent/core/defaults.js +1 -0
- package/dist/pi/coding-agent/core/diagnostics.js +1 -0
- package/dist/pi/coding-agent/core/event-bus.js +24 -0
- package/dist/pi/coding-agent/core/exec.js +74 -0
- package/dist/pi/coding-agent/core/export-html/ansi-to-html.js +248 -0
- package/dist/pi/coding-agent/core/export-html/index.js +225 -0
- package/dist/pi/coding-agent/core/export-html/tool-renderer.js +107 -0
- package/dist/pi/coding-agent/core/extensions/index.js +8 -0
- package/dist/pi/coding-agent/core/extensions/loader.js +485 -0
- package/dist/pi/coding-agent/core/extensions/runner.js +824 -0
- package/dist/pi/coding-agent/core/extensions/types.js +44 -0
- package/dist/pi/coding-agent/core/extensions/wrapper.js +21 -0
- package/dist/pi/coding-agent/core/footer-data-provider.js +309 -0
- package/dist/pi/coding-agent/core/http-dispatcher.js +47 -0
- package/dist/pi/coding-agent/core/index.js +11 -0
- package/dist/pi/coding-agent/core/keybindings.js +294 -0
- package/dist/pi/coding-agent/core/messages.js +122 -0
- package/dist/pi/coding-agent/core/model-registry.js +728 -0
- package/dist/pi/coding-agent/core/model-resolver.js +494 -0
- package/dist/pi/coding-agent/core/output-guard.js +58 -0
- package/dist/pi/coding-agent/core/package-manager.js +2020 -0
- package/dist/pi/coding-agent/core/prompt-templates.js +237 -0
- package/dist/pi/coding-agent/core/provider-display-names.js +32 -0
- package/dist/pi/coding-agent/core/resolve-config-value.js +125 -0
- package/dist/pi/coding-agent/core/resource-loader.js +733 -0
- package/dist/pi/coding-agent/core/sdk.js +282 -0
- package/dist/pi/coding-agent/core/session-cwd.js +37 -0
- package/dist/pi/coding-agent/core/session-manager.js +1146 -0
- package/dist/pi/coding-agent/core/settings-manager.js +794 -0
- package/dist/pi/coding-agent/core/skills.js +386 -0
- package/dist/pi/coding-agent/core/slash-commands.js +24 -0
- package/dist/pi/coding-agent/core/source-info.js +18 -0
- package/dist/pi/coding-agent/core/system-prompt.js +122 -0
- package/dist/pi/coding-agent/core/telemetry.js +8 -0
- package/dist/pi/coding-agent/core/timings.js +30 -0
- package/dist/pi/coding-agent/core/tools/bash.js +341 -0
- package/dist/pi/coding-agent/core/tools/edit-diff.js +344 -0
- package/dist/pi/coding-agent/core/tools/edit.js +324 -0
- package/dist/pi/coding-agent/core/tools/file-mutation-queue.js +36 -0
- package/dist/pi/coding-agent/core/tools/find.js +297 -0
- package/dist/pi/coding-agent/core/tools/grep.js +303 -0
- package/dist/pi/coding-agent/core/tools/index.js +111 -0
- package/dist/pi/coding-agent/core/tools/ls.js +168 -0
- package/dist/pi/coding-agent/core/tools/output-accumulator.js +183 -0
- package/dist/pi/coding-agent/core/tools/path-utils.js +61 -0
- package/dist/pi/coding-agent/core/tools/read.js +288 -0
- package/dist/pi/coding-agent/core/tools/render-utils.js +48 -0
- package/dist/pi/coding-agent/core/tools/tool-definition-wrapper.js +33 -0
- package/dist/pi/coding-agent/core/tools/truncate.js +214 -0
- package/dist/pi/coding-agent/core/tools/write.js +212 -0
- package/dist/pi/coding-agent/index.js +41 -0
- package/dist/pi/coding-agent/main.js +5 -0
- package/dist/pi/coding-agent/migrations.js +280 -0
- package/dist/pi/coding-agent/modes/index.js +7 -0
- package/dist/pi/coding-agent/modes/interactive/components/diff.js +132 -0
- package/dist/pi/coding-agent/modes/interactive/components/keybinding-hints.js +35 -0
- package/dist/pi/coding-agent/modes/interactive/components/visual-truncate.js +32 -0
- package/dist/pi/coding-agent/modes/interactive/interactive-mode.js +3 -0
- package/dist/pi/coding-agent/modes/interactive/theme/theme.js +1023 -0
- package/dist/pi/coding-agent/modes/print-mode.js +130 -0
- package/dist/pi/coding-agent/modes/rpc/jsonl.js +48 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-client.js +409 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-mode.js +600 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-types.js +7 -0
- package/dist/pi/coding-agent/utils/ansi.js +51 -0
- package/dist/pi/coding-agent/utils/changelog.js +86 -0
- package/dist/pi/coding-agent/utils/child-process.js +87 -0
- package/dist/pi/coding-agent/utils/clipboard-image.js +244 -0
- package/dist/pi/coding-agent/utils/clipboard-native.js +13 -0
- package/dist/pi/coding-agent/utils/clipboard.js +116 -0
- package/dist/pi/coding-agent/utils/exif-orientation.js +157 -0
- package/dist/pi/coding-agent/utils/frontmatter.js +25 -0
- package/dist/pi/coding-agent/utils/fs-watch.js +24 -0
- package/dist/pi/coding-agent/utils/git.js +162 -0
- package/dist/pi/coding-agent/utils/html.js +39 -0
- package/dist/pi/coding-agent/utils/image-convert.js +38 -0
- package/dist/pi/coding-agent/utils/image-resize.js +136 -0
- package/dist/pi/coding-agent/utils/mime.js +68 -0
- package/dist/pi/coding-agent/utils/paths.js +91 -0
- package/dist/pi/coding-agent/utils/photon.js +120 -0
- package/dist/pi/coding-agent/utils/pi-user-agent.js +4 -0
- package/dist/pi/coding-agent/utils/shell.js +194 -0
- package/dist/pi/coding-agent/utils/sleep.js +16 -0
- package/dist/pi/coding-agent/utils/syntax-highlight.js +117 -0
- package/dist/pi/coding-agent/utils/tools-manager.js +327 -0
- package/dist/pi/coding-agent/utils/version-check.js +81 -0
- package/dist/pi/coding-agent/utils/windows-self-update.js +76 -0
- package/dist/pi/tui/autocomplete.js +631 -0
- package/dist/pi/tui/components/box.js +103 -0
- package/dist/pi/tui/components/cancellable-loader.js +34 -0
- package/dist/pi/tui/components/editor.js +1915 -0
- package/dist/pi/tui/components/image.js +88 -0
- package/dist/pi/tui/components/input.js +425 -0
- package/dist/pi/tui/components/loader.js +68 -0
- package/dist/pi/tui/components/markdown.js +633 -0
- package/dist/pi/tui/components/select-list.js +158 -0
- package/dist/pi/tui/components/settings-list.js +184 -0
- package/dist/pi/tui/components/spacer.js +22 -0
- package/dist/pi/tui/components/text.js +88 -0
- package/dist/pi/tui/components/truncated-text.js +50 -0
- package/dist/pi/tui/editor-component.js +1 -0
- package/dist/pi/tui/fuzzy.js +109 -0
- package/dist/pi/tui/index.js +31 -0
- package/dist/pi/tui/keybindings.js +173 -0
- package/dist/pi/tui/keys.js +1172 -0
- package/dist/pi/tui/kill-ring.js +43 -0
- package/dist/pi/tui/stdin-buffer.js +360 -0
- package/dist/pi/tui/terminal-image.js +335 -0
- package/dist/pi/tui/terminal.js +324 -0
- package/dist/pi/tui/tui.js +1076 -0
- package/dist/pi/tui/undo-stack.js +24 -0
- package/dist/pi/tui/utils.js +1016 -0
- package/dist/relay/dispatcher.js +30 -0
- package/dist/server/handlers/queue.js +52 -0
- package/dist/server/pi-bridge.js +9 -1
- package/dist/server/session-stream.js +76 -111
- package/dist/server/storage.js +154 -2
- package/dist/server/title-generator.js +14 -153
- package/package.json +24 -6
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Print mode (single-shot): Send prompts, output result, exit.
|
|
3
|
+
*
|
|
4
|
+
* Used for:
|
|
5
|
+
* - `pi -p "prompt"` - text output
|
|
6
|
+
* - `pi --mode json "prompt"` - JSON event stream
|
|
7
|
+
*/
|
|
8
|
+
import { flushRawStdout, writeRawStdout } from "../core/output-guard.js";
|
|
9
|
+
import { killTrackedDetachedChildren } from "../utils/shell.js";
|
|
10
|
+
/**
|
|
11
|
+
* Run in print (single-shot) mode.
|
|
12
|
+
* Sends prompts to the agent and outputs the result.
|
|
13
|
+
*/
|
|
14
|
+
export async function runPrintMode(runtimeHost, options) {
|
|
15
|
+
const { mode, messages = [], initialMessage, initialImages } = options;
|
|
16
|
+
let exitCode = 0;
|
|
17
|
+
let session = runtimeHost.session;
|
|
18
|
+
let unsubscribe;
|
|
19
|
+
let disposed = false;
|
|
20
|
+
const signalCleanupHandlers = [];
|
|
21
|
+
const disposeRuntime = async () => {
|
|
22
|
+
if (disposed)
|
|
23
|
+
return;
|
|
24
|
+
disposed = true;
|
|
25
|
+
unsubscribe?.();
|
|
26
|
+
await runtimeHost.dispose();
|
|
27
|
+
};
|
|
28
|
+
const registerSignalHandlers = () => {
|
|
29
|
+
const signals = ["SIGTERM"];
|
|
30
|
+
if (process.platform !== "win32") {
|
|
31
|
+
signals.push("SIGHUP");
|
|
32
|
+
}
|
|
33
|
+
for (const signal of signals) {
|
|
34
|
+
const handler = () => {
|
|
35
|
+
killTrackedDetachedChildren();
|
|
36
|
+
void disposeRuntime().finally(() => {
|
|
37
|
+
process.exit(signal === "SIGHUP" ? 129 : 143);
|
|
38
|
+
});
|
|
39
|
+
};
|
|
40
|
+
process.on(signal, handler);
|
|
41
|
+
signalCleanupHandlers.push(() => process.off(signal, handler));
|
|
42
|
+
}
|
|
43
|
+
};
|
|
44
|
+
registerSignalHandlers();
|
|
45
|
+
runtimeHost.setRebindSession(async () => {
|
|
46
|
+
await rebindSession();
|
|
47
|
+
});
|
|
48
|
+
const rebindSession = async () => {
|
|
49
|
+
session = runtimeHost.session;
|
|
50
|
+
await session.bindExtensions({
|
|
51
|
+
commandContextActions: {
|
|
52
|
+
waitForIdle: () => session.agent.waitForIdle(),
|
|
53
|
+
newSession: async (newSessionOptions) => runtimeHost.newSession(newSessionOptions),
|
|
54
|
+
fork: async (entryId, forkOptions) => {
|
|
55
|
+
const result = await runtimeHost.fork(entryId, forkOptions);
|
|
56
|
+
return { cancelled: result.cancelled };
|
|
57
|
+
},
|
|
58
|
+
navigateTree: async (targetId, navigateOptions) => {
|
|
59
|
+
const result = await session.navigateTree(targetId, {
|
|
60
|
+
summarize: navigateOptions?.summarize,
|
|
61
|
+
customInstructions: navigateOptions?.customInstructions,
|
|
62
|
+
replaceInstructions: navigateOptions?.replaceInstructions,
|
|
63
|
+
label: navigateOptions?.label,
|
|
64
|
+
});
|
|
65
|
+
return { cancelled: result.cancelled };
|
|
66
|
+
},
|
|
67
|
+
switchSession: async (sessionPath, switchOptions) => {
|
|
68
|
+
return runtimeHost.switchSession(sessionPath, switchOptions);
|
|
69
|
+
},
|
|
70
|
+
reload: async () => {
|
|
71
|
+
await session.reload();
|
|
72
|
+
},
|
|
73
|
+
},
|
|
74
|
+
onError: (err) => {
|
|
75
|
+
console.error(`Extension error (${err.extensionPath}): ${err.error}`);
|
|
76
|
+
},
|
|
77
|
+
});
|
|
78
|
+
unsubscribe?.();
|
|
79
|
+
unsubscribe = session.subscribe((event) => {
|
|
80
|
+
if (mode === "json") {
|
|
81
|
+
writeRawStdout(`${JSON.stringify(event)}\n`);
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
};
|
|
85
|
+
try {
|
|
86
|
+
if (mode === "json") {
|
|
87
|
+
const header = session.sessionManager.getHeader();
|
|
88
|
+
if (header) {
|
|
89
|
+
writeRawStdout(`${JSON.stringify(header)}\n`);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
await rebindSession();
|
|
93
|
+
if (initialMessage) {
|
|
94
|
+
await session.prompt(initialMessage, { images: initialImages });
|
|
95
|
+
}
|
|
96
|
+
for (const message of messages) {
|
|
97
|
+
await session.prompt(message);
|
|
98
|
+
}
|
|
99
|
+
if (mode === "text") {
|
|
100
|
+
const state = session.state;
|
|
101
|
+
const lastMessage = state.messages[state.messages.length - 1];
|
|
102
|
+
if (lastMessage?.role === "assistant") {
|
|
103
|
+
const assistantMsg = lastMessage;
|
|
104
|
+
if (assistantMsg.stopReason === "error" || assistantMsg.stopReason === "aborted") {
|
|
105
|
+
console.error(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);
|
|
106
|
+
exitCode = 1;
|
|
107
|
+
}
|
|
108
|
+
else {
|
|
109
|
+
for (const content of assistantMsg.content) {
|
|
110
|
+
if (content.type === "text") {
|
|
111
|
+
writeRawStdout(`${content.text}\n`);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
return exitCode;
|
|
118
|
+
}
|
|
119
|
+
catch (error) {
|
|
120
|
+
console.error(error instanceof Error ? error.message : String(error));
|
|
121
|
+
return 1;
|
|
122
|
+
}
|
|
123
|
+
finally {
|
|
124
|
+
for (const cleanup of signalCleanupHandlers) {
|
|
125
|
+
cleanup();
|
|
126
|
+
}
|
|
127
|
+
await disposeRuntime();
|
|
128
|
+
await flushRawStdout();
|
|
129
|
+
}
|
|
130
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { StringDecoder } from "node:string_decoder";
|
|
2
|
+
/**
|
|
3
|
+
* Serialize a single strict JSONL record.
|
|
4
|
+
*
|
|
5
|
+
* Framing is LF-only. Payload strings may contain other Unicode separators such as
|
|
6
|
+
* U+2028 and U+2029. Clients must split records on `\n` only.
|
|
7
|
+
*/
|
|
8
|
+
export function serializeJsonLine(value) {
|
|
9
|
+
return `${JSON.stringify(value)}\n`;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Attach an LF-only JSONL reader to a stream.
|
|
13
|
+
*
|
|
14
|
+
* This intentionally does not use Node readline. Readline splits on additional
|
|
15
|
+
* Unicode separators that are valid inside JSON strings and therefore does not
|
|
16
|
+
* implement strict JSONL framing.
|
|
17
|
+
*/
|
|
18
|
+
export function attachJsonlLineReader(stream, onLine) {
|
|
19
|
+
const decoder = new StringDecoder("utf8");
|
|
20
|
+
let buffer = "";
|
|
21
|
+
const emitLine = (line) => {
|
|
22
|
+
onLine(line.endsWith("\r") ? line.slice(0, -1) : line);
|
|
23
|
+
};
|
|
24
|
+
const onData = (chunk) => {
|
|
25
|
+
buffer += typeof chunk === "string" ? chunk : decoder.write(chunk);
|
|
26
|
+
while (true) {
|
|
27
|
+
const newlineIndex = buffer.indexOf("\n");
|
|
28
|
+
if (newlineIndex === -1) {
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
emitLine(buffer.slice(0, newlineIndex));
|
|
32
|
+
buffer = buffer.slice(newlineIndex + 1);
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
const onEnd = () => {
|
|
36
|
+
buffer += decoder.end();
|
|
37
|
+
if (buffer.length > 0) {
|
|
38
|
+
emitLine(buffer);
|
|
39
|
+
buffer = "";
|
|
40
|
+
}
|
|
41
|
+
};
|
|
42
|
+
stream.on("data", onData);
|
|
43
|
+
stream.on("end", onEnd);
|
|
44
|
+
return () => {
|
|
45
|
+
stream.off("data", onData);
|
|
46
|
+
stream.off("end", onEnd);
|
|
47
|
+
};
|
|
48
|
+
}
|
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* RPC Client for programmatic access to the coding agent.
|
|
3
|
+
*
|
|
4
|
+
* Spawns the agent in RPC mode and provides a typed API for all operations.
|
|
5
|
+
*/
|
|
6
|
+
import { spawn } from "node:child_process";
|
|
7
|
+
import { attachJsonlLineReader, serializeJsonLine } from "./jsonl.js";
|
|
8
|
+
// ============================================================================
|
|
9
|
+
// RPC Client
|
|
10
|
+
// ============================================================================
|
|
11
|
+
export class RpcClient {
|
|
12
|
+
process = null;
|
|
13
|
+
stopReadingStdout = null;
|
|
14
|
+
eventListeners = [];
|
|
15
|
+
pendingRequests = new Map();
|
|
16
|
+
requestId = 0;
|
|
17
|
+
stderr = "";
|
|
18
|
+
options;
|
|
19
|
+
constructor(options = {}) {
|
|
20
|
+
this.options = options;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Start the RPC agent process.
|
|
24
|
+
*/
|
|
25
|
+
async start() {
|
|
26
|
+
if (this.process) {
|
|
27
|
+
throw new Error("Client already started");
|
|
28
|
+
}
|
|
29
|
+
const cliPath = this.options.cliPath ?? "dist/cli.js";
|
|
30
|
+
const args = ["--mode", "rpc"];
|
|
31
|
+
if (this.options.provider) {
|
|
32
|
+
args.push("--provider", this.options.provider);
|
|
33
|
+
}
|
|
34
|
+
if (this.options.model) {
|
|
35
|
+
args.push("--model", this.options.model);
|
|
36
|
+
}
|
|
37
|
+
if (this.options.args) {
|
|
38
|
+
args.push(...this.options.args);
|
|
39
|
+
}
|
|
40
|
+
this.process = spawn("node", [cliPath, ...args], {
|
|
41
|
+
cwd: this.options.cwd,
|
|
42
|
+
env: { ...process.env, ...this.options.env },
|
|
43
|
+
stdio: ["pipe", "pipe", "pipe"],
|
|
44
|
+
});
|
|
45
|
+
// Collect stderr for debugging
|
|
46
|
+
this.process.stderr?.on("data", (data) => {
|
|
47
|
+
this.stderr += data.toString();
|
|
48
|
+
process.stderr.write(data);
|
|
49
|
+
});
|
|
50
|
+
// Set up strict JSONL reader for stdout.
|
|
51
|
+
this.stopReadingStdout = attachJsonlLineReader(this.process.stdout, (line) => {
|
|
52
|
+
this.handleLine(line);
|
|
53
|
+
});
|
|
54
|
+
// Wait a moment for process to initialize
|
|
55
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
56
|
+
if (this.process.exitCode !== null) {
|
|
57
|
+
throw new Error(`Agent process exited immediately with code ${this.process.exitCode}. Stderr: ${this.stderr}`);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Stop the RPC agent process.
|
|
62
|
+
*/
|
|
63
|
+
async stop() {
|
|
64
|
+
if (!this.process)
|
|
65
|
+
return;
|
|
66
|
+
this.stopReadingStdout?.();
|
|
67
|
+
this.stopReadingStdout = null;
|
|
68
|
+
this.process.kill("SIGTERM");
|
|
69
|
+
// Wait for process to exit
|
|
70
|
+
await new Promise((resolve) => {
|
|
71
|
+
const timeout = setTimeout(() => {
|
|
72
|
+
this.process?.kill("SIGKILL");
|
|
73
|
+
resolve();
|
|
74
|
+
}, 1000);
|
|
75
|
+
this.process?.on("exit", () => {
|
|
76
|
+
clearTimeout(timeout);
|
|
77
|
+
resolve();
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
this.process = null;
|
|
81
|
+
this.pendingRequests.clear();
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Subscribe to agent events.
|
|
85
|
+
*/
|
|
86
|
+
onEvent(listener) {
|
|
87
|
+
this.eventListeners.push(listener);
|
|
88
|
+
return () => {
|
|
89
|
+
const index = this.eventListeners.indexOf(listener);
|
|
90
|
+
if (index !== -1) {
|
|
91
|
+
this.eventListeners.splice(index, 1);
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Get collected stderr output (useful for debugging).
|
|
97
|
+
*/
|
|
98
|
+
getStderr() {
|
|
99
|
+
return this.stderr;
|
|
100
|
+
}
|
|
101
|
+
// =========================================================================
|
|
102
|
+
// Command Methods
|
|
103
|
+
// =========================================================================
|
|
104
|
+
/**
|
|
105
|
+
* Send a prompt to the agent.
|
|
106
|
+
* Returns immediately after sending; use onEvent() to receive streaming events.
|
|
107
|
+
* Use waitForIdle() to wait for completion.
|
|
108
|
+
*/
|
|
109
|
+
async prompt(message, images) {
|
|
110
|
+
await this.send({ type: "prompt", message, images });
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Queue a steering message to interrupt the agent mid-run.
|
|
114
|
+
*/
|
|
115
|
+
async steer(message, images) {
|
|
116
|
+
await this.send({ type: "steer", message, images });
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Queue a follow-up message to be processed after the agent finishes.
|
|
120
|
+
*/
|
|
121
|
+
async followUp(message, images) {
|
|
122
|
+
await this.send({ type: "follow_up", message, images });
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Abort current operation.
|
|
126
|
+
*/
|
|
127
|
+
async abort() {
|
|
128
|
+
await this.send({ type: "abort" });
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Start a new session, optionally with parent tracking.
|
|
132
|
+
* @param parentSession - Optional parent session path for lineage tracking
|
|
133
|
+
* @returns Object with `cancelled: true` if an extension cancelled the new session
|
|
134
|
+
*/
|
|
135
|
+
async newSession(parentSession) {
|
|
136
|
+
const response = await this.send({ type: "new_session", parentSession });
|
|
137
|
+
return this.getData(response);
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Get current session state.
|
|
141
|
+
*/
|
|
142
|
+
async getState() {
|
|
143
|
+
const response = await this.send({ type: "get_state" });
|
|
144
|
+
return this.getData(response);
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Set model by provider and ID.
|
|
148
|
+
*/
|
|
149
|
+
async setModel(provider, modelId) {
|
|
150
|
+
const response = await this.send({ type: "set_model", provider, modelId });
|
|
151
|
+
return this.getData(response);
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Cycle to next model.
|
|
155
|
+
*/
|
|
156
|
+
async cycleModel() {
|
|
157
|
+
const response = await this.send({ type: "cycle_model" });
|
|
158
|
+
return this.getData(response);
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Get list of available models.
|
|
162
|
+
*/
|
|
163
|
+
async getAvailableModels() {
|
|
164
|
+
const response = await this.send({ type: "get_available_models" });
|
|
165
|
+
return this.getData(response).models;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Set thinking level.
|
|
169
|
+
*/
|
|
170
|
+
async setThinkingLevel(level) {
|
|
171
|
+
await this.send({ type: "set_thinking_level", level });
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Cycle thinking level.
|
|
175
|
+
*/
|
|
176
|
+
async cycleThinkingLevel() {
|
|
177
|
+
const response = await this.send({ type: "cycle_thinking_level" });
|
|
178
|
+
return this.getData(response);
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Set steering mode.
|
|
182
|
+
*/
|
|
183
|
+
async setSteeringMode(mode) {
|
|
184
|
+
await this.send({ type: "set_steering_mode", mode });
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Set follow-up mode.
|
|
188
|
+
*/
|
|
189
|
+
async setFollowUpMode(mode) {
|
|
190
|
+
await this.send({ type: "set_follow_up_mode", mode });
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* Compact session context.
|
|
194
|
+
*/
|
|
195
|
+
async compact(customInstructions) {
|
|
196
|
+
const response = await this.send({ type: "compact", customInstructions });
|
|
197
|
+
return this.getData(response);
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Set auto-compaction enabled/disabled.
|
|
201
|
+
*/
|
|
202
|
+
async setAutoCompaction(enabled) {
|
|
203
|
+
await this.send({ type: "set_auto_compaction", enabled });
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Set auto-retry enabled/disabled.
|
|
207
|
+
*/
|
|
208
|
+
async setAutoRetry(enabled) {
|
|
209
|
+
await this.send({ type: "set_auto_retry", enabled });
|
|
210
|
+
}
|
|
211
|
+
/**
|
|
212
|
+
* Abort in-progress retry.
|
|
213
|
+
*/
|
|
214
|
+
async abortRetry() {
|
|
215
|
+
await this.send({ type: "abort_retry" });
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Execute a bash command.
|
|
219
|
+
*/
|
|
220
|
+
async bash(command) {
|
|
221
|
+
const response = await this.send({ type: "bash", command });
|
|
222
|
+
return this.getData(response);
|
|
223
|
+
}
|
|
224
|
+
/**
|
|
225
|
+
* Abort running bash command.
|
|
226
|
+
*/
|
|
227
|
+
async abortBash() {
|
|
228
|
+
await this.send({ type: "abort_bash" });
|
|
229
|
+
}
|
|
230
|
+
/**
|
|
231
|
+
* Get session statistics.
|
|
232
|
+
*/
|
|
233
|
+
async getSessionStats() {
|
|
234
|
+
const response = await this.send({ type: "get_session_stats" });
|
|
235
|
+
return this.getData(response);
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* Export session to HTML.
|
|
239
|
+
*/
|
|
240
|
+
async exportHtml(outputPath) {
|
|
241
|
+
const response = await this.send({ type: "export_html", outputPath });
|
|
242
|
+
return this.getData(response);
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* Switch to a different session file.
|
|
246
|
+
* @returns Object with `cancelled: true` if an extension cancelled the switch
|
|
247
|
+
*/
|
|
248
|
+
async switchSession(sessionPath) {
|
|
249
|
+
const response = await this.send({ type: "switch_session", sessionPath });
|
|
250
|
+
return this.getData(response);
|
|
251
|
+
}
|
|
252
|
+
/**
|
|
253
|
+
* Fork from a specific message.
|
|
254
|
+
* @returns Object with `text` (the message text) and `cancelled` (if extension cancelled)
|
|
255
|
+
*/
|
|
256
|
+
async fork(entryId) {
|
|
257
|
+
const response = await this.send({ type: "fork", entryId });
|
|
258
|
+
return this.getData(response);
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Clone the current active branch into a new session.
|
|
262
|
+
* @returns Object with `cancelled: true` if an extension cancelled the clone
|
|
263
|
+
*/
|
|
264
|
+
async clone() {
|
|
265
|
+
const response = await this.send({ type: "clone" });
|
|
266
|
+
return this.getData(response);
|
|
267
|
+
}
|
|
268
|
+
/**
|
|
269
|
+
* Get messages available for forking.
|
|
270
|
+
*/
|
|
271
|
+
async getForkMessages() {
|
|
272
|
+
const response = await this.send({ type: "get_fork_messages" });
|
|
273
|
+
return this.getData(response).messages;
|
|
274
|
+
}
|
|
275
|
+
/**
|
|
276
|
+
* Get text of last assistant message.
|
|
277
|
+
*/
|
|
278
|
+
async getLastAssistantText() {
|
|
279
|
+
const response = await this.send({ type: "get_last_assistant_text" });
|
|
280
|
+
return this.getData(response).text;
|
|
281
|
+
}
|
|
282
|
+
/**
|
|
283
|
+
* Set the session display name.
|
|
284
|
+
*/
|
|
285
|
+
async setSessionName(name) {
|
|
286
|
+
await this.send({ type: "set_session_name", name });
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* Get all messages in the session.
|
|
290
|
+
*/
|
|
291
|
+
async getMessages() {
|
|
292
|
+
const response = await this.send({ type: "get_messages" });
|
|
293
|
+
return this.getData(response).messages;
|
|
294
|
+
}
|
|
295
|
+
/**
|
|
296
|
+
* Get available commands (extension commands, prompt templates, skills).
|
|
297
|
+
*/
|
|
298
|
+
async getCommands() {
|
|
299
|
+
const response = await this.send({ type: "get_commands" });
|
|
300
|
+
return this.getData(response).commands;
|
|
301
|
+
}
|
|
302
|
+
// =========================================================================
|
|
303
|
+
// Helpers
|
|
304
|
+
// =========================================================================
|
|
305
|
+
/**
|
|
306
|
+
* Wait for agent to become idle (no streaming).
|
|
307
|
+
* Resolves when agent_end event is received.
|
|
308
|
+
*/
|
|
309
|
+
waitForIdle(timeout = 60000) {
|
|
310
|
+
return new Promise((resolve, reject) => {
|
|
311
|
+
const timer = setTimeout(() => {
|
|
312
|
+
unsubscribe();
|
|
313
|
+
reject(new Error(`Timeout waiting for agent to become idle. Stderr: ${this.stderr}`));
|
|
314
|
+
}, timeout);
|
|
315
|
+
const unsubscribe = this.onEvent((event) => {
|
|
316
|
+
if (event.type === "agent_end") {
|
|
317
|
+
clearTimeout(timer);
|
|
318
|
+
unsubscribe();
|
|
319
|
+
resolve();
|
|
320
|
+
}
|
|
321
|
+
});
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
/**
|
|
325
|
+
* Collect events until agent becomes idle.
|
|
326
|
+
*/
|
|
327
|
+
collectEvents(timeout = 60000) {
|
|
328
|
+
return new Promise((resolve, reject) => {
|
|
329
|
+
const events = [];
|
|
330
|
+
const timer = setTimeout(() => {
|
|
331
|
+
unsubscribe();
|
|
332
|
+
reject(new Error(`Timeout collecting events. Stderr: ${this.stderr}`));
|
|
333
|
+
}, timeout);
|
|
334
|
+
const unsubscribe = this.onEvent((event) => {
|
|
335
|
+
events.push(event);
|
|
336
|
+
if (event.type === "agent_end") {
|
|
337
|
+
clearTimeout(timer);
|
|
338
|
+
unsubscribe();
|
|
339
|
+
resolve(events);
|
|
340
|
+
}
|
|
341
|
+
});
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
/**
|
|
345
|
+
* Send prompt and wait for completion, returning all events.
|
|
346
|
+
*/
|
|
347
|
+
async promptAndWait(message, images, timeout = 60000) {
|
|
348
|
+
const eventsPromise = this.collectEvents(timeout);
|
|
349
|
+
await this.prompt(message, images);
|
|
350
|
+
return eventsPromise;
|
|
351
|
+
}
|
|
352
|
+
// =========================================================================
|
|
353
|
+
// Internal
|
|
354
|
+
// =========================================================================
|
|
355
|
+
handleLine(line) {
|
|
356
|
+
try {
|
|
357
|
+
const data = JSON.parse(line);
|
|
358
|
+
// Check if it's a response to a pending request
|
|
359
|
+
if (data.type === "response" && data.id && this.pendingRequests.has(data.id)) {
|
|
360
|
+
const pending = this.pendingRequests.get(data.id);
|
|
361
|
+
this.pendingRequests.delete(data.id);
|
|
362
|
+
pending.resolve(data);
|
|
363
|
+
return;
|
|
364
|
+
}
|
|
365
|
+
// Otherwise it's an event
|
|
366
|
+
for (const listener of this.eventListeners) {
|
|
367
|
+
listener(data);
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
catch {
|
|
371
|
+
// Ignore non-JSON lines
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
async send(command) {
|
|
375
|
+
if (!this.process?.stdin) {
|
|
376
|
+
throw new Error("Client not started");
|
|
377
|
+
}
|
|
378
|
+
const id = `req_${++this.requestId}`;
|
|
379
|
+
const fullCommand = { ...command, id };
|
|
380
|
+
return new Promise((resolve, reject) => {
|
|
381
|
+
this.pendingRequests.set(id, { resolve, reject });
|
|
382
|
+
const timeout = setTimeout(() => {
|
|
383
|
+
this.pendingRequests.delete(id);
|
|
384
|
+
reject(new Error(`Timeout waiting for response to ${command.type}. Stderr: ${this.stderr}`));
|
|
385
|
+
}, 30000);
|
|
386
|
+
this.pendingRequests.set(id, {
|
|
387
|
+
resolve: (response) => {
|
|
388
|
+
clearTimeout(timeout);
|
|
389
|
+
resolve(response);
|
|
390
|
+
},
|
|
391
|
+
reject: (error) => {
|
|
392
|
+
clearTimeout(timeout);
|
|
393
|
+
reject(error);
|
|
394
|
+
},
|
|
395
|
+
});
|
|
396
|
+
this.process.stdin.write(serializeJsonLine(fullCommand));
|
|
397
|
+
});
|
|
398
|
+
}
|
|
399
|
+
getData(response) {
|
|
400
|
+
if (!response.success) {
|
|
401
|
+
const errorResponse = response;
|
|
402
|
+
throw new Error(errorResponse.error);
|
|
403
|
+
}
|
|
404
|
+
// Type assertion: we trust response.data matches T based on the command sent.
|
|
405
|
+
// This is safe because each public method specifies the correct T for its command.
|
|
406
|
+
const successResponse = response;
|
|
407
|
+
return successResponse.data;
|
|
408
|
+
}
|
|
409
|
+
}
|