@code-yeongyu/senpi 2026.6.3 → 2026.6.4
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 +19 -9
- package/README.md +2 -0
- package/dist/bun/cli.d.ts.map +1 -1
- package/dist/bun/cli.js +1 -1
- package/dist/bun/cli.js.map +1 -1
- package/dist/cli/args.d.ts.map +1 -1
- package/dist/cli/args.js +2 -0
- package/dist/cli/args.js.map +1 -1
- package/dist/cli-main.d.ts +3 -0
- package/dist/cli-main.d.ts.map +1 -0
- package/dist/cli-main.js +10 -0
- package/dist/cli-main.js.map +1 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +46 -13
- package/dist/cli.js.map +1 -1
- package/dist/core/auth-storage.d.ts.map +1 -1
- package/dist/core/auth-storage.js +4 -3
- package/dist/core/auth-storage.js.map +1 -1
- package/dist/core/export-html/template.js +19 -6
- package/dist/core/model-resolver.d.ts.map +1 -1
- package/dist/core/model-resolver.js +2 -0
- package/dist/core/model-resolver.js.map +1 -1
- package/dist/core/package-manager.d.ts +2 -0
- package/dist/core/package-manager.d.ts.map +1 -1
- package/dist/core/package-manager.js +22 -6
- package/dist/core/package-manager.js.map +1 -1
- package/dist/core/provider-display-names.d.ts.map +1 -1
- package/dist/core/provider-display-names.js +2 -0
- package/dist/core/provider-display-names.js.map +1 -1
- package/dist/modes/interactive/components/login-dialog.d.ts +0 -1
- package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
- package/dist/modes/interactive/components/login-dialog.js +3 -12
- package/dist/modes/interactive/components/login-dialog.js.map +1 -1
- package/dist/self-update-bootstrap.d.ts +19 -0
- package/dist/self-update-bootstrap.d.ts.map +1 -0
- package/dist/self-update-bootstrap.js +160 -0
- package/dist/self-update-bootstrap.js.map +1 -0
- package/dist/senpi +46 -13
- package/dist/utils/git.d.ts.map +1 -1
- package/dist/utils/git.js +54 -22
- package/dist/utils/git.js.map +1 -1
- package/dist/utils/open-browser.d.ts +9 -0
- package/dist/utils/open-browser.d.ts.map +1 -0
- package/dist/utils/open-browser.js +22 -0
- package/dist/utils/open-browser.js.map +1 -0
- package/docs/containerization.md +111 -0
- package/docs/docs.json +4 -0
- package/docs/extensions.md +2 -0
- package/docs/index.md +1 -0
- package/docs/providers.md +3 -0
- package/examples/extensions/README.md +1 -0
- package/examples/extensions/gondolin/index.ts +531 -0
- package/examples/extensions/gondolin/package-lock.json +185 -0
- package/examples/extensions/gondolin/package.json +19 -0
- package/node_modules/@earendil-works/pi-agent-core/package.json +2 -2
- package/node_modules/@earendil-works/pi-ai/README.md +5 -1
- package/node_modules/@earendil-works/pi-ai/dist/env-api-keys.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/env-api-keys.js +2 -0
- package/node_modules/@earendil-works/pi-ai/dist/env-api-keys.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts +251 -48
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js +208 -44
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.js +25 -8
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/types.d.ts +3 -3
- package/node_modules/@earendil-works/pi-ai/dist/types.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/types.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.js +13 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/openai-codex.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/openai-codex.js +4 -2
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/openai-codex.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/package.json +1 -1
- package/node_modules/@earendil-works/pi-tui/package.json +1 -1
- package/npm-shrinkwrap.json +12 -12
- package/package.json +5 -4
|
@@ -0,0 +1,531 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Gondolin Tool Routing Example
|
|
3
|
+
*
|
|
4
|
+
* Runs pi's built-in tools inside a local Gondolin micro-VM. The host working
|
|
5
|
+
* directory is mounted at /workspace in the guest. File changes under
|
|
6
|
+
* /workspace write through to the host; other guest filesystem changes are
|
|
7
|
+
* isolated to the VM.
|
|
8
|
+
*
|
|
9
|
+
* Setup:
|
|
10
|
+
* cd packages/coding-agent/examples/extensions/gondolin
|
|
11
|
+
* npm install --ignore-scripts
|
|
12
|
+
*
|
|
13
|
+
* Usage:
|
|
14
|
+
* cd /path/to/project
|
|
15
|
+
* pi -e /path/to/pi/packages/coding-agent/examples/extensions/gondolin
|
|
16
|
+
*
|
|
17
|
+
* Requirements:
|
|
18
|
+
* - Node.js >= 23.6.0 for @earendil-works/gondolin
|
|
19
|
+
* - QEMU installed (for example, `brew install qemu` on macOS)
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import path from "node:path";
|
|
23
|
+
import { RealFSProvider, VM } from "@earendil-works/gondolin";
|
|
24
|
+
import type { ExtensionAPI, ExtensionContext } from "@earendil-works/pi-coding-agent";
|
|
25
|
+
import {
|
|
26
|
+
type BashOperations,
|
|
27
|
+
createBashTool,
|
|
28
|
+
createEditTool,
|
|
29
|
+
createFindTool,
|
|
30
|
+
createGrepTool,
|
|
31
|
+
createLsTool,
|
|
32
|
+
createReadTool,
|
|
33
|
+
createWriteTool,
|
|
34
|
+
DEFAULT_MAX_BYTES,
|
|
35
|
+
type EditOperations,
|
|
36
|
+
type FindOperations,
|
|
37
|
+
formatSize,
|
|
38
|
+
type GrepToolDetails,
|
|
39
|
+
type GrepToolInput,
|
|
40
|
+
type LsOperations,
|
|
41
|
+
type ReadOperations,
|
|
42
|
+
truncateHead,
|
|
43
|
+
truncateLine,
|
|
44
|
+
type WriteOperations,
|
|
45
|
+
} from "@earendil-works/pi-coding-agent";
|
|
46
|
+
|
|
47
|
+
const GUEST_WORKSPACE = "/workspace";
|
|
48
|
+
const DEFAULT_GREP_LIMIT = 100;
|
|
49
|
+
|
|
50
|
+
type TextToolResult<TDetails> = {
|
|
51
|
+
content: Array<{ type: "text"; text: string }>;
|
|
52
|
+
details: TDetails | undefined;
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
function stripAtPrefix(value: string): string {
|
|
56
|
+
return value.startsWith("@") ? value.slice(1) : value;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
function toPosix(value: string): string {
|
|
60
|
+
return value.split(path.sep).join(path.posix.sep);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function isInsideHostPath(root: string, value: string): boolean {
|
|
64
|
+
const relativePath = path.relative(root, value);
|
|
65
|
+
return relativePath === "" || (!relativePath.startsWith("..") && !path.isAbsolute(relativePath));
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function hostPathToGuest(localCwd: string, hostPath: string): string {
|
|
69
|
+
const relativePath = path.relative(localCwd, hostPath);
|
|
70
|
+
if (!isInsideHostPath(localCwd, hostPath)) return toPosix(hostPath);
|
|
71
|
+
return relativePath ? path.posix.join(GUEST_WORKSPACE, toPosix(relativePath)) : GUEST_WORKSPACE;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function toGuestPath(localCwd: string, inputPath: string): string {
|
|
75
|
+
const trimmed = stripAtPrefix(inputPath.trim());
|
|
76
|
+
if (!trimmed) return GUEST_WORKSPACE;
|
|
77
|
+
if (path.isAbsolute(trimmed)) {
|
|
78
|
+
if (isInsideHostPath(localCwd, trimmed)) return hostPathToGuest(localCwd, trimmed);
|
|
79
|
+
return path.posix.resolve("/", toPosix(trimmed));
|
|
80
|
+
}
|
|
81
|
+
return path.posix.resolve(GUEST_WORKSPACE, toPosix(trimmed));
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
function createGondolinReadOps(vm: VM, localCwd: string): ReadOperations {
|
|
85
|
+
return {
|
|
86
|
+
readFile: async (filePath) => vm.fs.readFile(toGuestPath(localCwd, filePath)),
|
|
87
|
+
access: async (filePath) => {
|
|
88
|
+
await vm.fs.access(toGuestPath(localCwd, filePath));
|
|
89
|
+
},
|
|
90
|
+
detectImageMimeType: async (filePath) => {
|
|
91
|
+
const ext = path.posix.extname(toGuestPath(localCwd, filePath)).toLowerCase();
|
|
92
|
+
if (ext === ".png") return "image/png";
|
|
93
|
+
if (ext === ".jpg" || ext === ".jpeg") return "image/jpeg";
|
|
94
|
+
if (ext === ".gif") return "image/gif";
|
|
95
|
+
if (ext === ".webp") return "image/webp";
|
|
96
|
+
return null;
|
|
97
|
+
},
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
function createGondolinWriteOps(vm: VM, localCwd: string): WriteOperations {
|
|
102
|
+
return {
|
|
103
|
+
writeFile: async (filePath, content) => {
|
|
104
|
+
await vm.fs.writeFile(toGuestPath(localCwd, filePath), content, { encoding: "utf8" });
|
|
105
|
+
},
|
|
106
|
+
mkdir: async (dirPath) => {
|
|
107
|
+
await vm.fs.mkdir(toGuestPath(localCwd, dirPath), { recursive: true });
|
|
108
|
+
},
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
function createGondolinEditOps(vm: VM, localCwd: string): EditOperations {
|
|
113
|
+
const readOps = createGondolinReadOps(vm, localCwd);
|
|
114
|
+
const writeOps = createGondolinWriteOps(vm, localCwd);
|
|
115
|
+
return {
|
|
116
|
+
readFile: readOps.readFile,
|
|
117
|
+
writeFile: writeOps.writeFile,
|
|
118
|
+
access: readOps.access,
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
function createGondolinLsOps(vm: VM, localCwd: string): LsOperations {
|
|
123
|
+
return {
|
|
124
|
+
exists: async (filePath) => {
|
|
125
|
+
try {
|
|
126
|
+
await vm.fs.access(toGuestPath(localCwd, filePath));
|
|
127
|
+
return true;
|
|
128
|
+
} catch {
|
|
129
|
+
return false;
|
|
130
|
+
}
|
|
131
|
+
},
|
|
132
|
+
stat: async (filePath) => vm.fs.stat(toGuestPath(localCwd, filePath)),
|
|
133
|
+
readdir: async (dirPath) => vm.fs.listDir(toGuestPath(localCwd, dirPath)),
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
async function walkGuestFiles(
|
|
138
|
+
vm: VM,
|
|
139
|
+
root: string,
|
|
140
|
+
visit: (guestPath: string, relativePath: string) => Promise<boolean>,
|
|
141
|
+
signal?: AbortSignal,
|
|
142
|
+
): Promise<boolean> {
|
|
143
|
+
if (signal?.aborted) throw new Error("Operation aborted");
|
|
144
|
+
const stat = await vm.fs.stat(root, { signal });
|
|
145
|
+
if (!stat.isDirectory()) return visit(root, path.posix.basename(root));
|
|
146
|
+
|
|
147
|
+
const walkDirectory = async (dir: string, relativeDir: string): Promise<boolean> => {
|
|
148
|
+
if (signal?.aborted) throw new Error("Operation aborted");
|
|
149
|
+
const entries = await vm.fs.listDir(dir, { signal });
|
|
150
|
+
for (const entry of entries) {
|
|
151
|
+
if (entry === ".git" || entry === "node_modules") continue;
|
|
152
|
+
const guestPath = path.posix.join(dir, entry);
|
|
153
|
+
const relativePath = relativeDir ? path.posix.join(relativeDir, entry) : entry;
|
|
154
|
+
let entryStat: Awaited<ReturnType<VM["fs"]["stat"]>>;
|
|
155
|
+
try {
|
|
156
|
+
entryStat = await vm.fs.stat(guestPath, { signal });
|
|
157
|
+
} catch {
|
|
158
|
+
continue;
|
|
159
|
+
}
|
|
160
|
+
if (entryStat.isDirectory()) {
|
|
161
|
+
if (!(await walkDirectory(guestPath, relativePath))) return false;
|
|
162
|
+
} else if (!(await visit(guestPath, relativePath))) {
|
|
163
|
+
return false;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
return true;
|
|
167
|
+
};
|
|
168
|
+
|
|
169
|
+
return walkDirectory(root, "");
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
function matchesToolGlob(relativePath: string, pattern: string): boolean {
|
|
173
|
+
const normalizedPattern = toPosix(pattern);
|
|
174
|
+
if (normalizedPattern.includes("/")) {
|
|
175
|
+
return (
|
|
176
|
+
path.posix.matchesGlob(relativePath, normalizedPattern) ||
|
|
177
|
+
path.posix.matchesGlob(relativePath, `**/${normalizedPattern}`)
|
|
178
|
+
);
|
|
179
|
+
}
|
|
180
|
+
return path.posix.matchesGlob(path.posix.basename(relativePath), normalizedPattern);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
function createGondolinFindOps(vm: VM, localCwd: string): FindOperations {
|
|
184
|
+
return {
|
|
185
|
+
exists: async (filePath) => {
|
|
186
|
+
try {
|
|
187
|
+
await vm.fs.access(toGuestPath(localCwd, filePath));
|
|
188
|
+
return true;
|
|
189
|
+
} catch {
|
|
190
|
+
return false;
|
|
191
|
+
}
|
|
192
|
+
},
|
|
193
|
+
glob: async (pattern, cwd, options) => {
|
|
194
|
+
const root = toGuestPath(localCwd, cwd);
|
|
195
|
+
const results: string[] = [];
|
|
196
|
+
await walkGuestFiles(vm, root, async (guestPath, relativePath) => {
|
|
197
|
+
if (results.length >= options.limit) return false;
|
|
198
|
+
if (matchesToolGlob(relativePath, pattern)) results.push(guestPath);
|
|
199
|
+
return results.length < options.limit;
|
|
200
|
+
});
|
|
201
|
+
return results;
|
|
202
|
+
},
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
function createLineMatcher(pattern: string, literal: boolean | undefined, ignoreCase: boolean | undefined) {
|
|
207
|
+
if (literal) {
|
|
208
|
+
const needle = ignoreCase ? pattern.toLowerCase() : pattern;
|
|
209
|
+
return (line: string) => (ignoreCase ? line.toLowerCase() : line).includes(needle);
|
|
210
|
+
}
|
|
211
|
+
const regex = new RegExp(pattern, ignoreCase ? "i" : undefined);
|
|
212
|
+
return (line: string) => regex.test(line);
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
function appendGrepBlock(params: {
|
|
216
|
+
outputLines: string[];
|
|
217
|
+
lines: string[];
|
|
218
|
+
relativePath: string;
|
|
219
|
+
lineIndex: number;
|
|
220
|
+
contextLines: number;
|
|
221
|
+
}): boolean {
|
|
222
|
+
let linesTruncated = false;
|
|
223
|
+
const start = params.contextLines > 0 ? Math.max(0, params.lineIndex - params.contextLines) : params.lineIndex;
|
|
224
|
+
const end =
|
|
225
|
+
params.contextLines > 0
|
|
226
|
+
? Math.min(params.lines.length - 1, params.lineIndex + params.contextLines)
|
|
227
|
+
: params.lineIndex;
|
|
228
|
+
|
|
229
|
+
for (let index = start; index <= end; index++) {
|
|
230
|
+
const rawLine = params.lines[index] ?? "";
|
|
231
|
+
const { text, wasTruncated } = truncateLine(rawLine.replace(/\r/g, ""));
|
|
232
|
+
if (wasTruncated) linesTruncated = true;
|
|
233
|
+
const separator = index === params.lineIndex ? ":" : "-";
|
|
234
|
+
params.outputLines.push(`${params.relativePath}${separator}${index + 1}${separator} ${text}`);
|
|
235
|
+
}
|
|
236
|
+
return linesTruncated;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
async function executeGondolinGrep(
|
|
240
|
+
vm: VM,
|
|
241
|
+
localCwd: string,
|
|
242
|
+
params: GrepToolInput,
|
|
243
|
+
signal?: AbortSignal,
|
|
244
|
+
): Promise<TextToolResult<GrepToolDetails>> {
|
|
245
|
+
const root = toGuestPath(localCwd, params.path ?? ".");
|
|
246
|
+
const rootStat = await vm.fs.stat(root, { signal });
|
|
247
|
+
const rootIsDirectory = rootStat.isDirectory();
|
|
248
|
+
const matcher = createLineMatcher(params.pattern, params.literal, params.ignoreCase);
|
|
249
|
+
const contextLines = params.context && params.context > 0 ? params.context : 0;
|
|
250
|
+
const effectiveLimit = Math.max(1, params.limit ?? DEFAULT_GREP_LIMIT);
|
|
251
|
+
const outputLines: string[] = [];
|
|
252
|
+
const details: GrepToolDetails = {};
|
|
253
|
+
let matchCount = 0;
|
|
254
|
+
let matchLimitReached = false;
|
|
255
|
+
let linesTruncated = false;
|
|
256
|
+
|
|
257
|
+
await walkGuestFiles(
|
|
258
|
+
vm,
|
|
259
|
+
root,
|
|
260
|
+
async (guestPath, relativePath) => {
|
|
261
|
+
if (matchCount >= effectiveLimit) return false;
|
|
262
|
+
if (params.glob && !matchesToolGlob(relativePath, params.glob)) return true;
|
|
263
|
+
let content: string;
|
|
264
|
+
try {
|
|
265
|
+
content = await vm.fs.readFile(guestPath, { encoding: "utf8", signal });
|
|
266
|
+
} catch {
|
|
267
|
+
return true;
|
|
268
|
+
}
|
|
269
|
+
const lines = content.replace(/\r\n/g, "\n").replace(/\r/g, "\n").split("\n");
|
|
270
|
+
const displayPath = rootIsDirectory ? relativePath : path.posix.basename(guestPath);
|
|
271
|
+
for (let index = 0; index < lines.length; index++) {
|
|
272
|
+
if (signal?.aborted) throw new Error("Operation aborted");
|
|
273
|
+
if (!matcher(lines[index] ?? "")) continue;
|
|
274
|
+
matchCount++;
|
|
275
|
+
if (appendGrepBlock({ outputLines, lines, relativePath: displayPath, lineIndex: index, contextLines })) {
|
|
276
|
+
linesTruncated = true;
|
|
277
|
+
}
|
|
278
|
+
if (matchCount >= effectiveLimit) {
|
|
279
|
+
matchLimitReached = true;
|
|
280
|
+
return false;
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
return true;
|
|
284
|
+
},
|
|
285
|
+
signal,
|
|
286
|
+
);
|
|
287
|
+
|
|
288
|
+
if (matchCount === 0) return { content: [{ type: "text", text: "No matches found" }], details: undefined };
|
|
289
|
+
|
|
290
|
+
const rawOutput = outputLines.join("\n");
|
|
291
|
+
const truncation = truncateHead(rawOutput, { maxLines: Number.MAX_SAFE_INTEGER });
|
|
292
|
+
const notices: string[] = [];
|
|
293
|
+
let output = truncation.content;
|
|
294
|
+
|
|
295
|
+
if (matchLimitReached) {
|
|
296
|
+
details.matchLimitReached = effectiveLimit;
|
|
297
|
+
notices.push(`${effectiveLimit} matches limit reached`);
|
|
298
|
+
}
|
|
299
|
+
if (linesTruncated) {
|
|
300
|
+
details.linesTruncated = true;
|
|
301
|
+
notices.push("long lines truncated");
|
|
302
|
+
}
|
|
303
|
+
if (truncation.truncated) {
|
|
304
|
+
details.truncation = truncation;
|
|
305
|
+
notices.push(`${formatSize(DEFAULT_MAX_BYTES)} limit reached`);
|
|
306
|
+
}
|
|
307
|
+
if (notices.length > 0) output += `\n\n[${notices.join(". ")}]`;
|
|
308
|
+
|
|
309
|
+
return {
|
|
310
|
+
content: [{ type: "text", text: output }],
|
|
311
|
+
details: Object.keys(details).length > 0 ? details : undefined,
|
|
312
|
+
};
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
function sanitizeEnv(env: NodeJS.ProcessEnv | undefined): Record<string, string> | undefined {
|
|
316
|
+
if (!env) return undefined;
|
|
317
|
+
const result: Record<string, string> = {};
|
|
318
|
+
for (const [key, value] of Object.entries(env)) {
|
|
319
|
+
if (typeof value === "string") result[key] = value;
|
|
320
|
+
}
|
|
321
|
+
return result;
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
function createGondolinBashOps(vm: VM, localCwd: string, shellPath: string): BashOperations {
|
|
325
|
+
return {
|
|
326
|
+
exec: async (command, cwd, { onData, signal, timeout, env }) => {
|
|
327
|
+
if (signal?.aborted) throw new Error("aborted");
|
|
328
|
+
const guestCwd = toGuestPath(localCwd, cwd);
|
|
329
|
+
const controller = new AbortController();
|
|
330
|
+
const onAbort = () => controller.abort();
|
|
331
|
+
signal?.addEventListener("abort", onAbort, { once: true });
|
|
332
|
+
|
|
333
|
+
let timedOut = false;
|
|
334
|
+
const timer =
|
|
335
|
+
timeout && timeout > 0
|
|
336
|
+
? setTimeout(() => {
|
|
337
|
+
timedOut = true;
|
|
338
|
+
controller.abort();
|
|
339
|
+
}, timeout * 1000)
|
|
340
|
+
: undefined;
|
|
341
|
+
|
|
342
|
+
try {
|
|
343
|
+
const proc = vm.exec([shellPath, "-lc", command], {
|
|
344
|
+
cwd: guestCwd,
|
|
345
|
+
env: sanitizeEnv(env),
|
|
346
|
+
signal: controller.signal,
|
|
347
|
+
stdout: "pipe",
|
|
348
|
+
stderr: "pipe",
|
|
349
|
+
});
|
|
350
|
+
for await (const chunk of proc.output()) onData(chunk.data);
|
|
351
|
+
const result = await proc;
|
|
352
|
+
return { exitCode: result.exitCode };
|
|
353
|
+
} catch (error) {
|
|
354
|
+
if (signal?.aborted) throw new Error("aborted");
|
|
355
|
+
if (timedOut) throw new Error(`timeout:${timeout}`);
|
|
356
|
+
throw error;
|
|
357
|
+
} finally {
|
|
358
|
+
if (timer) clearTimeout(timer);
|
|
359
|
+
signal?.removeEventListener("abort", onAbort);
|
|
360
|
+
}
|
|
361
|
+
},
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
export default function (pi: ExtensionAPI) {
|
|
366
|
+
const localCwd = process.cwd();
|
|
367
|
+
const localRead = createReadTool(localCwd);
|
|
368
|
+
const localWrite = createWriteTool(localCwd);
|
|
369
|
+
const localEdit = createEditTool(localCwd);
|
|
370
|
+
const localBash = createBashTool(localCwd);
|
|
371
|
+
const localGrep = createGrepTool(localCwd);
|
|
372
|
+
const localFind = createFindTool(localCwd);
|
|
373
|
+
const localLs = createLsTool(localCwd);
|
|
374
|
+
|
|
375
|
+
let vm: VM | undefined;
|
|
376
|
+
let vmStarting: Promise<VM> | undefined;
|
|
377
|
+
let shellPath = "/bin/sh";
|
|
378
|
+
|
|
379
|
+
async function startVm(ctx?: ExtensionContext): Promise<VM> {
|
|
380
|
+
ctx?.ui.setStatus("gondolin", ctx.ui.theme.fg("accent", `Gondolin: starting ${GUEST_WORKSPACE}`));
|
|
381
|
+
const created = await VM.create({
|
|
382
|
+
sessionLabel: `pi ${path.basename(localCwd)}`,
|
|
383
|
+
vfs: {
|
|
384
|
+
mounts: {
|
|
385
|
+
[GUEST_WORKSPACE]: new RealFSProvider(localCwd),
|
|
386
|
+
},
|
|
387
|
+
},
|
|
388
|
+
});
|
|
389
|
+
const bashProbe = await created.exec(["/bin/sh", "-lc", "command -v bash || true"]);
|
|
390
|
+
shellPath = bashProbe.stdout.trim() || "/bin/sh";
|
|
391
|
+
vm = created;
|
|
392
|
+
ctx?.ui.setStatus(
|
|
393
|
+
"gondolin",
|
|
394
|
+
ctx.ui.theme.fg("accent", `Gondolin: ${created.id.slice(0, 8)} (${GUEST_WORKSPACE})`),
|
|
395
|
+
);
|
|
396
|
+
ctx?.ui.notify(`Gondolin VM ready. ${localCwd} is mounted at ${GUEST_WORKSPACE}.`, "info");
|
|
397
|
+
return created;
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
async function ensureVm(ctx?: ExtensionContext): Promise<VM> {
|
|
401
|
+
if (vm) return vm;
|
|
402
|
+
if (!vmStarting) {
|
|
403
|
+
vmStarting = startVm(ctx).finally(() => {
|
|
404
|
+
vmStarting = undefined;
|
|
405
|
+
});
|
|
406
|
+
}
|
|
407
|
+
return vmStarting;
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
pi.on("session_start", async (_event, ctx) => {
|
|
411
|
+
await ensureVm(ctx);
|
|
412
|
+
});
|
|
413
|
+
|
|
414
|
+
pi.on("session_shutdown", async (_event, ctx) => {
|
|
415
|
+
const activeVm = vm;
|
|
416
|
+
vm = undefined;
|
|
417
|
+
vmStarting = undefined;
|
|
418
|
+
if (!activeVm) return;
|
|
419
|
+
ctx.ui.setStatus("gondolin", ctx.ui.theme.fg("muted", "Gondolin: stopping"));
|
|
420
|
+
try {
|
|
421
|
+
await activeVm.close();
|
|
422
|
+
} finally {
|
|
423
|
+
ctx.ui.setStatus("gondolin", undefined);
|
|
424
|
+
}
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
pi.registerCommand("gondolin", {
|
|
428
|
+
description: "Show Gondolin VM status",
|
|
429
|
+
handler: async (_args, ctx) => {
|
|
430
|
+
const activeVm = await ensureVm(ctx);
|
|
431
|
+
ctx.ui.notify(
|
|
432
|
+
[
|
|
433
|
+
`Gondolin VM: ${activeVm.id}`,
|
|
434
|
+
`Host workspace: ${localCwd}`,
|
|
435
|
+
`Guest workspace: ${GUEST_WORKSPACE}`,
|
|
436
|
+
`Shell: ${shellPath}`,
|
|
437
|
+
].join("\n"),
|
|
438
|
+
"info",
|
|
439
|
+
);
|
|
440
|
+
},
|
|
441
|
+
});
|
|
442
|
+
|
|
443
|
+
pi.registerTool({
|
|
444
|
+
...localRead,
|
|
445
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
446
|
+
const activeVm = await ensureVm(ctx);
|
|
447
|
+
const tool = createReadTool(GUEST_WORKSPACE, {
|
|
448
|
+
operations: createGondolinReadOps(activeVm, localCwd),
|
|
449
|
+
});
|
|
450
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
451
|
+
},
|
|
452
|
+
});
|
|
453
|
+
|
|
454
|
+
pi.registerTool({
|
|
455
|
+
...localWrite,
|
|
456
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
457
|
+
const activeVm = await ensureVm(ctx);
|
|
458
|
+
const tool = createWriteTool(GUEST_WORKSPACE, {
|
|
459
|
+
operations: createGondolinWriteOps(activeVm, localCwd),
|
|
460
|
+
});
|
|
461
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
462
|
+
},
|
|
463
|
+
});
|
|
464
|
+
|
|
465
|
+
pi.registerTool({
|
|
466
|
+
...localEdit,
|
|
467
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
468
|
+
const activeVm = await ensureVm(ctx);
|
|
469
|
+
const tool = createEditTool(GUEST_WORKSPACE, {
|
|
470
|
+
operations: createGondolinEditOps(activeVm, localCwd),
|
|
471
|
+
});
|
|
472
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
473
|
+
},
|
|
474
|
+
});
|
|
475
|
+
|
|
476
|
+
pi.registerTool({
|
|
477
|
+
...localBash,
|
|
478
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
479
|
+
const activeVm = await ensureVm(ctx);
|
|
480
|
+
const tool = createBashTool(GUEST_WORKSPACE, {
|
|
481
|
+
operations: createGondolinBashOps(activeVm, localCwd, shellPath),
|
|
482
|
+
});
|
|
483
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
484
|
+
},
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
pi.registerTool({
|
|
488
|
+
...localLs,
|
|
489
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
490
|
+
const activeVm = await ensureVm(ctx);
|
|
491
|
+
const tool = createLsTool(GUEST_WORKSPACE, {
|
|
492
|
+
operations: createGondolinLsOps(activeVm, localCwd),
|
|
493
|
+
});
|
|
494
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
495
|
+
},
|
|
496
|
+
});
|
|
497
|
+
|
|
498
|
+
pi.registerTool({
|
|
499
|
+
...localFind,
|
|
500
|
+
async execute(id, params, signal, onUpdate, ctx) {
|
|
501
|
+
const activeVm = await ensureVm(ctx);
|
|
502
|
+
const tool = createFindTool(GUEST_WORKSPACE, {
|
|
503
|
+
operations: createGondolinFindOps(activeVm, localCwd),
|
|
504
|
+
});
|
|
505
|
+
return tool.execute(id, params, signal, onUpdate);
|
|
506
|
+
},
|
|
507
|
+
});
|
|
508
|
+
|
|
509
|
+
pi.registerTool({
|
|
510
|
+
...localGrep,
|
|
511
|
+
async execute(_id, params, signal, _onUpdate, ctx) {
|
|
512
|
+
const activeVm = await ensureVm(ctx);
|
|
513
|
+
return executeGondolinGrep(activeVm, localCwd, params, signal);
|
|
514
|
+
},
|
|
515
|
+
});
|
|
516
|
+
|
|
517
|
+
pi.on("user_bash", async (_event, ctx) => {
|
|
518
|
+
const activeVm = await ensureVm(ctx);
|
|
519
|
+
return { operations: createGondolinBashOps(activeVm, localCwd, shellPath) };
|
|
520
|
+
});
|
|
521
|
+
|
|
522
|
+
pi.on("before_agent_start", async (event, ctx) => {
|
|
523
|
+
await ensureVm(ctx);
|
|
524
|
+
const localLine = `Current working directory: ${localCwd}`;
|
|
525
|
+
const guestLine = `Current working directory: ${GUEST_WORKSPACE} (Gondolin VM; host workspace mounted from ${localCwd})`;
|
|
526
|
+
const systemPrompt = event.systemPrompt.includes(localLine)
|
|
527
|
+
? event.systemPrompt.replace(localLine, guestLine)
|
|
528
|
+
: `${event.systemPrompt}\n\n${guestLine}`;
|
|
529
|
+
return { systemPrompt };
|
|
530
|
+
});
|
|
531
|
+
}
|