experimental-agent 0.0.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/dist/index.js ADDED
@@ -0,0 +1,3078 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __esm = (fn, res) => function __init() {
9
+ return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res;
10
+ };
11
+ var __export = (target, all2) => {
12
+ for (var name in all2)
13
+ __defProp(target, name, { get: all2[name], enumerable: true });
14
+ };
15
+ var __copyProps = (to, from, except, desc) => {
16
+ if (from && typeof from === "object" || typeof from === "function") {
17
+ for (let key of __getOwnPropNames(from))
18
+ if (!__hasOwnProp.call(to, key) && key !== except)
19
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
20
+ }
21
+ return to;
22
+ };
23
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
24
+ // If the importer is in node compatibility mode or this is not an ESM
25
+ // file that has been converted to a CommonJS file using a Babel-
26
+ // compatible transform (i.e. "__esModule" has not been set), then set
27
+ // "default" to the CommonJS "module.exports" for node compatibility.
28
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
29
+ mod
30
+ ));
31
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
32
+
33
+ // src/sandbox/process-manager.ts
34
+ var process_manager_exports = {};
35
+ __export(process_manager_exports, {
36
+ createProcessManager: () => createProcessManager
37
+ });
38
+ var DEFAULT_WAIT_UNTIL, RUN_SCRIPT, createProcessManager;
39
+ var init_process_manager = __esm({
40
+ "src/sandbox/process-manager.ts"() {
41
+ "use strict";
42
+ DEFAULT_WAIT_UNTIL = 3e4;
43
+ RUN_SCRIPT = `#!/bin/sh
44
+ # Process runner for Agent SDK
45
+ # All parameters passed via environment variables to avoid escaping issues
46
+ # Output is base64 encoded to avoid JSON escaping issues
47
+
48
+ AGENT_DIR="$HOME/.agent"
49
+ CWD_FILE="$AGENT_DIR/cwd/$SESSION_ID"
50
+
51
+ # Get current working directory (per-session)
52
+ if [ -f "$CWD_FILE" ]; then
53
+ CWD=$(cat "$CWD_FILE")
54
+ else
55
+ CWD="$HOME"
56
+ fi
57
+
58
+ # Decode command
59
+ CMD=$(echo "$CMD_BASE64" | base64 -d)
60
+
61
+ # Background mode (waitUntil = 0)
62
+ if [ "$WAIT_UNTIL" = "0" ]; then
63
+ cd "$CWD" || cd "$HOME"
64
+
65
+ # Generate a simple numeric ID for output file
66
+ OUTPUT_ID="$(date +%s)$$"
67
+ OUTPUT_FILE="$AGENT_DIR/outputs/$OUTPUT_ID.log"
68
+
69
+ # Double-fork to fully detach the process
70
+ (
71
+ (
72
+ eval "$CMD"
73
+ pwd > "$CWD_FILE" 2>/dev/null
74
+ ) > "$OUTPUT_FILE" 2>&1 < /dev/null &
75
+ echo $! > "$AGENT_DIR/pids/$OUTPUT_ID.pid"
76
+ )
77
+
78
+ # Small delay to ensure PID file is written
79
+ sleep 0.05
80
+
81
+ # Read PID
82
+ if [ -f "$AGENT_DIR/pids/$OUTPUT_ID.pid" ]; then
83
+ PID=$(cat "$AGENT_DIR/pids/$OUTPUT_ID.pid")
84
+ else
85
+ PID=0
86
+ fi
87
+
88
+ # Output JSON result (no output for background, so no encoding needed)
89
+ printf '{"pid":%d,"outputBase64":"","exitCode":-1,"status":"running","outputFile":"%s","cwd":"%s"}\\n' \\
90
+ "$PID" "$OUTPUT_FILE" "$CWD"
91
+ exit 0
92
+ fi
93
+
94
+ # Foreground mode (waitUntil > 0) - run directly and capture output
95
+ cd "$CWD" || cd "$HOME"
96
+
97
+ # Run command, capture output and exit code
98
+ OUTPUT=$(eval "$CMD" 2>&1)
99
+ EXIT_CODE=$?
100
+
101
+ # Update CWD after command
102
+ NEW_CWD=$(pwd)
103
+ echo "$NEW_CWD" > "$CWD_FILE"
104
+
105
+ # Determine status
106
+ if [ $EXIT_CODE -eq 0 ]; then
107
+ STATUS="completed"
108
+ else
109
+ STATUS="failed"
110
+ fi
111
+
112
+ # Base64 encode output to avoid ALL escaping issues
113
+ OUTPUT_BASE64=$(printf '%s' "$OUTPUT" | base64 | tr -d '\\n')
114
+
115
+ # Output JSON result with base64-encoded output
116
+ printf '{"pid":0,"outputBase64":"%s","exitCode":%d,"status":"%s","outputFile":"","cwd":"%s"}\\n' \\
117
+ "$OUTPUT_BASE64" "$EXIT_CODE" "$STATUS" "$NEW_CWD"
118
+ `;
119
+ createProcessManager = (opts) => {
120
+ const { sandbox, sessionId } = opts;
121
+ let initialized = false;
122
+ const init = async () => {
123
+ if (initialized) {
124
+ return;
125
+ }
126
+ const result = await sandbox.exec({
127
+ command: "sh",
128
+ args: [
129
+ "-c",
130
+ `mkdir -p $HOME/.agent/bin $HOME/.agent/pids $HOME/.agent/outputs $HOME/.agent/cwd
131
+ cat > $HOME/.agent/bin/run.sh << 'SCRIPT_EOF'
132
+ ${RUN_SCRIPT}
133
+ SCRIPT_EOF
134
+ chmod +x $HOME/.agent/bin/run.sh`
135
+ ]
136
+ });
137
+ if (result instanceof Error) {
138
+ throw result;
139
+ }
140
+ await result.result;
141
+ initialized = true;
142
+ };
143
+ const run = async (opts2) => {
144
+ await init();
145
+ const { command, waitUntil = DEFAULT_WAIT_UNTIL } = opts2;
146
+ const cmdBase64 = Buffer.from(command, "utf-8").toString("base64");
147
+ const result = await sandbox.exec({
148
+ command: "sh",
149
+ args: [
150
+ "-c",
151
+ `CMD_BASE64="${cmdBase64}" WAIT_UNTIL="${waitUntil}" SESSION_ID="${sessionId}" $HOME/.agent/bin/run.sh`
152
+ ]
153
+ });
154
+ if (result instanceof Error) {
155
+ throw result;
156
+ }
157
+ const { stdout } = await result.result;
158
+ const parsed = JSON.parse(stdout.trim());
159
+ const output = parsed.outputBase64 ? Buffer.from(parsed.outputBase64, "base64").toString("utf-8") : "";
160
+ return {
161
+ pid: parsed.pid,
162
+ output,
163
+ exitCode: parsed.exitCode,
164
+ status: parsed.status,
165
+ cwd: parsed.cwd,
166
+ outputFile: parsed.outputFile
167
+ };
168
+ };
169
+ const getCwd = async () => {
170
+ await init();
171
+ const result = await sandbox.exec({
172
+ command: "sh",
173
+ args: [
174
+ "-c",
175
+ `cat $HOME/.agent/cwd/${sessionId} 2>/dev/null || echo "$HOME"`
176
+ ]
177
+ });
178
+ if (result instanceof Error) {
179
+ return process.cwd();
180
+ }
181
+ const { stdout } = await result.result;
182
+ return stdout.trim() || process.cwd();
183
+ };
184
+ return {
185
+ init,
186
+ run,
187
+ getCwd
188
+ };
189
+ };
190
+ }
191
+ });
192
+
193
+ // src/errors.ts
194
+ var errore, SessionNotFoundError, SessionError, SandboxNotFoundError, StorageError, SandboxError, MessageNotFoundError;
195
+ var init_errors = __esm({
196
+ "src/errors.ts"() {
197
+ "use strict";
198
+ errore = __toESM(require("errore"));
199
+ SessionNotFoundError = class extends errore.createTaggedError({
200
+ name: "SessionNotFoundError",
201
+ message: "Session $id not found"
202
+ }) {
203
+ };
204
+ SessionError = class extends errore.createTaggedError({
205
+ name: "SessionError",
206
+ message: "Session $id failed: $reason"
207
+ }) {
208
+ };
209
+ SandboxNotFoundError = class extends errore.createTaggedError({
210
+ name: "SandboxNotFoundError",
211
+ message: "Sandbox $id not found"
212
+ }) {
213
+ };
214
+ StorageError = class extends errore.createTaggedError({
215
+ name: "StorageError",
216
+ message: "$reason"
217
+ }) {
218
+ };
219
+ SandboxError = class extends errore.createTaggedError({
220
+ name: "SandboxError",
221
+ message: "$reason"
222
+ }) {
223
+ };
224
+ MessageNotFoundError = class extends errore.createTaggedError({
225
+ name: "MessageNotFoundError",
226
+ message: "Message $id not found"
227
+ }) {
228
+ };
229
+ }
230
+ });
231
+
232
+ // src/utils/paginate.ts
233
+ function paginate(opts) {
234
+ const { items, cursor, limit = 50 } = opts;
235
+ const startIndex = cursor ? items.findIndex((m) => m.id === cursor) + 1 : 0;
236
+ const sliced = items.slice(startIndex, startIndex + limit);
237
+ const nextCursor = startIndex + limit < items.length ? sliced.at(-1)?.id ?? null : null;
238
+ return { items: sliced, nextCursor };
239
+ }
240
+ var init_paginate = __esm({
241
+ "src/utils/paginate.ts"() {
242
+ "use strict";
243
+ }
244
+ });
245
+
246
+ // src/storage/bindings/local-fs-handlers.ts
247
+ function createFilesystemHandlers(basePath) {
248
+ const sessionDir = (0, import_node_path.join)(basePath, "session");
249
+ const messageDir = (0, import_node_path.join)(basePath, "message");
250
+ const partDir = (0, import_node_path.join)(basePath, "part");
251
+ const sandboxDir = (0, import_node_path.join)(basePath, "sandbox");
252
+ const commandDir = (0, import_node_path.join)(basePath, "command");
253
+ async function ensureDir(dir) {
254
+ await (0, import_promises.mkdir)(dir, { recursive: true });
255
+ }
256
+ async function readJson(filePath) {
257
+ try {
258
+ const content = await (0, import_promises.readFile)(filePath, "utf-8");
259
+ return JSON.parse(content);
260
+ } catch {
261
+ return null;
262
+ }
263
+ }
264
+ async function writeJsonFile(filePath, data) {
265
+ await ensureDir((0, import_node_path.dirname)(filePath));
266
+ await (0, import_promises.writeFile)(filePath, JSON.stringify(data, null, 2));
267
+ }
268
+ async function readAllFromDir(dir) {
269
+ try {
270
+ const files = await (0, import_promises.readdir)(dir);
271
+ const results = await Promise.all(
272
+ files.filter((f) => f.endsWith(".json")).map((f) => readJson((0, import_node_path.join)(dir, f)))
273
+ );
274
+ return results.filter((r) => r !== null);
275
+ } catch {
276
+ return [];
277
+ }
278
+ }
279
+ return {
280
+ "session.get": async ({ id }) => {
281
+ const session = await readJson((0, import_node_path.join)(sessionDir, `${id}.json`));
282
+ return session ?? null;
283
+ },
284
+ "session.set": async (session) => {
285
+ const now = Date.now();
286
+ const sessionPath = (0, import_node_path.join)(sessionDir, `${session.id}.json`);
287
+ const existing = await readJson(sessionPath);
288
+ const newSession = {
289
+ ...session,
290
+ tags: session.tags ?? existing?.tags ?? {},
291
+ createdAt: existing?.createdAt ?? session.createdAt ?? now,
292
+ updatedAt: now
293
+ };
294
+ await writeJsonFile(sessionPath, newSession);
295
+ return newSession;
296
+ },
297
+ "session.list": async ({ tags, cursor, limit }) => {
298
+ const allSessions = await readAllFromDir(sessionDir);
299
+ let filtered = allSessions;
300
+ if (tags && Object.keys(tags).length > 0) {
301
+ filtered = filtered.filter((s) => {
302
+ const sessionTags = s.tags ?? {};
303
+ return Object.entries(tags).every(
304
+ ([key, value]) => sessionTags[key] === value
305
+ );
306
+ });
307
+ }
308
+ filtered.sort((a, b) => a.createdAt - b.createdAt);
309
+ return paginate({ items: filtered, cursor, limit });
310
+ },
311
+ "session.listBySandbox": async ({ sandboxId, tags, cursor, limit }) => {
312
+ const allSessions = await readAllFromDir(sessionDir);
313
+ let filtered = allSessions.filter((s) => s.sandboxId === sandboxId);
314
+ if (tags && Object.keys(tags).length > 0) {
315
+ filtered = filtered.filter((s) => {
316
+ const sessionTags = s.tags ?? {};
317
+ return Object.entries(tags).every(
318
+ ([key, value]) => sessionTags[key] === value
319
+ );
320
+ });
321
+ }
322
+ filtered.sort((a, b) => a.createdAt - b.createdAt);
323
+ return paginate({ items: filtered, cursor, limit });
324
+ },
325
+ "session.tag.set": async ({ sessionId, tags }) => {
326
+ const sessionPath = (0, import_node_path.join)(sessionDir, `${sessionId}.json`);
327
+ const existing = await readJson(sessionPath);
328
+ if (!existing) {
329
+ throw new Error(`Session ${sessionId} not found`);
330
+ }
331
+ const mergedTags = { ...existing.tags, ...tags };
332
+ const now = Date.now();
333
+ const updatedSession = {
334
+ ...existing,
335
+ tags: mergedTags,
336
+ updatedAt: now
337
+ };
338
+ await writeJsonFile(sessionPath, updatedSession);
339
+ return updatedSession;
340
+ },
341
+ "message.get": async ({ id }) => {
342
+ return await readJson((0, import_node_path.join)(messageDir, `${id}.json`));
343
+ },
344
+ "message.set": async (message) => {
345
+ await writeJsonFile((0, import_node_path.join)(messageDir, `${message.id}.json`), message);
346
+ return message;
347
+ },
348
+ "message.list": async ({ sessionId, cursor, limit }) => {
349
+ const allMessages = await readAllFromDir(messageDir);
350
+ const filtered = allMessages.filter((m) => m.sessionId === sessionId).sort((a, b) => a.createdAt - b.createdAt);
351
+ return paginate({ items: filtered, cursor, limit });
352
+ },
353
+ "part.listByMessage": async ({ messageId, cursor, limit }) => {
354
+ const allParts = await readAllFromDir(partDir);
355
+ const filtered = allParts.filter((p) => p.messageId === messageId).sort((a, b) => a.index - b.index);
356
+ return paginate({ items: filtered, cursor, limit });
357
+ },
358
+ "part.listBySession": async ({ sessionId, cursor, limit }) => {
359
+ const allParts = await readAllFromDir(partDir);
360
+ const filtered = allParts.filter((p) => p.sessionId === sessionId).sort((a, b) => {
361
+ if (a.messageId !== b.messageId) {
362
+ return a.messageId.localeCompare(b.messageId);
363
+ }
364
+ return a.index - b.index;
365
+ });
366
+ return paginate({ items: filtered, cursor, limit });
367
+ },
368
+ "part.set": async (part) => {
369
+ await writeJsonFile((0, import_node_path.join)(partDir, `${part.id}.json`), part);
370
+ return part;
371
+ },
372
+ "sandbox.get": async ({ key }) => {
373
+ const safeName = Buffer.from(key).toString("base64url");
374
+ const sandboxPath = (0, import_node_path.join)(sandboxDir, `${safeName}.json`);
375
+ const data = await readJson(sandboxPath);
376
+ if (!data) {
377
+ return null;
378
+ }
379
+ return data;
380
+ },
381
+ "sandbox.set": async (record) => {
382
+ const safeName = Buffer.from(record.id).toString("base64url");
383
+ const sandboxPath = (0, import_node_path.join)(sandboxDir, `${safeName}.json`);
384
+ await writeJsonFile(sandboxPath, record);
385
+ },
386
+ "sandbox.getBySession": async ({ sessionId }) => {
387
+ const allSandboxes = await readAllFromDir(sandboxDir);
388
+ const matching = allSandboxes.filter(
389
+ (s) => s.id.startsWith(`${sessionId}-`)
390
+ );
391
+ if (matching.length === 0) {
392
+ return null;
393
+ }
394
+ matching.sort(
395
+ (a, b) => (b.lastActivityAt ?? 0) - (a.lastActivityAt ?? 0)
396
+ );
397
+ return matching[0];
398
+ },
399
+ "command.get": async ({ id }) => {
400
+ return await readJson((0, import_node_path.join)(commandDir, `${id}.json`));
401
+ },
402
+ "command.set": async (command) => {
403
+ await writeJsonFile((0, import_node_path.join)(commandDir, `${command.id}.json`), command);
404
+ return command;
405
+ },
406
+ "command.list": async ({ sessionId, includeFinished, cursor, limit }) => {
407
+ const allCommands = await readAllFromDir(commandDir);
408
+ let filtered = allCommands.filter((c) => c.sessionId === sessionId);
409
+ if (!includeFinished) {
410
+ filtered = filtered.filter((c) => c.status === "running");
411
+ }
412
+ filtered.sort((a, b) => a.startedAt - b.startedAt);
413
+ return paginate({ items: filtered, cursor, limit });
414
+ }
415
+ };
416
+ }
417
+ var import_promises, import_node_path;
418
+ var init_local_fs_handlers = __esm({
419
+ "src/storage/bindings/local-fs-handlers.ts"() {
420
+ "use strict";
421
+ import_promises = require("fs/promises");
422
+ import_node_path = require("path");
423
+ init_paginate();
424
+ }
425
+ });
426
+
427
+ // src/storage/bindings/local.ts
428
+ var local_exports = {};
429
+ __export(local_exports, {
430
+ startLocalStorage: () => startLocalStorage,
431
+ stopLocalStorage: () => stopLocalStorage
432
+ });
433
+ function startLocalStorage(opts) {
434
+ if (startPromise) {
435
+ return startPromise;
436
+ }
437
+ startPromise = new Promise((resolve) => {
438
+ const handlers = createFilesystemHandlers(
439
+ opts?.path ?? DEFAULT_STORAGE_DIR
440
+ );
441
+ const app = new import_hono.Hono();
442
+ app.post("/", async (c) => {
443
+ const body = await c.req.json();
444
+ const response = await handleStorageRpc(body, handlers);
445
+ return c.json(response);
446
+ });
447
+ const port = opts?.port ?? 0;
448
+ const server = (0, import_node_server.serve)({
449
+ fetch: app.fetch,
450
+ port
451
+ });
452
+ server.on("listening", () => {
453
+ const address = server.address();
454
+ const actualPort = typeof address === "object" ? address?.port : port;
455
+ const url = `http://localhost:${actualPort}`;
456
+ serverInstance = { url, server };
457
+ console.log(`[agent] Local storage server started at ${url}`);
458
+ resolve(url);
459
+ });
460
+ });
461
+ return startPromise;
462
+ }
463
+ function stopLocalStorage() {
464
+ if (serverInstance) {
465
+ serverInstance.server.close();
466
+ serverInstance = null;
467
+ startPromise = null;
468
+ }
469
+ }
470
+ var import_node_server, import_hono, DEFAULT_STORAGE_DIR, startPromise, serverInstance;
471
+ var init_local = __esm({
472
+ "src/storage/bindings/local.ts"() {
473
+ "use strict";
474
+ import_node_server = require("@hono/node-server");
475
+ import_hono = require("hono");
476
+ init_storage();
477
+ init_local_fs_handlers();
478
+ DEFAULT_STORAGE_DIR = ".agent-storage";
479
+ startPromise = null;
480
+ serverInstance = null;
481
+ if (typeof process !== "undefined") {
482
+ process.on("SIGTERM", stopLocalStorage);
483
+ process.on("SIGINT", stopLocalStorage);
484
+ }
485
+ }
486
+ });
487
+
488
+ // src/storage/bindings/vercel.ts
489
+ var vercel_exports = {};
490
+ __export(vercel_exports, {
491
+ getVercelStorageConfig: () => getVercelStorageConfig
492
+ });
493
+ async function getVercelStorageConfig() {
494
+ let token = null;
495
+ try {
496
+ const { getVercelOidcToken } = await import("@vercel/oidc");
497
+ token = await getVercelOidcToken();
498
+ } catch {
499
+ }
500
+ return {
501
+ url: VERCEL_STORAGE_URL,
502
+ headers: token ? { Authorization: `Bearer ${token}` } : {}
503
+ };
504
+ }
505
+ var VERCEL_STORAGE_URL;
506
+ var init_vercel = __esm({
507
+ "src/storage/bindings/vercel.ts"() {
508
+ "use strict";
509
+ VERCEL_STORAGE_URL = "https://agent-sdk-storage-nine.labs.vercel.dev/api/storage";
510
+ }
511
+ });
512
+
513
+ // src/storage/client.ts
514
+ var client_exports = {};
515
+ __export(client_exports, {
516
+ getStorage: () => getStorage,
517
+ getStorageClient: () => getStorageClient
518
+ });
519
+ function getStorageClient({ url, headers }) {
520
+ async function rpc(method, params) {
521
+ const res = await fetch(url, {
522
+ method: "POST",
523
+ headers: { "Content-Type": "application/json", ...headers },
524
+ body: JSON.stringify({ method, params })
525
+ });
526
+ if (!res.ok) {
527
+ throw new StorageError({
528
+ reason: `HTTP ${res.status}: ${res.statusText}`
529
+ });
530
+ }
531
+ const json = await res.json();
532
+ if ("error" in json) {
533
+ throw new StorageError({ reason: json.error.message });
534
+ }
535
+ return json.result;
536
+ }
537
+ return {
538
+ session: {
539
+ get: async (id) => {
540
+ try {
541
+ const result = await rpc("session.get", { id });
542
+ if (result === null) {
543
+ return new SessionNotFoundError({ id });
544
+ }
545
+ return result;
546
+ } catch (e) {
547
+ if (e instanceof StorageError) {
548
+ return e;
549
+ }
550
+ return new StorageError({ reason: String(e), cause: e });
551
+ }
552
+ },
553
+ set: async (session) => {
554
+ try {
555
+ return await rpc("session.set", session);
556
+ } catch (e) {
557
+ if (e instanceof StorageError) {
558
+ return e;
559
+ }
560
+ return new StorageError({ reason: String(e), cause: e });
561
+ }
562
+ },
563
+ list: async (opts) => {
564
+ try {
565
+ return await rpc("session.list", {
566
+ tags: opts?.tags,
567
+ order: opts?.order,
568
+ cursor: opts?.cursor,
569
+ limit: opts?.limit
570
+ });
571
+ } catch (e) {
572
+ if (e instanceof StorageError) {
573
+ return e;
574
+ }
575
+ return new StorageError({ reason: String(e), cause: e });
576
+ }
577
+ },
578
+ tag: {
579
+ set: async ({ sessionId, tags }) => {
580
+ try {
581
+ return await rpc("session.tag.set", { sessionId, tags });
582
+ } catch (e) {
583
+ if (e instanceof StorageError) {
584
+ return e;
585
+ }
586
+ return new StorageError({ reason: String(e), cause: e });
587
+ }
588
+ }
589
+ }
590
+ },
591
+ message: {
592
+ list: async (sessionId, opts) => {
593
+ try {
594
+ return await rpc("message.list", {
595
+ sessionId,
596
+ cursor: opts?.cursor,
597
+ limit: opts?.limit
598
+ });
599
+ } catch (e) {
600
+ if (e instanceof StorageError) {
601
+ return e;
602
+ }
603
+ return new StorageError({ reason: String(e), cause: e });
604
+ }
605
+ },
606
+ get: async (id) => {
607
+ try {
608
+ const result = await rpc("message.get", { id });
609
+ if (result === null) {
610
+ return new MessageNotFoundError({ id });
611
+ }
612
+ return result;
613
+ } catch (e) {
614
+ if (e instanceof StorageError) {
615
+ return e;
616
+ }
617
+ return new StorageError({ reason: String(e), cause: e });
618
+ }
619
+ },
620
+ set: async (message) => {
621
+ try {
622
+ return await rpc("message.set", message);
623
+ } catch (e) {
624
+ if (e instanceof StorageError) {
625
+ return e;
626
+ }
627
+ return new StorageError({ reason: String(e), cause: e });
628
+ }
629
+ }
630
+ },
631
+ part: {
632
+ listByMessage: async (messageId, opts) => {
633
+ try {
634
+ const result = await rpc("part.listByMessage", {
635
+ messageId,
636
+ cursor: opts?.cursor,
637
+ limit: opts?.limit
638
+ });
639
+ return result;
640
+ } catch (e) {
641
+ if (e instanceof StorageError) {
642
+ return e;
643
+ }
644
+ return new StorageError({ reason: String(e), cause: e });
645
+ }
646
+ },
647
+ listBySession: async (sessionId, opts) => {
648
+ try {
649
+ const result = await rpc("part.listBySession", {
650
+ sessionId,
651
+ cursor: opts?.cursor,
652
+ limit: opts?.limit
653
+ });
654
+ return result;
655
+ } catch (e) {
656
+ if (e instanceof StorageError) {
657
+ return e;
658
+ }
659
+ return new StorageError({ reason: String(e), cause: e });
660
+ }
661
+ },
662
+ set: async (part) => {
663
+ try {
664
+ return await rpc("part.set", part);
665
+ } catch (e) {
666
+ if (e instanceof StorageError) {
667
+ return e;
668
+ }
669
+ return new StorageError({ reason: String(e), cause: e });
670
+ }
671
+ }
672
+ },
673
+ sandbox: {
674
+ get: async (key) => {
675
+ try {
676
+ const result = await rpc("sandbox.get", { key });
677
+ if (result === null) {
678
+ return new SandboxNotFoundError({ id: key });
679
+ }
680
+ return result;
681
+ } catch (e) {
682
+ if (e instanceof StorageError) {
683
+ return e;
684
+ }
685
+ return new StorageError({ reason: String(e), cause: e });
686
+ }
687
+ },
688
+ getBySession: async (sessionId) => {
689
+ try {
690
+ const result = await rpc("sandbox.getBySession", { sessionId });
691
+ if (result === null) {
692
+ return new SandboxNotFoundError({ id: sessionId });
693
+ }
694
+ return result;
695
+ } catch (e) {
696
+ if (e instanceof StorageError) {
697
+ return e;
698
+ }
699
+ return new StorageError({ reason: String(e), cause: e });
700
+ }
701
+ },
702
+ set: async (record) => {
703
+ try {
704
+ await rpc("sandbox.set", record);
705
+ return record;
706
+ } catch (e) {
707
+ if (e instanceof StorageError) {
708
+ return e;
709
+ }
710
+ return new StorageError({ reason: String(e), cause: e });
711
+ }
712
+ }
713
+ },
714
+ command: {
715
+ get: async (id) => {
716
+ try {
717
+ return await rpc("command.get", { id });
718
+ } catch (e) {
719
+ if (e instanceof StorageError) {
720
+ return e;
721
+ }
722
+ return new StorageError({ reason: String(e), cause: e });
723
+ }
724
+ },
725
+ set: async (command) => {
726
+ try {
727
+ return await rpc("command.set", command);
728
+ } catch (e) {
729
+ if (e instanceof StorageError) {
730
+ return e;
731
+ }
732
+ return new StorageError({ reason: String(e), cause: e });
733
+ }
734
+ },
735
+ list: async (sessionId, opts) => {
736
+ try {
737
+ const result = await rpc("command.list", {
738
+ sessionId,
739
+ includeFinished: opts?.includeFinished,
740
+ cursor: opts?.cursor,
741
+ limit: opts?.limit
742
+ });
743
+ return result;
744
+ } catch (e) {
745
+ if (e instanceof StorageError) {
746
+ return e;
747
+ }
748
+ return new StorageError({ reason: String(e), cause: e });
749
+ }
750
+ }
751
+ }
752
+ };
753
+ }
754
+ async function resolveStorageUrl(config) {
755
+ switch (config.type) {
756
+ case "local": {
757
+ const { startLocalStorage: startLocalStorage2 } = await Promise.resolve().then(() => (init_local(), local_exports));
758
+ return { url: await startLocalStorage2({ path: config.path }) };
759
+ }
760
+ case "vercel": {
761
+ const { getVercelStorageConfig: getVercelStorageConfig2 } = await Promise.resolve().then(() => (init_vercel(), vercel_exports));
762
+ return getVercelStorageConfig2();
763
+ }
764
+ case "custom": {
765
+ return { url: config.url, headers: config.headers };
766
+ }
767
+ default: {
768
+ config;
769
+ throw new Error(
770
+ `Unknown storage type: ${config.type}`
771
+ );
772
+ }
773
+ }
774
+ }
775
+ function getStorage(storageConfig) {
776
+ let clientPromise = null;
777
+ const getClient = () => {
778
+ if (!clientPromise) {
779
+ clientPromise = resolveStorageUrl(storageConfig).then(
780
+ (resolved) => getStorageClient(resolved)
781
+ );
782
+ }
783
+ return clientPromise;
784
+ };
785
+ return {
786
+ session: {
787
+ get: async (id) => (await getClient()).session.get(id),
788
+ set: async (session) => (await getClient()).session.set(session),
789
+ list: async (opts) => (await getClient()).session.list(opts),
790
+ tag: {
791
+ set: async (opts) => (await getClient()).session.tag.set(opts)
792
+ }
793
+ },
794
+ message: {
795
+ list: async (sessionId, opts) => (await getClient()).message.list(sessionId, opts),
796
+ get: async (id) => (await getClient()).message.get(id),
797
+ set: async (message) => (await getClient()).message.set(message)
798
+ },
799
+ part: {
800
+ listByMessage: async (messageId, opts) => (await getClient()).part.listByMessage(messageId, opts),
801
+ listBySession: async (sessionId, opts) => (await getClient()).part.listBySession(sessionId, opts),
802
+ set: async (part) => (await getClient()).part.set(part)
803
+ },
804
+ sandbox: {
805
+ get: async (key) => (await getClient()).sandbox.get(key),
806
+ set: async (opts) => (await getClient()).sandbox.set(opts),
807
+ getBySession: async (sessionId) => (await getClient()).sandbox.getBySession(sessionId)
808
+ },
809
+ command: {
810
+ get: async (id) => (await getClient()).command.get(id),
811
+ set: async (command) => (await getClient()).command.set(command),
812
+ list: async (sessionId, opts) => (await getClient()).command.list(sessionId, opts)
813
+ }
814
+ };
815
+ }
816
+ var init_client = __esm({
817
+ "src/storage/client.ts"() {
818
+ "use strict";
819
+ init_errors();
820
+ }
821
+ });
822
+
823
+ // src/storage/rpc.ts
824
+ function ListResultSchema(itemSchema) {
825
+ return import_zod2.z.object({
826
+ items: import_zod2.z.array(itemSchema),
827
+ nextCursor: import_zod2.z.string().nullable()
828
+ });
829
+ }
830
+ var import_zod2, VercelLifecycleConfigSchema, SandboxConfigSchema, SessionSchema, MessageSchema, PartSchema, CommandResultSchema, SandboxRecordSchema, CommandBaseSchema, CommandSchema, methods;
831
+ var init_rpc = __esm({
832
+ "src/storage/rpc.ts"() {
833
+ "use strict";
834
+ import_zod2 = require("zod");
835
+ VercelLifecycleConfigSchema = import_zod2.z.object({
836
+ stopAfterInactiveMs: import_zod2.z.number().optional(),
837
+ snapshotBeforeTimeoutMs: import_zod2.z.number().optional(),
838
+ snapshotId: import_zod2.z.string().optional(),
839
+ autoStart: import_zod2.z.boolean().optional()
840
+ });
841
+ SandboxConfigSchema = import_zod2.z.discriminatedUnion("type", [
842
+ import_zod2.z.object({
843
+ type: import_zod2.z.literal("vercel"),
844
+ resources: import_zod2.z.object({ vcpus: import_zod2.z.number() }).optional(),
845
+ ports: import_zod2.z.array(import_zod2.z.number()).optional(),
846
+ lifecycle: VercelLifecycleConfigSchema.optional()
847
+ }),
848
+ import_zod2.z.object({
849
+ type: import_zod2.z.literal("local"),
850
+ path: import_zod2.z.string().optional()
851
+ }),
852
+ import_zod2.z.object({
853
+ type: import_zod2.z.literal("custom"),
854
+ url: import_zod2.z.string(),
855
+ headers: import_zod2.z.record(import_zod2.z.string(), import_zod2.z.string()).optional()
856
+ })
857
+ ]);
858
+ SessionSchema = import_zod2.z.object({
859
+ id: import_zod2.z.string(),
860
+ createdAt: import_zod2.z.number(),
861
+ updatedAt: import_zod2.z.number(),
862
+ runId: import_zod2.z.string().nullable(),
863
+ lastMessageId: import_zod2.z.string().nullable(),
864
+ tags: import_zod2.z.record(import_zod2.z.string(), import_zod2.z.unknown()).nullable(),
865
+ instructions: import_zod2.z.string().nullable(),
866
+ model: import_zod2.z.string().nullable(),
867
+ mcp: import_zod2.z.unknown().nullable(),
868
+ // TODO
869
+ sandboxId: import_zod2.z.string().nullable(),
870
+ skillsDir: import_zod2.z.array(import_zod2.z.string()).nullable(),
871
+ hookToken: import_zod2.z.string().nullable()
872
+ });
873
+ MessageSchema = import_zod2.z.object({
874
+ id: import_zod2.z.string(),
875
+ sessionId: import_zod2.z.string(),
876
+ role: import_zod2.z.enum(["user", "assistant", "system"]),
877
+ createdAt: import_zod2.z.number(),
878
+ completedAt: import_zod2.z.number().nullable(),
879
+ mcpContext: import_zod2.z.unknown().nullable()
880
+ });
881
+ PartSchema = import_zod2.z.object({
882
+ id: import_zod2.z.string(),
883
+ messageId: import_zod2.z.string(),
884
+ sessionId: import_zod2.z.string(),
885
+ index: import_zod2.z.number(),
886
+ part: import_zod2.z.unknown()
887
+ });
888
+ CommandResultSchema = import_zod2.z.object({
889
+ stdout: import_zod2.z.string(),
890
+ stderr: import_zod2.z.string(),
891
+ exitCode: import_zod2.z.number(),
892
+ completedAt: import_zod2.z.number()
893
+ });
894
+ SandboxRecordSchema = import_zod2.z.object({
895
+ id: import_zod2.z.string(),
896
+ config: SandboxConfigSchema,
897
+ createdAt: import_zod2.z.number().nullable(),
898
+ lastActivityAt: import_zod2.z.number().nullable(),
899
+ acquiringLockId: import_zod2.z.string().nullable(),
900
+ acquiringLockAt: import_zod2.z.number().nullable(),
901
+ providerMetadata: import_zod2.z.discriminatedUnion("provider", [
902
+ import_zod2.z.object({
903
+ provider: import_zod2.z.literal("vercel"),
904
+ sandboxId: import_zod2.z.string().nullable(),
905
+ snapshotId: import_zod2.z.string().nullable()
906
+ })
907
+ ]).optional()
908
+ });
909
+ CommandBaseSchema = import_zod2.z.object({
910
+ id: import_zod2.z.string(),
911
+ sessionId: import_zod2.z.string(),
912
+ command: import_zod2.z.string(),
913
+ args: import_zod2.z.array(import_zod2.z.string()).optional(),
914
+ startedAt: import_zod2.z.number()
915
+ });
916
+ CommandSchema = import_zod2.z.discriminatedUnion("status", [
917
+ CommandBaseSchema.extend({ status: import_zod2.z.literal("running") }),
918
+ CommandBaseSchema.extend({
919
+ status: import_zod2.z.literal("completed"),
920
+ result: CommandResultSchema
921
+ }),
922
+ CommandBaseSchema.extend({
923
+ status: import_zod2.z.literal("killed"),
924
+ result: CommandResultSchema.optional()
925
+ }),
926
+ CommandBaseSchema.extend({
927
+ status: import_zod2.z.literal("failed"),
928
+ result: CommandResultSchema
929
+ })
930
+ ]);
931
+ methods = {
932
+ "session.get": {
933
+ params: import_zod2.z.object({ id: import_zod2.z.string() }),
934
+ result: SessionSchema.nullable()
935
+ },
936
+ "session.set": {
937
+ params: SessionSchema,
938
+ result: SessionSchema
939
+ },
940
+ "session.list": {
941
+ params: import_zod2.z.object({
942
+ tags: import_zod2.z.record(import_zod2.z.string(), import_zod2.z.unknown()).optional(),
943
+ order: import_zod2.z.enum([
944
+ "createdAt_asc",
945
+ "createdAt_desc",
946
+ "updatedAt_asc",
947
+ "updatedAt_desc"
948
+ ]).optional(),
949
+ cursor: import_zod2.z.string().optional(),
950
+ limit: import_zod2.z.number().optional()
951
+ }),
952
+ result: ListResultSchema(SessionSchema)
953
+ },
954
+ "session.listBySandbox": {
955
+ params: import_zod2.z.object({
956
+ sandboxId: import_zod2.z.string(),
957
+ tags: import_zod2.z.record(import_zod2.z.string(), import_zod2.z.unknown()).optional(),
958
+ order: import_zod2.z.enum([
959
+ "createdAt_asc",
960
+ "createdAt_desc",
961
+ "updatedAt_asc",
962
+ "updatedAt_desc"
963
+ ]).optional(),
964
+ cursor: import_zod2.z.string().optional(),
965
+ limit: import_zod2.z.number().optional()
966
+ }),
967
+ result: ListResultSchema(SessionSchema)
968
+ },
969
+ "session.tag.set": {
970
+ params: import_zod2.z.object({
971
+ sessionId: import_zod2.z.string(),
972
+ tags: import_zod2.z.record(import_zod2.z.string(), import_zod2.z.unknown())
973
+ }),
974
+ result: SessionSchema
975
+ },
976
+ "message.get": {
977
+ params: import_zod2.z.object({ id: import_zod2.z.string() }),
978
+ result: MessageSchema.nullable()
979
+ },
980
+ "message.set": {
981
+ params: MessageSchema,
982
+ result: MessageSchema
983
+ },
984
+ "message.list": {
985
+ params: import_zod2.z.object({
986
+ sessionId: import_zod2.z.string(),
987
+ cursor: import_zod2.z.string().optional(),
988
+ limit: import_zod2.z.number().optional()
989
+ }),
990
+ result: ListResultSchema(MessageSchema)
991
+ },
992
+ "part.listByMessage": {
993
+ params: import_zod2.z.object({
994
+ messageId: import_zod2.z.string(),
995
+ cursor: import_zod2.z.string().optional(),
996
+ limit: import_zod2.z.number().optional()
997
+ }),
998
+ result: ListResultSchema(PartSchema)
999
+ },
1000
+ "part.listBySession": {
1001
+ params: import_zod2.z.object({
1002
+ sessionId: import_zod2.z.string(),
1003
+ cursor: import_zod2.z.string().optional(),
1004
+ limit: import_zod2.z.number().optional()
1005
+ }),
1006
+ result: ListResultSchema(PartSchema)
1007
+ },
1008
+ "part.set": {
1009
+ params: PartSchema,
1010
+ result: PartSchema
1011
+ },
1012
+ "sandbox.get": {
1013
+ params: import_zod2.z.object({ key: import_zod2.z.string() }),
1014
+ result: SandboxRecordSchema.nullable()
1015
+ },
1016
+ "sandbox.getBySession": {
1017
+ params: import_zod2.z.object({ sessionId: import_zod2.z.string() }),
1018
+ result: SandboxRecordSchema.nullable()
1019
+ },
1020
+ "sandbox.set": {
1021
+ params: SandboxRecordSchema,
1022
+ result: import_zod2.z.void()
1023
+ },
1024
+ "command.get": {
1025
+ params: import_zod2.z.object({ id: import_zod2.z.string() }),
1026
+ result: CommandSchema.nullable()
1027
+ },
1028
+ "command.set": {
1029
+ params: CommandSchema,
1030
+ result: CommandSchema
1031
+ },
1032
+ "command.list": {
1033
+ params: import_zod2.z.object({
1034
+ sessionId: import_zod2.z.string(),
1035
+ includeFinished: import_zod2.z.boolean().optional(),
1036
+ cursor: import_zod2.z.string().optional(),
1037
+ limit: import_zod2.z.number().optional()
1038
+ }),
1039
+ result: ListResultSchema(CommandSchema)
1040
+ }
1041
+ };
1042
+ }
1043
+ });
1044
+
1045
+ // src/storage/handler.ts
1046
+ async function handleStorageRpc(body, handlers) {
1047
+ const { method, params } = body;
1048
+ if (!(method in methods)) {
1049
+ return {
1050
+ error: { code: "METHOD_NOT_FOUND", message: `Unknown method: ${method}` }
1051
+ };
1052
+ }
1053
+ const methodName = method;
1054
+ const schema = methods[methodName].params;
1055
+ const parsed = schema.safeParse(params);
1056
+ if (!parsed.success) {
1057
+ return {
1058
+ error: {
1059
+ code: "INVALID_PARAMS",
1060
+ message: parsed.error.issues.map((i) => `${i.path.join(".")}: ${i.message}`).join(", ")
1061
+ }
1062
+ };
1063
+ }
1064
+ try {
1065
+ const handler = handlers[methodName];
1066
+ const result = await handler(parsed.data);
1067
+ return { result };
1068
+ } catch (e) {
1069
+ return {
1070
+ error: {
1071
+ code: "INTERNAL_ERROR",
1072
+ message: e instanceof Error ? e.message : String(e)
1073
+ }
1074
+ };
1075
+ }
1076
+ }
1077
+ var init_handler = __esm({
1078
+ "src/storage/handler.ts"() {
1079
+ "use strict";
1080
+ init_rpc();
1081
+ }
1082
+ });
1083
+
1084
+ // src/storage/index.ts
1085
+ var storage_exports = {};
1086
+ __export(storage_exports, {
1087
+ CommandResultSchema: () => CommandResultSchema,
1088
+ CommandSchema: () => CommandSchema,
1089
+ MessageSchema: () => MessageSchema,
1090
+ PartSchema: () => PartSchema,
1091
+ SandboxConfigSchema: () => SandboxConfigSchema,
1092
+ SandboxRecordSchema: () => SandboxRecordSchema,
1093
+ SessionSchema: () => SessionSchema,
1094
+ getStorage: () => getStorage,
1095
+ handleStorageRpc: () => handleStorageRpc,
1096
+ methods: () => methods
1097
+ });
1098
+ var init_storage = __esm({
1099
+ "src/storage/index.ts"() {
1100
+ "use strict";
1101
+ init_client();
1102
+ init_handler();
1103
+ init_rpc();
1104
+ }
1105
+ });
1106
+
1107
+ // src/sandbox/write-files.ts
1108
+ async function writeFiles(opts) {
1109
+ const { sandbox, files, destPath } = opts;
1110
+ if (files.length === 0) {
1111
+ return;
1112
+ }
1113
+ const filePaths = files.map((file) => path.posix.join(destPath, file.path));
1114
+ const parentDirs = [...new Set(filePaths.map((p) => path.posix.dirname(p)))];
1115
+ const shellScripts = filePaths.filter((p) => p.endsWith(".sh"));
1116
+ const mkdirResult = await sandbox.exec({
1117
+ command: "mkdir",
1118
+ args: ["-p", ...parentDirs]
1119
+ });
1120
+ if (mkdirResult instanceof Error) {
1121
+ throw mkdirResult;
1122
+ }
1123
+ await mkdirResult.result;
1124
+ const CHUNK_SIZE = 5e4;
1125
+ for (let i = 0; i < files.length; i++) {
1126
+ const file = files[i];
1127
+ const fullPath = filePaths[i];
1128
+ const base64Content = toBase64(file.content);
1129
+ if (base64Content.length < CHUNK_SIZE) {
1130
+ const marker = `EOF_${i}`;
1131
+ const execResult = await sandbox.exec({
1132
+ command: "bash",
1133
+ args: [
1134
+ "-c",
1135
+ `base64 -d > ${quote(fullPath)} << '${marker}'
1136
+ ${base64Content}
1137
+ ${marker}`
1138
+ ]
1139
+ });
1140
+ if (execResult instanceof Error) {
1141
+ throw execResult;
1142
+ }
1143
+ const { exitCode, stderr } = await execResult.result;
1144
+ if (exitCode !== 0) {
1145
+ throw new Error(
1146
+ `writeFiles failed with exit code ${exitCode}: ${stderr}`
1147
+ );
1148
+ }
1149
+ } else {
1150
+ const tempB64 = `/tmp/chunk-${Date.now()}-${i}.b64`;
1151
+ const clearResult = await sandbox.exec({
1152
+ command: "bash",
1153
+ args: ["-c", `> ${quote(tempB64)}`]
1154
+ });
1155
+ if (clearResult instanceof Error) {
1156
+ throw clearResult;
1157
+ }
1158
+ await clearResult.result;
1159
+ for (let offset = 0; offset < base64Content.length; offset += CHUNK_SIZE) {
1160
+ const chunk = base64Content.slice(offset, offset + CHUNK_SIZE);
1161
+ const marker = `CHUNK_${offset}`;
1162
+ const appendResult = await sandbox.exec({
1163
+ command: "bash",
1164
+ args: [
1165
+ "-c",
1166
+ `cat >> ${quote(tempB64)} << '${marker}'
1167
+ ${chunk}
1168
+ ${marker}`
1169
+ ]
1170
+ });
1171
+ if (appendResult instanceof Error) {
1172
+ throw appendResult;
1173
+ }
1174
+ const { exitCode: exitCode2, stderr: stderr2 } = await appendResult.result;
1175
+ if (exitCode2 !== 0) {
1176
+ throw new Error(
1177
+ `writeFiles chunk failed with exit code ${exitCode2}: ${stderr2}`
1178
+ );
1179
+ }
1180
+ }
1181
+ const decodeResult = await sandbox.exec({
1182
+ command: "bash",
1183
+ args: [
1184
+ "-c",
1185
+ `base64 -d < ${quote(tempB64)} > ${quote(fullPath)} && rm -f ${quote(tempB64)}`
1186
+ ]
1187
+ });
1188
+ if (decodeResult instanceof Error) {
1189
+ throw decodeResult;
1190
+ }
1191
+ const { exitCode, stderr } = await decodeResult.result;
1192
+ if (exitCode !== 0) {
1193
+ throw new Error(
1194
+ `writeFiles decode failed with exit code ${exitCode}: ${stderr}`
1195
+ );
1196
+ }
1197
+ }
1198
+ }
1199
+ if (shellScripts.length > 0) {
1200
+ const chmodResult = await sandbox.exec({
1201
+ command: "chmod",
1202
+ args: ["+x", ...shellScripts]
1203
+ });
1204
+ if (chmodResult instanceof Error) {
1205
+ throw chmodResult;
1206
+ }
1207
+ await chmodResult.result;
1208
+ }
1209
+ }
1210
+ function toBase64(content) {
1211
+ if (typeof content === "string") {
1212
+ return Buffer.from(content).toString("base64");
1213
+ }
1214
+ return content.toString("base64");
1215
+ }
1216
+ function quote(s) {
1217
+ return `'${s.replace(/'/g, "'\\''")}'`;
1218
+ }
1219
+ var path;
1220
+ var init_write_files = __esm({
1221
+ "src/sandbox/write-files.ts"() {
1222
+ "use strict";
1223
+ path = __toESM(require("path"));
1224
+ }
1225
+ });
1226
+
1227
+ // src/sandbox/bindings/local.ts
1228
+ var import_node_child_process, errore2, import_ulid, localSandbox;
1229
+ var init_local2 = __esm({
1230
+ "src/sandbox/bindings/local.ts"() {
1231
+ "use strict";
1232
+ import_node_child_process = require("child_process");
1233
+ errore2 = __toESM(require("errore"));
1234
+ import_ulid = require("ulid");
1235
+ init_errors();
1236
+ init_write_files();
1237
+ localSandbox = ({
1238
+ sandboxRecord
1239
+ }) => {
1240
+ const config = sandboxRecord.config;
1241
+ const basePath = config.path ?? process.cwd();
1242
+ const processes = /* @__PURE__ */ new Map();
1243
+ const sandbox = {
1244
+ id: sandboxRecord.id,
1245
+ config: sandboxRecord.config,
1246
+ exec: ({ command, args, signal }) => {
1247
+ return errore2.tryAsync({
1248
+ try: () => {
1249
+ const commandId = `command_${(0, import_ulid.ulid)()}`;
1250
+ const child = (0, import_node_child_process.spawn)(command, args, {
1251
+ cwd: basePath,
1252
+ signal
1253
+ });
1254
+ processes.set(commandId, child);
1255
+ let stdout = "";
1256
+ let stderr = "";
1257
+ const logQueue = [];
1258
+ let logResolve = null;
1259
+ let closed = false;
1260
+ child.stdout.on("data", (data) => {
1261
+ const str = String(data);
1262
+ stdout += str;
1263
+ logQueue.push({ stream: "stdout", data: str });
1264
+ logResolve?.();
1265
+ });
1266
+ child.stderr.on("data", (data) => {
1267
+ const str = String(data);
1268
+ stderr += str;
1269
+ logQueue.push({ stream: "stderr", data: str });
1270
+ logResolve?.();
1271
+ });
1272
+ const result = new Promise((resolve, reject) => {
1273
+ child.on("error", (err) => {
1274
+ processes.delete(commandId);
1275
+ closed = true;
1276
+ logResolve?.();
1277
+ reject(err);
1278
+ });
1279
+ child.on("close", (code) => {
1280
+ processes.delete(commandId);
1281
+ closed = true;
1282
+ logResolve?.();
1283
+ resolve({ stdout, stderr, exitCode: code ?? 0 });
1284
+ });
1285
+ });
1286
+ async function* logs() {
1287
+ while (!closed || logQueue.length > 0) {
1288
+ const entry = logQueue.shift();
1289
+ if (entry) {
1290
+ yield entry;
1291
+ } else if (!closed) {
1292
+ await new Promise((resolve) => {
1293
+ logResolve = resolve;
1294
+ });
1295
+ logResolve = null;
1296
+ }
1297
+ }
1298
+ }
1299
+ return Promise.resolve({ commandId, logs, result });
1300
+ },
1301
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1302
+ });
1303
+ },
1304
+ getDomain: (port) => {
1305
+ return Promise.resolve(`http://localhost:${port}`);
1306
+ },
1307
+ kill: async ({ commandId, storage }) => {
1308
+ const child = processes.get(commandId);
1309
+ if (!child) {
1310
+ return new SandboxError({
1311
+ reason: `Command ${commandId} not found or already finished`
1312
+ });
1313
+ }
1314
+ child.kill("SIGTERM");
1315
+ const cmd = await storage.command.get(commandId);
1316
+ if (cmd instanceof Error) {
1317
+ return new SandboxError({ reason: cmd.message, cause: cmd });
1318
+ }
1319
+ if (cmd && cmd.status === "running") {
1320
+ const result = await storage.command.set({
1321
+ ...cmd,
1322
+ status: "killed"
1323
+ });
1324
+ if (result instanceof Error) {
1325
+ return new SandboxError({ reason: result.message, cause: result });
1326
+ }
1327
+ }
1328
+ },
1329
+ writeFiles: (opts) => writeFiles({ sandbox, ...opts })
1330
+ };
1331
+ return sandbox;
1332
+ };
1333
+ }
1334
+ });
1335
+
1336
+ // src/sandbox/bindings/vercel.ts
1337
+ var errore3, import_sandbox, VERCEL_MAX_TIMEOUT_MS, LOCK_TIMEOUT_MS, LOCK_POLL_INTERVAL_MS, getTestCredentials, createPromises, ACTIVITY_THROTTLE_MS, lastActivitySent, DEFAULT_VCPUS, vercelSandbox;
1338
+ var init_vercel2 = __esm({
1339
+ "src/sandbox/bindings/vercel.ts"() {
1340
+ "use strict";
1341
+ errore3 = __toESM(require("errore"));
1342
+ import_sandbox = require("sandbox");
1343
+ init_errors();
1344
+ init_storage();
1345
+ init_write_files();
1346
+ VERCEL_MAX_TIMEOUT_MS = 5 * 60 * 60 * 1e3;
1347
+ LOCK_TIMEOUT_MS = 3e4;
1348
+ LOCK_POLL_INTERVAL_MS = 200;
1349
+ getTestCredentials = () => process.env.NODE_ENV === "test" ? {
1350
+ token: process.env.TEST_VERCEL_TOKEN,
1351
+ teamId: process.env.TEST_VERCEL_TEAM_ID,
1352
+ projectId: process.env.TEST_VERCEL_PROJECT_ID
1353
+ } : {};
1354
+ createPromises = /* @__PURE__ */ new Map();
1355
+ ACTIVITY_THROTTLE_MS = 1e4;
1356
+ lastActivitySent = /* @__PURE__ */ new Map();
1357
+ DEFAULT_VCPUS = 2;
1358
+ vercelSandbox = ({
1359
+ sandboxRecord,
1360
+ storageConfig,
1361
+ enableLifecycleWorkflow = true,
1362
+ storage: storageOverride
1363
+ }) => {
1364
+ const { id, config } = sandboxRecord;
1365
+ const vcpus = config.resources?.vcpus ?? DEFAULT_VCPUS;
1366
+ const ports = config.ports;
1367
+ const storage = storageOverride ?? getStorage(storageConfig);
1368
+ const initialVercel = sandboxRecord.providerMetadata?.provider === "vercel" ? sandboxRecord.providerMetadata : null;
1369
+ let sandboxPromise = null;
1370
+ async function pollForSandboxId() {
1371
+ const deadline = Date.now() + LOCK_TIMEOUT_MS;
1372
+ while (Date.now() < deadline) {
1373
+ await new Promise((r) => setTimeout(r, LOCK_POLL_INTERVAL_MS));
1374
+ const record = await storage.sandbox.get(id);
1375
+ if (record instanceof Error) {
1376
+ return new SandboxError({ reason: record.message, cause: record });
1377
+ }
1378
+ const vercelSandboxId = record?.providerMetadata?.provider === "vercel" ? record.providerMetadata.sandboxId : null;
1379
+ if (vercelSandboxId) {
1380
+ return vercelSandboxId;
1381
+ }
1382
+ if (!record?.acquiringLockAt) {
1383
+ break;
1384
+ }
1385
+ }
1386
+ return new SandboxError({
1387
+ reason: "Timed out waiting for sandbox creation by another process"
1388
+ });
1389
+ }
1390
+ async function createSandboxFromSnapshot(snapshotId) {
1391
+ return await errore3.tryAsync({
1392
+ try: async () => {
1393
+ const sandbox2 = await import_sandbox.Sandbox.create({
1394
+ source: { type: "snapshot", snapshotId },
1395
+ resources: { vcpus },
1396
+ timeout: VERCEL_MAX_TIMEOUT_MS,
1397
+ ports,
1398
+ ...getTestCredentials()
1399
+ });
1400
+ const now = Date.now();
1401
+ await storage.sandbox.set({
1402
+ id,
1403
+ config,
1404
+ createdAt: now,
1405
+ lastActivityAt: now,
1406
+ acquiringLockId: null,
1407
+ acquiringLockAt: null,
1408
+ providerMetadata: {
1409
+ provider: "vercel",
1410
+ sandboxId: sandbox2.sandboxId,
1411
+ snapshotId
1412
+ }
1413
+ });
1414
+ return sandbox2.sandboxId;
1415
+ },
1416
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1417
+ });
1418
+ }
1419
+ async function createFreshSandbox() {
1420
+ return await errore3.tryAsync({
1421
+ try: async () => {
1422
+ const sandbox2 = await import_sandbox.Sandbox.create({
1423
+ resources: { vcpus },
1424
+ timeout: VERCEL_MAX_TIMEOUT_MS,
1425
+ ports,
1426
+ ...getTestCredentials()
1427
+ });
1428
+ const now = Date.now();
1429
+ await storage.sandbox.set({
1430
+ id,
1431
+ config,
1432
+ createdAt: now,
1433
+ lastActivityAt: now,
1434
+ acquiringLockId: null,
1435
+ acquiringLockAt: null,
1436
+ providerMetadata: {
1437
+ provider: "vercel",
1438
+ sandboxId: sandbox2.sandboxId,
1439
+ snapshotId: null
1440
+ }
1441
+ });
1442
+ return sandbox2.sandboxId;
1443
+ },
1444
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1445
+ });
1446
+ }
1447
+ async function doGetOrCreateSandboxId() {
1448
+ if (initialVercel?.sandboxId) {
1449
+ return initialVercel.sandboxId;
1450
+ }
1451
+ const existing = await storage.sandbox.get(id);
1452
+ if (existing instanceof Error) {
1453
+ return new SandboxError({ reason: existing.message, cause: existing });
1454
+ }
1455
+ const existingVercel = existing?.providerMetadata?.provider === "vercel" ? existing.providerMetadata : null;
1456
+ if (existingVercel?.sandboxId) {
1457
+ return existingVercel.sandboxId;
1458
+ }
1459
+ const hasActiveLock = existing?.acquiringLockId && existing.acquiringLockAt && Date.now() - existing.acquiringLockAt < LOCK_TIMEOUT_MS;
1460
+ if (hasActiveLock) {
1461
+ return pollForSandboxId();
1462
+ }
1463
+ const lockId = crypto.randomUUID();
1464
+ const now = Date.now();
1465
+ await storage.sandbox.set({
1466
+ id,
1467
+ config,
1468
+ createdAt: existing?.createdAt ?? sandboxRecord.createdAt,
1469
+ lastActivityAt: existing?.lastActivityAt ?? sandboxRecord.lastActivityAt,
1470
+ acquiringLockId: lockId,
1471
+ acquiringLockAt: now,
1472
+ providerMetadata: {
1473
+ provider: "vercel",
1474
+ sandboxId: null,
1475
+ snapshotId: existingVercel?.snapshotId ?? initialVercel?.snapshotId ?? null
1476
+ }
1477
+ });
1478
+ const afterLock = await storage.sandbox.get(id);
1479
+ if (afterLock instanceof Error) {
1480
+ return new SandboxError({ reason: afterLock.message, cause: afterLock });
1481
+ }
1482
+ if (afterLock?.acquiringLockId !== lockId) {
1483
+ return pollForSandboxId();
1484
+ }
1485
+ const snapshotId = existingVercel?.snapshotId ?? initialVercel?.snapshotId ?? config.lifecycle?.snapshotId;
1486
+ if (snapshotId) {
1487
+ const result = await createSandboxFromSnapshot(snapshotId);
1488
+ if (!(result instanceof Error)) {
1489
+ return result;
1490
+ }
1491
+ }
1492
+ return createFreshSandbox();
1493
+ }
1494
+ function getOrCreateSandboxId() {
1495
+ const cached = createPromises.get(id);
1496
+ if (cached) {
1497
+ return cached;
1498
+ }
1499
+ const promise = doGetOrCreateSandboxId().finally(() => {
1500
+ createPromises.delete(id);
1501
+ });
1502
+ createPromises.set(id, promise);
1503
+ return promise;
1504
+ }
1505
+ async function doGetSandbox() {
1506
+ const vercelSandboxId = await getOrCreateSandboxId();
1507
+ if (vercelSandboxId instanceof Error) {
1508
+ return vercelSandboxId;
1509
+ }
1510
+ return errore3.tryAsync({
1511
+ try: () => import_sandbox.Sandbox.get({
1512
+ sandboxId: vercelSandboxId,
1513
+ ...getTestCredentials()
1514
+ }),
1515
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1516
+ });
1517
+ }
1518
+ function getSandbox2() {
1519
+ if (!sandboxPromise) {
1520
+ sandboxPromise = doGetSandbox();
1521
+ }
1522
+ return sandboxPromise;
1523
+ }
1524
+ async function updateLastActivity() {
1525
+ const now = Date.now();
1526
+ const lastSent = lastActivitySent.get(id);
1527
+ if (lastSent && now - lastSent < ACTIVITY_THROTTLE_MS) {
1528
+ return;
1529
+ }
1530
+ lastActivitySent.set(id, now);
1531
+ const existing = await storage.sandbox.get(id);
1532
+ if (existing instanceof Error || !existing) {
1533
+ return;
1534
+ }
1535
+ const existingVercel = existing.providerMetadata?.provider === "vercel" ? existing.providerMetadata : null;
1536
+ await storage.sandbox.set({
1537
+ id: existing.id,
1538
+ config: existing.config,
1539
+ createdAt: existing.createdAt,
1540
+ lastActivityAt: now,
1541
+ acquiringLockId: null,
1542
+ acquiringLockAt: null,
1543
+ providerMetadata: existingVercel ?? {
1544
+ provider: "vercel",
1545
+ sandboxId: null,
1546
+ snapshotId: null
1547
+ }
1548
+ });
1549
+ if (enableLifecycleWorkflow) {
1550
+ }
1551
+ }
1552
+ const lifecycle = {
1553
+ start: async () => {
1554
+ const sandbox2 = await getSandbox2();
1555
+ if (sandbox2 instanceof Error) {
1556
+ return sandbox2;
1557
+ }
1558
+ await updateLastActivity();
1559
+ return sandbox2.status;
1560
+ },
1561
+ snapshot: async () => {
1562
+ const sandbox2 = await getSandbox2();
1563
+ if (sandbox2 instanceof Error) {
1564
+ return sandbox2;
1565
+ }
1566
+ return errore3.tryAsync({
1567
+ try: async () => {
1568
+ const existing = await storage.sandbox.get(id);
1569
+ const snapshot = await sandbox2.snapshot();
1570
+ await storage.sandbox.set({
1571
+ id,
1572
+ config,
1573
+ createdAt: existing instanceof Error ? null : existing?.createdAt ?? null,
1574
+ lastActivityAt: existing instanceof Error ? null : existing?.lastActivityAt ?? null,
1575
+ acquiringLockId: null,
1576
+ acquiringLockAt: null,
1577
+ providerMetadata: {
1578
+ provider: "vercel",
1579
+ sandboxId: null,
1580
+ snapshotId: snapshot.snapshotId
1581
+ }
1582
+ });
1583
+ return { snapshotId: snapshot.snapshotId };
1584
+ },
1585
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1586
+ });
1587
+ },
1588
+ stop: async () => {
1589
+ const sandbox2 = await getSandbox2();
1590
+ if (sandbox2 instanceof Error) {
1591
+ return sandbox2;
1592
+ }
1593
+ return errore3.tryAsync({
1594
+ try: async () => {
1595
+ await sandbox2.stop();
1596
+ const existing = await storage.sandbox.get(id);
1597
+ if (existing instanceof Error || !existing) {
1598
+ return void 0;
1599
+ }
1600
+ await storage.sandbox.set({
1601
+ id: existing.id,
1602
+ config: existing.config,
1603
+ createdAt: existing.createdAt,
1604
+ lastActivityAt: existing.lastActivityAt,
1605
+ acquiringLockId: null,
1606
+ acquiringLockAt: null,
1607
+ providerMetadata: {
1608
+ provider: "vercel",
1609
+ sandboxId: null,
1610
+ snapshotId: null
1611
+ }
1612
+ });
1613
+ return void 0;
1614
+ },
1615
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1616
+ });
1617
+ },
1618
+ getStatus: async () => {
1619
+ const sandbox2 = await getSandbox2();
1620
+ if (sandbox2 instanceof Error) {
1621
+ return sandbox2;
1622
+ }
1623
+ return sandbox2.status;
1624
+ },
1625
+ getCreatedAt: async () => {
1626
+ const sandbox2 = await getSandbox2();
1627
+ if (sandbox2 instanceof Error) {
1628
+ return sandbox2;
1629
+ }
1630
+ return sandbox2.createdAt;
1631
+ },
1632
+ getRemainingTimeout: async () => {
1633
+ const sandbox2 = await getSandbox2();
1634
+ if (sandbox2 instanceof Error) {
1635
+ return sandbox2;
1636
+ }
1637
+ return sandbox2.timeout;
1638
+ }
1639
+ };
1640
+ const sandbox = {
1641
+ id,
1642
+ config,
1643
+ exec: async ({ command, args, signal }) => {
1644
+ const instance = await getSandbox2();
1645
+ if (instance instanceof Error) {
1646
+ return instance;
1647
+ }
1648
+ const updatePromise = updateLastActivity();
1649
+ const execResult = await errore3.tryAsync({
1650
+ try: async () => {
1651
+ const output = await instance.runCommand(command, args, { signal });
1652
+ let stdout = "";
1653
+ let stderr = "";
1654
+ const logBuffer = [];
1655
+ const state = {
1656
+ resolve: null,
1657
+ consumed: false
1658
+ };
1659
+ const consumeLogs = (async () => {
1660
+ for await (const log of output.logs()) {
1661
+ const entry = log.stream === "stdout" ? { stream: "stdout", data: log.data } : { stream: "stderr", data: log.data };
1662
+ if (log.stream === "stdout") {
1663
+ stdout += log.data;
1664
+ } else {
1665
+ stderr += log.data;
1666
+ }
1667
+ logBuffer.push(entry);
1668
+ state.resolve?.();
1669
+ }
1670
+ state.consumed = true;
1671
+ state.resolve?.();
1672
+ })();
1673
+ async function* logs() {
1674
+ let index = 0;
1675
+ while (!state.consumed || index < logBuffer.length) {
1676
+ if (index < logBuffer.length) {
1677
+ yield logBuffer[index++];
1678
+ } else {
1679
+ await new Promise((resolve) => {
1680
+ state.resolve = resolve;
1681
+ });
1682
+ state.resolve = null;
1683
+ }
1684
+ }
1685
+ }
1686
+ const result = consumeLogs.then(() => ({
1687
+ stdout,
1688
+ stderr,
1689
+ exitCode: output.exitCode
1690
+ }));
1691
+ return { commandId: output.cmdId, logs, result };
1692
+ },
1693
+ catch: (e) => new SandboxError({ reason: String(e), cause: e })
1694
+ });
1695
+ await updatePromise;
1696
+ return execResult;
1697
+ },
1698
+ getDomain: async (port) => {
1699
+ const sandbox2 = await getSandbox2();
1700
+ if (sandbox2 instanceof Error) {
1701
+ return sandbox2;
1702
+ }
1703
+ try {
1704
+ return sandbox2.domain(port);
1705
+ } catch (e) {
1706
+ return new SandboxError({ reason: String(e), cause: e });
1707
+ }
1708
+ },
1709
+ kill: async ({ commandId, storage: cmdStorage }) => {
1710
+ const instance = await getSandbox2();
1711
+ if (instance instanceof Error) {
1712
+ return instance;
1713
+ }
1714
+ const cmd = await cmdStorage.command.get(commandId);
1715
+ if (cmd instanceof Error) {
1716
+ return new SandboxError({ reason: cmd.message, cause: cmd });
1717
+ }
1718
+ if (cmd && cmd.status === "running") {
1719
+ const result = await cmdStorage.command.set({
1720
+ ...cmd,
1721
+ status: "killed"
1722
+ });
1723
+ if (result instanceof Error) {
1724
+ return new SandboxError({ reason: result.message, cause: result });
1725
+ }
1726
+ }
1727
+ return void 0;
1728
+ },
1729
+ writeFiles: (opts) => writeFiles({ sandbox, ...opts }),
1730
+ lifecycle
1731
+ };
1732
+ if (config.lifecycle?.autoStart !== false) {
1733
+ sandboxPromise = doGetSandbox();
1734
+ }
1735
+ return sandbox;
1736
+ };
1737
+ }
1738
+ });
1739
+
1740
+ // src/sandbox/client.ts
1741
+ function getSandbox({
1742
+ sandboxRecord,
1743
+ storageConfig
1744
+ }) {
1745
+ const { config } = sandboxRecord;
1746
+ switch (config.type) {
1747
+ case "local":
1748
+ return localSandbox({
1749
+ sandboxRecord,
1750
+ storageConfig
1751
+ });
1752
+ case "vercel":
1753
+ return vercelSandbox({
1754
+ sandboxRecord,
1755
+ storageConfig
1756
+ });
1757
+ case "custom":
1758
+ throw new Error("Custom sandboxes are not supported");
1759
+ default:
1760
+ config;
1761
+ throw new Error(
1762
+ `Unknown sandbox type: ${// biome-ignore lint/suspicious/noExplicitAny: .
1763
+ config.type}`
1764
+ );
1765
+ }
1766
+ }
1767
+ var init_client2 = __esm({
1768
+ "src/sandbox/client.ts"() {
1769
+ "use strict";
1770
+ init_local2();
1771
+ init_vercel2();
1772
+ }
1773
+ });
1774
+
1775
+ // src/sandbox/index.ts
1776
+ var sandbox_exports = {};
1777
+ __export(sandbox_exports, {
1778
+ getSandbox: () => getSandbox
1779
+ });
1780
+ var init_sandbox = __esm({
1781
+ "src/sandbox/index.ts"() {
1782
+ "use strict";
1783
+ init_client2();
1784
+ }
1785
+ });
1786
+
1787
+ // src/index.ts
1788
+ var src_exports = {};
1789
+ __export(src_exports, {
1790
+ MessageNotFoundError: () => MessageNotFoundError,
1791
+ SandboxError: () => SandboxError,
1792
+ SandboxNotFoundError: () => SandboxNotFoundError,
1793
+ SessionNotFoundError: () => SessionNotFoundError,
1794
+ StorageError: () => StorageError,
1795
+ agent: () => agent,
1796
+ handleStorageRpc: () => handleStorageRpc,
1797
+ mcp: () => mcp
1798
+ });
1799
+ module.exports = __toCommonJS(src_exports);
1800
+
1801
+ // src/client.ts
1802
+ var import_better_all = require("better-all");
1803
+ var errore4 = __toESM(require("errore"));
1804
+ var import_fast_deep_equal = __toESM(require("fast-deep-equal"));
1805
+ var import_ulid3 = require("ulid");
1806
+ var import_api = require("workflow/api");
1807
+
1808
+ // src/agent-workflow.ts
1809
+ var import_workflow2 = require("workflow");
1810
+
1811
+ // src/agent-workflow-steps.ts
1812
+ var import_ai2 = require("ai");
1813
+ var import_ulid2 = require("ulid");
1814
+ var import_workflow = require("workflow");
1815
+
1816
+ // src/skills/parser.ts
1817
+ function parseSkillFrontmatter(content) {
1818
+ const trimmed = content.trim();
1819
+ if (!trimmed.startsWith("---")) {
1820
+ return null;
1821
+ }
1822
+ const endMarkerIndex = trimmed.indexOf("---", 3);
1823
+ if (endMarkerIndex === -1) {
1824
+ return null;
1825
+ }
1826
+ const frontmatterBlock = trimmed.slice(3, endMarkerIndex).trim();
1827
+ const parsed = parseSimpleYaml(frontmatterBlock);
1828
+ if (!(parsed.name && parsed.description)) {
1829
+ return null;
1830
+ }
1831
+ return {
1832
+ name: String(parsed.name),
1833
+ description: String(parsed.description)
1834
+ };
1835
+ }
1836
+ function parseSimpleYaml(yaml) {
1837
+ const result = {};
1838
+ for (const line of yaml.split("\n")) {
1839
+ const trimmedLine = line.trim();
1840
+ if (!trimmedLine || trimmedLine.startsWith("#")) {
1841
+ continue;
1842
+ }
1843
+ const colonIndex = trimmedLine.indexOf(":");
1844
+ if (colonIndex === -1) {
1845
+ continue;
1846
+ }
1847
+ const key = trimmedLine.slice(0, colonIndex).trim();
1848
+ let value = trimmedLine.slice(colonIndex + 1).trim();
1849
+ if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
1850
+ value = value.slice(1, -1);
1851
+ }
1852
+ if (key) {
1853
+ result[key] = value;
1854
+ }
1855
+ }
1856
+ return result;
1857
+ }
1858
+ function normalizeSkillsDirs(skillsDir) {
1859
+ if (!skillsDir) {
1860
+ return [];
1861
+ }
1862
+ return Array.isArray(skillsDir) ? skillsDir : [skillsDir];
1863
+ }
1864
+
1865
+ // src/skills/discover.ts
1866
+ async function discoverSkillsInSandbox(opts) {
1867
+ const { sandbox, skillsDirs, debug } = opts;
1868
+ const summaries = [];
1869
+ const seenNames = /* @__PURE__ */ new Set();
1870
+ for (const skillsDir of skillsDirs) {
1871
+ const dirSummaries = await discoverSkillsInDirectory({
1872
+ sandbox,
1873
+ skillsDir,
1874
+ debug
1875
+ });
1876
+ for (const summary of dirSummaries) {
1877
+ if (!seenNames.has(summary.name)) {
1878
+ seenNames.add(summary.name);
1879
+ summaries.push(summary);
1880
+ }
1881
+ }
1882
+ }
1883
+ return summaries;
1884
+ }
1885
+ async function discoverSkillsInDirectory(opts) {
1886
+ const { sandbox, skillsDir, debug } = opts;
1887
+ const skillPaths = await findSkillFiles({ sandbox, skillsDir, debug });
1888
+ if (skillPaths.length === 0) {
1889
+ return [];
1890
+ }
1891
+ const summaries = [];
1892
+ for (const skillMdPath of skillPaths) {
1893
+ const summary = await parseSkillFile({ sandbox, skillMdPath, debug });
1894
+ if (summary) {
1895
+ summaries.push(summary);
1896
+ }
1897
+ }
1898
+ return summaries;
1899
+ }
1900
+ async function findSkillFiles(opts) {
1901
+ const { sandbox, skillsDir, debug } = opts;
1902
+ if (debug) {
1903
+ console.log(`[discover] Finding skills in: ${skillsDir}`);
1904
+ }
1905
+ const execResult = await sandbox.exec({
1906
+ command: "find",
1907
+ args: [skillsDir, "-name", "SKILL.md", "-type", "f"]
1908
+ });
1909
+ if (execResult instanceof Error) {
1910
+ if (debug) {
1911
+ console.warn(
1912
+ `[discover] Failed to scan skills directory "${skillsDir}": ${execResult.message}`
1913
+ );
1914
+ }
1915
+ return [];
1916
+ }
1917
+ const { stdout, stderr, exitCode } = await execResult.result;
1918
+ if (debug) {
1919
+ console.log(
1920
+ `[discover] find result: exitCode=${exitCode}, stdout="${stdout.trim()}", stderr="${stderr.trim()}"`
1921
+ );
1922
+ }
1923
+ if (exitCode !== 0) {
1924
+ if (debug) {
1925
+ console.warn(
1926
+ `[discover] Skills directory not found or inaccessible: ${skillsDir}`
1927
+ );
1928
+ }
1929
+ return [];
1930
+ }
1931
+ const paths = stdout.trim().split("\n").filter((p) => p.length > 0);
1932
+ if (debug) {
1933
+ console.log("[discover] Found skill paths:", paths);
1934
+ }
1935
+ return paths;
1936
+ }
1937
+ async function parseSkillFile(opts) {
1938
+ const { sandbox, skillMdPath, debug } = opts;
1939
+ const execResult = await sandbox.exec({
1940
+ command: "cat",
1941
+ args: [skillMdPath]
1942
+ });
1943
+ if (execResult instanceof Error) {
1944
+ if (debug) {
1945
+ console.warn(
1946
+ `[discover] Failed to read skill file "${skillMdPath}": ${execResult.message}`
1947
+ );
1948
+ }
1949
+ return null;
1950
+ }
1951
+ const { stdout, exitCode } = await execResult.result;
1952
+ if (exitCode !== 0) {
1953
+ if (debug) {
1954
+ console.warn(`[discover] Could not read skill file: ${skillMdPath}`);
1955
+ }
1956
+ return null;
1957
+ }
1958
+ const parsed = parseSkillFrontmatter(stdout);
1959
+ if (!parsed) {
1960
+ if (debug) {
1961
+ console.warn(
1962
+ `[discover] Invalid or missing frontmatter in: ${skillMdPath}`
1963
+ );
1964
+ }
1965
+ return null;
1966
+ }
1967
+ return {
1968
+ name: parsed.name,
1969
+ description: parsed.description,
1970
+ skillMdPath
1971
+ };
1972
+ }
1973
+
1974
+ // src/tools/index.ts
1975
+ var import_ai = require("ai");
1976
+ var import_zod = require("zod");
1977
+ function getTools(context) {
1978
+ return {
1979
+ Read: (0, import_ai.tool)({
1980
+ description: "Reads a file and returns its contents with metadata. For files over 200 lines, automatically shows first 100 lines unless a specific line range is provided. Use startLine and endLine parameters to read specific portions of large files.",
1981
+ inputSchema: import_zod.z.object({
1982
+ path: import_zod.z.string().describe("Path to the file relative to workspace root"),
1983
+ startLine: import_zod.z.number().optional().describe(
1984
+ "Starting line number (1-indexed). If provided with endLine, reads exact range regardless of file size."
1985
+ ),
1986
+ endLine: import_zod.z.number().optional().describe(
1987
+ "Ending line number (1-indexed, inclusive). If provided with startLine, reads exact range regardless of file size."
1988
+ )
1989
+ }),
1990
+ outputSchema: import_zod.z.object({
1991
+ content: import_zod.z.string().describe("File content"),
1992
+ metadata: import_zod.z.object({
1993
+ totalLines: import_zod.z.number().describe("Total number of lines in the file"),
1994
+ linesShown: import_zod.z.number().describe("Number of lines included in this response"),
1995
+ startLine: import_zod.z.number().describe("First line number shown (1-indexed)"),
1996
+ endLine: import_zod.z.number().describe("Last line number shown (1-indexed)"),
1997
+ isPaginated: import_zod.z.boolean().describe("Whether this is a partial view of the file"),
1998
+ fileSize: import_zod.z.string().describe("Human-readable file size (e.g., '2.5K', '1.2M')"),
1999
+ path: import_zod.z.string().describe("Path to the file relative to workspace root")
2000
+ })
2001
+ }),
2002
+ execute: async ({ path: path2, startLine, endLine }) => {
2003
+ const filePath = path2.startsWith("/") ? path2.slice(1) : path2;
2004
+ const result = await context.sandbox.exec({
2005
+ command: "bash",
2006
+ args: [
2007
+ "-c",
2008
+ `
2009
+ set -e
2010
+ FILE="$1"
2011
+ START_LINE="$2"
2012
+ END_LINE="$3"
2013
+
2014
+ # Resolve symlinks and check file exists
2015
+ if [ -L "$FILE" ]; then
2016
+ RESOLVED=$(readlink -f "$FILE" 2>/dev/null || echo "")
2017
+ if [ -z "$RESOLVED" ] || [ ! -e "$RESOLVED" ]; then
2018
+ echo "Error: Broken symlink - $FILE points to non-existent target" >&2
2019
+ exit 1
2020
+ fi
2021
+ FILE="$RESOLVED"
2022
+ elif [ ! -e "$FILE" ]; then
2023
+ echo "Error: File not found - $FILE" >&2
2024
+ exit 1
2025
+ fi
2026
+
2027
+ # Get metadata (count actual lines, not just newlines)
2028
+ TOTAL_LINES=$(awk 'END{print NR}' "$FILE")
2029
+ FILE_SIZE=$(ls -lh "$FILE" | awk '{print $5}')
2030
+
2031
+ # Determine range
2032
+ PAGE_SIZE=100
2033
+ if [ -n "$START_LINE" ] && [ -n "$END_LINE" ]; then
2034
+ # Both provided - use exact range
2035
+ ACTUAL_START=$START_LINE
2036
+ ACTUAL_END=$END_LINE
2037
+ elif [ -n "$START_LINE" ]; then
2038
+ # Only startLine - read PAGE_SIZE lines from there
2039
+ ACTUAL_START=$START_LINE
2040
+ ACTUAL_END=$((START_LINE + PAGE_SIZE - 1))
2041
+ [ "$ACTUAL_END" -gt "$TOTAL_LINES" ] && ACTUAL_END=$TOTAL_LINES
2042
+ elif [ -n "$END_LINE" ]; then
2043
+ # Only endLine - read from beginning to endLine
2044
+ ACTUAL_START=1
2045
+ ACTUAL_END=$END_LINE
2046
+ elif [ "$TOTAL_LINES" -gt 200 ]; then
2047
+ # No range, large file - paginate
2048
+ ACTUAL_START=1
2049
+ ACTUAL_END=$PAGE_SIZE
2050
+ else
2051
+ # No range, small file - show all
2052
+ ACTUAL_START=1
2053
+ ACTUAL_END=$TOTAL_LINES
2054
+ fi
2055
+
2056
+ # Output metadata first (separated by ||| for parsing)
2057
+ echo "$TOTAL_LINES|$FILE_SIZE|$ACTUAL_START|$ACTUAL_END"
2058
+ echo "|||CONTENT|||"
2059
+
2060
+ # Read content
2061
+ if [ "$ACTUAL_START" -eq 1 ] && [ "$ACTUAL_END" -eq "$TOTAL_LINES" ]; then
2062
+ cat "$FILE"
2063
+ else
2064
+ sed -n "\${ACTUAL_START},\${ACTUAL_END}p" "$FILE"
2065
+ fi
2066
+ `,
2067
+ "--",
2068
+ filePath,
2069
+ startLine?.toString() || "",
2070
+ endLine?.toString() || ""
2071
+ ]
2072
+ });
2073
+ if (result instanceof Error) {
2074
+ console.error("[Read Tool]", result);
2075
+ throw result;
2076
+ }
2077
+ const { stdout, stderr } = await result.result;
2078
+ if (stderr) {
2079
+ console.error(`[Read Tool] Error: ${stderr}`);
2080
+ return {
2081
+ content: `Error: ${stderr}`,
2082
+ metadata: {
2083
+ totalLines: 0,
2084
+ linesShown: 0,
2085
+ startLine: 0,
2086
+ endLine: 0,
2087
+ isPaginated: false,
2088
+ fileSize: "0",
2089
+ path: filePath
2090
+ }
2091
+ };
2092
+ }
2093
+ const [metadataLine, ...rest] = stdout.split("|||CONTENT|||");
2094
+ const content = rest.join("|||CONTENT|||").trimStart();
2095
+ const [totalLinesStr, fileSize, actualStartStr, actualEndStr] = metadataLine.trim().split("|");
2096
+ const totalLines = Number.parseInt(totalLinesStr, 10);
2097
+ const actualStart = Number.parseInt(actualStartStr, 10);
2098
+ const actualEnd = Number.parseInt(actualEndStr, 10);
2099
+ if (Number.isNaN(totalLines) || Number.isNaN(actualStart) || Number.isNaN(actualEnd)) {
2100
+ console.error(
2101
+ `[Read Tool] Failed to parse metadata: ${metadataLine}`
2102
+ );
2103
+ return {
2104
+ content: `Error: Failed to parse file metadata. Raw output: ${stdout.slice(
2105
+ 0,
2106
+ 500
2107
+ )}`,
2108
+ metadata: {
2109
+ totalLines: 0,
2110
+ linesShown: 0,
2111
+ startLine: 0,
2112
+ endLine: 0,
2113
+ isPaginated: false,
2114
+ fileSize: "unknown",
2115
+ path: filePath
2116
+ }
2117
+ };
2118
+ }
2119
+ return {
2120
+ content,
2121
+ metadata: {
2122
+ totalLines,
2123
+ linesShown: Math.max(0, actualEnd - actualStart + 1),
2124
+ startLine: actualStart,
2125
+ endLine: actualEnd,
2126
+ isPaginated: actualEnd < totalLines,
2127
+ fileSize: fileSize || "unknown",
2128
+ path: filePath
2129
+ }
2130
+ };
2131
+ }
2132
+ }),
2133
+ Grep: (0, import_ai.tool)({
2134
+ description: "Search for patterns in files using ripgrep. Supports regex patterns, file type filtering, and context lines. Returns matching lines with file paths and line numbers. Use this to find code patterns, function definitions, imports, etc.",
2135
+ inputSchema: import_zod.z.object({
2136
+ pattern: import_zod.z.string().describe("Regex pattern to search for (ripgrep syntax)"),
2137
+ path: import_zod.z.string().optional().describe(
2138
+ "Path to search in (defaults to workspace root). Can be a file or directory."
2139
+ ),
2140
+ fileType: import_zod.z.string().optional().describe(
2141
+ "File type to filter by (e.g., 'ts', 'js', 'py', 'md'). Uses ripgrep's built-in type filters."
2142
+ ),
2143
+ glob: import_zod.z.string().optional().describe(
2144
+ "Glob pattern to filter files (e.g., '*.tsx', 'src/**/*.ts')"
2145
+ ),
2146
+ caseSensitive: import_zod.z.boolean().optional().default(true).describe("Whether search is case-sensitive (default: true)"),
2147
+ contextLines: import_zod.z.number().optional().describe(
2148
+ "Number of context lines to show before and after each match"
2149
+ ),
2150
+ maxCount: import_zod.z.number().optional().describe(
2151
+ "Maximum number of matches per file (useful for limiting output)"
2152
+ ),
2153
+ filesWithMatches: import_zod.z.boolean().optional().default(false).describe(
2154
+ "Only show file paths that contain matches, not the matching lines themselves"
2155
+ )
2156
+ }),
2157
+ outputSchema: import_zod.z.object({
2158
+ matches: import_zod.z.string().describe(
2159
+ "Search results with file paths, line numbers, and matching content"
2160
+ ),
2161
+ summary: import_zod.z.object({
2162
+ matchCount: import_zod.z.number().describe("Number of matches found"),
2163
+ fileCount: import_zod.z.number().describe("Number of files containing matches"),
2164
+ searchPath: import_zod.z.string().describe("Path that was searched"),
2165
+ pattern: import_zod.z.string().describe("Pattern that was searched for")
2166
+ })
2167
+ }),
2168
+ execute: async ({
2169
+ pattern,
2170
+ path: path2,
2171
+ fileType,
2172
+ glob,
2173
+ caseSensitive,
2174
+ contextLines,
2175
+ maxCount,
2176
+ filesWithMatches
2177
+ }) => {
2178
+ let searchPath = path2 ?? ".";
2179
+ if (searchPath.startsWith("/")) {
2180
+ searchPath = searchPath.slice(1);
2181
+ }
2182
+ const args = [];
2183
+ args.push("--line-number");
2184
+ args.push("--heading");
2185
+ args.push("--color", "never");
2186
+ if (!caseSensitive) {
2187
+ args.push("-i");
2188
+ }
2189
+ if (fileType) {
2190
+ args.push("--type", fileType);
2191
+ }
2192
+ if (glob) {
2193
+ args.push("--glob", glob);
2194
+ }
2195
+ if (contextLines !== void 0) {
2196
+ args.push("-C", String(contextLines));
2197
+ }
2198
+ if (maxCount !== void 0) {
2199
+ args.push("--max-count", String(maxCount));
2200
+ }
2201
+ if (filesWithMatches) {
2202
+ args.push("--files-with-matches");
2203
+ }
2204
+ args.push("--", pattern, searchPath);
2205
+ const result = await context.sandbox.exec({ command: "rg", args });
2206
+ if (result instanceof Error) {
2207
+ console.error("[Grep Tool]", result);
2208
+ throw result;
2209
+ }
2210
+ const { stdout, stderr } = await result.result;
2211
+ if (stderr && !stderr.toLowerCase().includes("no matches")) {
2212
+ console.error(`[Grep Tool] Warning: ${stderr}`);
2213
+ }
2214
+ const MAX_GREP_OUTPUT_CHARS = 5e4;
2215
+ let finalOutput = stdout;
2216
+ let wasTruncated = false;
2217
+ if (finalOutput.length > MAX_GREP_OUTPUT_CHARS) {
2218
+ finalOutput = finalOutput.slice(0, MAX_GREP_OUTPUT_CHARS) + "\n\n[Output truncated - use more specific pattern or path]";
2219
+ wasTruncated = true;
2220
+ }
2221
+ const lines = finalOutput.trim().split("\n").filter((l) => l.length > 0);
2222
+ const fileCount = filesWithMatches ? lines.length : new Set(
2223
+ lines.filter((l) => !l.startsWith(" ") && l.includes(":")).map((l) => l.split(":")[0])
2224
+ ).size;
2225
+ return {
2226
+ matches: finalOutput || "(no matches found)",
2227
+ summary: {
2228
+ matchCount: filesWithMatches ? 0 : lines.filter((l) => l.includes(":")).length,
2229
+ fileCount,
2230
+ searchPath,
2231
+ pattern,
2232
+ wasTruncated
2233
+ }
2234
+ };
2235
+ }
2236
+ }),
2237
+ List: (0, import_ai.tool)({
2238
+ description: "Recursively list directory contents. Use this to understand the codebase structure, find files, or explore directories. Control depth to balance detail vs. overview. Depth 1 shows immediate children, depth 2 includes subdirectories, etc.",
2239
+ inputSchema: import_zod.z.object({
2240
+ path: import_zod.z.string().optional().describe("Path to list (defaults to workspace root)"),
2241
+ depth: import_zod.z.number().optional().describe(
2242
+ "Maximum depth to traverse. Choose based on context: 1-2 for quick overview, 3-4 for detailed exploration, 5+ for comprehensive mapping"
2243
+ ),
2244
+ includeHidden: import_zod.z.boolean().optional().default(false).describe(
2245
+ "Include hidden files and directories (those starting with '.')"
2246
+ ),
2247
+ filesOnly: import_zod.z.boolean().optional().default(false).describe("Only show files, not directories"),
2248
+ pattern: import_zod.z.string().optional().describe("Glob pattern to filter results (e.g., '*.ts', '*test*')")
2249
+ }),
2250
+ outputSchema: import_zod.z.object({
2251
+ listing: import_zod.z.string().describe(
2252
+ "Directory tree listing showing paths relative to search root"
2253
+ ),
2254
+ summary: import_zod.z.object({
2255
+ totalItems: import_zod.z.number().describe("Total number of items found"),
2256
+ totalFiles: import_zod.z.number().describe("Total number of files found"),
2257
+ totalDirs: import_zod.z.number().describe("Total number of directories found"),
2258
+ searchPath: import_zod.z.string().describe("Path that was listed"),
2259
+ depth: import_zod.z.number().optional().describe("Maximum depth used (if specified)")
2260
+ })
2261
+ }),
2262
+ execute: async ({ path: path2, depth, includeHidden, filesOnly, pattern }) => {
2263
+ const searchPath = path2 ?? ".";
2264
+ const result = await context.sandbox.exec({
2265
+ command: "bash",
2266
+ args: [
2267
+ "-c",
2268
+ `
2269
+ set -e
2270
+ SEARCH_PATH="$1"
2271
+ DEPTH="$2"
2272
+ INCLUDE_HIDDEN="$3"
2273
+ FILES_ONLY="$4"
2274
+ PATTERN="$5"
2275
+
2276
+ # Build find command arguments
2277
+ FIND_ARGS=""
2278
+ [ -n "$DEPTH" ] && FIND_ARGS="$FIND_ARGS -maxdepth $DEPTH"
2279
+ [ "$INCLUDE_HIDDEN" != "true" ] && FIND_ARGS="$FIND_ARGS ! -path '*/.*'"
2280
+ [ "$FILES_ONLY" = "true" ] && FIND_ARGS="$FIND_ARGS -type f"
2281
+ [ -n "$PATTERN" ] && FIND_ARGS="$FIND_ARGS -name '$PATTERN'"
2282
+
2283
+ # Get listing
2284
+ LISTING=$(eval "find '$SEARCH_PATH' $FIND_ARGS" 2>/dev/null | sort)
2285
+
2286
+ # Get counts
2287
+ COUNT_ARGS=""
2288
+ [ -n "$DEPTH" ] && COUNT_ARGS="$COUNT_ARGS -maxdepth $DEPTH"
2289
+ [ "$INCLUDE_HIDDEN" != "true" ] && COUNT_ARGS="$COUNT_ARGS ! -path '*/.*'"
2290
+
2291
+ FILE_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type f" 2>/dev/null | wc -l)
2292
+ DIR_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type d" 2>/dev/null | wc -l)
2293
+
2294
+ # Output: counts first, then listing
2295
+ echo "$FILE_COUNT|$DIR_COUNT"
2296
+ echo "|||LISTING|||"
2297
+ echo "$LISTING" | sed "s|^$SEARCH_PATH|.|"
2298
+ `,
2299
+ "--",
2300
+ searchPath,
2301
+ depth?.toString() || "",
2302
+ includeHidden ? "true" : "false",
2303
+ filesOnly ? "true" : "false",
2304
+ pattern || ""
2305
+ ]
2306
+ });
2307
+ if (result instanceof Error) {
2308
+ console.error("[List Tool]", result);
2309
+ throw result;
2310
+ }
2311
+ const { stdout, stderr } = await result.result;
2312
+ if (stderr) {
2313
+ console.warn(`[List Tool] stderr: ${stderr}`);
2314
+ }
2315
+ const [countsLine, ...rest] = stdout.split("|||LISTING|||");
2316
+ const listing = rest.join("|||LISTING|||").trim();
2317
+ const [fileCountStr, dirCountStr] = countsLine.trim().split("|");
2318
+ const totalFiles = Number.parseInt(fileCountStr, 10) || 0;
2319
+ const totalDirs = Number.parseInt(dirCountStr, 10) || 0;
2320
+ const lines = listing.split("\n").filter((l) => l.length > 0);
2321
+ return {
2322
+ listing,
2323
+ summary: {
2324
+ totalItems: lines.length,
2325
+ totalFiles,
2326
+ totalDirs,
2327
+ searchPath,
2328
+ depth
2329
+ }
2330
+ };
2331
+ }
2332
+ }),
2333
+ Bash: (0, import_ai.tool)({
2334
+ description: "Executes a bash command inside the workspace. CWD persists between commands within a session. Use waitUntil:0 for background processes (dev servers).",
2335
+ inputSchema: import_zod.z.object({
2336
+ command: import_zod.z.string().describe("The shell command to execute"),
2337
+ waitUntil: import_zod.z.number().optional().describe(
2338
+ "Max ms to wait for completion (default: 30000). Use 0 to run in background and return immediately."
2339
+ )
2340
+ }),
2341
+ outputSchema: import_zod.z.object({
2342
+ pid: import_zod.z.number().describe(
2343
+ "System PID (0 for foreground, >0 for background - use to kill)"
2344
+ ),
2345
+ output: import_zod.z.string().describe("Command stdout+stderr combined (empty for background)"),
2346
+ exitCode: import_zod.z.number().describe("Exit code (-1 for background/running)"),
2347
+ status: import_zod.z.enum(["running", "completed", "failed"]).describe("Process status"),
2348
+ cwd: import_zod.z.string().describe("Current working directory after command"),
2349
+ outputFile: import_zod.z.string().describe("Path to output log (for background processes)")
2350
+ }),
2351
+ execute: async ({ command, waitUntil }) => {
2352
+ const { createProcessManager: createProcessManager2 } = await Promise.resolve().then(() => (init_process_manager(), process_manager_exports));
2353
+ const processManager = createProcessManager2({
2354
+ sandbox: context.sandbox,
2355
+ sessionId: context.input.sessionId,
2356
+ generateId: () => crypto.randomUUID()
2357
+ });
2358
+ await processManager.init();
2359
+ return processManager.run({ command, waitUntil });
2360
+ }
2361
+ }),
2362
+ ExecuteMCPTool: (0, import_ai.tool)({
2363
+ description: "Execute a tool from an MCP server. Use this to call custom tools provided by the application.",
2364
+ inputSchema: import_zod.z.object({
2365
+ server: import_zod.z.string().describe("MCP server name"),
2366
+ tool: import_zod.z.string().describe("Tool name to execute"),
2367
+ input: import_zod.z.unknown().describe("Tool input matching the tool's schema")
2368
+ }),
2369
+ outputSchema: import_zod.z.object({
2370
+ result: import_zod.z.unknown().describe("Tool execution result"),
2371
+ truncated: import_zod.z.boolean().optional().describe("Whether the result was truncated due to size"),
2372
+ totalChars: import_zod.z.number().optional().describe("Total characters in full output (only if truncated)"),
2373
+ returnedChars: import_zod.z.number().optional().describe("Characters returned in this response (only if truncated)"),
2374
+ fullOutputPath: import_zod.z.string().optional().describe("Path to full output file if truncated")
2375
+ }),
2376
+ execute: async ({ server, tool: toolName, input }) => {
2377
+ const serverConfig = context.mcp?.find((s) => s.name === server);
2378
+ if (!serverConfig) {
2379
+ throw new Error(`Unknown MCP server: ${server}`);
2380
+ }
2381
+ const parsedInput = typeof input === "string" ? JSON.parse(input) : input;
2382
+ const timestamp = Date.now();
2383
+ const outputDir = `.agent/mcp/${server}/.outputs`;
2384
+ const outputFile = `${outputDir}/${timestamp}-${toolName}.txt`;
2385
+ const [res] = await Promise.all([
2386
+ fetch(serverConfig.url, {
2387
+ method: "POST",
2388
+ headers: { "Content-Type": "application/json" },
2389
+ body: JSON.stringify({
2390
+ server,
2391
+ tool: toolName,
2392
+ input: parsedInput,
2393
+ sessionId: context.input.sessionId,
2394
+ messageId: context.event.assistantMessageId,
2395
+ hookToken: context.event.hookToken
2396
+ })
2397
+ }),
2398
+ context.sandbox.exec({ command: "mkdir", args: ["-p", outputDir] })
2399
+ ]);
2400
+ if (!res.ok) {
2401
+ const error = await res.text();
2402
+ throw new Error(`MCP tool call failed: ${error}`);
2403
+ }
2404
+ const json = await res.json();
2405
+ if (json.error) {
2406
+ throw new Error(`MCP tool call failed: ${json.error.message}`);
2407
+ }
2408
+ if (json.result === void 0) {
2409
+ throw new Error("MCP tool call failed: No result in response");
2410
+ }
2411
+ const result = json.result;
2412
+ const MAX_OUTPUT_CHARS = 24e3;
2413
+ const resultStr = JSON.stringify(result, null, 2);
2414
+ if (resultStr.length <= MAX_OUTPUT_CHARS) {
2415
+ return { result };
2416
+ }
2417
+ await context.sandbox.writeFiles({
2418
+ files: [{ path: `${timestamp}-${toolName}.txt`, content: resultStr }],
2419
+ destPath: outputDir
2420
+ });
2421
+ const truncatedResult = resultStr.slice(0, MAX_OUTPUT_CHARS) + `
2422
+
2423
+ [Output truncated at ~6k tokens. Full output saved to: ${outputFile}]`;
2424
+ return {
2425
+ result: truncatedResult,
2426
+ truncated: true,
2427
+ totalChars: resultStr.length,
2428
+ returnedChars: MAX_OUTPUT_CHARS,
2429
+ fullOutputPath: outputFile
2430
+ };
2431
+ }
2432
+ })
2433
+ };
2434
+ }
2435
+
2436
+ // src/utils/ui.ts
2437
+ function assembleUIMessages(opts) {
2438
+ const partsByMessage = /* @__PURE__ */ new Map();
2439
+ for (const part of opts.parts) {
2440
+ const existing = partsByMessage.get(part.messageId) ?? [];
2441
+ existing.push(part);
2442
+ partsByMessage.set(part.messageId, existing);
2443
+ }
2444
+ return opts.messages.map((m) => {
2445
+ const messageParts = partsByMessage.get(m.id) ?? [];
2446
+ messageParts.sort((a, b) => a.index - b.index);
2447
+ return {
2448
+ id: m.id,
2449
+ role: m.role,
2450
+ parts: messageParts.map((p) => p.part)
2451
+ };
2452
+ });
2453
+ }
2454
+
2455
+ // src/agent-workflow-steps.ts
2456
+ var BASE_SYSTEM_PROMPT = "You are an AI assistant with basic tools to interact with your environment. Explore and work freely.";
2457
+ function joinPromptSections(...sections) {
2458
+ return sections.filter((s) => s?.trim()).join("\n\n");
2459
+ }
2460
+ function buildSkillsContext(skills) {
2461
+ if (skills.length === 0) {
2462
+ return "";
2463
+ }
2464
+ const skillLines = skills.map((s) => `- ${s.name}: ${s.description}
2465
+ Path: ${s.skillMdPath}`).join("\n");
2466
+ return `## Available Skills
2467
+ ${skillLines}
2468
+
2469
+ You can use the Read tool to read any skill's SKILL.md file to learn more about it.`;
2470
+ }
2471
+ function buildMcpToolsContext(mcp2) {
2472
+ if (!mcp2 || mcp2.length === 0) {
2473
+ return "";
2474
+ }
2475
+ const serverDocs = mcp2.filter(
2476
+ (s) => Array.isArray(s.tools) && s.tools.length > 0
2477
+ ).map((server) => {
2478
+ const toolDocs = server.tools.map((t) => {
2479
+ let doc = ` - ${t.name}: ${t.description}
2480
+ Input: ${JSON.stringify(t.inputSchema)}`;
2481
+ if (t.outputSchema) {
2482
+ doc += `
2483
+ Output: ${JSON.stringify(t.outputSchema)}`;
2484
+ }
2485
+ return doc;
2486
+ }).join("\n");
2487
+ return `### ${server.name}
2488
+ ${server.description ?? ""}
2489
+ ${toolDocs}`;
2490
+ }).join("\n\n");
2491
+ if (!serverDocs) {
2492
+ return "";
2493
+ }
2494
+ return `## MCP Tools
2495
+ Use the ExecuteMCPTool to call these custom tools:
2496
+
2497
+ ${serverDocs}`;
2498
+ }
2499
+ async function completeMessageStep({
2500
+ assistantMessageId,
2501
+ input,
2502
+ writable
2503
+ }) {
2504
+ "use step";
2505
+ const { getStorage: getStorage2 } = await Promise.resolve().then(() => (init_storage(), storage_exports));
2506
+ const storage = getStorage2(input.storageConfig);
2507
+ const message = await storage.message.get(assistantMessageId);
2508
+ if (message instanceof Error) {
2509
+ throw message;
2510
+ }
2511
+ if (!message) {
2512
+ throw new Error(`Message ${assistantMessageId} not found`);
2513
+ }
2514
+ const result = await storage.message.set({
2515
+ ...message,
2516
+ completedAt: Date.now()
2517
+ });
2518
+ if (result instanceof Error) {
2519
+ throw result;
2520
+ }
2521
+ await writable.close();
2522
+ }
2523
+ async function streamTextStep({
2524
+ assistantMessageId,
2525
+ input,
2526
+ event,
2527
+ writable,
2528
+ lastPartIndex
2529
+ }) {
2530
+ "use step";
2531
+ const { getStorage: getStorage2 } = await Promise.resolve().then(() => (init_client(), client_exports));
2532
+ const { getSandbox: getSandbox2 } = await Promise.resolve().then(() => (init_sandbox(), sandbox_exports));
2533
+ const storage = getStorage2(input.storageConfig);
2534
+ const session = await storage.session.get(input.sessionId);
2535
+ if (session instanceof Error) {
2536
+ throw session;
2537
+ }
2538
+ session;
2539
+ const sandboxRecord = session.sandboxId ? await storage.sandbox.get(session.sandboxId) : null;
2540
+ if (sandboxRecord instanceof Error) {
2541
+ throw sandboxRecord;
2542
+ }
2543
+ if (!sandboxRecord) {
2544
+ throw new import_workflow.FatalError(`Sandbox not found for session ${input.sessionId}`);
2545
+ }
2546
+ const sandbox = getSandbox2({
2547
+ sandboxRecord,
2548
+ storageConfig: input.storageConfig
2549
+ });
2550
+ const [messagesResult, partsResult, skills] = await Promise.all([
2551
+ storage.message.list(input.sessionId),
2552
+ storage.part.listBySession(input.sessionId),
2553
+ session.skillsDir && session.skillsDir.length > 0 ? discoverSkillsInSandbox({ sandbox, skillsDirs: session.skillsDir }) : Promise.resolve([])
2554
+ ]);
2555
+ if (messagesResult instanceof Error) {
2556
+ throw messagesResult;
2557
+ }
2558
+ if (partsResult instanceof Error) {
2559
+ throw partsResult;
2560
+ }
2561
+ const uiMessages = assembleUIMessages({
2562
+ messages: messagesResult.items,
2563
+ parts: partsResult.items
2564
+ });
2565
+ const mcp2 = session.mcp;
2566
+ const systemPrompt = joinPromptSections(
2567
+ BASE_SYSTEM_PROMPT,
2568
+ session.instructions,
2569
+ buildSkillsContext(skills),
2570
+ buildMcpToolsContext(mcp2)
2571
+ );
2572
+ if (!session.model) {
2573
+ throw new import_workflow.FatalError("Session model is not set");
2574
+ }
2575
+ const result = (0, import_ai2.streamText)({
2576
+ messages: await (0, import_ai2.convertToModelMessages)(uiMessages),
2577
+ tools: getTools({ input, event, mcp: mcp2, sandbox }),
2578
+ system: systemPrompt,
2579
+ model: session.model
2580
+ });
2581
+ const stepParts = [];
2582
+ await result.toUIMessageStream({
2583
+ generateMessageId: () => assistantMessageId,
2584
+ onFinish: ({ messages }) => {
2585
+ for (const m of messages) {
2586
+ if (m.role === "assistant") {
2587
+ stepParts.push(...m.parts);
2588
+ }
2589
+ }
2590
+ }
2591
+ }).pipeTo(writable, { preventClose: true });
2592
+ await Promise.all(
2593
+ stepParts.map(async (uiPart, index) => {
2594
+ const result2 = await storage.part.set({
2595
+ id: `part_${(0, import_ulid2.ulid)()}`,
2596
+ index: lastPartIndex + index,
2597
+ messageId: assistantMessageId,
2598
+ sessionId: input.sessionId,
2599
+ part: uiPart
2600
+ });
2601
+ if (result2 instanceof Error) {
2602
+ throw result2;
2603
+ }
2604
+ return result2;
2605
+ })
2606
+ );
2607
+ return {
2608
+ finishReason: await result.finishReason,
2609
+ lastPartIndex: lastPartIndex + stepParts.length
2610
+ };
2611
+ }
2612
+
2613
+ // src/agent-workflow.ts
2614
+ var agentMessageHook = (0, import_workflow2.defineHook)();
2615
+ async function agentWorkflow({
2616
+ input,
2617
+ event
2618
+ }) {
2619
+ "use workflow";
2620
+ const messageHook = agentMessageHook.create({ token: input.sessionId });
2621
+ const iterator = messageHook[Symbol.asyncIterator]();
2622
+ let pendingNext = iterator.next();
2623
+ await onMessage({ event, input }).catch((e) => {
2624
+ if (import_workflow2.FatalError.is(e)) {
2625
+ console.error("Message processing failed permanently:", e.message);
2626
+ return;
2627
+ }
2628
+ throw e;
2629
+ });
2630
+ while (true) {
2631
+ const result = await pendingNext;
2632
+ if (result.done) {
2633
+ console.error("Unexpected: message hook iterator done");
2634
+ break;
2635
+ }
2636
+ await onMessage({ event: result.value, input }).catch((e) => {
2637
+ if (import_workflow2.FatalError.is(e)) {
2638
+ console.error("Message processing failed permanently:", e.message);
2639
+ return;
2640
+ }
2641
+ throw e;
2642
+ });
2643
+ pendingNext = iterator.next();
2644
+ }
2645
+ }
2646
+ async function onMessage({
2647
+ event,
2648
+ input
2649
+ }) {
2650
+ const writable = (0, import_workflow2.getWritable)({ namespace: event.assistantMessageId });
2651
+ let finishReason;
2652
+ let lastPartIndex = 0;
2653
+ while (finishReason !== "stop") {
2654
+ try {
2655
+ const result = await streamTextStep({
2656
+ assistantMessageId: event.assistantMessageId,
2657
+ writable,
2658
+ input,
2659
+ event,
2660
+ lastPartIndex
2661
+ });
2662
+ finishReason = result.finishReason;
2663
+ lastPartIndex = result.lastPartIndex;
2664
+ } catch (err) {
2665
+ console.error(err);
2666
+ throw err;
2667
+ }
2668
+ }
2669
+ await completeMessageStep({
2670
+ assistantMessageId: event.assistantMessageId,
2671
+ input,
2672
+ writable
2673
+ });
2674
+ }
2675
+
2676
+ // src/client.ts
2677
+ init_errors();
2678
+
2679
+ // src/mcp/index.ts
2680
+ var import_zod3 = require("zod");
2681
+ function serializeMcpConfig(mcp2) {
2682
+ return Object.entries(mcp2).map(([name, server]) => ({
2683
+ name,
2684
+ url: server.url,
2685
+ description: server.description,
2686
+ tools: Object.entries(server.tools).map(([toolName, tool2]) => ({
2687
+ name: toolName,
2688
+ description: tool2.description,
2689
+ inputSchema: import_zod3.z.toJSONSchema(tool2.inputSchema),
2690
+ outputSchema: tool2.outputSchema ? import_zod3.z.toJSONSchema(tool2.outputSchema) : void 0
2691
+ }))
2692
+ }));
2693
+ }
2694
+ function mcp(opts) {
2695
+ return opts;
2696
+ }
2697
+
2698
+ // src/client.ts
2699
+ init_sandbox();
2700
+ init_storage();
2701
+
2702
+ // src/utils/message.ts
2703
+ function toMessageAndParts({
2704
+ sessionId,
2705
+ id,
2706
+ input,
2707
+ generateId,
2708
+ defaultRole
2709
+ }) {
2710
+ const messageId = id ?? generateId("message");
2711
+ if (typeof input === "string") {
2712
+ return {
2713
+ message: {
2714
+ id: messageId,
2715
+ sessionId,
2716
+ role: defaultRole,
2717
+ createdAt: Date.now(),
2718
+ completedAt: Date.now(),
2719
+ mcpContext: null
2720
+ },
2721
+ parts: [
2722
+ {
2723
+ sessionId,
2724
+ messageId,
2725
+ id: generateId("part"),
2726
+ index: 0,
2727
+ part: { type: "text", text: input }
2728
+ }
2729
+ ]
2730
+ };
2731
+ }
2732
+ return {
2733
+ message: {
2734
+ sessionId,
2735
+ id: messageId,
2736
+ createdAt: Date.now(),
2737
+ completedAt: Date.now(),
2738
+ mcpContext: null,
2739
+ role: input.role ?? defaultRole
2740
+ },
2741
+ parts: input.parts.map((part, index) => ({
2742
+ sessionId,
2743
+ messageId,
2744
+ id: generateId("part"),
2745
+ index,
2746
+ part
2747
+ }))
2748
+ };
2749
+ }
2750
+
2751
+ // src/client.ts
2752
+ var agent = (options) => {
2753
+ const storageConfig = options.storage ?? { type: "vercel" };
2754
+ const storage = getStorage(storageConfig);
2755
+ const generateId = options.generateId ?? ((resource) => `${resource}_${(0, import_ulid3.ulid)()}`);
2756
+ const sandbox = async (sandboxId, config) => {
2757
+ const existingRecord = await storage.sandbox.get(sandboxId);
2758
+ if (existingRecord instanceof Error) {
2759
+ if (!(existingRecord instanceof SandboxNotFoundError)) {
2760
+ throw existingRecord;
2761
+ }
2762
+ const sandboxRecord = await storage.sandbox.set({
2763
+ id: sandboxId,
2764
+ acquiringLockAt: null,
2765
+ acquiringLockId: null,
2766
+ createdAt: Date.now(),
2767
+ lastActivityAt: Date.now(),
2768
+ config: config ?? { type: "vercel" }
2769
+ });
2770
+ if (sandboxRecord instanceof Error) {
2771
+ throw sandboxRecord;
2772
+ }
2773
+ return getSandbox({ storageConfig, sandboxRecord });
2774
+ }
2775
+ const newConfig = config ?? existingRecord.config;
2776
+ if (!(0, import_fast_deep_equal.default)(existingRecord.config, newConfig)) {
2777
+ throw new SandboxError({
2778
+ reason: "sandbox config changes are not supported"
2779
+ });
2780
+ }
2781
+ return getSandbox({ storageConfig, sandboxRecord: existingRecord });
2782
+ };
2783
+ return {
2784
+ session: async (sessionId, sessionOptions) => {
2785
+ let session = await storage.session.get(sessionId);
2786
+ const sandboxInput = sessionOptions?.sandbox ?? options.sandbox;
2787
+ const sandboxId = typeof sandboxInput === "string" ? sandboxInput : (
2788
+ // biome-ignore lint/style/noNestedTernary: ok
2789
+ session instanceof Error ? generateId("sandbox") : session.sandboxId ?? generateId("sandbox")
2790
+ );
2791
+ const sandboxPromise = sandbox(
2792
+ sandboxId,
2793
+ typeof sandboxInput === "string" ? void 0 : sandboxInput
2794
+ );
2795
+ const mcpConfig = options.mcp || sessionOptions?.mcp ? { ...options.mcp, ...sessionOptions?.mcp } : null;
2796
+ const skillsDir = sessionOptions?.skillsDir ?? options.skillsDir ?? null;
2797
+ if (session instanceof Error) {
2798
+ if (session instanceof SessionNotFoundError) {
2799
+ const instructions = sessionOptions?.instructions ?? options.instructions ?? null;
2800
+ const model = sessionOptions?.model ?? options.model ?? null;
2801
+ if (!model) {
2802
+ throw new SessionError({
2803
+ id: sessionId,
2804
+ reason: "Model is not set"
2805
+ });
2806
+ }
2807
+ session = await storage.session.set({
2808
+ id: sessionId,
2809
+ createdAt: Date.now(),
2810
+ updatedAt: Date.now(),
2811
+ lastMessageId: null,
2812
+ runId: null,
2813
+ tags: sessionOptions?.tags ?? null,
2814
+ instructions,
2815
+ model,
2816
+ mcp: mcpConfig ? serializeMcpConfig(mcpConfig) : null,
2817
+ sandboxId,
2818
+ skillsDir: skillsDir ? normalizeSkillsDirs(skillsDir) : null,
2819
+ hookToken: null
2820
+ });
2821
+ } else {
2822
+ throw session;
2823
+ }
2824
+ } else {
2825
+ const updated = {
2826
+ ...session,
2827
+ ...sessionOptions,
2828
+ mcp: mcpConfig ? serializeMcpConfig(mcpConfig) : session.mcp,
2829
+ sandboxId,
2830
+ skillsDir: skillsDir ? normalizeSkillsDirs(skillsDir) : null
2831
+ };
2832
+ if (!(0, import_fast_deep_equal.default)(updated, session)) {
2833
+ session = await storage.session.set(updated);
2834
+ if (session instanceof Error) {
2835
+ throw session;
2836
+ }
2837
+ }
2838
+ }
2839
+ return {
2840
+ send: async ({
2841
+ input,
2842
+ mcpContext
2843
+ }) => {
2844
+ const newMessages = [];
2845
+ const newParts = [];
2846
+ if (Array.isArray(input)) {
2847
+ for (const i of input) {
2848
+ const { message, parts } = toMessageAndParts({
2849
+ sessionId,
2850
+ input: i,
2851
+ defaultRole: "user",
2852
+ generateId: () => generateId("message")
2853
+ });
2854
+ newMessages.push(message);
2855
+ newParts.push(...parts);
2856
+ }
2857
+ } else {
2858
+ const { message, parts } = toMessageAndParts({
2859
+ sessionId,
2860
+ input,
2861
+ defaultRole: "user",
2862
+ generateId: () => generateId("message")
2863
+ });
2864
+ newMessages.push(message);
2865
+ newParts.push(...parts);
2866
+ }
2867
+ const assistantMessageId = generateId("message");
2868
+ newMessages.push({
2869
+ id: assistantMessageId,
2870
+ sessionId,
2871
+ role: "assistant",
2872
+ createdAt: Date.now(),
2873
+ completedAt: null,
2874
+ mcpContext
2875
+ });
2876
+ const now = Date.now();
2877
+ const session2 = await storage.session.get(sessionId);
2878
+ if (session2 instanceof Error) {
2879
+ throw session2;
2880
+ }
2881
+ const agentInput = {
2882
+ storageConfig,
2883
+ sessionId
2884
+ };
2885
+ const event = {
2886
+ assistantMessageId,
2887
+ hookToken: crypto.randomUUID()
2888
+ };
2889
+ return await errore4.tryAsync({
2890
+ try: async () => {
2891
+ await (0, import_better_all.all)({
2892
+ async saveMessages() {
2893
+ await Promise.all(newMessages.map(storage.message.set));
2894
+ },
2895
+ async saveParts() {
2896
+ await Promise.all(newParts.map(storage.part.set));
2897
+ },
2898
+ async resumeOrStartAgentWorkflow() {
2899
+ if (session2.runId) {
2900
+ await agentMessageHook.resume(sessionId, event);
2901
+ } else {
2902
+ const startResult = await (0, import_api.start)(agentWorkflow, [
2903
+ { input: agentInput, event }
2904
+ ]);
2905
+ session2.runId = startResult.runId;
2906
+ }
2907
+ },
2908
+ async updateSession() {
2909
+ await this.$.resumeOrStartAgentWorkflow;
2910
+ const result = await storage.session.set({
2911
+ ...session2,
2912
+ updatedAt: now,
2913
+ lastMessageId: assistantMessageId,
2914
+ hookToken: event.hookToken
2915
+ });
2916
+ if (result instanceof Error) {
2917
+ throw result;
2918
+ }
2919
+ return result;
2920
+ }
2921
+ });
2922
+ },
2923
+ catch: (e) => {
2924
+ if (e instanceof Error) {
2925
+ throw e;
2926
+ }
2927
+ return new SessionError({
2928
+ id: sessionId,
2929
+ reason: `failed to send: ${String(e)}`,
2930
+ cause: e
2931
+ });
2932
+ }
2933
+ });
2934
+ },
2935
+ stream: async (opts) => {
2936
+ const session2 = await storage.session.get(sessionId);
2937
+ if (session2 instanceof Error) {
2938
+ return session2;
2939
+ }
2940
+ const messageId = opts?.messageId ?? session2.lastMessageId;
2941
+ if (!messageId) {
2942
+ return new SessionError({
2943
+ id: sessionId,
2944
+ reason: "No message associated with session"
2945
+ });
2946
+ }
2947
+ if (!session2.runId) {
2948
+ return new SessionError({
2949
+ id: sessionId,
2950
+ reason: "No run associated with session"
2951
+ });
2952
+ }
2953
+ const run = (0, import_api.getRun)(session2.runId);
2954
+ return run.getReadable({ namespace: messageId });
2955
+ },
2956
+ ui: async () => {
2957
+ const [messagesResult, partsResult] = await Promise.all([
2958
+ storage.message.list(sessionId),
2959
+ storage.part.listBySession(sessionId)
2960
+ ]);
2961
+ if (messagesResult instanceof Error) {
2962
+ return messagesResult;
2963
+ }
2964
+ if (partsResult instanceof Error) {
2965
+ return partsResult;
2966
+ }
2967
+ const messages = assembleUIMessages({
2968
+ messages: messagesResult.items,
2969
+ parts: partsResult.items
2970
+ });
2971
+ const lastMessage = messagesResult.items.at(-1);
2972
+ const isStreaming = lastMessage?.role === "assistant" && lastMessage.completedAt === null;
2973
+ const streamingMessageId = isStreaming ? lastMessage.id : null;
2974
+ return { messages, streamingMessageId };
2975
+ },
2976
+ tag: {
2977
+ list: async () => {
2978
+ const session2 = await storage.session.get(sessionId);
2979
+ if (session2 instanceof Error) {
2980
+ return session2;
2981
+ }
2982
+ return session2.tags ?? {};
2983
+ },
2984
+ get: async (key) => {
2985
+ const session2 = await storage.session.get(sessionId);
2986
+ if (session2 instanceof Error) {
2987
+ return session2;
2988
+ }
2989
+ return session2.tags?.[key];
2990
+ },
2991
+ set: async (key, value) => {
2992
+ const result = await storage.session.tag.set({
2993
+ sessionId,
2994
+ tags: { [key]: value }
2995
+ });
2996
+ if (result instanceof Error) {
2997
+ return result;
2998
+ }
2999
+ return void 0;
3000
+ },
3001
+ setMany: async (tags) => {
3002
+ const result = await storage.session.tag.set({
3003
+ sessionId,
3004
+ tags
3005
+ });
3006
+ if (result instanceof Error) {
3007
+ return result;
3008
+ }
3009
+ return void 0;
3010
+ }
3011
+ },
3012
+ sandbox: await sandboxPromise
3013
+ };
3014
+ },
3015
+ sandbox,
3016
+ storage,
3017
+ handleMcpToolCall: async (body, handlers) => {
3018
+ const { server, tool: tool2, input, sessionId, messageId, hookToken } = body;
3019
+ const handlerKey = `${server}/${tool2}`;
3020
+ const handler = handlers[handlerKey];
3021
+ if (!handler) {
3022
+ return {
3023
+ error: {
3024
+ code: "HANDLER_NOT_FOUND",
3025
+ message: `No handler for ${handlerKey}`
3026
+ }
3027
+ };
3028
+ }
3029
+ const session = await storage.session.get(sessionId);
3030
+ if (session instanceof Error) {
3031
+ return {
3032
+ error: { code: "SESSION_NOT_FOUND", message: "Session not found" }
3033
+ };
3034
+ }
3035
+ if (!session.hookToken || session.hookToken !== hookToken) {
3036
+ return {
3037
+ error: { code: "INVALID_TOKEN", message: "Invalid hook token" }
3038
+ };
3039
+ }
3040
+ const message = await storage.message.get(messageId);
3041
+ if (message instanceof Error) {
3042
+ return {
3043
+ error: { code: "MESSAGE_NOT_FOUND", message: "Message not found" }
3044
+ };
3045
+ }
3046
+ const mcpContext = message.mcpContext;
3047
+ const serverContext = mcpContext?.[server];
3048
+ const headers = serverContext?.headers ?? {};
3049
+ try {
3050
+ const result = await handler({ input, headers, body });
3051
+ return { result };
3052
+ } catch (e) {
3053
+ return {
3054
+ error: {
3055
+ code: "EXECUTION_ERROR",
3056
+ message: e instanceof Error ? e.message : String(e)
3057
+ }
3058
+ };
3059
+ }
3060
+ }
3061
+ };
3062
+ };
3063
+
3064
+ // src/index.ts
3065
+ init_errors();
3066
+ init_storage();
3067
+ // Annotate the CommonJS export names for ESM import in node:
3068
+ 0 && (module.exports = {
3069
+ MessageNotFoundError,
3070
+ SandboxError,
3071
+ SandboxNotFoundError,
3072
+ SessionNotFoundError,
3073
+ StorageError,
3074
+ agent,
3075
+ handleStorageRpc,
3076
+ mcp
3077
+ });
3078
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/sandbox/process-manager.ts", "../src/errors.ts", "../src/utils/paginate.ts", "../src/storage/bindings/local-fs-handlers.ts", "../src/storage/bindings/local.ts", "../src/storage/bindings/vercel.ts", "../src/storage/client.ts", "../src/storage/rpc.ts", "../src/storage/handler.ts", "../src/storage/index.ts", "../src/sandbox/write-files.ts", "../src/sandbox/bindings/local.ts", "../src/sandbox/bindings/vercel.ts", "../src/sandbox/client.ts", "../src/sandbox/index.ts", "../src/index.ts", "../src/client.ts", "../src/agent-workflow.ts", "../src/agent-workflow-steps.ts", "../src/skills/parser.ts", "../src/skills/discover.ts", "../src/tools/index.ts", "../src/utils/ui.ts", "../src/mcp/index.ts", "../src/utils/message.ts"],
  "sourcesContent": ["import type { BashResult, Sandbox } from \"./types\";\n\nconst DEFAULT_WAIT_UNTIL = 30000;\n\n/**\n * Shell script that handles all process execution logic.\n * Installed to ~/.agent/bin/run.sh during init().\n */\nconst RUN_SCRIPT = `#!/bin/sh\n# Process runner for Agent SDK\n# All parameters passed via environment variables to avoid escaping issues\n# Output is base64 encoded to avoid JSON escaping issues\n\nAGENT_DIR=\"$HOME/.agent\"\nCWD_FILE=\"$AGENT_DIR/cwd/$SESSION_ID\"\n\n# Get current working directory (per-session)\nif [ -f \"$CWD_FILE\" ]; then\n  CWD=$(cat \"$CWD_FILE\")\nelse\n  CWD=\"$HOME\"\nfi\n\n# Decode command\nCMD=$(echo \"$CMD_BASE64\" | base64 -d)\n\n# Background mode (waitUntil = 0)\nif [ \"$WAIT_UNTIL\" = \"0\" ]; then\n  cd \"$CWD\" || cd \"$HOME\"\n\n  # Generate a simple numeric ID for output file\n  OUTPUT_ID=\"$(date +%s)$$\"\n  OUTPUT_FILE=\"$AGENT_DIR/outputs/$OUTPUT_ID.log\"\n\n  # Double-fork to fully detach the process\n  (\n    (\n      eval \"$CMD\"\n      pwd > \"$CWD_FILE\" 2>/dev/null\n    ) > \"$OUTPUT_FILE\" 2>&1 < /dev/null &\n    echo $! > \"$AGENT_DIR/pids/$OUTPUT_ID.pid\"\n  )\n\n  # Small delay to ensure PID file is written\n  sleep 0.05\n\n  # Read PID\n  if [ -f \"$AGENT_DIR/pids/$OUTPUT_ID.pid\" ]; then\n    PID=$(cat \"$AGENT_DIR/pids/$OUTPUT_ID.pid\")\n  else\n    PID=0\n  fi\n\n  # Output JSON result (no output for background, so no encoding needed)\n  printf '{\"pid\":%d,\"outputBase64\":\"\",\"exitCode\":-1,\"status\":\"running\",\"outputFile\":\"%s\",\"cwd\":\"%s\"}\\\\n' \\\\\n    \"$PID\" \"$OUTPUT_FILE\" \"$CWD\"\n  exit 0\nfi\n\n# Foreground mode (waitUntil > 0) - run directly and capture output\ncd \"$CWD\" || cd \"$HOME\"\n\n# Run command, capture output and exit code\nOUTPUT=$(eval \"$CMD\" 2>&1)\nEXIT_CODE=$?\n\n# Update CWD after command\nNEW_CWD=$(pwd)\necho \"$NEW_CWD\" > \"$CWD_FILE\"\n\n# Determine status\nif [ $EXIT_CODE -eq 0 ]; then\n  STATUS=\"completed\"\nelse\n  STATUS=\"failed\"\nfi\n\n# Base64 encode output to avoid ALL escaping issues\nOUTPUT_BASE64=$(printf '%s' \"$OUTPUT\" | base64 | tr -d '\\\\n')\n\n# Output JSON result with base64-encoded output\nprintf '{\"pid\":0,\"outputBase64\":\"%s\",\"exitCode\":%d,\"status\":\"%s\",\"outputFile\":\"\",\"cwd\":\"%s\"}\\\\n' \\\\\n  \"$OUTPUT_BASE64\" \"$EXIT_CODE\" \"$STATUS\" \"$NEW_CWD\"\n`;\n\ntype RunOptions = {\n  command: string;\n  /**\n   * Max ms to wait for completion.\n   * - 0: Background mode - return immediately with PID\n   * - >0: Foreground mode - wait for completion (default: 30000)\n   */\n  waitUntil?: number;\n};\n\n/**\n * Process manager that uses a single shell script to handle all execution logic.\n * Reduces round-trips to the sandbox by doing everything in one exec call.\n */\nexport const createProcessManager = (opts: {\n  sandbox: Sandbox;\n  sessionId: string;\n  generateId: () => string;\n}) => {\n  const { sandbox, sessionId } = opts;\n  let initialized = false;\n\n  const init = async (): Promise<void> => {\n    if (initialized) {\n      return;\n    }\n\n    const result = await sandbox.exec({\n      command: \"sh\",\n      args: [\n        \"-c\",\n        `mkdir -p $HOME/.agent/bin $HOME/.agent/pids $HOME/.agent/outputs $HOME/.agent/cwd\ncat > $HOME/.agent/bin/run.sh << 'SCRIPT_EOF'\n${RUN_SCRIPT}\nSCRIPT_EOF\nchmod +x $HOME/.agent/bin/run.sh`,\n      ],\n    });\n\n    if (result instanceof Error) {\n      throw result;\n    }\n    await result.result;\n    initialized = true;\n  };\n\n  const run = async (opts: RunOptions): Promise<BashResult> => {\n    await init();\n\n    const { command, waitUntil = DEFAULT_WAIT_UNTIL } = opts;\n    const cmdBase64 = Buffer.from(command, \"utf-8\").toString(\"base64\");\n\n    const result = await sandbox.exec({\n      command: \"sh\",\n      args: [\n        \"-c\",\n        `CMD_BASE64=\"${cmdBase64}\" WAIT_UNTIL=\"${waitUntil}\" SESSION_ID=\"${sessionId}\" $HOME/.agent/bin/run.sh`,\n      ],\n    });\n\n    if (result instanceof Error) {\n      throw result;\n    }\n\n    const { stdout } = await result.result;\n    const parsed = JSON.parse(stdout.trim()) as {\n      pid: number;\n      outputBase64: string;\n      exitCode: number;\n      status: \"running\" | \"completed\" | \"failed\";\n      outputFile: string;\n      cwd: string;\n    };\n\n    // Decode base64 output\n    const output = parsed.outputBase64\n      ? Buffer.from(parsed.outputBase64, \"base64\").toString(\"utf-8\")\n      : \"\";\n\n    return {\n      pid: parsed.pid,\n      output,\n      exitCode: parsed.exitCode,\n      status: parsed.status,\n      cwd: parsed.cwd,\n      outputFile: parsed.outputFile,\n    };\n  };\n\n  const getCwd = async (): Promise<string> => {\n    await init();\n\n    const result = await sandbox.exec({\n      command: \"sh\",\n      args: [\n        \"-c\",\n        `cat $HOME/.agent/cwd/${sessionId} 2>/dev/null || echo \"$HOME\"`,\n      ],\n    });\n\n    if (result instanceof Error) {\n      return process.cwd();\n    }\n\n    const { stdout } = await result.result;\n    return stdout.trim() || process.cwd();\n  };\n\n  return {\n    init,\n    run,\n    getCwd,\n  };\n};\n\nexport type ProcessManager = ReturnType<typeof createProcessManager>;\n", "import * as errore from \"errore\";\n\nexport class SessionNotFoundError extends errore.createTaggedError({\n  name: \"SessionNotFoundError\",\n  message: \"Session $id not found\",\n}) {}\n\nexport class SessionError extends errore.createTaggedError({\n  name: \"SessionError\",\n  message: \"Session $id failed: $reason\",\n}) {}\n\nexport class SandboxNotFoundError extends errore.createTaggedError({\n  name: \"SandboxNotFoundError\",\n  message: \"Sandbox $id not found\",\n}) {}\n\nexport class StorageError extends errore.createTaggedError({\n  name: \"StorageError\",\n  message: \"$reason\",\n}) {}\n\nexport class SandboxError extends errore.createTaggedError({\n  name: \"SandboxError\",\n  message: \"$reason\",\n}) {}\n\nexport class MessageNotFoundError extends errore.createTaggedError({\n  name: \"MessageNotFoundError\",\n  message: \"Message $id not found\",\n}) {}\n", "import type { ListResult } from \"../storage\";\n\nexport function paginate<T extends { id: string }>(opts: {\n  items: T[];\n  cursor?: string;\n  limit?: number;\n}): ListResult<T> {\n  const { items, cursor, limit = 50 } = opts;\n  const startIndex = cursor ? items.findIndex((m) => m.id === cursor) + 1 : 0;\n  const sliced = items.slice(startIndex, startIndex + limit);\n  const nextCursor =\n    startIndex + limit < items.length ? (sliced.at(-1)?.id ?? null) : null;\n  return { items: sliced, nextCursor };\n}\n", "import { mkdir, readdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport { paginate } from \"../../utils/paginate\";\nimport type {\n  Command,\n  Handlers,\n  Message,\n  Part,\n  SandboxRecord,\n  Session,\n} from \"..\";\n\nexport function createFilesystemHandlers(basePath: string): Handlers {\n  const sessionDir = join(basePath, \"session\");\n  const messageDir = join(basePath, \"message\");\n  const partDir = join(basePath, \"part\");\n  const sandboxDir = join(basePath, \"sandbox\");\n  const commandDir = join(basePath, \"command\");\n\n  async function ensureDir(dir: string) {\n    await mkdir(dir, { recursive: true });\n  }\n\n  async function readJson<T>(filePath: string): Promise<T | null> {\n    try {\n      const content = await readFile(filePath, \"utf-8\");\n      return JSON.parse(content) as T;\n    } catch {\n      return null;\n    }\n  }\n\n  async function writeJsonFile(filePath: string, data: unknown) {\n    await ensureDir(dirname(filePath));\n    await writeFile(filePath, JSON.stringify(data, null, 2));\n  }\n\n  async function readAllFromDir<T>(dir: string): Promise<T[]> {\n    try {\n      const files = await readdir(dir);\n      const results = await Promise.all(\n        files\n          .filter((f) => f.endsWith(\".json\"))\n          .map((f) => readJson<T>(join(dir, f)))\n      );\n      return results.filter((r): r is NonNullable<typeof r> => r !== null);\n    } catch {\n      return [];\n    }\n  }\n\n  return {\n    \"session.get\": async ({ id }) => {\n      const session = await readJson<Session>(join(sessionDir, `${id}.json`));\n      return session ?? null;\n    },\n\n    \"session.set\": async (session) => {\n      const now = Date.now();\n      const sessionPath = join(sessionDir, `${session.id}.json`);\n      const existing = await readJson<Session>(sessionPath);\n      const newSession: Session = {\n        ...session,\n        tags: session.tags ?? existing?.tags ?? {},\n        createdAt: existing?.createdAt ?? session.createdAt ?? now,\n        updatedAt: now,\n      };\n      await writeJsonFile(sessionPath, newSession);\n      return newSession;\n    },\n\n    \"session.list\": async ({ tags, cursor, limit }) => {\n      const allSessions = await readAllFromDir<Session>(sessionDir);\n      let filtered = allSessions;\n      if (tags && Object.keys(tags).length > 0) {\n        filtered = filtered.filter((s) => {\n          const sessionTags = s.tags ?? {};\n          return Object.entries(tags).every(\n            ([key, value]) => sessionTags[key] === value\n          );\n        });\n      }\n      filtered.sort((a, b) => a.createdAt - b.createdAt);\n      return paginate({ items: filtered, cursor, limit });\n    },\n\n    \"session.listBySandbox\": async ({ sandboxId, tags, cursor, limit }) => {\n      const allSessions = await readAllFromDir<Session>(sessionDir);\n      let filtered = allSessions.filter((s) => s.sandboxId === sandboxId);\n      if (tags && Object.keys(tags).length > 0) {\n        filtered = filtered.filter((s) => {\n          const sessionTags = s.tags ?? {};\n          return Object.entries(tags).every(\n            ([key, value]) => sessionTags[key] === value\n          );\n        });\n      }\n      filtered.sort((a, b) => a.createdAt - b.createdAt);\n      return paginate({ items: filtered, cursor, limit });\n    },\n\n    \"session.tag.set\": async ({ sessionId, tags }) => {\n      const sessionPath = join(sessionDir, `${sessionId}.json`);\n      const existing = await readJson<Session>(sessionPath);\n      if (!existing) {\n        throw new Error(`Session ${sessionId} not found`);\n      }\n      const mergedTags = { ...existing.tags, ...tags };\n      const now = Date.now();\n      const updatedSession: Session = {\n        ...existing,\n        tags: mergedTags,\n        updatedAt: now,\n      };\n      await writeJsonFile(sessionPath, updatedSession);\n      return updatedSession;\n    },\n\n    \"message.get\": async ({ id }) => {\n      return await readJson<Message>(join(messageDir, `${id}.json`));\n    },\n\n    \"message.set\": async (message) => {\n      await writeJsonFile(join(messageDir, `${message.id}.json`), message);\n      return message;\n    },\n\n    \"message.list\": async ({ sessionId, cursor, limit }) => {\n      const allMessages = await readAllFromDir<Message>(messageDir);\n      const filtered = allMessages\n        .filter((m) => m.sessionId === sessionId)\n        .sort((a, b) => a.createdAt - b.createdAt);\n      return paginate({ items: filtered, cursor, limit });\n    },\n\n    \"part.listByMessage\": async ({ messageId, cursor, limit }) => {\n      const allParts = await readAllFromDir<Part>(partDir);\n      const filtered = allParts\n        .filter((p) => p.messageId === messageId)\n        .sort((a, b) => a.index - b.index);\n      return paginate({ items: filtered, cursor, limit });\n    },\n\n    \"part.listBySession\": async ({ sessionId, cursor, limit }) => {\n      const allParts = await readAllFromDir<Part>(partDir);\n      const filtered = allParts\n        .filter((p) => p.sessionId === sessionId)\n        .sort((a, b) => {\n          if (a.messageId !== b.messageId) {\n            return a.messageId.localeCompare(b.messageId);\n          }\n          return a.index - b.index;\n        });\n      return paginate({ items: filtered, cursor, limit });\n    },\n\n    \"part.set\": async (part) => {\n      await writeJsonFile(join(partDir, `${part.id}.json`), part);\n      return part;\n    },\n\n    \"sandbox.get\": async ({ key }) => {\n      const safeName = Buffer.from(key).toString(\"base64url\");\n      const sandboxPath = join(sandboxDir, `${safeName}.json`);\n      const data = await readJson<SandboxRecord>(sandboxPath);\n      if (!data) {\n        return null;\n      }\n      return data;\n    },\n\n    \"sandbox.set\": async (record) => {\n      const safeName = Buffer.from(record.id).toString(\"base64url\");\n      const sandboxPath = join(sandboxDir, `${safeName}.json`);\n      await writeJsonFile(sandboxPath, record);\n    },\n\n    \"sandbox.getBySession\": async ({ sessionId }) => {\n      const allSandboxes = await readAllFromDir<SandboxRecord>(sandboxDir);\n      const matching = allSandboxes.filter((s) =>\n        s.id.startsWith(`${sessionId}-`)\n      );\n      if (matching.length === 0) {\n        return null;\n      }\n      matching.sort(\n        (a, b) => (b.lastActivityAt ?? 0) - (a.lastActivityAt ?? 0)\n      );\n      return matching[0];\n    },\n\n    \"command.get\": async ({ id }) => {\n      return await readJson<Command>(join(commandDir, `${id}.json`));\n    },\n\n    \"command.set\": async (command) => {\n      await writeJsonFile(join(commandDir, `${command.id}.json`), command);\n      return command;\n    },\n\n    \"command.list\": async ({ sessionId, includeFinished, cursor, limit }) => {\n      const allCommands = await readAllFromDir<Command>(commandDir);\n      let filtered = allCommands.filter((c) => c.sessionId === sessionId);\n      if (!includeFinished) {\n        filtered = filtered.filter((c) => c.status === \"running\");\n      }\n      filtered.sort((a, b) => a.startedAt - b.startedAt);\n      return paginate({ items: filtered, cursor, limit });\n    },\n  };\n}\n", "import { type ServerType, serve } from \"@hono/node-server\";\nimport { Hono } from \"hono\";\nimport { handleStorageRpc } from \"..\";\nimport { createFilesystemHandlers } from \"./local-fs-handlers\";\n\nconst DEFAULT_STORAGE_DIR = \".agent-storage\";\n\nlet startPromise: Promise<string> | null = null;\nlet serverInstance: { url: string; server: ServerType } | null = null;\n\nexport function startLocalStorage(opts?: {\n  path?: string;\n  port?: number;\n}): Promise<string> {\n  if (startPromise) {\n    return startPromise;\n  }\n\n  startPromise = new Promise((resolve) => {\n    const handlers = createFilesystemHandlers(\n      opts?.path ?? DEFAULT_STORAGE_DIR\n    );\n    const app = new Hono();\n\n    app.post(\"/\", async (c) => {\n      const body = await c.req.json();\n      const response = await handleStorageRpc(body, handlers);\n      return c.json(response);\n    });\n\n    const port = opts?.port ?? 0;\n    const server = serve({\n      fetch: app.fetch,\n      port,\n    });\n\n    server.on(\"listening\", () => {\n      const address = server.address();\n      const actualPort = typeof address === \"object\" ? address?.port : port;\n      const url = `http://localhost:${actualPort}`;\n\n      serverInstance = { url, server };\n      console.log(`[agent] Local storage server started at ${url}`);\n      resolve(url);\n    });\n  });\n\n  return startPromise;\n}\n\nexport function stopLocalStorage(): void {\n  if (serverInstance) {\n    serverInstance.server.close();\n    serverInstance = null;\n    startPromise = null;\n  }\n}\n\nif (typeof process !== \"undefined\") {\n  process.on(\"SIGTERM\", stopLocalStorage);\n  process.on(\"SIGINT\", stopLocalStorage);\n}\n", "const VERCEL_STORAGE_URL =\n  \"https://agent-sdk-storage-nine.labs.vercel.dev/api/storage\";\n\nexport type VercelStorageConfig = {\n  url: string;\n  headers: Record<string, string>;\n};\n\nexport async function getVercelStorageConfig(): Promise<VercelStorageConfig> {\n  let token: string | null = null;\n\n  try {\n    const { getVercelOidcToken } = await import(\"@vercel/oidc\");\n    token = await getVercelOidcToken();\n  } catch {\n    // OIDC not available (local dev, non-Vercel environment)\n  }\n\n  return {\n    url: VERCEL_STORAGE_URL,\n    headers: token ? { Authorization: `Bearer ${token}` } : {},\n  };\n}\n", "import type { z } from \"zod\";\nimport {\n  MessageNotFoundError,\n  SandboxNotFoundError,\n  SessionNotFoundError,\n  StorageError,\n} from \"../errors\";\nimport type { StorageMethods } from \"./rpc\";\nimport type {\n  Command,\n  ListResult,\n  Part,\n  ResolvedStorage,\n  Storage,\n  StorageConfig,\n} from \"./types\";\n\ntype RpcResponse<T> =\n  | { result: T }\n  | { error: { code: string; message: string } };\n\nexport function getStorageClient({ url, headers }: ResolvedStorage): Storage {\n  async function rpc<M extends keyof StorageMethods>(\n    method: M,\n    params: z.infer<StorageMethods[M][\"params\"]>\n  ): Promise<z.infer<StorageMethods[M][\"result\"]>> {\n    const res = await fetch(url, {\n      method: \"POST\",\n      headers: { \"Content-Type\": \"application/json\", ...headers },\n      body: JSON.stringify({ method, params }),\n    });\n\n    if (!res.ok) {\n      throw new StorageError({\n        reason: `HTTP ${res.status}: ${res.statusText}`,\n      });\n    }\n\n    const json = (await res.json()) as RpcResponse<\n      z.infer<StorageMethods[M][\"result\"]>\n    >;\n\n    if (\"error\" in json) {\n      throw new StorageError({ reason: json.error.message });\n    }\n\n    return json.result;\n  }\n\n  return {\n    session: {\n      get: async (id) => {\n        try {\n          const result = await rpc(\"session.get\", { id });\n          if (result === null) {\n            return new SessionNotFoundError({ id });\n          }\n          return result;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      set: async (session) => {\n        try {\n          return await rpc(\"session.set\", session);\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      list: async (opts) => {\n        try {\n          return await rpc(\"session.list\", {\n            tags: opts?.tags,\n            order: opts?.order,\n            cursor: opts?.cursor,\n            limit: opts?.limit,\n          });\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      tag: {\n        set: async ({ sessionId, tags }) => {\n          try {\n            return await rpc(\"session.tag.set\", { sessionId, tags });\n          } catch (e) {\n            if (e instanceof StorageError) {\n              return e;\n            }\n            return new StorageError({ reason: String(e), cause: e });\n          }\n        },\n      },\n    },\n    message: {\n      list: async (sessionId, opts) => {\n        try {\n          return await rpc(\"message.list\", {\n            sessionId,\n            cursor: opts?.cursor,\n            limit: opts?.limit,\n          });\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      get: async (id) => {\n        try {\n          const result = await rpc(\"message.get\", { id });\n          if (result === null) {\n            return new MessageNotFoundError({ id });\n          }\n          return result;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      set: async (message) => {\n        try {\n          return await rpc(\"message.set\", message);\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n    },\n    part: {\n      listByMessage: async (messageId, opts) => {\n        try {\n          const result = await rpc(\"part.listByMessage\", {\n            messageId,\n            cursor: opts?.cursor,\n            limit: opts?.limit,\n          });\n          return result as ListResult<Part>;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      listBySession: async (sessionId, opts) => {\n        try {\n          const result = await rpc(\"part.listBySession\", {\n            sessionId,\n            cursor: opts?.cursor,\n            limit: opts?.limit,\n          });\n          return result as ListResult<Part>;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      set: async (part) => {\n        try {\n          return (await rpc(\"part.set\", part)) as Part;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n    },\n    sandbox: {\n      get: async (key) => {\n        try {\n          const result = await rpc(\"sandbox.get\", { key });\n          if (result === null) {\n            return new SandboxNotFoundError({ id: key });\n          }\n          return result;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      getBySession: async (sessionId) => {\n        try {\n          const result = await rpc(\"sandbox.getBySession\", { sessionId });\n          if (result === null) {\n            return new SandboxNotFoundError({ id: sessionId });\n          }\n          return result;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      set: async (record) => {\n        try {\n          await rpc(\"sandbox.set\", record);\n          return record;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n    },\n    command: {\n      get: async (id) => {\n        try {\n          return (await rpc(\"command.get\", { id })) as Command | null;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      set: async (command) => {\n        try {\n          return (await rpc(\"command.set\", command)) as Command;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n      list: async (sessionId, opts) => {\n        try {\n          const result = await rpc(\"command.list\", {\n            sessionId,\n            includeFinished: opts?.includeFinished,\n            cursor: opts?.cursor,\n            limit: opts?.limit,\n          });\n          return result as ListResult<Command>;\n        } catch (e) {\n          if (e instanceof StorageError) {\n            return e;\n          }\n          return new StorageError({ reason: String(e), cause: e });\n        }\n      },\n    },\n  };\n}\n\nasync function resolveStorageUrl(\n  config: StorageConfig\n): Promise<ResolvedStorage> {\n  switch (config.type) {\n    case \"local\": {\n      const { startLocalStorage } = await import(\"./bindings/local\");\n      return { url: await startLocalStorage({ path: config.path }) };\n    }\n    case \"vercel\": {\n      const { getVercelStorageConfig } = await import(\"./bindings/vercel\");\n      return getVercelStorageConfig();\n    }\n    case \"custom\": {\n      return { url: config.url, headers: config.headers };\n    }\n    default: {\n      config satisfies never;\n      throw new Error(\n        `Unknown storage type: ${(config as { type: string }).type}`\n      );\n    }\n  }\n}\n\nexport function getStorage(storageConfig: StorageConfig): Storage {\n  let clientPromise: Promise<Storage> | null = null;\n  const getClient = () => {\n    if (!clientPromise) {\n      clientPromise = resolveStorageUrl(storageConfig).then((resolved) =>\n        getStorageClient(resolved)\n      );\n    }\n    return clientPromise;\n  };\n\n  return {\n    session: {\n      get: async (id) => (await getClient()).session.get(id),\n      set: async (session) => (await getClient()).session.set(session),\n      list: async (opts) => (await getClient()).session.list(opts),\n      tag: {\n        set: async (opts) => (await getClient()).session.tag.set(opts),\n      },\n    },\n    message: {\n      list: async (sessionId, opts) =>\n        (await getClient()).message.list(sessionId, opts),\n      get: async (id) => (await getClient()).message.get(id),\n      set: async (message) => (await getClient()).message.set(message),\n    },\n    part: {\n      listByMessage: async (messageId, opts) =>\n        (await getClient()).part.listByMessage(messageId, opts),\n      listBySession: async (sessionId, opts) =>\n        (await getClient()).part.listBySession(sessionId, opts),\n      set: async (part) => (await getClient()).part.set(part),\n    },\n    sandbox: {\n      get: async (key) => (await getClient()).sandbox.get(key),\n      set: async (opts) => (await getClient()).sandbox.set(opts),\n      getBySession: async (sessionId) =>\n        (await getClient()).sandbox.getBySession(sessionId),\n    },\n    command: {\n      get: async (id) => (await getClient()).command.get(id),\n      set: async (command) => (await getClient()).command.set(command),\n      list: async (sessionId, opts) =>\n        (await getClient()).command.list(sessionId, opts),\n    },\n  };\n}\n", "import { z } from \"zod\";\n\nconst VercelLifecycleConfigSchema = z.object({\n  stopAfterInactiveMs: z.number().optional(),\n  snapshotBeforeTimeoutMs: z.number().optional(),\n  snapshotId: z.string().optional(),\n  autoStart: z.boolean().optional(),\n});\n\nexport const SandboxConfigSchema = z.discriminatedUnion(\"type\", [\n  z.object({\n    type: z.literal(\"vercel\"),\n    resources: z.object({ vcpus: z.number() }).optional(),\n    ports: z.array(z.number()).optional(),\n    lifecycle: VercelLifecycleConfigSchema.optional(),\n  }),\n  z.object({\n    type: z.literal(\"local\"),\n    path: z.string().optional(),\n  }),\n  z.object({\n    type: z.literal(\"custom\"),\n    url: z.string(),\n    headers: z.record(z.string(), z.string()).optional(),\n  }),\n]);\n\nexport type SandboxConfig = z.infer<typeof SandboxConfigSchema>;\n\nexport const SessionSchema = z.object({\n  id: z.string(),\n  createdAt: z.number(),\n  updatedAt: z.number(),\n  runId: z.string().nullable(),\n  lastMessageId: z.string().nullable(),\n  tags: z.record(z.string(), z.unknown()).nullable(),\n  instructions: z.string().nullable(),\n  model: z.string().nullable(),\n  mcp: z.unknown().nullable(), // TODO\n  sandboxId: z.string().nullable(),\n  skillsDir: z.array(z.string()).nullable(),\n  hookToken: z.string().nullable(),\n});\n\nexport const MessageSchema = z.object({\n  id: z.string(),\n  sessionId: z.string(),\n  role: z.enum([\"user\", \"assistant\", \"system\"]),\n  createdAt: z.number(),\n  completedAt: z.number().nullable(),\n  mcpContext: z.unknown().nullable(),\n});\n\n/**\n * The `part` field corresponds to `UIMessage[\"parts\"][number]` from the `ai` package.\n * We use `z.unknown()` because the exact shape varies by part type (text, tool-call,\n * tool-result, etc.) and may evolve with the AI SDK. The RPC layer just passes it through.\n */\nexport const PartSchema = z.object({\n  id: z.string(),\n  messageId: z.string(),\n  sessionId: z.string(),\n  index: z.number(),\n  part: z.unknown(),\n});\n\nexport const CommandResultSchema = z.object({\n  stdout: z.string(),\n  stderr: z.string(),\n  exitCode: z.number(),\n  completedAt: z.number(),\n});\n\nexport const SandboxRecordSchema = z.object({\n  id: z.string(),\n  config: SandboxConfigSchema,\n  createdAt: z.number().nullable(),\n  lastActivityAt: z.number().nullable(),\n  acquiringLockId: z.string().nullable(),\n  acquiringLockAt: z.number().nullable(),\n  providerMetadata: z\n    .discriminatedUnion(\"provider\", [\n      z.object({\n        provider: z.literal(\"vercel\"),\n        sandboxId: z.string().nullable(),\n        snapshotId: z.string().nullable(),\n      }),\n    ])\n    .optional(),\n});\n\nconst CommandBaseSchema = z.object({\n  id: z.string(),\n  sessionId: z.string(),\n  command: z.string(),\n  args: z.array(z.string()).optional(),\n  startedAt: z.number(),\n});\n\nexport const CommandSchema = z.discriminatedUnion(\"status\", [\n  CommandBaseSchema.extend({ status: z.literal(\"running\") }),\n  CommandBaseSchema.extend({\n    status: z.literal(\"completed\"),\n    result: CommandResultSchema,\n  }),\n  CommandBaseSchema.extend({\n    status: z.literal(\"killed\"),\n    result: CommandResultSchema.optional(),\n  }),\n  CommandBaseSchema.extend({\n    status: z.literal(\"failed\"),\n    result: CommandResultSchema,\n  }),\n]);\n\nfunction ListResultSchema<T extends z.ZodTypeAny>(itemSchema: T) {\n  return z.object({\n    items: z.array(itemSchema),\n    nextCursor: z.string().nullable(),\n  });\n}\n\nexport const methods = {\n  \"session.get\": {\n    params: z.object({ id: z.string() }),\n    result: SessionSchema.nullable(),\n  },\n  \"session.set\": {\n    params: SessionSchema,\n    result: SessionSchema,\n  },\n  \"session.list\": {\n    params: z.object({\n      tags: z.record(z.string(), z.unknown()).optional(),\n      order: z\n        .enum([\n          \"createdAt_asc\",\n          \"createdAt_desc\",\n          \"updatedAt_asc\",\n          \"updatedAt_desc\",\n        ])\n        .optional(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(SessionSchema),\n  },\n  \"session.listBySandbox\": {\n    params: z.object({\n      sandboxId: z.string(),\n      tags: z.record(z.string(), z.unknown()).optional(),\n      order: z\n        .enum([\n          \"createdAt_asc\",\n          \"createdAt_desc\",\n          \"updatedAt_asc\",\n          \"updatedAt_desc\",\n        ])\n        .optional(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(SessionSchema),\n  },\n  \"session.tag.set\": {\n    params: z.object({\n      sessionId: z.string(),\n      tags: z.record(z.string(), z.unknown()),\n    }),\n    result: SessionSchema,\n  },\n  \"message.get\": {\n    params: z.object({ id: z.string() }),\n    result: MessageSchema.nullable(),\n  },\n  \"message.set\": {\n    params: MessageSchema,\n    result: MessageSchema,\n  },\n  \"message.list\": {\n    params: z.object({\n      sessionId: z.string(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(MessageSchema),\n  },\n  \"part.listByMessage\": {\n    params: z.object({\n      messageId: z.string(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(PartSchema),\n  },\n  \"part.listBySession\": {\n    params: z.object({\n      sessionId: z.string(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(PartSchema),\n  },\n  \"part.set\": {\n    params: PartSchema,\n    result: PartSchema,\n  },\n  \"sandbox.get\": {\n    params: z.object({ key: z.string() }),\n    result: SandboxRecordSchema.nullable(),\n  },\n  \"sandbox.getBySession\": {\n    params: z.object({ sessionId: z.string() }),\n    result: SandboxRecordSchema.nullable(),\n  },\n  \"sandbox.set\": {\n    params: SandboxRecordSchema,\n    result: z.void(),\n  },\n  \"command.get\": {\n    params: z.object({ id: z.string() }),\n    result: CommandSchema.nullable(),\n  },\n  \"command.set\": {\n    params: CommandSchema,\n    result: CommandSchema,\n  },\n  \"command.list\": {\n    params: z.object({\n      sessionId: z.string(),\n      includeFinished: z.boolean().optional(),\n      cursor: z.string().optional(),\n      limit: z.number().optional(),\n    }),\n    result: ListResultSchema(CommandSchema),\n  },\n} as const;\n\nexport type StorageMethods = typeof methods;\nexport type MethodName = keyof StorageMethods;\n", "import type { z } from \"zod\";\nimport { type MethodName, methods, type StorageMethods } from \"./rpc\";\n\nexport type Handlers = {\n  [K in MethodName]: (\n    params: z.infer<StorageMethods[K][\"params\"]>\n  ) => Promise<z.infer<StorageMethods[K][\"result\"]>>;\n};\n\nexport type RpcRequest = {\n  method: string;\n  params: unknown;\n};\n\nexport type RpcSuccessResponse<T = unknown> = {\n  result: T;\n};\n\nexport type RpcErrorResponse = {\n  error: { code: string; message: string };\n};\n\nexport type RpcResponse<T = unknown> = RpcSuccessResponse<T> | RpcErrorResponse;\n\nexport async function handleStorageRpc(\n  body: RpcRequest,\n  handlers: Handlers\n): Promise<RpcResponse> {\n  const { method, params } = body;\n\n  if (!(method in methods)) {\n    return {\n      error: { code: \"METHOD_NOT_FOUND\", message: `Unknown method: ${method}` },\n    };\n  }\n\n  const methodName = method as MethodName;\n  const schema = methods[methodName].params;\n  const parsed = schema.safeParse(params);\n\n  if (!parsed.success) {\n    return {\n      error: {\n        code: \"INVALID_PARAMS\",\n        message: parsed.error.issues\n          .map((i) => `${i.path.join(\".\")}: ${i.message}`)\n          .join(\", \"),\n      },\n    };\n  }\n\n  try {\n    const handler = handlers[methodName] as (\n      params: unknown\n    ) => Promise<unknown>;\n    const result = await handler(parsed.data);\n    return { result };\n  } catch (e) {\n    return {\n      error: {\n        code: \"INTERNAL_ERROR\",\n        message: e instanceof Error ? e.message : String(e),\n      },\n    };\n  }\n}\n", "export { getStorage } from \"./client\";\nexport {\n  type Handlers,\n  handleStorageRpc,\n  type RpcErrorResponse,\n  type RpcRequest,\n  type RpcResponse,\n  type RpcSuccessResponse,\n} from \"./handler\";\nexport {\n  CommandResultSchema,\n  CommandSchema,\n  MessageSchema,\n  type MethodName,\n  methods,\n  PartSchema,\n  type SandboxConfig,\n  SandboxConfigSchema,\n  SandboxRecordSchema,\n  SessionSchema,\n  type StorageMethods,\n} from \"./rpc\";\nexport type {\n  Command,\n  ListResult,\n  Message,\n  Part,\n  ResolvedStorage,\n  SandboxRecord,\n  Session,\n  Storage,\n  StorageConfig,\n} from \"./types\";\n", "import * as path from \"node:path\";\nimport type { UploadableFile } from \"../skills/types\";\nimport type { Sandbox } from \"./types\";\n\n/**\n * Writes files to a sandbox at the specified destination path.\n * Shell scripts (.sh files) are automatically made executable.\n *\n * For small files (<100KB total), uses single exec with heredoc.\n * For large files, writes base64 chunks then decodes to avoid ARG_MAX limits.\n */\nexport async function writeFiles(opts: {\n  sandbox: Pick<Sandbox, \"exec\">;\n  files: UploadableFile[];\n  destPath: string;\n}): Promise<void> {\n  const { sandbox, files, destPath } = opts;\n\n  if (files.length === 0) {\n    return;\n  }\n\n  const filePaths = files.map((file) => path.posix.join(destPath, file.path));\n  const parentDirs = [...new Set(filePaths.map((p) => path.posix.dirname(p)))];\n  const shellScripts = filePaths.filter((p) => p.endsWith(\".sh\"));\n\n  const mkdirResult = await sandbox.exec({\n    command: \"mkdir\",\n    args: [\"-p\", ...parentDirs],\n  });\n  if (mkdirResult instanceof Error) {\n    throw mkdirResult;\n  }\n  await mkdirResult.result;\n\n  const CHUNK_SIZE = 50_000;\n\n  for (let i = 0; i < files.length; i++) {\n    const file = files[i];\n    const fullPath = filePaths[i];\n    const base64Content = toBase64(file.content);\n\n    if (base64Content.length < CHUNK_SIZE) {\n      const marker = `EOF_${i}`;\n      const execResult = await sandbox.exec({\n        command: \"bash\",\n        args: [\n          \"-c\",\n          `base64 -d > ${quote(fullPath)} << '${marker}'\n${base64Content}\n${marker}`,\n        ],\n      });\n\n      if (execResult instanceof Error) {\n        throw execResult;\n      }\n\n      const { exitCode, stderr } = await execResult.result;\n      if (exitCode !== 0) {\n        throw new Error(\n          `writeFiles failed with exit code ${exitCode}: ${stderr}`\n        );\n      }\n    } else {\n      const tempB64 = `/tmp/chunk-${Date.now()}-${i}.b64`;\n\n      const clearResult = await sandbox.exec({\n        command: \"bash\",\n        args: [\"-c\", `> ${quote(tempB64)}`],\n      });\n      if (clearResult instanceof Error) {\n        throw clearResult;\n      }\n      await clearResult.result;\n\n      for (let offset = 0; offset < base64Content.length; offset += CHUNK_SIZE) {\n        const chunk = base64Content.slice(offset, offset + CHUNK_SIZE);\n        const marker = `CHUNK_${offset}`;\n        const appendResult = await sandbox.exec({\n          command: \"bash\",\n          args: [\n            \"-c\",\n            `cat >> ${quote(tempB64)} << '${marker}'\n${chunk}\n${marker}`,\n          ],\n        });\n\n        if (appendResult instanceof Error) {\n          throw appendResult;\n        }\n\n        const { exitCode, stderr } = await appendResult.result;\n        if (exitCode !== 0) {\n          throw new Error(\n            `writeFiles chunk failed with exit code ${exitCode}: ${stderr}`\n          );\n        }\n      }\n\n      const decodeResult = await sandbox.exec({\n        command: \"bash\",\n        args: [\n          \"-c\",\n          `base64 -d < ${quote(tempB64)} > ${quote(fullPath)} && rm -f ${quote(tempB64)}`,\n        ],\n      });\n\n      if (decodeResult instanceof Error) {\n        throw decodeResult;\n      }\n\n      const { exitCode, stderr } = await decodeResult.result;\n      if (exitCode !== 0) {\n        throw new Error(\n          `writeFiles decode failed with exit code ${exitCode}: ${stderr}`\n        );\n      }\n    }\n  }\n\n  if (shellScripts.length > 0) {\n    const chmodResult = await sandbox.exec({\n      command: \"chmod\",\n      args: [\"+x\", ...shellScripts],\n    });\n    if (chmodResult instanceof Error) {\n      throw chmodResult;\n    }\n    await chmodResult.result;\n  }\n}\n\nfunction toBase64(content: string | Buffer): string {\n  if (typeof content === \"string\") {\n    return Buffer.from(content).toString(\"base64\");\n  }\n  return content.toString(\"base64\");\n}\n\nfunction quote(s: string): string {\n  return `'${s.replace(/'/g, \"'\\\\''\")}'`;\n}\n", "import type { ChildProcess } from \"node:child_process\";\nimport { spawn } from \"node:child_process\";\nimport * as errore from \"errore\";\nimport { ulid } from \"ulid\";\nimport { SandboxError } from \"../../errors\";\nimport type { SandboxRecord, StorageConfig } from \"../../storage/types\";\nimport type { LogEntry, Sandbox } from \"../types\";\nimport { writeFiles } from \"../write-files\";\n\nexport const localSandbox = ({\n  sandboxRecord,\n}: {\n  sandboxRecord: SandboxRecord & { config: { type: \"local\" } };\n  storageConfig: StorageConfig;\n}): Sandbox => {\n  const config = sandboxRecord.config;\n  const basePath = config.path ?? process.cwd();\n  const processes = new Map<string, ChildProcess>();\n\n  const sandbox: Sandbox = {\n    id: sandboxRecord.id,\n    config: sandboxRecord.config,\n    exec: ({ command, args, signal }) => {\n      return errore.tryAsync({\n        try: () => {\n          const commandId = `command_${ulid()}`;\n\n          const child = spawn(command, args, {\n            cwd: basePath,\n            signal,\n          });\n\n          processes.set(commandId, child);\n\n          let stdout = \"\";\n          let stderr = \"\";\n          const logQueue: LogEntry[] = [];\n          let logResolve: (() => void) | null = null;\n          let closed = false;\n\n          child.stdout.on(\"data\", (data: string | Buffer) => {\n            const str = String(data);\n            stdout += str;\n            logQueue.push({ stream: \"stdout\", data: str });\n            logResolve?.();\n          });\n\n          child.stderr.on(\"data\", (data: string | Buffer) => {\n            const str = String(data);\n            stderr += str;\n            logQueue.push({ stream: \"stderr\", data: str });\n            logResolve?.();\n          });\n\n          const result = new Promise<{\n            stdout: string;\n            stderr: string;\n            exitCode: number;\n          }>((resolve, reject) => {\n            child.on(\"error\", (err) => {\n              processes.delete(commandId);\n              closed = true;\n              logResolve?.();\n              reject(err);\n            });\n\n            child.on(\"close\", (code: number | null) => {\n              processes.delete(commandId);\n              closed = true;\n              logResolve?.();\n              resolve({ stdout, stderr, exitCode: code ?? 0 });\n            });\n          });\n\n          async function* logs(): AsyncIterable<LogEntry> {\n            while (!closed || logQueue.length > 0) {\n              const entry = logQueue.shift();\n              if (entry) {\n                yield entry;\n              } else if (!closed) {\n                await new Promise<void>((resolve) => {\n                  logResolve = resolve;\n                });\n                logResolve = null;\n              }\n            }\n          }\n\n          return Promise.resolve({ commandId, logs, result });\n        },\n        catch: (e: unknown) =>\n          new SandboxError({ reason: String(e), cause: e }),\n      });\n    },\n\n    getDomain: (port) => {\n      return Promise.resolve(`http://localhost:${port}`);\n    },\n\n    kill: async ({ commandId, storage }) => {\n      const child = processes.get(commandId);\n      if (!child) {\n        return new SandboxError({\n          reason: `Command ${commandId} not found or already finished`,\n        });\n      }\n\n      child.kill(\"SIGTERM\");\n\n      const cmd = await storage.command.get(commandId);\n      if (cmd instanceof Error) {\n        return new SandboxError({ reason: cmd.message, cause: cmd });\n      }\n      if (cmd && cmd.status === \"running\") {\n        const result = await storage.command.set({\n          ...cmd,\n          status: \"killed\",\n        });\n        if (result instanceof Error) {\n          return new SandboxError({ reason: result.message, cause: result });\n        }\n      }\n    },\n\n    writeFiles: (opts) => writeFiles({ sandbox, ...opts }),\n  };\n\n  return sandbox;\n};\n", "import * as errore from \"errore\";\nimport { Sandbox as VercelSandboxSDK } from \"sandbox\";\nimport { SandboxError } from \"../../errors\";\nimport {\n  getStorage,\n  type SandboxRecord,\n  type Storage,\n  type StorageConfig,\n} from \"../../storage\";\nimport type {\n  LogEntry,\n  Sandbox,\n  SandboxLifecycle,\n  SandboxStatus,\n} from \"../types\";\nimport { writeFiles } from \"../write-files\";\n\nexport type VercelSandboxCreateOptions = Extract<\n  Parameters<typeof VercelSandboxSDK.create>[0],\n  // biome-ignore lint/complexity/noBannedTypes: .\n  {}\n>;\n\nexport const VERCEL_MAX_TIMEOUT_MS = 5 * 60 * 60 * 1000; // 5 hours\nconst LOCK_TIMEOUT_MS = 30_000; // 30 seconds - if lock older than this, consider it stale\nconst LOCK_POLL_INTERVAL_MS = 200;\n\nconst getTestCredentials = () =>\n  process.env.NODE_ENV === \"test\"\n    ? {\n        token: process.env.TEST_VERCEL_TOKEN,\n        teamId: process.env.TEST_VERCEL_TEAM_ID,\n        projectId: process.env.TEST_VERCEL_PROJECT_ID,\n      }\n    : {};\n\n/**\n * Module-level cache for in-flight sandbox creation promises.\n * Prevents parallel requests within the same process from creating duplicate sandboxes.\n */\nconst createPromises = new Map<string, Promise<SandboxError | string>>();\n\nconst ACTIVITY_THROTTLE_MS = 10_000;\nconst lastActivitySent = new Map<string, number>();\n\nconst DEFAULT_VCPUS = 2;\n\nexport const vercelSandbox = ({\n  sandboxRecord,\n  storageConfig,\n  enableLifecycleWorkflow = true,\n  storage: storageOverride,\n}: {\n  sandboxRecord: SandboxRecord & { config: { type: \"vercel\" } };\n  storageConfig: StorageConfig;\n  enableLifecycleWorkflow?: boolean;\n  storage?: Storage;\n}): Sandbox => {\n  const { id, config } = sandboxRecord;\n  const vcpus = config.resources?.vcpus ?? DEFAULT_VCPUS;\n  const ports = config.ports;\n  const storage = storageOverride ?? getStorage(storageConfig);\n  const initialVercel =\n    sandboxRecord.providerMetadata?.provider === \"vercel\"\n      ? sandboxRecord.providerMetadata\n      : null;\n\n  type SandboxInstance = Awaited<ReturnType<typeof VercelSandboxSDK.get>>;\n  let sandboxPromise: Promise<SandboxError | SandboxInstance> | null = null;\n\n  async function pollForSandboxId(): Promise<SandboxError | string> {\n    const deadline = Date.now() + LOCK_TIMEOUT_MS;\n    while (Date.now() < deadline) {\n      await new Promise((r) => setTimeout(r, LOCK_POLL_INTERVAL_MS));\n      const record = await storage.sandbox.get(id);\n      if (record instanceof Error) {\n        return new SandboxError({ reason: record.message, cause: record });\n      }\n      const vercelSandboxId =\n        record?.providerMetadata?.provider === \"vercel\"\n          ? record.providerMetadata.sandboxId\n          : null;\n      if (vercelSandboxId) {\n        return vercelSandboxId;\n      }\n      if (!record?.acquiringLockAt) {\n        break;\n      }\n    }\n    return new SandboxError({\n      reason: \"Timed out waiting for sandbox creation by another process\",\n    });\n  }\n\n  async function createSandboxFromSnapshot(\n    snapshotId: string\n  ): Promise<SandboxError | string> {\n    return await errore.tryAsync({\n      try: async () => {\n        const sandbox = await VercelSandboxSDK.create({\n          source: { type: \"snapshot\", snapshotId },\n          resources: { vcpus },\n          timeout: VERCEL_MAX_TIMEOUT_MS,\n          ports,\n          ...getTestCredentials(),\n        });\n        const now = Date.now();\n        await storage.sandbox.set({\n          id,\n          config,\n          createdAt: now,\n          lastActivityAt: now,\n          acquiringLockId: null,\n          acquiringLockAt: null,\n          providerMetadata: {\n            provider: \"vercel\",\n            sandboxId: sandbox.sandboxId,\n            snapshotId,\n          },\n        });\n        return sandbox.sandboxId;\n      },\n      catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n    });\n  }\n\n  async function createFreshSandbox(): Promise<SandboxError | string> {\n    return await errore.tryAsync({\n      try: async () => {\n        const sandbox = await VercelSandboxSDK.create({\n          resources: { vcpus },\n          timeout: VERCEL_MAX_TIMEOUT_MS,\n          ports,\n          ...getTestCredentials(),\n        });\n        const now = Date.now();\n        await storage.sandbox.set({\n          id,\n          config,\n          createdAt: now,\n          lastActivityAt: now,\n          acquiringLockId: null,\n          acquiringLockAt: null,\n          providerMetadata: {\n            provider: \"vercel\",\n            sandboxId: sandbox.sandboxId,\n            snapshotId: null,\n          },\n        });\n        return sandbox.sandboxId;\n      },\n      catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n    });\n  }\n\n  async function doGetOrCreateSandboxId(): Promise<SandboxError | string> {\n    if (initialVercel?.sandboxId) {\n      return initialVercel.sandboxId;\n    }\n\n    const existing = await storage.sandbox.get(id);\n    if (existing instanceof Error) {\n      return new SandboxError({ reason: existing.message, cause: existing });\n    }\n\n    const existingVercel =\n      existing?.providerMetadata?.provider === \"vercel\"\n        ? existing.providerMetadata\n        : null;\n\n    if (existingVercel?.sandboxId) {\n      return existingVercel.sandboxId;\n    }\n\n    const hasActiveLock =\n      existing?.acquiringLockId &&\n      existing.acquiringLockAt &&\n      Date.now() - existing.acquiringLockAt < LOCK_TIMEOUT_MS;\n\n    if (hasActiveLock) {\n      return pollForSandboxId();\n    }\n\n    const lockId = crypto.randomUUID();\n    const now = Date.now();\n    await storage.sandbox.set({\n      id,\n      config,\n      createdAt: existing?.createdAt ?? sandboxRecord.createdAt,\n      lastActivityAt: existing?.lastActivityAt ?? sandboxRecord.lastActivityAt,\n      acquiringLockId: lockId,\n      acquiringLockAt: now,\n      providerMetadata: {\n        provider: \"vercel\",\n        sandboxId: null,\n        snapshotId:\n          existingVercel?.snapshotId ?? initialVercel?.snapshotId ?? null,\n      },\n    });\n\n    const afterLock = await storage.sandbox.get(id);\n    if (afterLock instanceof Error) {\n      return new SandboxError({ reason: afterLock.message, cause: afterLock });\n    }\n    if (afterLock?.acquiringLockId !== lockId) {\n      return pollForSandboxId();\n    }\n\n    const snapshotId =\n      existingVercel?.snapshotId ??\n      initialVercel?.snapshotId ??\n      config.lifecycle?.snapshotId;\n    if (snapshotId) {\n      const result = await createSandboxFromSnapshot(snapshotId);\n      if (!(result instanceof Error)) {\n        return result;\n      }\n    }\n\n    return createFreshSandbox();\n  }\n\n  function getOrCreateSandboxId(): Promise<SandboxError | string> {\n    const cached = createPromises.get(id);\n    if (cached) {\n      return cached;\n    }\n\n    const promise = doGetOrCreateSandboxId().finally(() => {\n      createPromises.delete(id);\n    });\n    createPromises.set(id, promise);\n    return promise;\n  }\n\n  async function doGetSandbox(): Promise<SandboxError | SandboxInstance> {\n    const vercelSandboxId = await getOrCreateSandboxId();\n    if (vercelSandboxId instanceof Error) {\n      return vercelSandboxId;\n    }\n\n    return errore.tryAsync({\n      try: () =>\n        VercelSandboxSDK.get({\n          sandboxId: vercelSandboxId,\n          ...getTestCredentials(),\n        }),\n      catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n    });\n  }\n\n  function getSandbox(): Promise<SandboxError | SandboxInstance> {\n    if (!sandboxPromise) {\n      sandboxPromise = doGetSandbox();\n    }\n    return sandboxPromise;\n  }\n\n  async function updateLastActivity(): Promise<void> {\n    const now = Date.now();\n    const lastSent = lastActivitySent.get(id);\n    if (lastSent && now - lastSent < ACTIVITY_THROTTLE_MS) {\n      return;\n    }\n    lastActivitySent.set(id, now);\n\n    const existing = await storage.sandbox.get(id);\n    if (existing instanceof Error || !existing) {\n      return;\n    }\n    const existingVercel =\n      existing.providerMetadata?.provider === \"vercel\"\n        ? existing.providerMetadata\n        : null;\n    await storage.sandbox.set({\n      id: existing.id,\n      config: existing.config,\n      createdAt: existing.createdAt,\n      lastActivityAt: now,\n      acquiringLockId: null,\n      acquiringLockAt: null,\n      providerMetadata: existingVercel ?? {\n        provider: \"vercel\",\n        sandboxId: null,\n        snapshotId: null,\n      },\n    });\n\n    if (enableLifecycleWorkflow) {\n      // todo re-enable once we figure out why workflow throws a bunch of errors in dev\n      // const lifecycleInput: SandboxLifecycleInput = {\n      //   id,\n      //   storageConfig,\n      //   initialConfig: config,\n      // };\n      // try {\n      //   await sandboxActivityHook.resume(id, {\n      //     type: \"activity\",\n      //     newConfig: config,\n      //   });\n      // } catch {\n      //   await start(sandboxLifecycleWorkflow, [{ input: lifecycleInput }]);\n      // }\n    }\n  }\n\n  const lifecycle: SandboxLifecycle = {\n    start: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n      await updateLastActivity();\n      return sandbox.status as SandboxStatus;\n    },\n\n    snapshot: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n\n      return errore.tryAsync({\n        try: async () => {\n          const existing = await storage.sandbox.get(id);\n          const snapshot = await sandbox.snapshot();\n          await storage.sandbox.set({\n            id,\n            config,\n            createdAt:\n              existing instanceof Error ? null : existing?.createdAt ?? null,\n            lastActivityAt:\n              existing instanceof Error\n                ? null\n                : existing?.lastActivityAt ?? null,\n            acquiringLockId: null,\n            acquiringLockAt: null,\n            providerMetadata: {\n              provider: \"vercel\",\n              sandboxId: null,\n              snapshotId: snapshot.snapshotId,\n            },\n          });\n          return { snapshotId: snapshot.snapshotId };\n        },\n        catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n      });\n    },\n\n    stop: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n\n      return errore.tryAsync({\n        try: async () => {\n          await sandbox.stop();\n          const existing = await storage.sandbox.get(id);\n          if (existing instanceof Error || !existing) {\n            return undefined;\n          }\n          await storage.sandbox.set({\n            id: existing.id,\n            config: existing.config,\n            createdAt: existing.createdAt,\n            lastActivityAt: existing.lastActivityAt,\n            acquiringLockId: null,\n            acquiringLockAt: null,\n            providerMetadata: {\n              provider: \"vercel\",\n              sandboxId: null,\n              snapshotId: null,\n            },\n          });\n          return undefined;\n        },\n        catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n      });\n    },\n\n    getStatus: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n      return sandbox.status as SandboxStatus;\n    },\n\n    getCreatedAt: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n      return sandbox.createdAt;\n    },\n\n    getRemainingTimeout: async () => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n      return sandbox.timeout;\n    },\n  };\n\n  const sandbox: Sandbox = {\n    id,\n    config,\n    exec: async ({ command, args, signal }) => {\n      const instance = await getSandbox();\n      if (instance instanceof Error) {\n        return instance;\n      }\n\n      const updatePromise = updateLastActivity();\n\n      const execResult = await errore.tryAsync({\n        try: async () => {\n          const output = await instance.runCommand(command, args, { signal });\n\n          let stdout = \"\";\n          let stderr = \"\";\n          const logBuffer: LogEntry[] = [];\n          const state = {\n            resolve: null as (() => void) | null,\n            consumed: false,\n          };\n\n          const consumeLogs = (async () => {\n            for await (const log of output.logs()) {\n              const entry: LogEntry =\n                log.stream === \"stdout\"\n                  ? { stream: \"stdout\", data: log.data }\n                  : { stream: \"stderr\", data: log.data };\n\n              if (log.stream === \"stdout\") {\n                stdout += log.data;\n              } else {\n                stderr += log.data;\n              }\n\n              logBuffer.push(entry);\n              state.resolve?.();\n            }\n            state.consumed = true;\n            state.resolve?.();\n          })();\n\n          async function* logs(): AsyncIterable<LogEntry> {\n            let index = 0;\n            while (!state.consumed || index < logBuffer.length) {\n              if (index < logBuffer.length) {\n                yield logBuffer[index++];\n              } else {\n                await new Promise<void>((resolve) => {\n                  state.resolve = resolve;\n                });\n                state.resolve = null;\n              }\n            }\n          }\n\n          const result = consumeLogs.then(() => ({\n            stdout,\n            stderr,\n            exitCode: output.exitCode,\n          }));\n\n          return { commandId: output.cmdId, logs, result };\n        },\n        catch: (e) => new SandboxError({ reason: String(e), cause: e }),\n      });\n\n      await updatePromise;\n      return execResult;\n    },\n\n    getDomain: async (port) => {\n      const sandbox = await getSandbox();\n      if (sandbox instanceof Error) {\n        return sandbox;\n      }\n\n      try {\n        return sandbox.domain(port);\n      } catch (e) {\n        return new SandboxError({ reason: String(e), cause: e });\n      }\n    },\n\n    kill: async ({ commandId, storage: cmdStorage }) => {\n      const instance = await getSandbox();\n      if (instance instanceof Error) {\n        return instance;\n      }\n\n      const cmd = await cmdStorage.command.get(commandId);\n      if (cmd instanceof Error) {\n        return new SandboxError({ reason: cmd.message, cause: cmd });\n      }\n      if (cmd && cmd.status === \"running\") {\n        const result = await cmdStorage.command.set({\n          ...cmd,\n          status: \"killed\",\n        });\n        if (result instanceof Error) {\n          return new SandboxError({ reason: result.message, cause: result });\n        }\n      }\n      return undefined;\n    },\n\n    writeFiles: (opts) => writeFiles({ sandbox, ...opts }),\n\n    lifecycle,\n  };\n\n  if (config.lifecycle?.autoStart !== false) {\n    sandboxPromise = doGetSandbox();\n  }\n\n  return sandbox;\n};\n", "import type { SandboxRecord, StorageConfig } from \"../storage\";\nimport { localSandbox } from \"./bindings/local\";\nimport { vercelSandbox } from \"./bindings/vercel\";\nimport type { Sandbox } from \"./types\";\n\nexport function getSandbox({\n  sandboxRecord,\n  storageConfig,\n}: {\n  storageConfig: StorageConfig;\n  sandboxRecord: SandboxRecord;\n}): Sandbox {\n  const { config } = sandboxRecord;\n  switch (config.type) {\n    case \"local\":\n      return localSandbox({\n        sandboxRecord: sandboxRecord as SandboxRecord & {\n          config: { type: \"local\" };\n        },\n        storageConfig,\n      });\n    case \"vercel\":\n      return vercelSandbox({\n        sandboxRecord: sandboxRecord as SandboxRecord & {\n          config: { type: \"vercel\" };\n        },\n        storageConfig,\n      });\n    case \"custom\":\n      throw new Error(\"Custom sandboxes are not supported\");\n    default:\n      config satisfies never;\n      throw new Error(\n        `Unknown sandbox type: ${\n          // biome-ignore lint/suspicious/noExplicitAny: .\n          (config as any).type\n        }`\n      );\n  }\n}\n", "export { getSandbox } from \"./client\";\nexport type {\n  ExecResult,\n  LogEntry,\n  Sandbox,\n  SandboxLifecycle,\n  SandboxLifecycleInput,\n  SandboxStatus,\n  Terminal,\n  TerminalMetadata,\n  VercelSandboxLifecycleConfig,\n} from \"./types\";\n", "export {\n  type AgentOptions,\n  agent,\n  type SendInput,\n  type TagsSchema,\n} from \"./client\";\nexport {\n  MessageNotFoundError,\n  SandboxError,\n  SandboxNotFoundError,\n  SessionNotFoundError,\n  StorageError,\n} from \"./errors\";\nexport { mcp } from \"./mcp\";\nexport type { ExecResult, Sandbox } from \"./sandbox\";\nexport type { SkillSummary, UploadableFile } from \"./skills/types\";\nexport {\n  type Handlers,\n  handleStorageRpc,\n  type ListResult,\n  type Message,\n  type Part,\n  type ResolvedStorage,\n  type RpcRequest,\n  type RpcResponse,\n  type SandboxConfig,\n  type SandboxRecord,\n  type Session,\n  type StorageConfig,\n} from \"./storage\";\n", "import type { GatewayModelId, UIMessage } from \"ai\";\nimport { all } from \"better-all\";\nimport * as errore from \"errore\";\nimport equal from \"fast-deep-equal\";\nimport { ulid } from \"ulid\";\nimport { getRun, start } from \"workflow/api\";\nimport {\n  type AgentInput,\n  type AgentMessageInput,\n  agentMessageHook,\n  agentWorkflow,\n} from \"./agent-workflow\";\nimport {\n  SandboxError,\n  SandboxNotFoundError,\n  SessionError,\n  SessionNotFoundError,\n} from \"./errors\";\nimport { type McpConfig, type McpSendOptions, serializeMcpConfig } from \"./mcp\";\nimport { getSandbox } from \"./sandbox\";\nimport { normalizeSkillsDirs } from \"./skills/parser\";\nimport type { SkillsDir } from \"./skills/types\";\nimport {\n  getStorage,\n  type Message,\n  type Part,\n  type Session,\n  type StorageConfig,\n} from \"./storage\";\nimport type { SandboxConfig } from \"./storage/rpc\";\nimport { toMessageAndParts } from \"./utils/message\";\nimport { assembleUIMessages } from \"./utils/ui\";\n\nexport type SendInput =\n  | string\n  | { role?: UIMessage[\"role\"]; parts: UIMessage[\"parts\"]; id?: string };\n\ntype SessionOptions<\n  // biome-ignore lint/complexity/noBannedTypes: needed for generic default\n  TMcp extends McpConfig = {},\n  TTags extends TagsSchema = TagsSchema\n> = {\n  model?: GatewayModelId;\n  instructions?: string;\n  sandbox?: SandboxConfig | string;\n  tags?: TTags;\n  skillsDir?: SkillsDir;\n  mcp?: TMcp;\n};\n\nexport type McpToolCallBody = {\n  server: string;\n  tool: string;\n  input: unknown;\n  sessionId: string;\n  messageId: string;\n  hookToken: string;\n};\n\n// biome-ignore lint/complexity/noBannedTypes: needed for generic default\nexport type AgentOptions<TMcp extends McpConfig = {}> = {\n  storage?: StorageConfig;\n  generateId?: (\n    resource: \"session\" | \"message\" | \"part\" | \"sandbox\" | \"command\"\n  ) => string;\n} & SessionOptions<TMcp>;\n\nexport type TagsSchema = Record<string, unknown>;\n\n// biome-ignore lint/complexity/noBannedTypes: needed for generic default\nexport const agent = <TMcp extends McpConfig = {}>(\n  options: AgentOptions<TMcp>\n) => {\n  const storageConfig = options.storage ?? { type: \"vercel\" };\n  const storage = getStorage(storageConfig);\n  const generateId =\n    options.generateId ?? ((resource) => `${resource}_${ulid()}`);\n\n  const sandbox = async (sandboxId: string, config?: SandboxConfig) => {\n    const existingRecord = await storage.sandbox.get(sandboxId);\n\n    if (existingRecord instanceof Error) {\n      if (!(existingRecord instanceof SandboxNotFoundError)) {\n        throw existingRecord;\n      }\n      const sandboxRecord = await storage.sandbox.set({\n        id: sandboxId,\n        acquiringLockAt: null,\n        acquiringLockId: null,\n        createdAt: Date.now(),\n        lastActivityAt: Date.now(),\n        config: config ?? { type: \"vercel\" },\n      });\n      if (sandboxRecord instanceof Error) {\n        throw sandboxRecord;\n      }\n      return getSandbox({ storageConfig, sandboxRecord });\n    }\n\n    const newConfig = config ?? existingRecord.config;\n    if (!equal(existingRecord.config, newConfig)) {\n      throw new SandboxError({\n        reason: \"sandbox config changes are not supported\",\n      });\n    }\n\n    return getSandbox({ storageConfig, sandboxRecord: existingRecord });\n  };\n\n  return {\n    session: async <\n      TTags extends TagsSchema = TagsSchema,\n      // biome-ignore lint/complexity/noBannedTypes: needed for generic default\n      TMcpOverride extends McpConfig = {}\n    >(\n      sessionId: string,\n      sessionOptions?: SessionOptions<TMcp & TMcpOverride, TTags>\n    ) => {\n      let session = await storage.session.get(sessionId);\n\n      /**\n       * normalize sandbox input to a sandbox ID and config\n       */\n      const sandboxInput = sessionOptions?.sandbox ?? options.sandbox;\n      const sandboxId =\n        typeof sandboxInput === \"string\"\n          ? sandboxInput\n          : // biome-ignore lint/style/noNestedTernary: ok\n          session instanceof Error\n          ? generateId(\"sandbox\")\n          : session.sandboxId ?? generateId(\"sandbox\");\n      const sandboxPromise = sandbox(\n        sandboxId,\n        typeof sandboxInput === \"string\" ? undefined : sandboxInput\n      );\n\n      const mcpConfig =\n        options.mcp || sessionOptions?.mcp\n          ? { ...options.mcp, ...sessionOptions?.mcp }\n          : null;\n      const skillsDir = sessionOptions?.skillsDir ?? options.skillsDir ?? null;\n\n      if (session instanceof Error) {\n        if (session instanceof SessionNotFoundError) {\n          const instructions =\n            sessionOptions?.instructions ?? options.instructions ?? null;\n          const model = sessionOptions?.model ?? options.model ?? null;\n          if (!model) {\n            throw new SessionError({\n              id: sessionId,\n              reason: \"Model is not set\",\n            });\n          }\n          session = await storage.session.set({\n            id: sessionId,\n            createdAt: Date.now(),\n            updatedAt: Date.now(),\n            lastMessageId: null,\n            runId: null,\n            tags: sessionOptions?.tags ?? null,\n            instructions,\n            model,\n            mcp: mcpConfig ? serializeMcpConfig(mcpConfig) : null,\n            sandboxId,\n            skillsDir: skillsDir ? normalizeSkillsDirs(skillsDir) : null,\n            hookToken: null,\n          });\n        } else {\n          throw session;\n        }\n      } else {\n        /**\n         * this is an existing session. we compare the options and update the session if necessary.\n         */\n        const updated: Session = {\n          ...session,\n          ...sessionOptions,\n          mcp: mcpConfig ? serializeMcpConfig(mcpConfig) : session.mcp,\n          sandboxId,\n          skillsDir: skillsDir ? normalizeSkillsDirs(skillsDir) : null,\n        };\n        if (!equal(updated, session)) {\n          session = await storage.session.set(updated);\n          if (session instanceof Error) {\n            throw session;\n          }\n        }\n      }\n\n      return {\n        send: async ({\n          input,\n          mcpContext,\n        }: {\n          input: SendInput | SendInput[];\n          mcpContext?: McpSendOptions<TMcp & TMcpOverride>;\n        }) => {\n          const newMessages: Message[] = [];\n          const newParts: Part[] = [];\n          if (Array.isArray(input)) {\n            for (const i of input) {\n              const { message, parts } = toMessageAndParts({\n                sessionId,\n                input: i,\n                defaultRole: \"user\",\n                generateId: () => generateId(\"message\"),\n              });\n\n              newMessages.push(message);\n              newParts.push(...parts);\n            }\n          } else {\n            const { message, parts } = toMessageAndParts({\n              sessionId,\n              input,\n              defaultRole: \"user\",\n              generateId: () => generateId(\"message\"),\n            });\n            newMessages.push(message);\n            newParts.push(...parts);\n          }\n          /**\n           * we also pre-create the assistant message\n           */\n          const assistantMessageId = generateId(\"message\");\n          newMessages.push({\n            id: assistantMessageId,\n            sessionId,\n            role: \"assistant\",\n            createdAt: Date.now(),\n            completedAt: null,\n            mcpContext,\n          });\n\n          const now = Date.now();\n          const session2 = await storage.session.get(sessionId);\n          if (session2 instanceof Error) {\n            throw session2;\n          }\n\n          const agentInput: AgentInput = {\n            storageConfig,\n            sessionId,\n          };\n          const event: AgentMessageInput = {\n            assistantMessageId,\n            hookToken: crypto.randomUUID(),\n          };\n\n          return await errore.tryAsync({\n            try: async () => {\n              await all({\n                async saveMessages() {\n                  await Promise.all(newMessages.map(storage.message.set));\n                },\n                async saveParts() {\n                  await Promise.all(newParts.map(storage.part.set));\n                },\n                async resumeOrStartAgentWorkflow() {\n                  if (session2.runId) {\n                    await agentMessageHook.resume(sessionId, event);\n                  } else {\n                    const startResult = await start(agentWorkflow, [\n                      { input: agentInput, event },\n                    ]);\n                    session2.runId = startResult.runId;\n                  }\n                },\n                async updateSession() {\n                  await this.$.resumeOrStartAgentWorkflow;\n                  const result = await storage.session.set({\n                    ...session2,\n                    updatedAt: now,\n                    lastMessageId: assistantMessageId,\n                    hookToken: event.hookToken,\n                  });\n                  if (result instanceof Error) {\n                    throw result;\n                  }\n                  return result;\n                },\n              });\n            },\n            catch: (e) => {\n              if (e instanceof Error) {\n                throw e;\n              }\n              return new SessionError({\n                id: sessionId,\n                reason: `failed to send: ${String(e)}`,\n                cause: e,\n              });\n            },\n          });\n        },\n\n        stream: async (opts?: { messageId?: string }) => {\n          const session = await storage.session.get(sessionId);\n          if (session instanceof Error) {\n            return session;\n          }\n          const messageId = opts?.messageId ?? session.lastMessageId;\n          if (!messageId) {\n            return new SessionError({\n              id: sessionId,\n              reason: \"No message associated with session\",\n            });\n          }\n          if (!session.runId) {\n            return new SessionError({\n              id: sessionId,\n              reason: \"No run associated with session\",\n            });\n          }\n          const run = getRun(session.runId);\n          return run.getReadable({ namespace: messageId });\n        },\n\n        ui: async () => {\n          const [messagesResult, partsResult] = await Promise.all([\n            storage.message.list(sessionId),\n            storage.part.listBySession(sessionId),\n          ]);\n\n          if (messagesResult instanceof Error) {\n            return messagesResult;\n          }\n          if (partsResult instanceof Error) {\n            return partsResult;\n          }\n\n          const messages = assembleUIMessages({\n            messages: messagesResult.items,\n            parts: partsResult.items,\n          });\n\n          const lastMessage = messagesResult.items.at(-1);\n          const isStreaming =\n            lastMessage?.role === \"assistant\" &&\n            lastMessage.completedAt === null;\n          const streamingMessageId = isStreaming ? lastMessage.id : null;\n\n          return { messages, streamingMessageId };\n        },\n        tag: {\n          list: async () => {\n            const session = await storage.session.get(sessionId);\n            if (session instanceof Error) {\n              return session;\n            }\n            return (session.tags ?? {}) as TTags;\n          },\n          get: async (key: string) => {\n            const session = await storage.session.get(sessionId);\n            if (session instanceof Error) {\n              return session;\n            }\n            return session.tags?.[key as string] as\n              | TTags[typeof key]\n              | undefined;\n          },\n          set: async (key: string, value: unknown) => {\n            const result = await storage.session.tag.set({\n              sessionId,\n              tags: { [key]: value } as Record<string, unknown>,\n            });\n            if (result instanceof Error) {\n              return result;\n            }\n            return undefined;\n          },\n          setMany: async (tags: Record<string, unknown>) => {\n            const result = await storage.session.tag.set({\n              sessionId,\n              tags: tags as Record<string, unknown>,\n            });\n            if (result instanceof Error) {\n              return result;\n            }\n            return undefined;\n          },\n        },\n        sandbox: await sandboxPromise,\n      };\n    },\n    sandbox,\n    storage,\n    handleMcpToolCall: async (\n      body: McpToolCallBody,\n      handlers: Record<\n        string,\n        (opts: {\n          input: unknown;\n          headers: Record<string, string>;\n          body: McpToolCallBody;\n        }) => Promise<unknown>\n      >\n    ): Promise<\n      { result: unknown } | { error: { code: string; message: string } }\n    > => {\n      const { server, tool, input, sessionId, messageId, hookToken } = body;\n\n      const handlerKey = `${server}/${tool}`;\n      const handler = handlers[handlerKey];\n      if (!handler) {\n        return {\n          error: {\n            code: \"HANDLER_NOT_FOUND\",\n            message: `No handler for ${handlerKey}`,\n          },\n        };\n      }\n\n      const session = await storage.session.get(sessionId);\n      if (session instanceof Error) {\n        return {\n          error: { code: \"SESSION_NOT_FOUND\", message: \"Session not found\" },\n        };\n      }\n\n      if (!session.hookToken || session.hookToken !== hookToken) {\n        return {\n          error: { code: \"INVALID_TOKEN\", message: \"Invalid hook token\" },\n        };\n      }\n\n      const message = await storage.message.get(messageId);\n      if (message instanceof Error) {\n        return {\n          error: { code: \"MESSAGE_NOT_FOUND\", message: \"Message not found\" },\n        };\n      }\n\n      const mcpContext = message.mcpContext as McpSendOptions<TMcp> | undefined;\n      const serverContext = mcpContext?.[server as keyof typeof mcpContext];\n      const headers = (serverContext?.headers ?? {}) as Record<string, string>;\n\n      try {\n        const result = await handler({ input, headers, body });\n        return { result };\n      } catch (e) {\n        return {\n          error: {\n            code: \"EXECUTION_ERROR\",\n            message: e instanceof Error ? e.message : String(e),\n          },\n        };\n      }\n    },\n  };\n};\n", "import type { FinishReason } from \"ai\";\nimport { defineHook, FatalError, getWritable } from \"workflow\";\nimport { completeMessageStep, streamTextStep } from \"./agent-workflow-steps\";\nimport type { StorageConfig } from \"./storage\";\n\nexport type AgentInput = {\n  sessionId: string;\n  storageConfig: StorageConfig;\n};\n\nexport type AgentMessageInput = {\n  assistantMessageId: string;\n  hookToken: string;\n};\n\nexport const agentMessageHook = defineHook<AgentMessageInput>();\n\nexport async function agentWorkflow({\n  input,\n  event,\n}: {\n  input: AgentInput;\n  event: AgentMessageInput;\n}) {\n  \"use workflow\";\n\n  const messageHook = agentMessageHook.create({ token: input.sessionId });\n  const iterator = messageHook[Symbol.asyncIterator]();\n  let pendingNext = iterator.next();\n\n  await onMessage({ event, input }).catch((e) => {\n    if (FatalError.is(e)) {\n      console.error(\"Message processing failed permanently:\", e.message);\n      return;\n    }\n    throw e;\n  });\n\n  while (true) {\n    const result = await pendingNext;\n\n    if (result.done) {\n      console.error(\"Unexpected: message hook iterator done\");\n      break;\n    }\n\n    await onMessage({ event: result.value, input }).catch((e) => {\n      if (FatalError.is(e)) {\n        console.error(\"Message processing failed permanently:\", e.message);\n        return;\n      }\n      throw e;\n    });\n    pendingNext = iterator.next();\n  }\n}\n\nasync function onMessage({\n  event,\n  input,\n}: {\n  event: AgentMessageInput;\n  input: AgentInput;\n}) {\n  const writable = getWritable({ namespace: event.assistantMessageId });\n\n  let finishReason: FinishReason | undefined;\n  let lastPartIndex = 0;\n\n  while (finishReason !== \"stop\") {\n    try {\n      const result = await streamTextStep({\n        assistantMessageId: event.assistantMessageId,\n        writable,\n        input,\n        event,\n        lastPartIndex,\n      });\n      finishReason = result.finishReason;\n      lastPartIndex = result.lastPartIndex;\n    } catch (err) {\n      console.error(err);\n      throw err;\n    }\n  }\n\n  await completeMessageStep({\n    assistantMessageId: event.assistantMessageId,\n    input,\n    writable,\n  });\n}\n", "import {\n  convertToModelMessages,\n  type FinishReason,\n  streamText,\n  type UIMessage,\n} from \"ai\";\nimport { ulid } from \"ulid\";\nimport { FatalError } from \"workflow\";\nimport type { McpServerSchema } from \"./mcp\";\nimport { discoverSkillsInSandbox } from \"./skills/discover\";\nimport type { SkillSummary } from \"./skills/types\";\nimport type { StorageConfig } from \"./storage\";\nimport { getTools } from \"./tools\";\nimport { assembleUIMessages } from \"./utils/ui\";\n\nconst BASE_SYSTEM_PROMPT =\n  \"You are an AI assistant with basic tools to interact with your environment. Explore and work freely.\";\n\nfunction joinPromptSections(\n  ...sections: (string | undefined | null)[]\n): string {\n  return sections.filter((s) => s?.trim()).join(\"\\n\\n\");\n}\n\nfunction buildSkillsContext(skills: SkillSummary[]): string {\n  if (skills.length === 0) {\n    return \"\";\n  }\n\n  const skillLines = skills\n    .map((s) => `- ${s.name}: ${s.description}\\n  Path: ${s.skillMdPath}`)\n    .join(\"\\n\");\n\n  return `## Available Skills\n${skillLines}\n\nYou can use the Read tool to read any skill's SKILL.md file to learn more about it.`;\n}\n\nfunction buildMcpToolsContext(mcp: McpServerSchema[] | null | undefined): string {\n  if (!mcp || mcp.length === 0) {\n    return \"\";\n  }\n\n  const serverDocs = mcp\n    .filter(\n      (\n        s\n      ): s is McpServerSchema & {\n        tools: NonNullable<McpServerSchema[\"tools\"]>;\n      } => Array.isArray(s.tools) && s.tools.length > 0\n    )\n    .map((server) => {\n      const toolDocs = server.tools\n        .map((t) => {\n          let doc = `  - ${t.name}: ${\n            t.description\n          }\\n    Input: ${JSON.stringify(t.inputSchema)}`;\n          if (t.outputSchema) {\n            doc += `\\n    Output: ${JSON.stringify(t.outputSchema)}`;\n          }\n          return doc;\n        })\n        .join(\"\\n\");\n      return `### ${server.name}\\n${server.description ?? \"\"}\\n${toolDocs}`;\n    })\n    .join(\"\\n\\n\");\n\n  if (!serverDocs) {\n    return \"\";\n  }\n\n  return `## MCP Tools\nUse the ExecuteMCPTool to call these custom tools:\n\n${serverDocs}`;\n}\n\nexport type AgentInput = {\n  sessionId: string;\n  storageConfig: StorageConfig;\n};\n\nexport type AgentMessageInput = {\n  assistantMessageId: string;\n  hookToken: string;\n};\n\nexport async function completeMessageStep({\n  assistantMessageId,\n  input,\n  writable,\n}: {\n  assistantMessageId: string;\n  input: AgentInput;\n  writable: WritableStream;\n}) {\n  \"use step\";\n\n  const { getStorage } = await import(\"./storage\");\n  const storage = getStorage(input.storageConfig);\n\n  const message = await storage.message.get(assistantMessageId);\n  if (message instanceof Error) {\n    throw message;\n  }\n  if (!message) {\n    throw new Error(`Message ${assistantMessageId} not found`);\n  }\n\n  const result = await storage.message.set({\n    ...message,\n    completedAt: Date.now(),\n  });\n  if (result instanceof Error) {\n    throw result;\n  }\n\n  await writable.close();\n}\n\nexport async function streamTextStep({\n  assistantMessageId,\n  input,\n  event,\n  writable,\n  lastPartIndex,\n}: {\n  assistantMessageId: string;\n  input: AgentInput;\n  event: AgentMessageInput;\n  writable: WritableStream;\n  lastPartIndex: number;\n}): Promise<{ finishReason: FinishReason; lastPartIndex: number }> {\n  \"use step\";\n\n  const { getStorage } = await import(\"./storage/client\");\n  const { getSandbox } = await import(\"./sandbox\");\n\n  const storage = getStorage(input.storageConfig);\n  const session = await storage.session.get(input.sessionId);\n\n  if (session instanceof Error) {\n    throw session;\n  }\n  session;\n\n  const sandboxRecord = session.sandboxId\n    ? await storage.sandbox.get(session.sandboxId)\n    : null;\n  if (sandboxRecord instanceof Error) {\n    throw sandboxRecord;\n  }\n  if (!sandboxRecord) {\n    throw new FatalError(`Sandbox not found for session ${input.sessionId}`);\n  }\n\n  const sandbox = getSandbox({\n    sandboxRecord,\n    storageConfig: input.storageConfig,\n  });\n\n  const [messagesResult, partsResult, skills] = await Promise.all([\n    storage.message.list(input.sessionId),\n    storage.part.listBySession(input.sessionId),\n    session.skillsDir && session.skillsDir.length > 0\n      ? discoverSkillsInSandbox({ sandbox, skillsDirs: session.skillsDir })\n      : Promise.resolve([]),\n  ]);\n\n  if (messagesResult instanceof Error) {\n    throw messagesResult;\n  }\n  if (partsResult instanceof Error) {\n    throw partsResult;\n  }\n\n  const uiMessages = assembleUIMessages({\n    messages: messagesResult.items,\n    parts: partsResult.items,\n  });\n\n  const mcp = session.mcp as McpServerSchema[] | null;\n\n  const systemPrompt = joinPromptSections(\n    BASE_SYSTEM_PROMPT,\n    session.instructions,\n    buildSkillsContext(skills),\n    buildMcpToolsContext(mcp)\n  );\n\n  if (!session.model) {\n    throw new FatalError(\"Session model is not set\");\n  }\n\n  const result = streamText({\n    messages: await convertToModelMessages(uiMessages),\n    tools: getTools({ input, event, mcp, sandbox }),\n    system: systemPrompt,\n    model: session.model,\n  });\n\n  const stepParts: UIMessage[\"parts\"] = [];\n  await result\n    .toUIMessageStream({\n      generateMessageId: () => assistantMessageId,\n      onFinish: ({ messages }) => {\n        for (const m of messages) {\n          if (m.role === \"assistant\") {\n            stepParts.push(...m.parts);\n          }\n        }\n      },\n    })\n    .pipeTo(writable, { preventClose: true });\n\n  await Promise.all(\n    stepParts.map(async (uiPart, index) => {\n      const result = await storage.part.set({\n        id: `part_${ulid()}`,\n        index: lastPartIndex + index,\n        messageId: assistantMessageId,\n        sessionId: input.sessionId,\n        part: uiPart,\n      });\n      if (result instanceof Error) {\n        throw result;\n      }\n      return result;\n    })\n  );\n\n  return {\n    finishReason: await result.finishReason,\n    lastPartIndex: lastPartIndex + stepParts.length,\n  };\n}\n", "import type { SkillsDir } from \"./types\";\n\n/**\n * Parsed frontmatter from a SKILL.md file.\n */\nexport type SkillFrontmatter = {\n  name: string;\n  description: string;\n};\n\n/**\n * Parses YAML frontmatter from a SKILL.md file content.\n * Frontmatter must be at the start of the file, delimited by `---` markers.\n *\n * @example\n * ```markdown\n * ---\n * name: csv\n * description: Analyze CSV data\n * ---\n * # CSV Skill\n * ...\n * ```\n *\n * @returns Parsed name and description, or null if frontmatter is missing/invalid\n */\nexport function parseSkillFrontmatter(\n  content: string\n): SkillFrontmatter | null {\n  const trimmed = content.trim();\n\n  if (!trimmed.startsWith(\"---\")) {\n    return null;\n  }\n\n  const endMarkerIndex = trimmed.indexOf(\"---\", 3);\n  if (endMarkerIndex === -1) {\n    return null;\n  }\n\n  const frontmatterBlock = trimmed.slice(3, endMarkerIndex).trim();\n  const parsed = parseSimpleYaml(frontmatterBlock);\n\n  if (!(parsed.name && parsed.description)) {\n    return null;\n  }\n\n  return {\n    name: String(parsed.name),\n    description: String(parsed.description),\n  };\n}\n\n/**\n * Parses simple YAML key-value pairs (single-line values only).\n * This avoids adding a full YAML parser dependency for basic frontmatter.\n */\nfunction parseSimpleYaml(yaml: string): Record<string, string> {\n  const result: Record<string, string> = {};\n\n  for (const line of yaml.split(\"\\n\")) {\n    const trimmedLine = line.trim();\n    if (!trimmedLine || trimmedLine.startsWith(\"#\")) {\n      continue;\n    }\n\n    const colonIndex = trimmedLine.indexOf(\":\");\n    if (colonIndex === -1) {\n      continue;\n    }\n\n    const key = trimmedLine.slice(0, colonIndex).trim();\n    let value = trimmedLine.slice(colonIndex + 1).trim();\n\n    // Remove surrounding quotes if present\n    if (\n      (value.startsWith('\"') && value.endsWith('\"')) ||\n      (value.startsWith(\"'\") && value.endsWith(\"'\"))\n    ) {\n      value = value.slice(1, -1);\n    }\n\n    if (key) {\n      result[key] = value;\n    }\n  }\n\n  return result;\n}\n\n/**\n * Normalizes skillsDir to an array of strings.\n */\nexport function normalizeSkillsDirs(skillsDir?: SkillsDir): string[] {\n  if (!skillsDir) {\n    return [];\n  }\n  return Array.isArray(skillsDir) ? skillsDir : [skillsDir];\n}\n", "import type { Sandbox } from \"../sandbox\";\nimport { parseSkillFrontmatter } from \"./parser\";\nimport type { SkillSummary } from \"./types\";\n\n/**\n * Discovers skills from directories inside the sandbox by finding and parsing SKILL.md files.\n * Scans each directory for subdirectories containing SKILL.md, extracts frontmatter metadata,\n * and returns summaries for use in the system prompt.\n *\n * @returns Array of skill summaries (deduplicated by name, first occurrence wins)\n */\nexport async function discoverSkillsInSandbox(opts: {\n  sandbox: Sandbox;\n  skillsDirs: string[];\n  debug?: boolean;\n}): Promise<SkillSummary[]> {\n  const { sandbox, skillsDirs, debug } = opts;\n  const summaries: SkillSummary[] = [];\n  const seenNames = new Set<string>();\n\n  for (const skillsDir of skillsDirs) {\n    const dirSummaries = await discoverSkillsInDirectory({\n      sandbox,\n      skillsDir,\n      debug,\n    });\n\n    for (const summary of dirSummaries) {\n      if (!seenNames.has(summary.name)) {\n        seenNames.add(summary.name);\n        summaries.push(summary);\n      }\n    }\n  }\n\n  return summaries;\n}\n\nasync function discoverSkillsInDirectory(opts: {\n  sandbox: Sandbox;\n  skillsDir: string;\n  debug?: boolean;\n}): Promise<SkillSummary[]> {\n  const { sandbox, skillsDir, debug } = opts;\n  const skillPaths = await findSkillFiles({ sandbox, skillsDir, debug });\n\n  if (skillPaths.length === 0) {\n    return [];\n  }\n\n  const summaries: SkillSummary[] = [];\n\n  for (const skillMdPath of skillPaths) {\n    const summary = await parseSkillFile({ sandbox, skillMdPath, debug });\n    if (summary) {\n      summaries.push(summary);\n    }\n  }\n\n  return summaries;\n}\n\nasync function findSkillFiles(opts: {\n  sandbox: Sandbox;\n  skillsDir: string;\n  debug?: boolean;\n}): Promise<string[]> {\n  const { sandbox, skillsDir, debug } = opts;\n  if (debug) {\n    console.log(`[discover] Finding skills in: ${skillsDir}`);\n  }\n  const execResult = await sandbox.exec({\n    command: \"find\",\n    args: [skillsDir, \"-name\", \"SKILL.md\", \"-type\", \"f\"],\n  });\n\n  if (execResult instanceof Error) {\n    if (debug) {\n      console.warn(\n        `[discover] Failed to scan skills directory \"${skillsDir}\": ${execResult.message}`\n      );\n    }\n    return [];\n  }\n\n  const { stdout, stderr, exitCode } = await execResult.result;\n  if (debug) {\n    console.log(\n      `[discover] find result: exitCode=${exitCode}, stdout=\"${stdout.trim()}\", stderr=\"${stderr.trim()}\"`\n    );\n  }\n\n  if (exitCode !== 0) {\n    if (debug) {\n      console.warn(\n        `[discover] Skills directory not found or inaccessible: ${skillsDir}`\n      );\n    }\n    return [];\n  }\n\n  const paths = stdout\n    .trim()\n    .split(\"\\n\")\n    .filter((p) => p.length > 0);\n  if (debug) {\n    console.log(\"[discover] Found skill paths:\", paths);\n  }\n  return paths;\n}\n\nasync function parseSkillFile(opts: {\n  sandbox: Sandbox;\n  skillMdPath: string;\n  debug?: boolean;\n}): Promise<SkillSummary | null> {\n  const { sandbox, skillMdPath, debug } = opts;\n  const execResult = await sandbox.exec({\n    command: \"cat\",\n    args: [skillMdPath],\n  });\n\n  if (execResult instanceof Error) {\n    if (debug) {\n      console.warn(\n        `[discover] Failed to read skill file \"${skillMdPath}\": ${execResult.message}`\n      );\n    }\n    return null;\n  }\n\n  const { stdout, exitCode } = await execResult.result;\n\n  if (exitCode !== 0) {\n    if (debug) {\n      console.warn(`[discover] Could not read skill file: ${skillMdPath}`);\n    }\n    return null;\n  }\n\n  const parsed = parseSkillFrontmatter(stdout);\n\n  if (!parsed) {\n    if (debug) {\n      console.warn(\n        `[discover] Invalid or missing frontmatter in: ${skillMdPath}`\n      );\n    }\n    return null;\n  }\n\n  return {\n    name: parsed.name,\n    description: parsed.description,\n    skillMdPath,\n  };\n}\n", "import { type ToolSet, tool } from \"ai\";\nimport { z } from \"zod\";\nimport type { AgentInput, AgentMessageInput } from \"../agent-workflow\";\nimport type { McpServerSchema } from \"../mcp\";\nimport type { Sandbox } from \"../sandbox\";\n\nexport type ToolContext = {\n  input: AgentInput;\n  event: AgentMessageInput;\n  mcp: McpServerSchema[] | null;\n  sandbox: Sandbox;\n  waitUntil?: (promise: Promise<unknown>) => void;\n};\n\nexport function getTools(context: ToolContext) {\n  return {\n    Read: tool({\n      description:\n        \"Reads a file and returns its contents with metadata. For files over 200 lines, automatically shows first 100 lines unless a specific line range is provided. Use startLine and endLine parameters to read specific portions of large files.\",\n      inputSchema: z.object({\n        path: z\n          .string()\n          .describe(\"Path to the file relative to workspace root\"),\n        startLine: z\n          .number()\n          .optional()\n          .describe(\n            \"Starting line number (1-indexed). If provided with endLine, reads exact range regardless of file size.\"\n          ),\n        endLine: z\n          .number()\n          .optional()\n          .describe(\n            \"Ending line number (1-indexed, inclusive). If provided with startLine, reads exact range regardless of file size.\"\n          ),\n      }),\n      outputSchema: z.object({\n        content: z.string().describe(\"File content\"),\n        metadata: z.object({\n          totalLines: z.number().describe(\"Total number of lines in the file\"),\n          linesShown: z\n            .number()\n            .describe(\"Number of lines included in this response\"),\n          startLine: z.number().describe(\"First line number shown (1-indexed)\"),\n          endLine: z.number().describe(\"Last line number shown (1-indexed)\"),\n          isPaginated: z\n            .boolean()\n            .describe(\"Whether this is a partial view of the file\"),\n          fileSize: z\n            .string()\n            .describe(\"Human-readable file size (e.g., '2.5K', '1.2M')\"),\n          path: z\n            .string()\n            .describe(\"Path to the file relative to workspace root\"),\n        }),\n      }),\n      execute: async ({ path, startLine, endLine }) => {\n        const filePath = path.startsWith(\"/\") ? path.slice(1) : path;\n\n        const result = await context.sandbox.exec({\n          command: \"bash\",\n          args: [\n            \"-c\",\n            `\n            set -e\n            FILE=\"$1\"\n            START_LINE=\"$2\"\n            END_LINE=\"$3\"\n\n            # Resolve symlinks and check file exists\n            if [ -L \"$FILE\" ]; then\n              RESOLVED=$(readlink -f \"$FILE\" 2>/dev/null || echo \"\")\n              if [ -z \"$RESOLVED\" ] || [ ! -e \"$RESOLVED\" ]; then\n                echo \"Error: Broken symlink - $FILE points to non-existent target\" >&2\n                exit 1\n              fi\n              FILE=\"$RESOLVED\"\n            elif [ ! -e \"$FILE\" ]; then\n              echo \"Error: File not found - $FILE\" >&2\n              exit 1\n            fi\n\n            # Get metadata (count actual lines, not just newlines)\n            TOTAL_LINES=$(awk 'END{print NR}' \"$FILE\")\n            FILE_SIZE=$(ls -lh \"$FILE\" | awk '{print $5}')\n\n            # Determine range\n            PAGE_SIZE=100\n            if [ -n \"$START_LINE\" ] && [ -n \"$END_LINE\" ]; then\n              # Both provided - use exact range\n              ACTUAL_START=$START_LINE\n              ACTUAL_END=$END_LINE\n            elif [ -n \"$START_LINE\" ]; then\n              # Only startLine - read PAGE_SIZE lines from there\n              ACTUAL_START=$START_LINE\n              ACTUAL_END=$((START_LINE + PAGE_SIZE - 1))\n              [ \"$ACTUAL_END\" -gt \"$TOTAL_LINES\" ] && ACTUAL_END=$TOTAL_LINES\n            elif [ -n \"$END_LINE\" ]; then\n              # Only endLine - read from beginning to endLine\n              ACTUAL_START=1\n              ACTUAL_END=$END_LINE\n            elif [ \"$TOTAL_LINES\" -gt 200 ]; then\n              # No range, large file - paginate\n              ACTUAL_START=1\n              ACTUAL_END=$PAGE_SIZE\n            else\n              # No range, small file - show all\n              ACTUAL_START=1\n              ACTUAL_END=$TOTAL_LINES\n            fi\n\n            # Output metadata first (separated by ||| for parsing)\n            echo \"$TOTAL_LINES|$FILE_SIZE|$ACTUAL_START|$ACTUAL_END\"\n            echo \"|||CONTENT|||\"\n\n            # Read content\n            if [ \"$ACTUAL_START\" -eq 1 ] && [ \"$ACTUAL_END\" -eq \"$TOTAL_LINES\" ]; then\n              cat \"$FILE\"\n            else\n              sed -n \"\\${ACTUAL_START},\\${ACTUAL_END}p\" \"$FILE\"\n            fi\n          `,\n            \"--\",\n            filePath,\n            startLine?.toString() || \"\",\n            endLine?.toString() || \"\",\n          ],\n        });\n\n        if (result instanceof Error) {\n          console.error(\"[Read Tool]\", result);\n          throw result;\n        }\n\n        const { stdout, stderr } = await result.result;\n\n        if (stderr) {\n          console.error(`[Read Tool] Error: ${stderr}`);\n          return {\n            content: `Error: ${stderr}`,\n            metadata: {\n              totalLines: 0,\n              linesShown: 0,\n              startLine: 0,\n              endLine: 0,\n              isPaginated: false,\n              fileSize: \"0\",\n              path: filePath,\n            },\n          };\n        }\n\n        const [metadataLine, ...rest] = stdout.split(\"|||CONTENT|||\");\n        const content = rest.join(\"|||CONTENT|||\").trimStart();\n        const [totalLinesStr, fileSize, actualStartStr, actualEndStr] =\n          metadataLine.trim().split(\"|\");\n\n        const totalLines = Number.parseInt(totalLinesStr, 10);\n        const actualStart = Number.parseInt(actualStartStr, 10);\n        const actualEnd = Number.parseInt(actualEndStr, 10);\n\n        if (\n          Number.isNaN(totalLines) ||\n          Number.isNaN(actualStart) ||\n          Number.isNaN(actualEnd)\n        ) {\n          console.error(\n            `[Read Tool] Failed to parse metadata: ${metadataLine}`\n          );\n          return {\n            content: `Error: Failed to parse file metadata. Raw output: ${stdout.slice(\n              0,\n              500\n            )}`,\n            metadata: {\n              totalLines: 0,\n              linesShown: 0,\n              startLine: 0,\n              endLine: 0,\n              isPaginated: false,\n              fileSize: \"unknown\",\n              path: filePath,\n            },\n          };\n        }\n\n        return {\n          content,\n          metadata: {\n            totalLines,\n            linesShown: Math.max(0, actualEnd - actualStart + 1),\n            startLine: actualStart,\n            endLine: actualEnd,\n            isPaginated: actualEnd < totalLines,\n            fileSize: fileSize || \"unknown\",\n            path: filePath,\n          },\n        };\n      },\n    }),\n    Grep: tool({\n      description:\n        \"Search for patterns in files using ripgrep. Supports regex patterns, file type filtering, and context lines. Returns matching lines with file paths and line numbers. Use this to find code patterns, function definitions, imports, etc.\",\n      inputSchema: z.object({\n        pattern: z\n          .string()\n          .describe(\"Regex pattern to search for (ripgrep syntax)\"),\n        path: z\n          .string()\n          .optional()\n          .describe(\n            \"Path to search in (defaults to workspace root). Can be a file or directory.\"\n          ),\n        fileType: z\n          .string()\n          .optional()\n          .describe(\n            \"File type to filter by (e.g., 'ts', 'js', 'py', 'md'). Uses ripgrep's built-in type filters.\"\n          ),\n        glob: z\n          .string()\n          .optional()\n          .describe(\n            \"Glob pattern to filter files (e.g., '*.tsx', 'src/**/*.ts')\"\n          ),\n        caseSensitive: z\n          .boolean()\n          .optional()\n          .default(true)\n          .describe(\"Whether search is case-sensitive (default: true)\"),\n        contextLines: z\n          .number()\n          .optional()\n          .describe(\n            \"Number of context lines to show before and after each match\"\n          ),\n        maxCount: z\n          .number()\n          .optional()\n          .describe(\n            \"Maximum number of matches per file (useful for limiting output)\"\n          ),\n        filesWithMatches: z\n          .boolean()\n          .optional()\n          .default(false)\n          .describe(\n            \"Only show file paths that contain matches, not the matching lines themselves\"\n          ),\n      }),\n      outputSchema: z.object({\n        matches: z\n          .string()\n          .describe(\n            \"Search results with file paths, line numbers, and matching content\"\n          ),\n        summary: z.object({\n          matchCount: z.number().describe(\"Number of matches found\"),\n          fileCount: z.number().describe(\"Number of files containing matches\"),\n          searchPath: z.string().describe(\"Path that was searched\"),\n          pattern: z.string().describe(\"Pattern that was searched for\"),\n        }),\n      }),\n      execute: async ({\n        pattern,\n        path,\n        fileType,\n        glob,\n        caseSensitive,\n        contextLines,\n        maxCount,\n        filesWithMatches,\n      }) => {\n        let searchPath = path ?? \".\";\n        if (searchPath.startsWith(\"/\")) {\n          searchPath = searchPath.slice(1);\n        }\n\n        const args: string[] = [];\n\n        args.push(\"--line-number\");\n        args.push(\"--heading\");\n        args.push(\"--color\", \"never\");\n\n        if (!caseSensitive) {\n          args.push(\"-i\");\n        }\n\n        if (fileType) {\n          args.push(\"--type\", fileType);\n        }\n\n        if (glob) {\n          args.push(\"--glob\", glob);\n        }\n\n        if (contextLines !== undefined) {\n          args.push(\"-C\", String(contextLines));\n        }\n\n        if (maxCount !== undefined) {\n          args.push(\"--max-count\", String(maxCount));\n        }\n\n        if (filesWithMatches) {\n          args.push(\"--files-with-matches\");\n        }\n\n        args.push(\"--\", pattern, searchPath);\n\n        const result = await context.sandbox.exec({ command: \"rg\", args });\n\n        if (result instanceof Error) {\n          console.error(\"[Grep Tool]\", result);\n          throw result;\n        }\n\n        const { stdout, stderr } = await result.result;\n\n        if (stderr && !stderr.toLowerCase().includes(\"no matches\")) {\n          console.error(`[Grep Tool] Warning: ${stderr}`);\n        }\n\n        // Truncate output to prevent \"input too long\" errors (50k chars \u2248 12.5k tokens)\n        const MAX_GREP_OUTPUT_CHARS = 50_000;\n        let finalOutput = stdout;\n        let wasTruncated = false;\n        if (finalOutput.length > MAX_GREP_OUTPUT_CHARS) {\n          finalOutput =\n            finalOutput.slice(0, MAX_GREP_OUTPUT_CHARS) +\n            \"\\n\\n[Output truncated - use more specific pattern or path]\";\n          wasTruncated = true;\n        }\n\n        const lines = finalOutput\n          .trim()\n          .split(\"\\n\")\n          .filter((l) => l.length > 0);\n        const fileCount = filesWithMatches\n          ? lines.length\n          : new Set(\n              lines\n                .filter((l) => !l.startsWith(\" \") && l.includes(\":\"))\n                .map((l) => l.split(\":\")[0])\n            ).size;\n\n        return {\n          matches: finalOutput || \"(no matches found)\",\n          summary: {\n            matchCount: filesWithMatches\n              ? 0\n              : lines.filter((l) => l.includes(\":\")).length,\n            fileCount,\n            searchPath,\n            pattern,\n            wasTruncated,\n          },\n        };\n      },\n    }),\n    List: tool({\n      description:\n        \"Recursively list directory contents. Use this to understand the codebase structure, find files, or explore directories. Control depth to balance detail vs. overview. Depth 1 shows immediate children, depth 2 includes subdirectories, etc.\",\n      inputSchema: z.object({\n        path: z\n          .string()\n          .optional()\n          .describe(\"Path to list (defaults to workspace root)\"),\n        depth: z\n          .number()\n          .optional()\n          .describe(\n            \"Maximum depth to traverse. Choose based on context: 1-2 for quick overview, 3-4 for detailed exploration, 5+ for comprehensive mapping\"\n          ),\n        includeHidden: z\n          .boolean()\n          .optional()\n          .default(false)\n          .describe(\n            \"Include hidden files and directories (those starting with '.')\"\n          ),\n        filesOnly: z\n          .boolean()\n          .optional()\n          .default(false)\n          .describe(\"Only show files, not directories\"),\n        pattern: z\n          .string()\n          .optional()\n          .describe(\"Glob pattern to filter results (e.g., '*.ts', '*test*')\"),\n      }),\n      outputSchema: z.object({\n        listing: z\n          .string()\n          .describe(\n            \"Directory tree listing showing paths relative to search root\"\n          ),\n        summary: z.object({\n          totalItems: z.number().describe(\"Total number of items found\"),\n          totalFiles: z.number().describe(\"Total number of files found\"),\n          totalDirs: z.number().describe(\"Total number of directories found\"),\n          searchPath: z.string().describe(\"Path that was listed\"),\n          depth: z\n            .number()\n            .optional()\n            .describe(\"Maximum depth used (if specified)\"),\n        }),\n      }),\n      execute: async ({ path, depth, includeHidden, filesOnly, pattern }) => {\n        const searchPath = path ?? \".\";\n\n        const result = await context.sandbox.exec({\n          command: \"bash\",\n          args: [\n            \"-c\",\n            `\n            set -e\n            SEARCH_PATH=\"$1\"\n            DEPTH=\"$2\"\n            INCLUDE_HIDDEN=\"$3\"\n            FILES_ONLY=\"$4\"\n            PATTERN=\"$5\"\n\n            # Build find command arguments\n            FIND_ARGS=\"\"\n            [ -n \"$DEPTH\" ] && FIND_ARGS=\"$FIND_ARGS -maxdepth $DEPTH\"\n            [ \"$INCLUDE_HIDDEN\" != \"true\" ] && FIND_ARGS=\"$FIND_ARGS ! -path '*/.*'\"\n            [ \"$FILES_ONLY\" = \"true\" ] && FIND_ARGS=\"$FIND_ARGS -type f\"\n            [ -n \"$PATTERN\" ] && FIND_ARGS=\"$FIND_ARGS -name '$PATTERN'\"\n\n            # Get listing\n            LISTING=$(eval \"find '$SEARCH_PATH' $FIND_ARGS\" 2>/dev/null | sort)\n\n            # Get counts\n            COUNT_ARGS=\"\"\n            [ -n \"$DEPTH\" ] && COUNT_ARGS=\"$COUNT_ARGS -maxdepth $DEPTH\"\n            [ \"$INCLUDE_HIDDEN\" != \"true\" ] && COUNT_ARGS=\"$COUNT_ARGS ! -path '*/.*'\"\n\n            FILE_COUNT=$(eval \"find '$SEARCH_PATH' $COUNT_ARGS -type f\" 2>/dev/null | wc -l)\n            DIR_COUNT=$(eval \"find '$SEARCH_PATH' $COUNT_ARGS -type d\" 2>/dev/null | wc -l)\n\n            # Output: counts first, then listing\n            echo \"$FILE_COUNT|$DIR_COUNT\"\n            echo \"|||LISTING|||\"\n            echo \"$LISTING\" | sed \"s|^$SEARCH_PATH|.|\"\n          `,\n            \"--\",\n            searchPath,\n            depth?.toString() || \"\",\n            includeHidden ? \"true\" : \"false\",\n            filesOnly ? \"true\" : \"false\",\n            pattern || \"\",\n          ],\n        });\n\n        if (result instanceof Error) {\n          console.error(\"[List Tool]\", result);\n          throw result;\n        }\n\n        const { stdout, stderr } = await result.result;\n\n        if (stderr) {\n          console.warn(`[List Tool] stderr: ${stderr}`);\n        }\n\n        const [countsLine, ...rest] = stdout.split(\"|||LISTING|||\");\n        const listing = rest.join(\"|||LISTING|||\").trim();\n        const [fileCountStr, dirCountStr] = countsLine.trim().split(\"|\");\n\n        const totalFiles = Number.parseInt(fileCountStr, 10) || 0;\n        const totalDirs = Number.parseInt(dirCountStr, 10) || 0;\n        const lines = listing.split(\"\\n\").filter((l) => l.length > 0);\n\n        return {\n          listing,\n          summary: {\n            totalItems: lines.length,\n            totalFiles,\n            totalDirs,\n            searchPath,\n            depth,\n          },\n        };\n      },\n    }),\n    Bash: tool({\n      description:\n        \"Executes a bash command inside the workspace. CWD persists between commands within a session. Use waitUntil:0 for background processes (dev servers).\",\n      inputSchema: z.object({\n        command: z.string().describe(\"The shell command to execute\"),\n        waitUntil: z\n          .number()\n          .optional()\n          .describe(\n            \"Max ms to wait for completion (default: 30000). Use 0 to run in background and return immediately.\"\n          ),\n      }),\n      outputSchema: z.object({\n        pid: z\n          .number()\n          .describe(\n            \"System PID (0 for foreground, >0 for background - use to kill)\"\n          ),\n        output: z\n          .string()\n          .describe(\"Command stdout+stderr combined (empty for background)\"),\n        exitCode: z.number().describe(\"Exit code (-1 for background/running)\"),\n        status: z\n          .enum([\"running\", \"completed\", \"failed\"])\n          .describe(\"Process status\"),\n        cwd: z.string().describe(\"Current working directory after command\"),\n        outputFile: z\n          .string()\n          .describe(\"Path to output log (for background processes)\"),\n      }),\n      execute: async ({ command, waitUntil }) => {\n        const { createProcessManager } = await import(\n          \"../sandbox/process-manager\"\n        );\n\n        const processManager = createProcessManager({\n          sandbox: context.sandbox,\n          sessionId: context.input.sessionId,\n          generateId: () => crypto.randomUUID(),\n        });\n\n        await processManager.init();\n        return processManager.run({ command, waitUntil });\n      },\n    }),\n    ExecuteMCPTool: tool({\n      description:\n        \"Execute a tool from an MCP server. Use this to call custom tools provided by the application.\",\n      inputSchema: z.object({\n        server: z.string().describe(\"MCP server name\"),\n        tool: z.string().describe(\"Tool name to execute\"),\n        input: z.unknown().describe(\"Tool input matching the tool's schema\"),\n      }),\n      outputSchema: z.object({\n        result: z.unknown().describe(\"Tool execution result\"),\n        truncated: z\n          .boolean()\n          .optional()\n          .describe(\"Whether the result was truncated due to size\"),\n        totalChars: z\n          .number()\n          .optional()\n          .describe(\"Total characters in full output (only if truncated)\"),\n        returnedChars: z\n          .number()\n          .optional()\n          .describe(\"Characters returned in this response (only if truncated)\"),\n        fullOutputPath: z\n          .string()\n          .optional()\n          .describe(\"Path to full output file if truncated\"),\n      }),\n      execute: async ({ server, tool: toolName, input }) => {\n        const serverConfig = context.mcp?.find((s) => s.name === server);\n        if (!serverConfig) {\n          throw new Error(`Unknown MCP server: ${server}`);\n        }\n\n        const parsedInput =\n          typeof input === \"string\" ? JSON.parse(input) : input;\n\n        const timestamp = Date.now();\n        const outputDir = `.agent/mcp/${server}/.outputs`;\n        const outputFile = `${outputDir}/${timestamp}-${toolName}.txt`;\n\n        const [res] = await Promise.all([\n          fetch(serverConfig.url, {\n            method: \"POST\",\n            headers: { \"Content-Type\": \"application/json\" },\n            body: JSON.stringify({\n              server,\n              tool: toolName,\n              input: parsedInput,\n              sessionId: context.input.sessionId,\n              messageId: context.event.assistantMessageId,\n              hookToken: context.event.hookToken,\n            }),\n          }),\n          context.sandbox.exec({ command: \"mkdir\", args: [\"-p\", outputDir] }),\n        ]);\n\n        if (!res.ok) {\n          const error = await res.text();\n          throw new Error(`MCP tool call failed: ${error}`);\n        }\n\n        const json = (await res.json()) as {\n          result?: unknown;\n          error?: { code: string; message: string };\n        };\n\n        if (json.error) {\n          throw new Error(`MCP tool call failed: ${json.error.message}`);\n        }\n\n        if (json.result === undefined) {\n          throw new Error(\"MCP tool call failed: No result in response\");\n        }\n\n        const result = json.result;\n        const MAX_OUTPUT_CHARS = 24_000;\n        const resultStr = JSON.stringify(result, null, 2);\n\n        if (resultStr.length <= MAX_OUTPUT_CHARS) {\n          return { result };\n        }\n\n        await context.sandbox.writeFiles({\n          files: [{ path: `${timestamp}-${toolName}.txt`, content: resultStr }],\n          destPath: outputDir,\n        });\n\n        const truncatedResult =\n          resultStr.slice(0, MAX_OUTPUT_CHARS) +\n          `\\n\\n[Output truncated at ~6k tokens. Full output saved to: ${outputFile}]`;\n\n        return {\n          result: truncatedResult,\n          truncated: true,\n          totalChars: resultStr.length,\n          returnedChars: MAX_OUTPUT_CHARS,\n          fullOutputPath: outputFile,\n        };\n      },\n    }),\n  } satisfies ToolSet;\n}\n\nexport type AgentTools = ReturnType<typeof getTools>;\nexport type AgentToolName = keyof AgentTools;\n", "import type { UIMessage } from \"ai\";\nimport type { Message, Part } from \"../storage\";\n\nexport function assembleUIMessages(opts: {\n  messages: Message[];\n  parts: Part[];\n}): UIMessage[] {\n  const partsByMessage = new Map<string, Part[]>();\n  for (const part of opts.parts) {\n    const existing = partsByMessage.get(part.messageId) ?? [];\n    existing.push(part);\n    partsByMessage.set(part.messageId, existing);\n  }\n\n  return opts.messages.map((m) => {\n    const messageParts = partsByMessage.get(m.id) ?? [];\n    messageParts.sort((a, b) => a.index - b.index);\n    return {\n      id: m.id,\n      role: m.role,\n      parts: messageParts.map((p) => p.part),\n    } satisfies UIMessage;\n  });\n}\n", "import { z } from \"zod\";\n\nexport type McpToolDef = {\n  description: string;\n  inputSchema: z.ZodType;\n  outputSchema?: z.ZodType;\n};\n\nexport type McpServerDef<\n  TTools extends Record<string, McpToolDef> = Record<string, McpToolDef>,\n  THeaders extends z.ZodType = z.ZodType,\n> = {\n  url: string;\n  description?: string;\n  headersSchema?: THeaders;\n  tools: TTools;\n};\n\nexport type McpConfig = Record<string, McpServerDef>;\n\nexport function serializeMcpConfig(mcp: McpConfig): McpServerSchema[] {\n  return Object.entries(mcp).map(([name, server]) => ({\n    name,\n    url: server.url,\n    description: server.description,\n    tools: Object.entries(server.tools).map(([toolName, tool]) => ({\n      name: toolName,\n      description: tool.description,\n      inputSchema: z.toJSONSchema(tool.inputSchema),\n      outputSchema: tool.outputSchema\n        ? z.toJSONSchema(tool.outputSchema)\n        : undefined,\n    })),\n  }));\n}\n\nexport type McpToolSchema = {\n  name: string;\n  description: string;\n  inputSchema: unknown;\n  outputSchema?: unknown;\n};\n\nexport type McpServerSchema = {\n  name: string;\n  url: string;\n  description?: string;\n  tools: McpToolSchema[];\n};\n\ntype ExtractHeaders<T> =\n  T extends McpServerDef<infer _TTools, infer THeaders>\n    ? THeaders extends z.ZodType\n      ? z.infer<THeaders>\n      : Record<string, string>\n    : Record<string, string>;\n\ntype HasRequiredHeaders<T extends McpServerDef> = T extends {\n  headersSchema: z.ZodType;\n}\n  ? true\n  : false;\n\ntype ServerSendOptions<T extends McpServerDef> =\n  HasRequiredHeaders<T> extends true\n    ?\n        | { headers: ExtractHeaders<T>; disabled?: false }\n        | { disabled: true; headers?: ExtractHeaders<T> }\n    : { headers?: ExtractHeaders<T>; disabled?: boolean };\n\nexport type McpSendOptions<TMcp extends McpConfig> = {\n  [K in keyof TMcp]?: ServerSendOptions<TMcp[K]>;\n};\n\nexport type McpToolCallRequest = {\n  server: string;\n  tool: string;\n  input: unknown;\n  messageId: string;\n  mcpToken: string;\n};\n\nexport type McpToolCallSuccessResponse<T = unknown> = {\n  result: T;\n};\n\nexport type McpToolCallErrorResponse = {\n  error: { code: string; message: string };\n};\n\nexport type McpToolCallResponse<T = unknown> =\n  | McpToolCallSuccessResponse<T>\n  | McpToolCallErrorResponse;\n\ntype ExtractInput<T extends McpToolDef> = T extends { inputSchema: infer S }\n  ? S extends z.ZodType\n    ? z.infer<S>\n    : unknown\n  : unknown;\n\ntype HandlerForTool<\n  TServer extends McpServerDef,\n  TTool extends McpToolDef,\n> = (opts: {\n  input: ExtractInput<TTool>;\n  headers: ExtractHeaders<TServer>;\n}) => Promise<unknown>;\n\ntype BuildHandlers<TMcp extends McpConfig> = {\n  [K in keyof TMcp & string]: {\n    [T in keyof TMcp[K][\"tools\"] & string]: {\n      key: `${K}/${T}`;\n      handler: HandlerForTool<TMcp[K], TMcp[K][\"tools\"][T]>;\n    };\n  }[keyof TMcp[K][\"tools\"] & string];\n}[keyof TMcp & string];\n\ntype UnionToIntersection<U> = (\n  U extends unknown\n    ? (k: U) => void\n    : never\n) extends (k: infer I) => void\n  ? I\n  : never;\n\ntype HandlersFromUnion<U> = UnionToIntersection<\n  U extends { key: infer K; handler: infer H }\n    ? K extends string\n      ? { [P in K]: H }\n      : never\n    : never\n>;\n\nexport type McpToolCallHandlers<TMcp extends McpConfig> = HandlersFromUnion<\n  BuildHandlers<TMcp>\n>;\n\nexport function mcp<\n  THeaders extends z.ZodType,\n  TTools extends Record<string, McpToolDef>,\n>(opts: {\n  url: string;\n  description?: string;\n  headersSchema?: THeaders;\n  tools: TTools;\n}): McpServerDef<TTools, THeaders> {\n  return opts;\n}\n", "import type { UIMessage } from \"ai\";\nimport type { AgentOptions, SendInput } from \"../client\";\nimport type { Message, Part } from \"../storage\";\n\nexport function toMessageAndParts({\n  sessionId,\n  id,\n  input,\n  generateId,\n  defaultRole,\n}: {\n  sessionId: string;\n  id?: string;\n  input: SendInput;\n  generateId: NonNullable<AgentOptions[\"generateId\"]>;\n  defaultRole: UIMessage[\"role\"];\n}): { message: Message; parts: Part[] } {\n  const messageId = id ?? generateId(\"message\");\n  if (typeof input === \"string\") {\n    return {\n      message: {\n        id: messageId,\n        sessionId,\n        role: defaultRole,\n        createdAt: Date.now(),\n        completedAt: Date.now(),\n        mcpContext: null,\n      },\n      parts: [\n        {\n          sessionId,\n          messageId,\n          id: generateId(\"part\"),\n          index: 0,\n          part: { type: \"text\", text: input },\n        },\n      ],\n    };\n  }\n  return {\n    message: {\n      sessionId,\n      id: messageId,\n      createdAt: Date.now(),\n      completedAt: Date.now(),\n      mcpContext: null,\n      role: input.role ?? defaultRole,\n    },\n    parts: input.parts.map((part, index) => ({\n      sessionId,\n      messageId,\n      id: generateId(\"part\"),\n      index,\n      part,\n    })),\n  };\n}\n"],
  "mappings": "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",
  "names": ["opts", "startLocalStorage", "getVercelStorageConfig", "import_zod", "exitCode", "stderr", "errore", "init_local", "errore", "init_vercel", "sandbox", "VercelSandboxSDK", "getSandbox", "init_client", "init_local", "init_vercel", "init_client", "errore", "import_ulid", "import_workflow", "import_ai", "import_ulid", "path", "createProcessManager", "mcp", "getStorage", "getSandbox", "result", "import_zod", "mcp", "tool", "equal", "session", "tool"]
}
