experimental-agent 0.7.1 → 0.8.0-alpha.2
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/chunk-DO4HKPHR.mjs +199 -0
- package/dist/chunk-EH4L2CPW.mjs +3779 -0
- package/dist/chunk-LMYEJDMD.mjs +123 -0
- package/dist/{client-D4AiVqFt.d.ts → client-DIanpEQ_.d.ts} +8 -53
- package/dist/{client-DCtmZCY8.d.mts → client-PuttJd6O.d.mts} +8 -53
- package/dist/framework-DH4Z1NLV.d.ts +61 -0
- package/dist/framework-c9fvEy0o.d.mts +61 -0
- package/dist/framework-runtime.d.mts +37 -0
- package/dist/framework-runtime.d.ts +37 -0
- package/dist/framework-runtime.js +5431 -0
- package/dist/framework-runtime.mjs +107 -0
- package/dist/framework.d.mts +11 -0
- package/dist/framework.d.ts +11 -0
- package/dist/framework.js +235 -0
- package/dist/framework.mjs +8 -0
- package/dist/index.d.mts +30 -7
- package/dist/index.d.ts +30 -7
- package/dist/index.js +50 -4
- package/dist/index.mjs +159 -3863
- package/dist/next/agents.d.mts +43 -0
- package/dist/next/agents.d.ts +43 -0
- package/dist/next/agents.js +364 -0
- package/dist/next/agents.mjs +137 -0
- package/dist/next/loader.js +18 -2
- package/dist/next/loader.mjs +1 -1
- package/dist/next.js +18 -2
- package/dist/next.mjs +1 -1
- package/dist/react.d.mts +2 -1
- package/dist/react.d.ts +2 -1
- package/dist/resolve-BZ26gcyj.d.ts +34 -0
- package/dist/resolve-Cma6YFgx.d.mts +34 -0
- package/dist/sandbox.mjs +2 -2
- package/dist/types-B3lbsOa7.d.mts +53 -0
- package/dist/types-B3lbsOa7.d.ts +53 -0
- package/package.json +29 -13
- package/dist/chunk-ZIXWGYE5.mjs +0 -107
package/dist/index.mjs
CHANGED
|
@@ -1,3892 +1,152 @@
|
|
|
1
1
|
import {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
} from "./chunk-
|
|
13
|
-
import {
|
|
14
|
-
|
|
15
|
-
} from "./chunk-
|
|
16
|
-
import "./chunk-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
import
|
|
20
|
-
|
|
21
|
-
// src/sandbox/handle.ts
|
|
22
|
-
import { ulid } from "ulid";
|
|
23
|
-
|
|
24
|
-
// src/agent/is-vercel.ts
|
|
25
|
-
var isVercel = async () => {
|
|
26
|
-
try {
|
|
27
|
-
const { getVercelOidcToken } = await import("@vercel/oidc");
|
|
28
|
-
return Boolean(await getVercelOidcToken());
|
|
29
|
-
} catch {
|
|
30
|
-
return false;
|
|
31
|
-
}
|
|
32
|
-
};
|
|
33
|
-
|
|
34
|
-
// src/errors.ts
|
|
35
|
-
import * as errore from "errore";
|
|
36
|
-
var SessionNotFoundError = class extends errore.createTaggedError({
|
|
37
|
-
name: "SessionNotFoundError",
|
|
38
|
-
message: "Session $id not found"
|
|
39
|
-
}) {
|
|
40
|
-
};
|
|
41
|
-
var SessionError = class extends errore.createTaggedError({
|
|
42
|
-
name: "SessionError",
|
|
43
|
-
message: "Session $id failed: $reason"
|
|
44
|
-
}) {
|
|
45
|
-
};
|
|
46
|
-
var SandboxNotFoundError = class extends errore.createTaggedError({
|
|
47
|
-
name: "SandboxNotFoundError",
|
|
48
|
-
message: "Sandbox $id not found"
|
|
49
|
-
}) {
|
|
50
|
-
};
|
|
51
|
-
var StorageError = class extends errore.createTaggedError({
|
|
52
|
-
name: "StorageError",
|
|
53
|
-
message: "$reason"
|
|
54
|
-
}) {
|
|
55
|
-
};
|
|
56
|
-
var StorageConflictError = class extends errore.createTaggedError({
|
|
57
|
-
name: "StorageConflictError",
|
|
58
|
-
message: "$reason"
|
|
59
|
-
}) {
|
|
60
|
-
};
|
|
61
|
-
var SandboxError = class extends errore.createTaggedError({
|
|
62
|
-
name: "SandboxError",
|
|
63
|
-
message: "$reason"
|
|
64
|
-
}) {
|
|
65
|
-
};
|
|
66
|
-
var MessageNotFoundError = class extends errore.createTaggedError({
|
|
67
|
-
name: "MessageNotFoundError",
|
|
68
|
-
message: "Message $id not found"
|
|
69
|
-
}) {
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
// src/sandbox/setup.ts
|
|
73
|
-
var POLL_INTERVAL_MS = 50;
|
|
74
|
-
var POLL_TIMEOUT_MS = 5 * 60 * 1e3;
|
|
75
|
-
function isSingleBinding(sandbox) {
|
|
76
|
-
return !Array.isArray(sandbox);
|
|
77
|
-
}
|
|
78
|
-
async function getDefaultBinding() {
|
|
79
|
-
if (await isVercel()) {
|
|
80
|
-
const { vercelSandbox } = await import("./vercel-7TYF67KD.mjs");
|
|
81
|
-
return vercelSandbox();
|
|
82
|
-
}
|
|
83
|
-
if (await isDockerAvailable()) {
|
|
84
|
-
const { dockerSandbox } = await import("./docker-OBUQX5ZI.mjs");
|
|
85
|
-
return dockerSandbox();
|
|
86
|
-
}
|
|
87
|
-
const { localSandbox } = await import("./local-4C3V4EFS.mjs");
|
|
88
|
-
return localSandbox();
|
|
89
|
-
}
|
|
90
|
-
async function isDockerAvailable() {
|
|
91
|
-
try {
|
|
92
|
-
const { execSync } = await import("child_process");
|
|
93
|
-
execSync("docker sandbox --help", { stdio: "ignore", timeout: 5e3 });
|
|
94
|
-
return true;
|
|
95
|
-
} catch {
|
|
96
|
-
return false;
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
async function resolveBinding(sandbox, type) {
|
|
100
|
-
if (!sandbox) {
|
|
101
|
-
const binding = await getDefaultBinding();
|
|
102
|
-
return { binding, bindingName: binding.type };
|
|
103
|
-
}
|
|
104
|
-
if (isSingleBinding(sandbox)) {
|
|
105
|
-
return { binding: sandbox, bindingName: sandbox.type };
|
|
106
|
-
}
|
|
107
|
-
if (type) {
|
|
108
|
-
const match = sandbox.find((b) => b.type === type);
|
|
109
|
-
if (!match) {
|
|
110
|
-
throw new SandboxError({
|
|
111
|
-
reason: `Unknown binding type: "${type}". Available: ${sandbox.map((b) => b.type).join(", ")}`
|
|
112
|
-
});
|
|
113
|
-
}
|
|
114
|
-
return { binding: match, bindingName: match.type };
|
|
115
|
-
}
|
|
116
|
-
if (sandbox.length === 0) {
|
|
117
|
-
throw new SandboxError({ reason: "No sandbox bindings configured" });
|
|
118
|
-
}
|
|
119
|
-
return { binding: sandbox[0], bindingName: sandbox[0].type };
|
|
120
|
-
}
|
|
121
|
-
async function resolveSandboxId(ctx) {
|
|
122
|
-
if (ctx.sandboxId) {
|
|
123
|
-
return ctx.sandboxId;
|
|
124
|
-
}
|
|
125
|
-
if (!ctx._sessionId) {
|
|
126
|
-
throw new Error(
|
|
127
|
-
"SandboxContext has no sandboxId and no sessionId to resolve from"
|
|
128
|
-
);
|
|
129
|
-
}
|
|
130
|
-
const record = await ctx.agent.storage.session.get(ctx._sessionId);
|
|
131
|
-
ctx.sandboxId = record?.sandboxId ?? ctx._sessionId;
|
|
132
|
-
return ctx.sandboxId;
|
|
133
|
-
}
|
|
134
|
-
async function setup(ctx, opts) {
|
|
135
|
-
"use step";
|
|
136
|
-
const sandboxId = await resolveSandboxId(ctx);
|
|
137
|
-
if (ctx._pendingSetup) {
|
|
138
|
-
return { done: ctx._pendingSetup };
|
|
139
|
-
}
|
|
140
|
-
const agent2 = ctx.agent;
|
|
141
|
-
const log2 = createLogger({
|
|
142
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
143
|
-
subsystem: "sandbox:setup",
|
|
144
|
-
context: { sandboxId }
|
|
145
|
-
});
|
|
146
|
-
const doneResolveBinding = log2.time("resolve binding", {
|
|
147
|
-
requestedType: opts?.type
|
|
148
|
-
});
|
|
149
|
-
const { binding, bindingName } = await resolveBinding(
|
|
150
|
-
agent2.options.sandbox,
|
|
151
|
-
opts?.type
|
|
152
|
-
);
|
|
153
|
-
doneResolveBinding();
|
|
154
|
-
if (opts?.cwd) {
|
|
155
|
-
ctx._cwd = opts.cwd;
|
|
156
|
-
}
|
|
157
|
-
const setupLog = log2.withContext({
|
|
158
|
-
binding: bindingName,
|
|
159
|
-
version: opts?.version ?? binding.defaults?.version
|
|
160
|
-
});
|
|
161
|
-
let resolveBlockingPromise = (value) => value;
|
|
162
|
-
let rejectBlockingPromise = (_reason) => {
|
|
163
|
-
};
|
|
164
|
-
const blockingPromise = new Promise((resolve2, reject) => {
|
|
165
|
-
resolveBlockingPromise = resolve2;
|
|
166
|
-
rejectBlockingPromise = reject;
|
|
167
|
-
});
|
|
168
|
-
ctx._pendingSetup = (async () => {
|
|
169
|
-
let sleepOldInstance;
|
|
170
|
-
const donePendingSetup = setupLog.time("pending setup");
|
|
171
|
-
try {
|
|
172
|
-
const doneStorageGet = setupLog.time("storage.sandbox.get");
|
|
173
|
-
const record = await agent2.storage.sandbox.get(sandboxId);
|
|
174
|
-
doneStorageGet();
|
|
175
|
-
const bindingDefaults = binding.defaults ?? {};
|
|
176
|
-
const effectiveVersion = opts?.version ?? bindingDefaults.version ?? null;
|
|
177
|
-
const effectiveRun = opts?.run ?? bindingDefaults.run;
|
|
178
|
-
const effectiveNetworkPolicy = opts?.networkPolicy ?? bindingDefaults.networkPolicy ?? null;
|
|
179
|
-
const effectiveCwd = ctx._cwd || opts?.cwd || bindingDefaults.cwd || null;
|
|
180
|
-
if (effectiveCwd) {
|
|
181
|
-
ctx._cwd = effectiveCwd;
|
|
182
|
-
}
|
|
183
|
-
const effectiveConfig = {
|
|
184
|
-
...bindingDefaults.config,
|
|
185
|
-
// biome-ignore lint/suspicious/noExplicitAny: opts is a union; config exists on binding-specific branches
|
|
186
|
-
...opts?.config
|
|
187
|
-
};
|
|
188
|
-
if (record) {
|
|
189
|
-
const versionMatches = !effectiveVersion || record.setup?.version === effectiveVersion;
|
|
190
|
-
if (versionMatches && record.setup?.completedAt) {
|
|
191
|
-
resolveBlockingPromise();
|
|
192
|
-
const doneConnectExisting = setupLog.time(
|
|
193
|
-
"connect existing sandbox",
|
|
194
|
-
{
|
|
195
|
-
setupState: "completed"
|
|
196
|
-
}
|
|
197
|
-
);
|
|
198
|
-
const instance = await binding.connect({
|
|
199
|
-
agent: agent2,
|
|
200
|
-
metadata: record.setup.metadata,
|
|
201
|
-
cwd: record.setup.cwd,
|
|
202
|
-
signal: opts?.signal,
|
|
203
|
-
log: setupLog
|
|
204
|
-
});
|
|
205
|
-
doneConnectExisting();
|
|
206
|
-
ctx._cwd = instance.cwd;
|
|
207
|
-
return instance;
|
|
208
|
-
}
|
|
209
|
-
if (versionMatches && !record.setup?.completedAt) {
|
|
210
|
-
resolveBlockingPromise();
|
|
211
|
-
const donePollUntilReady = setupLog.time("poll until ready");
|
|
212
|
-
await pollUntilReady(ctx, opts?.signal);
|
|
213
|
-
donePollUntilReady();
|
|
214
|
-
const doneStorageGetPostPoll = setupLog.time(
|
|
215
|
-
"storage.sandbox.get (post-poll)"
|
|
216
|
-
);
|
|
217
|
-
const ready = await agent2.storage.sandbox.get(sandboxId);
|
|
218
|
-
doneStorageGetPostPoll();
|
|
219
|
-
if (!ready) {
|
|
220
|
-
throw new SandboxNotFoundError({ id: sandboxId });
|
|
221
|
-
}
|
|
222
|
-
const doneConnectExisting = setupLog.time(
|
|
223
|
-
"connect existing sandbox",
|
|
224
|
-
{
|
|
225
|
-
setupState: "post-poll"
|
|
226
|
-
}
|
|
227
|
-
);
|
|
228
|
-
const instance = await binding.connect({
|
|
229
|
-
agent: agent2,
|
|
230
|
-
metadata: ready.setup.metadata,
|
|
231
|
-
cwd: ready.setup.cwd,
|
|
232
|
-
signal: opts?.signal,
|
|
233
|
-
log: setupLog
|
|
234
|
-
});
|
|
235
|
-
doneConnectExisting();
|
|
236
|
-
ctx._cwd = instance.cwd;
|
|
237
|
-
return instance;
|
|
238
|
-
}
|
|
239
|
-
if (record.setup.metadata) {
|
|
240
|
-
const oldInstanceMetadata = record.setup.metadata;
|
|
241
|
-
sleepOldInstance = (async () => {
|
|
242
|
-
const doneStopOldInstance = setupLog.time(
|
|
243
|
-
"stop old sandbox instance"
|
|
244
|
-
);
|
|
245
|
-
try {
|
|
246
|
-
const old = await binding.connect({
|
|
247
|
-
agent: agent2,
|
|
248
|
-
metadata: oldInstanceMetadata,
|
|
249
|
-
log: setupLog
|
|
250
|
-
});
|
|
251
|
-
await old.stop();
|
|
252
|
-
} catch {
|
|
253
|
-
doneStopOldInstance();
|
|
254
|
-
return;
|
|
255
|
-
}
|
|
256
|
-
doneStopOldInstance();
|
|
257
|
-
})();
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
const now = Date.now();
|
|
261
|
-
const setup_ = {
|
|
262
|
-
binding: bindingName,
|
|
263
|
-
version: effectiveVersion ?? record?.setup?.version ?? null,
|
|
264
|
-
networkPolicy: effectiveNetworkPolicy ?? record?.setup?.networkPolicy ?? null,
|
|
265
|
-
completedAt: null,
|
|
266
|
-
metadata: null,
|
|
267
|
-
cwd: null
|
|
268
|
-
};
|
|
269
|
-
const newRecord = {
|
|
270
|
-
id: sandboxId,
|
|
271
|
-
setup: setup_,
|
|
272
|
-
createdAt: record?.createdAt ?? now,
|
|
273
|
-
lastActiveAt: now
|
|
274
|
-
};
|
|
275
|
-
const doneStorageSetPending = setupLog.time(
|
|
276
|
-
"storage.sandbox.set (setup pending)"
|
|
277
|
-
);
|
|
278
|
-
await agent2.storage.sandbox.set(sandboxId, newRecord);
|
|
279
|
-
doneStorageSetPending();
|
|
280
|
-
resolveBlockingPromise();
|
|
281
|
-
const doneBindingCreate = setupLog.time("binding.create");
|
|
282
|
-
const result = await binding.create({
|
|
283
|
-
cwd: ctx._cwd ?? ".",
|
|
284
|
-
agent: agent2,
|
|
285
|
-
setup: {
|
|
286
|
-
...setup_,
|
|
287
|
-
run: effectiveRun,
|
|
288
|
-
config: effectiveConfig
|
|
289
|
-
},
|
|
290
|
-
record: newRecord,
|
|
291
|
-
signal: opts?.signal,
|
|
292
|
-
log: setupLog
|
|
293
|
-
});
|
|
294
|
-
doneBindingCreate();
|
|
295
|
-
setup_.completedAt = Date.now();
|
|
296
|
-
setup_.metadata = result.metadata;
|
|
297
|
-
setup_.cwd = result.instance.cwd;
|
|
298
|
-
const doneStorageUpdateComplete = setupLog.time(
|
|
299
|
-
"storage.sandbox.update (setup complete)"
|
|
300
|
-
);
|
|
301
|
-
await agent2.storage.sandbox.update(sandboxId, {
|
|
302
|
-
setup: setup_,
|
|
303
|
-
lastActiveAt: Date.now()
|
|
304
|
-
});
|
|
305
|
-
doneStorageUpdateComplete();
|
|
306
|
-
ctx._cwd = result.instance.cwd;
|
|
307
|
-
return result.instance;
|
|
308
|
-
} catch (err) {
|
|
309
|
-
ctx._pendingSetup = void 0;
|
|
310
|
-
rejectBlockingPromise(err);
|
|
311
|
-
throw err;
|
|
312
|
-
} finally {
|
|
313
|
-
const doneFinalize = setupLog.time("finalize setup");
|
|
314
|
-
await sleepOldInstance;
|
|
315
|
-
doneFinalize();
|
|
316
|
-
donePendingSetup();
|
|
317
|
-
}
|
|
318
|
-
})();
|
|
319
|
-
const doneWaitForUnblock = setupLog.time("wait for setup unblock");
|
|
320
|
-
await blockingPromise;
|
|
321
|
-
doneWaitForUnblock();
|
|
322
|
-
return { done: ctx._pendingSetup };
|
|
323
|
-
}
|
|
324
|
-
async function resolve(ctx, signal) {
|
|
325
|
-
const sandboxId = await resolveSandboxId(ctx);
|
|
326
|
-
const agent2 = ctx.agent;
|
|
327
|
-
const log2 = createLogger({
|
|
328
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
329
|
-
subsystem: "sandbox:resolve",
|
|
330
|
-
context: { sandboxId }
|
|
331
|
-
});
|
|
332
|
-
if (ctx._pendingSetup) {
|
|
333
|
-
return await ctx._pendingSetup;
|
|
334
|
-
}
|
|
335
|
-
let record = await agent2.storage.sandbox.get(sandboxId);
|
|
336
|
-
if (!record) {
|
|
337
|
-
const { done } = await setup(ctx, { signal });
|
|
338
|
-
return await done;
|
|
339
|
-
}
|
|
340
|
-
if (!record.setup?.completedAt) {
|
|
341
|
-
await pollUntilReady(ctx, signal);
|
|
342
|
-
record = await agent2.storage.sandbox.get(sandboxId);
|
|
343
|
-
}
|
|
344
|
-
if (!record) {
|
|
345
|
-
throw new SandboxNotFoundError({ id: sandboxId });
|
|
346
|
-
}
|
|
347
|
-
const { binding } = await resolveBinding(
|
|
348
|
-
ctx.agent.options.sandbox,
|
|
349
|
-
record.setup.binding
|
|
350
|
-
);
|
|
351
|
-
agent2.storage.sandbox.update(sandboxId, { lastActiveAt: Date.now() }).catch((error) => {
|
|
352
|
-
log2.error("Failed to update sandbox lastActiveAt", { error });
|
|
353
|
-
});
|
|
354
|
-
const instance = await binding.connect({
|
|
355
|
-
agent: agent2,
|
|
356
|
-
metadata: record.setup.metadata,
|
|
357
|
-
cwd: record.setup.cwd,
|
|
358
|
-
signal,
|
|
359
|
-
log: log2
|
|
360
|
-
});
|
|
361
|
-
ctx._cwd = instance.cwd;
|
|
362
|
-
return instance;
|
|
363
|
-
}
|
|
364
|
-
async function pollUntilReady(ctx, signal) {
|
|
365
|
-
const id = ctx.sandboxId;
|
|
366
|
-
const deadline = Date.now() + POLL_TIMEOUT_MS;
|
|
367
|
-
while (Date.now() < deadline) {
|
|
368
|
-
signal?.throwIfAborted();
|
|
369
|
-
const record = await ctx.agent.storage.sandbox.get(id);
|
|
370
|
-
if (record?.setup?.completedAt) {
|
|
371
|
-
return;
|
|
372
|
-
}
|
|
373
|
-
await new Promise((r) => setTimeout(r, POLL_INTERVAL_MS));
|
|
374
|
-
}
|
|
375
|
-
throw new SandboxError({
|
|
376
|
-
reason: `Timed out waiting for sandbox "${id}" setup to complete`
|
|
377
|
-
});
|
|
378
|
-
}
|
|
379
|
-
|
|
380
|
-
// src/sandbox/exec.ts
|
|
381
|
-
async function exec(sandboxContext, ...opts) {
|
|
382
|
-
const sbx = await resolve(sandboxContext);
|
|
383
|
-
return await sbx.exec(...opts);
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
// src/sandbox/get-domain.ts
|
|
387
|
-
async function getDomain(sandboxContext, ...opts) {
|
|
388
|
-
"use step";
|
|
389
|
-
const sbx = await resolve(sandboxContext);
|
|
390
|
-
return await sbx.getDomain(...opts);
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
// src/sandbox/get-status.ts
|
|
394
|
-
async function getStatus(sandboxContext, ...opts) {
|
|
395
|
-
"use step";
|
|
396
|
-
const sbx = await resolve(sandboxContext);
|
|
397
|
-
return await sbx.getStatus(...opts);
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
// src/sandbox/kill.ts
|
|
401
|
-
async function kill(sandboxContext, ...opts) {
|
|
402
|
-
"use step";
|
|
403
|
-
const sbx = await resolve(sandboxContext);
|
|
404
|
-
return await sbx.kill(...opts);
|
|
405
|
-
}
|
|
406
|
-
|
|
407
|
-
// src/sandbox/read-file.ts
|
|
408
|
-
async function readFile(sandboxContext, ...opts) {
|
|
409
|
-
"use step";
|
|
410
|
-
const sbx = await resolve(sandboxContext);
|
|
411
|
-
return await sbx.readFile(...opts);
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
// src/sandbox/snapshot.ts
|
|
415
|
-
async function snapshot(sandboxContext, ...opts) {
|
|
416
|
-
"use step";
|
|
417
|
-
const sbx = await resolve(sandboxContext);
|
|
418
|
-
return await sbx.snapshot(...opts);
|
|
419
|
-
}
|
|
420
|
-
|
|
421
|
-
// src/sandbox/start.ts
|
|
422
|
-
async function start(sandboxContext, ...opts) {
|
|
423
|
-
"use step";
|
|
424
|
-
const sbx = await resolve(sandboxContext);
|
|
425
|
-
return await sbx.start(...opts);
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
// src/sandbox/stop.ts
|
|
429
|
-
async function stop(sandboxContext, ...opts) {
|
|
430
|
-
"use step";
|
|
431
|
-
const sbx = await resolve(sandboxContext);
|
|
432
|
-
return await sbx.stop(...opts);
|
|
433
|
-
}
|
|
434
|
-
|
|
435
|
-
// src/sandbox/update-network-policy.ts
|
|
436
|
-
async function updateNetworkPolicy(sandboxContext, ...opts) {
|
|
437
|
-
"use step";
|
|
438
|
-
const sbx = await resolve(sandboxContext);
|
|
439
|
-
return await sbx.updateNetworkPolicy(...opts);
|
|
440
|
-
}
|
|
441
|
-
|
|
442
|
-
// src/sandbox/write-files.ts
|
|
443
|
-
async function writeFiles(sandboxContext, ...opts) {
|
|
444
|
-
"use step";
|
|
445
|
-
const sbx = await resolve(sandboxContext);
|
|
446
|
-
return await sbx.writeFiles(...opts);
|
|
447
|
-
}
|
|
448
|
-
|
|
449
|
-
// src/sandbox/handle.ts
|
|
450
|
-
function sandboxHandle(agent2, id) {
|
|
451
|
-
const sandboxId = typeof id === "string" ? id : id?.id ?? `sandbox_${ulid()}`;
|
|
452
|
-
const contexts = getSandboxContexts(agent2);
|
|
453
|
-
let sandboxContext = contexts.get(sandboxId);
|
|
454
|
-
if (!sandboxContext) {
|
|
455
|
-
sandboxContext = { agent: agent2, sandboxId };
|
|
456
|
-
contexts.set(sandboxId, sandboxContext);
|
|
457
|
-
}
|
|
458
|
-
return {
|
|
459
|
-
id: sandboxId,
|
|
460
|
-
get cwd() {
|
|
461
|
-
return sandboxContext._cwd ?? ".";
|
|
462
|
-
},
|
|
463
|
-
setup: setup.bind(null, sandboxContext),
|
|
464
|
-
exec: exec.bind(null, sandboxContext),
|
|
465
|
-
readFile: readFile.bind(null, sandboxContext),
|
|
466
|
-
writeFiles: writeFiles.bind(null, sandboxContext),
|
|
467
|
-
getDomain: getDomain.bind(null, sandboxContext),
|
|
468
|
-
updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),
|
|
469
|
-
start: start.bind(null, sandboxContext),
|
|
470
|
-
stop: stop.bind(null, sandboxContext),
|
|
471
|
-
kill: kill.bind(null, sandboxContext),
|
|
472
|
-
snapshot: snapshot.bind(null, sandboxContext),
|
|
473
|
-
getStatus: getStatus.bind(null, sandboxContext)
|
|
474
|
-
};
|
|
475
|
-
}
|
|
476
|
-
function lazySandboxHandle(agent2, sessionId) {
|
|
477
|
-
const sandboxContext = {
|
|
478
|
-
agent: agent2,
|
|
479
|
-
sandboxId: null,
|
|
480
|
-
_sessionId: sessionId
|
|
481
|
-
};
|
|
482
|
-
return {
|
|
483
|
-
setup: setup.bind(null, sandboxContext),
|
|
484
|
-
exec: exec.bind(null, sandboxContext),
|
|
485
|
-
readFile: readFile.bind(null, sandboxContext),
|
|
486
|
-
writeFiles: writeFiles.bind(null, sandboxContext),
|
|
487
|
-
getDomain: getDomain.bind(null, sandboxContext),
|
|
488
|
-
updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),
|
|
489
|
-
start: start.bind(null, sandboxContext),
|
|
490
|
-
stop: stop.bind(null, sandboxContext),
|
|
491
|
-
kill: kill.bind(null, sandboxContext),
|
|
492
|
-
snapshot: snapshot.bind(null, sandboxContext),
|
|
493
|
-
getStatus: getStatus.bind(null, sandboxContext)
|
|
494
|
-
};
|
|
495
|
-
}
|
|
496
|
-
|
|
497
|
-
// src/session/handle.ts
|
|
498
|
-
import { ulid as ulid4 } from "ulid";
|
|
499
|
-
|
|
500
|
-
// src/utils/ui.ts
|
|
501
|
-
function isSubToolPart(part) {
|
|
502
|
-
const p = part.part;
|
|
503
|
-
return typeof p?.toolCallId === "string" && p.toolCallId.startsWith("js_tc_");
|
|
504
|
-
}
|
|
505
|
-
function isStreamingContent(part) {
|
|
506
|
-
return typeof part === "object" && part !== null && "type" in part && (part.type === "text" || part.type === "reasoning");
|
|
507
|
-
}
|
|
508
|
-
function applyInterruptCutoff({
|
|
509
|
-
parts,
|
|
510
|
-
lastPart
|
|
511
|
-
}) {
|
|
512
|
-
return parts.filter((p) => p.index <= lastPart.index).map((p) => {
|
|
513
|
-
if (p.index !== lastPart.index) {
|
|
514
|
-
return p;
|
|
515
|
-
}
|
|
516
|
-
if (isStreamingContent(lastPart.part)) {
|
|
517
|
-
return { ...p, part: lastPart.part };
|
|
518
|
-
}
|
|
519
|
-
return p;
|
|
520
|
-
});
|
|
521
|
-
}
|
|
522
|
-
function assembleUIMessages(opts) {
|
|
523
|
-
let filtered = opts.messages;
|
|
524
|
-
if (opts.until !== void 0) {
|
|
525
|
-
const until = opts.until;
|
|
526
|
-
filtered = filtered.filter((m) => m.createdAt <= until);
|
|
527
|
-
}
|
|
528
|
-
if (!opts.includeQueued) {
|
|
529
|
-
filtered = filtered.filter((m) => m.startedAt !== null);
|
|
530
|
-
}
|
|
531
|
-
filtered = [...filtered].sort((a, b) => a.createdAt - b.createdAt);
|
|
532
|
-
const parts = opts.excludeSubToolParts ? opts.parts.filter((p) => !isSubToolPart(p)) : opts.parts;
|
|
533
|
-
const partsByMessage = /* @__PURE__ */ new Map();
|
|
534
|
-
for (const part of parts) {
|
|
535
|
-
const messageId = String(part.messageId);
|
|
536
|
-
const existing = partsByMessage.get(messageId) ?? [];
|
|
537
|
-
existing.push(part);
|
|
538
|
-
partsByMessage.set(messageId, existing);
|
|
539
|
-
}
|
|
540
|
-
return filtered.map((m) => {
|
|
541
|
-
const messageId = String(m.id);
|
|
542
|
-
let messageParts = partsByMessage.get(messageId) ?? [];
|
|
543
|
-
messageParts.sort((a, b) => a.index - b.index);
|
|
544
|
-
if (m.interruptedLastPart != null) {
|
|
545
|
-
messageParts = applyInterruptCutoff({
|
|
546
|
-
parts: messageParts,
|
|
547
|
-
lastPart: m.interruptedLastPart
|
|
548
|
-
});
|
|
549
|
-
}
|
|
550
|
-
return {
|
|
551
|
-
id: messageId,
|
|
552
|
-
role: m.role,
|
|
553
|
-
parts: messageParts.map((p) => p.part),
|
|
554
|
-
...m.metadata != null ? { metadata: m.metadata } : {}
|
|
555
|
-
};
|
|
556
|
-
}).filter((m) => m.parts.length > 0);
|
|
557
|
-
}
|
|
558
|
-
|
|
559
|
-
// src/utils/usage.ts
|
|
560
|
-
function computeSessionUsage(messages) {
|
|
561
|
-
const byMessageId = {};
|
|
562
|
-
for (const m of messages) {
|
|
563
|
-
byMessageId[m.id] = m.usage?.summary ?? null;
|
|
564
|
-
}
|
|
565
|
-
const summaries = messages.map((m) => m.usage?.summary).filter((s) => s !== void 0);
|
|
566
|
-
const total = {
|
|
567
|
-
model: summaries[0]?.model ?? "unknown",
|
|
568
|
-
inputTokens: summaries.reduce((acc, s) => acc + s.inputTokens, 0),
|
|
569
|
-
outputTokens: summaries.reduce((acc, s) => acc + s.outputTokens, 0),
|
|
570
|
-
totalTokens: summaries.reduce((acc, s) => acc + s.totalTokens, 0),
|
|
571
|
-
cacheReadTokens: summaries.reduce((acc, s) => acc + s.cacheReadTokens, 0),
|
|
572
|
-
cacheWriteTokens: summaries.reduce((acc, s) => acc + s.cacheWriteTokens, 0),
|
|
573
|
-
reasoningTokens: summaries.reduce((acc, s) => acc + s.reasoningTokens, 0),
|
|
574
|
-
stepCount: summaries.reduce((acc, s) => acc + s.stepCount, 0),
|
|
575
|
-
messageCount: summaries.length
|
|
576
|
-
};
|
|
577
|
-
return { total, byMessageId };
|
|
578
|
-
}
|
|
579
|
-
|
|
580
|
-
// src/session/history.ts
|
|
581
|
-
async function history(sessionContext) {
|
|
582
|
-
"use step";
|
|
583
|
-
const [session, messages, parts] = await Promise.all([
|
|
584
|
-
sessionContext.agent.storage.session.get(sessionContext.sessionId),
|
|
585
|
-
sessionContext.agent.storage.message.listBySession(
|
|
586
|
-
sessionContext.sessionId
|
|
587
|
-
),
|
|
588
|
-
sessionContext.agent.storage.part.listBySession(sessionContext.sessionId)
|
|
589
|
-
]);
|
|
590
|
-
const lastAssistant = messages.find(
|
|
591
|
-
(m) => m.id === session?.lastMessageId && m.role === "assistant"
|
|
592
|
-
) ?? messages.filter((m) => m.role === "assistant").at(-1);
|
|
593
|
-
const isStreaming = lastAssistant?.completedAt === null && lastAssistant?.interruptedAt === null;
|
|
594
|
-
const streamingMessageId = isStreaming ? lastAssistant.id : null;
|
|
595
|
-
const visibleUntil = lastAssistant?.createdAt;
|
|
596
|
-
const assembled = assembleUIMessages({
|
|
597
|
-
messages,
|
|
598
|
-
parts,
|
|
599
|
-
includeQueued: visibleUntil !== void 0,
|
|
600
|
-
until: visibleUntil
|
|
601
|
-
});
|
|
602
|
-
const usage2 = computeSessionUsage(messages);
|
|
603
|
-
return { messages: assembled, streamingMessageId, usage: usage2 };
|
|
604
|
-
}
|
|
605
|
-
|
|
606
|
-
// src/session/interrupt.ts
|
|
607
|
-
async function interrupt(sessionContext, opts) {
|
|
608
|
-
"use step";
|
|
609
|
-
const messages = await sessionContext.agent.storage.message.listBySession(
|
|
610
|
-
sessionContext.sessionId
|
|
611
|
-
);
|
|
612
|
-
const now = Date.now();
|
|
613
|
-
let pendingAssistantMessages = messages.filter(
|
|
614
|
-
(m) => m.role === "assistant" && m.completedAt === null && m.interruptedAt === null
|
|
615
|
-
);
|
|
616
|
-
if (pendingAssistantMessages.length === 0) {
|
|
617
|
-
const MAX_RETROACTIVE_MS = 5e3;
|
|
618
|
-
const latest = messages.filter(
|
|
619
|
-
(m) => m.role === "assistant" && m.interruptedAt === null && m.completedAt !== null && now - m.completedAt < MAX_RETROACTIVE_MS
|
|
620
|
-
).sort((a, b) => b.createdAt - a.createdAt).at(0);
|
|
621
|
-
if (latest) {
|
|
622
|
-
pendingAssistantMessages = [latest];
|
|
623
|
-
}
|
|
624
|
-
}
|
|
625
|
-
await Promise.all(
|
|
626
|
-
pendingAssistantMessages.map(
|
|
627
|
-
(m) => sessionContext.agent.storage.message.set(m.id, {
|
|
628
|
-
...m,
|
|
629
|
-
interruptedAt: now,
|
|
630
|
-
interruptedLastPart: opts?.lastPart ?? null
|
|
631
|
-
})
|
|
632
|
-
)
|
|
633
|
-
);
|
|
634
|
-
const messageIds = new Set(pendingAssistantMessages.map((m) => m.id));
|
|
635
|
-
if (messageIds.size === 0) {
|
|
636
|
-
return;
|
|
637
|
-
}
|
|
638
|
-
const allParts = await sessionContext.agent.storage.part.listBySession(
|
|
639
|
-
sessionContext.sessionId
|
|
640
|
-
);
|
|
641
|
-
for (const p of allParts) {
|
|
642
|
-
if (!messageIds.has(p.messageId)) {
|
|
643
|
-
continue;
|
|
644
|
-
}
|
|
645
|
-
if (p.part && "state" in p.part && p.part.state === "approval-requested" && "approval" in p.part) {
|
|
646
|
-
const approval = p.part.approval;
|
|
647
|
-
if (approval?.id) {
|
|
648
|
-
await sessionContext.agent.storage.part.set(p.id, {
|
|
649
|
-
...p,
|
|
650
|
-
part: {
|
|
651
|
-
...p.part,
|
|
652
|
-
state: "output-denied",
|
|
653
|
-
approval: {
|
|
654
|
-
...approval,
|
|
655
|
-
approved: false,
|
|
656
|
-
reason: "interrupted"
|
|
657
|
-
}
|
|
658
|
-
}
|
|
659
|
-
});
|
|
660
|
-
}
|
|
661
|
-
}
|
|
662
|
-
}
|
|
663
|
-
}
|
|
664
|
-
|
|
665
|
-
// src/session/send.ts
|
|
666
|
-
import { all } from "better-all";
|
|
667
|
-
import * as errore2 from "errore";
|
|
668
|
-
import { ulid as ulid3 } from "ulid";
|
|
669
|
-
|
|
670
|
-
// src/session/ai-loop/stream-text.ts
|
|
671
|
-
import {
|
|
672
|
-
convertToModelMessages,
|
|
673
|
-
createUIMessageStream,
|
|
674
|
-
stepCountIs,
|
|
675
|
-
streamText as streamTextAi
|
|
676
|
-
} from "ai";
|
|
677
|
-
|
|
678
|
-
// src/skills/parser.ts
|
|
679
|
-
function parseSkillFrontmatter(content) {
|
|
680
|
-
const trimmed = content.trim();
|
|
681
|
-
if (!trimmed.startsWith("---")) {
|
|
682
|
-
return null;
|
|
683
|
-
}
|
|
684
|
-
const endMarkerIndex = trimmed.indexOf("---", 3);
|
|
685
|
-
if (endMarkerIndex === -1) {
|
|
686
|
-
return null;
|
|
687
|
-
}
|
|
688
|
-
const frontmatterBlock = trimmed.slice(3, endMarkerIndex).trim();
|
|
689
|
-
const parsed = parseSimpleYaml(frontmatterBlock);
|
|
690
|
-
if (!(parsed.name && parsed.description)) {
|
|
691
|
-
return null;
|
|
692
|
-
}
|
|
693
|
-
return {
|
|
694
|
-
name: String(parsed.name),
|
|
695
|
-
description: String(parsed.description)
|
|
696
|
-
};
|
|
697
|
-
}
|
|
698
|
-
function parseSimpleYaml(yaml) {
|
|
699
|
-
const result = {};
|
|
700
|
-
for (const line of yaml.split("\n")) {
|
|
701
|
-
const trimmedLine = line.trim();
|
|
702
|
-
if (!trimmedLine || trimmedLine.startsWith("#")) {
|
|
703
|
-
continue;
|
|
704
|
-
}
|
|
705
|
-
const colonIndex = trimmedLine.indexOf(":");
|
|
706
|
-
if (colonIndex === -1) {
|
|
707
|
-
continue;
|
|
708
|
-
}
|
|
709
|
-
const key = trimmedLine.slice(0, colonIndex).trim();
|
|
710
|
-
let value = trimmedLine.slice(colonIndex + 1).trim();
|
|
711
|
-
if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
|
|
712
|
-
value = value.slice(1, -1);
|
|
713
|
-
}
|
|
714
|
-
if (key) {
|
|
715
|
-
result[key] = value;
|
|
716
|
-
}
|
|
717
|
-
}
|
|
718
|
-
return result;
|
|
719
|
-
}
|
|
720
|
-
function normalizeSkills(skills) {
|
|
721
|
-
if (!skills || skills.length === 0) {
|
|
722
|
-
return [];
|
|
723
|
-
}
|
|
724
|
-
return skills.map((skill) => {
|
|
725
|
-
if (typeof skill !== "object" || skill === null) {
|
|
726
|
-
throw new Error(
|
|
727
|
-
"Invalid skills entry: expected an object with a `type` field."
|
|
728
|
-
);
|
|
729
|
-
}
|
|
730
|
-
if ("type" in skill && skill.type) {
|
|
731
|
-
return skill;
|
|
732
|
-
}
|
|
733
|
-
throw new Error("Invalid skills entry: expected `type`.");
|
|
734
|
-
});
|
|
735
|
-
}
|
|
736
|
-
|
|
737
|
-
// src/skills/discover.ts
|
|
738
|
-
async function discoverSkillsInSandbox(opts) {
|
|
739
|
-
const { agent: agent2, sandbox, skillsDirs, sessionId } = opts;
|
|
740
|
-
const baseLog = createLogger({
|
|
741
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
742
|
-
subsystem: "skills"
|
|
743
|
-
});
|
|
744
|
-
const log2 = sessionId ? baseLog.withContext({ sessionId }) : baseLog;
|
|
745
|
-
const done = log2.time("discoverSkillsInSandbox");
|
|
746
|
-
const summaries = [];
|
|
747
|
-
const seenNames = /* @__PURE__ */ new Set();
|
|
748
|
-
for (const skillsDir of skillsDirs) {
|
|
749
|
-
const dirDone = log2.time("scan directory", { dir: skillsDir });
|
|
750
|
-
const dirSummaries = await discoverSkillsInDirectory({
|
|
751
|
-
sandbox,
|
|
752
|
-
skillsDir,
|
|
753
|
-
log: log2
|
|
754
|
-
});
|
|
755
|
-
dirDone({ count: dirSummaries.length });
|
|
756
|
-
for (const summary of dirSummaries) {
|
|
757
|
-
if (!seenNames.has(summary.name)) {
|
|
758
|
-
seenNames.add(summary.name);
|
|
759
|
-
summaries.push(summary);
|
|
760
|
-
}
|
|
761
|
-
}
|
|
762
|
-
}
|
|
763
|
-
done({ totalSkills: summaries.length });
|
|
764
|
-
return summaries;
|
|
765
|
-
}
|
|
766
|
-
async function discoverSkillsInDirectory(opts) {
|
|
767
|
-
const { sandbox, skillsDir, log: log2 } = opts;
|
|
768
|
-
const skillPaths = await findSkillFiles({ sandbox, skillsDir, log: log2 });
|
|
769
|
-
if (skillPaths.length === 0) {
|
|
770
|
-
return [];
|
|
771
|
-
}
|
|
772
|
-
const summaries = [];
|
|
773
|
-
for (const skillMdPath of skillPaths) {
|
|
774
|
-
const summary = await parseSkillFile({ sandbox, skillMdPath, log: log2 });
|
|
775
|
-
if (summary) {
|
|
776
|
-
summaries.push(summary);
|
|
777
|
-
}
|
|
778
|
-
}
|
|
779
|
-
return summaries;
|
|
780
|
-
}
|
|
781
|
-
async function findSkillFiles(opts) {
|
|
782
|
-
const { sandbox, skillsDir, log: log2 } = opts;
|
|
783
|
-
const execResult = await sandbox.exec({
|
|
784
|
-
command: "find",
|
|
785
|
-
args: [skillsDir, "-name", "SKILL.md", "-type", "f"]
|
|
786
|
-
});
|
|
787
|
-
if (execResult instanceof Error) {
|
|
788
|
-
log2.warn("failed to scan skills directory", {
|
|
789
|
-
dir: skillsDir,
|
|
790
|
-
error: execResult.message
|
|
791
|
-
});
|
|
792
|
-
return [];
|
|
793
|
-
}
|
|
794
|
-
const { stdout, exitCode } = await execResult.result;
|
|
795
|
-
if (exitCode !== 0) {
|
|
796
|
-
log2.warn("skills directory not found or inaccessible", { dir: skillsDir });
|
|
797
|
-
return [];
|
|
798
|
-
}
|
|
799
|
-
return stdout.trim().split("\n").filter((p) => p.length > 0);
|
|
800
|
-
}
|
|
801
|
-
async function parseSkillFile(opts) {
|
|
802
|
-
const { sandbox, skillMdPath, log: log2 } = opts;
|
|
803
|
-
const execResult = await sandbox.exec({
|
|
804
|
-
command: "cat",
|
|
805
|
-
args: [skillMdPath]
|
|
806
|
-
});
|
|
807
|
-
if (execResult instanceof Error) {
|
|
808
|
-
log2.warn("failed to read skill file", {
|
|
809
|
-
path: skillMdPath,
|
|
810
|
-
error: execResult.message
|
|
811
|
-
});
|
|
812
|
-
return null;
|
|
813
|
-
}
|
|
814
|
-
const { stdout, exitCode } = await execResult.result;
|
|
815
|
-
if (exitCode !== 0) {
|
|
816
|
-
log2.warn("could not read skill file", { path: skillMdPath });
|
|
817
|
-
return null;
|
|
818
|
-
}
|
|
819
|
-
const parsed = parseSkillFrontmatter(stdout);
|
|
820
|
-
if (!parsed) {
|
|
821
|
-
log2.warn("invalid or missing frontmatter", { path: skillMdPath });
|
|
822
|
-
return null;
|
|
823
|
-
}
|
|
824
|
-
return {
|
|
825
|
-
name: parsed.name,
|
|
826
|
-
description: parsed.description,
|
|
827
|
-
skillMdPath
|
|
828
|
-
};
|
|
829
|
-
}
|
|
830
|
-
|
|
831
|
-
// src/skills/resolve-host.ts
|
|
832
|
-
async function collectFiles(opts) {
|
|
833
|
-
const { readdir, readFile: readFile2, stat } = await import("fs/promises");
|
|
834
|
-
const { resolve: resolve2, posix: pathPosix } = await import("path");
|
|
835
|
-
const fileStat = await stat(opts.absolutePath);
|
|
836
|
-
if (fileStat.isFile()) {
|
|
837
|
-
const content = await readFile2(opts.absolutePath);
|
|
838
|
-
return [{ path: opts.relativePrefix ?? "SKILL.md", content }];
|
|
839
|
-
}
|
|
840
|
-
if (!fileStat.isDirectory()) {
|
|
841
|
-
return [];
|
|
842
|
-
}
|
|
843
|
-
const entries = await readdir(opts.absolutePath, { withFileTypes: true });
|
|
844
|
-
const files = [];
|
|
845
|
-
for (const entry of entries) {
|
|
846
|
-
const nextAbs = resolve2(opts.absolutePath, entry.name);
|
|
847
|
-
const nextRel = opts.relativePrefix ? pathPosix.join(opts.relativePrefix, entry.name) : entry.name;
|
|
848
|
-
if (entry.isDirectory()) {
|
|
849
|
-
files.push(
|
|
850
|
-
...await collectFiles({
|
|
851
|
-
absolutePath: nextAbs,
|
|
852
|
-
relativePrefix: nextRel
|
|
853
|
-
})
|
|
854
|
-
);
|
|
855
|
-
continue;
|
|
856
|
-
}
|
|
857
|
-
if (entry.isFile()) {
|
|
858
|
-
files.push({
|
|
859
|
-
path: nextRel,
|
|
860
|
-
content: await readFile2(nextAbs)
|
|
861
|
-
});
|
|
862
|
-
}
|
|
863
|
-
}
|
|
864
|
-
return files;
|
|
865
|
-
}
|
|
866
|
-
async function stableDirName(inputPath) {
|
|
867
|
-
const { createHash } = await import("crypto");
|
|
868
|
-
const hash = createHash("sha1").update(inputPath).digest("hex").slice(0, 10);
|
|
869
|
-
return `host-${hash}`;
|
|
870
|
-
}
|
|
871
|
-
async function resolveHostSkill(opts) {
|
|
872
|
-
const { resolve: resolve2, posix: pathPosix } = await import("path");
|
|
873
|
-
const hostPath = resolve2(process.cwd(), opts.skill.path);
|
|
874
|
-
const files = await collectFiles({ absolutePath: hostPath });
|
|
875
|
-
const dirName = await stableDirName(hostPath);
|
|
876
|
-
const destPath = pathPosix.join(".agent", "skills", ".materialized", dirName);
|
|
877
|
-
if (files.length === 0) {
|
|
878
|
-
return destPath;
|
|
879
|
-
}
|
|
880
|
-
await opts.sandbox.writeFiles({
|
|
881
|
-
files,
|
|
882
|
-
destPath
|
|
883
|
-
});
|
|
884
|
-
return destPath;
|
|
885
|
-
}
|
|
886
|
-
|
|
887
|
-
// src/skills/resolve-inline.ts
|
|
888
|
-
function slugify(value) {
|
|
889
|
-
return value.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/(^-|-$)/g, "");
|
|
890
|
-
}
|
|
891
|
-
async function stableInlineDirName(name) {
|
|
892
|
-
const { createHash } = await import("crypto");
|
|
893
|
-
const slug = slugify(name) || "inline-skill";
|
|
894
|
-
const hash = createHash("sha1").update(name).digest("hex").slice(0, 8);
|
|
895
|
-
return `${slug}-${hash}`;
|
|
896
|
-
}
|
|
897
|
-
function buildSkillMarkdown(skill) {
|
|
898
|
-
return `---
|
|
899
|
-
name: ${skill.name}
|
|
900
|
-
description: ${skill.description}
|
|
901
|
-
---
|
|
902
|
-
|
|
903
|
-
${skill.instructions.trim()}
|
|
904
|
-
`;
|
|
905
|
-
}
|
|
906
|
-
async function resolveInlineSkill(opts) {
|
|
907
|
-
const { posix: pathPosix } = await import("path");
|
|
908
|
-
const dirName = await stableInlineDirName(opts.skill.name);
|
|
909
|
-
const destPath = pathPosix.join(".agent", "skills", ".materialized", dirName);
|
|
910
|
-
await opts.sandbox.writeFiles({
|
|
911
|
-
destPath,
|
|
912
|
-
files: [
|
|
913
|
-
{
|
|
914
|
-
path: "SKILL.md",
|
|
915
|
-
content: buildSkillMarkdown(opts.skill)
|
|
916
|
-
}
|
|
917
|
-
]
|
|
918
|
-
});
|
|
919
|
-
return destPath;
|
|
920
|
-
}
|
|
921
|
-
|
|
922
|
-
// src/skills/resolve-remote.ts
|
|
923
|
-
function shellQuote(value) {
|
|
924
|
-
return `'${value.replace(/'/g, `'\\''`)}'`;
|
|
925
|
-
}
|
|
926
|
-
async function stableRepoDirName(skill) {
|
|
927
|
-
const { createHash } = await import("crypto");
|
|
928
|
-
const hashInput = `${skill.repo}::${skill.ref ?? "HEAD"}`;
|
|
929
|
-
const hash = createHash("sha1").update(hashInput).digest("hex").slice(0, 12);
|
|
930
|
-
return `git-${hash}`;
|
|
931
|
-
}
|
|
932
|
-
async function resolveRemoteSkill(opts) {
|
|
933
|
-
const { posix: pathPosix } = await import("path");
|
|
934
|
-
const dirName = await stableRepoDirName(opts.skill);
|
|
935
|
-
const cloneDir = pathPosix.join(".agent", "skills", ".materialized", dirName);
|
|
936
|
-
const quotedDir = shellQuote(cloneDir);
|
|
937
|
-
const quotedRepo = shellQuote(opts.skill.repo);
|
|
938
|
-
const quotedRef = shellQuote(opts.skill.ref ?? "HEAD");
|
|
939
|
-
const syncCommand = [
|
|
940
|
-
"set -euo pipefail",
|
|
941
|
-
`if [ ! -d ${quotedDir}/.git ]; then`,
|
|
942
|
-
` rm -rf ${quotedDir}`,
|
|
943
|
-
` mkdir -p ${quotedDir}`,
|
|
944
|
-
` git clone --depth 1 ${quotedRepo} ${quotedDir}`,
|
|
945
|
-
"fi",
|
|
946
|
-
`if [ ${quotedRef} != 'HEAD' ]; then`,
|
|
947
|
-
` git -C ${quotedDir} fetch --depth 1 origin ${quotedRef}`,
|
|
948
|
-
` git -C ${quotedDir} checkout FETCH_HEAD`,
|
|
949
|
-
"fi"
|
|
950
|
-
].join("\n");
|
|
951
|
-
const execResult = await opts.sandbox.exec({
|
|
952
|
-
command: "bash",
|
|
953
|
-
args: ["-lc", syncCommand]
|
|
954
|
-
});
|
|
955
|
-
const { exitCode, stderr } = await execResult.result;
|
|
956
|
-
if (exitCode !== 0) {
|
|
957
|
-
throw new Error(
|
|
958
|
-
`Failed to materialize git skill from ${opts.skill.repo}: ${stderr || "unknown error"}`
|
|
959
|
-
);
|
|
960
|
-
}
|
|
961
|
-
let resolvedDir = cloneDir;
|
|
962
|
-
if (opts.skill.path) {
|
|
963
|
-
resolvedDir = pathPosix.join(resolvedDir, opts.skill.path);
|
|
964
|
-
}
|
|
965
|
-
if (opts.skill.name) {
|
|
966
|
-
resolvedDir = pathPosix.join(resolvedDir, opts.skill.name);
|
|
967
|
-
}
|
|
968
|
-
return resolvedDir;
|
|
969
|
-
}
|
|
970
|
-
|
|
971
|
-
// src/skills/resolve.ts
|
|
972
|
-
async function resolveSkillInputsToSandboxDirs(opts) {
|
|
973
|
-
const baseLog = createLogger({
|
|
974
|
-
config: { ...opts.agent.options.logging, name: opts.agent.name },
|
|
975
|
-
subsystem: "skills"
|
|
976
|
-
});
|
|
977
|
-
const log2 = opts.sessionId ? baseLog.withContext({ sessionId: opts.sessionId }) : baseLog;
|
|
978
|
-
const entries = normalizeSkills(opts.skills);
|
|
979
|
-
const resolved = [];
|
|
980
|
-
for (const entry of entries) {
|
|
981
|
-
if (entry.type === "sandbox") {
|
|
982
|
-
resolved.push(entry.path);
|
|
983
|
-
continue;
|
|
984
|
-
}
|
|
985
|
-
if (entry.type === "host") {
|
|
986
|
-
resolved.push(
|
|
987
|
-
await resolveHostSkill({
|
|
988
|
-
sandbox: opts.sandbox,
|
|
989
|
-
skill: entry
|
|
990
|
-
})
|
|
991
|
-
);
|
|
992
|
-
continue;
|
|
993
|
-
}
|
|
994
|
-
if (entry.type === "git") {
|
|
995
|
-
resolved.push(
|
|
996
|
-
await resolveRemoteSkill({
|
|
997
|
-
sandbox: opts.sandbox,
|
|
998
|
-
skill: entry
|
|
999
|
-
})
|
|
1000
|
-
);
|
|
1001
|
-
continue;
|
|
1002
|
-
}
|
|
1003
|
-
if (entry.type === "inline") {
|
|
1004
|
-
resolved.push(
|
|
1005
|
-
await resolveInlineSkill({
|
|
1006
|
-
sandbox: opts.sandbox,
|
|
1007
|
-
skill: entry
|
|
1008
|
-
})
|
|
1009
|
-
);
|
|
1010
|
-
continue;
|
|
1011
|
-
}
|
|
1012
|
-
log2.warn("unknown skill source entry encountered", { entry });
|
|
1013
|
-
}
|
|
1014
|
-
return resolved;
|
|
1015
|
-
}
|
|
1016
|
-
|
|
1017
|
-
// src/tools/index.ts
|
|
1018
|
-
import {
|
|
1019
|
-
tool as tool3
|
|
1020
|
-
} from "ai";
|
|
1021
|
-
import { ulid as ulid2 } from "ulid";
|
|
1022
|
-
import { z as z2 } from "zod";
|
|
1023
|
-
|
|
1024
|
-
// src/tools/javascript.ts
|
|
1025
|
-
import { tool } from "ai";
|
|
1026
|
-
import { z } from "zod";
|
|
1027
|
-
var REQUEST_MARKER_START = "__TOOL_REQUEST__";
|
|
1028
|
-
var REQUEST_MARKER_END = "__TOOL_REQUEST_END__";
|
|
1029
|
-
var EXECUTION_TIMEOUT_MS = 5 * 60 * 1e3;
|
|
1030
|
-
var RUNNER_SCRIPT = `
|
|
1031
|
-
import { readFileSync, existsSync, unlinkSync, mkdirSync } from 'node:fs';
|
|
1032
|
-
import { join, dirname } from 'node:path';
|
|
1033
|
-
import { fileURLToPath } from 'node:url';
|
|
1034
|
-
|
|
1035
|
-
const SCRIPT_DIR = dirname(fileURLToPath(import.meta.url));
|
|
1036
|
-
const TOOL_CALL_TIMEOUT_MS = 300_000;
|
|
1037
|
-
|
|
1038
|
-
let reqCounter = 0;
|
|
1039
|
-
|
|
1040
|
-
async function callTool(name, input) {
|
|
1041
|
-
const id = 'req_' + (++reqCounter) + '_' + Date.now();
|
|
1042
|
-
const responseFile = join(SCRIPT_DIR, id + '.response.json');
|
|
1043
|
-
|
|
1044
|
-
const request = JSON.stringify({ id, tool: name, input });
|
|
1045
|
-
process.stderr.write('${REQUEST_MARKER_START}' + request + '${REQUEST_MARKER_END}' + '\\n');
|
|
1046
|
-
|
|
1047
|
-
const start = Date.now();
|
|
1048
|
-
while (!existsSync(responseFile)) {
|
|
1049
|
-
if (Date.now() - start > TOOL_CALL_TIMEOUT_MS) {
|
|
1050
|
-
throw new Error('Tool call ' + name + ' timed out waiting for response');
|
|
1051
|
-
}
|
|
1052
|
-
await new Promise(r => setTimeout(r, 30));
|
|
1053
|
-
}
|
|
1054
|
-
|
|
1055
|
-
let raw;
|
|
1056
|
-
try {
|
|
1057
|
-
raw = readFileSync(responseFile, 'utf8');
|
|
1058
|
-
} catch {
|
|
1059
|
-
await new Promise(r => setTimeout(r, 50));
|
|
1060
|
-
raw = readFileSync(responseFile, 'utf8');
|
|
1061
|
-
}
|
|
1062
|
-
|
|
1063
|
-
const response = JSON.parse(raw);
|
|
1064
|
-
try { unlinkSync(responseFile); } catch {}
|
|
1065
|
-
|
|
1066
|
-
if (response.error) {
|
|
1067
|
-
throw new Error(response.error);
|
|
1068
|
-
}
|
|
1069
|
-
return response.result;
|
|
1070
|
-
}
|
|
1071
|
-
|
|
1072
|
-
const code = readFileSync(join(SCRIPT_DIR, 'code.js'), 'utf8');
|
|
1073
|
-
const toolNames = JSON.parse(readFileSync(join(SCRIPT_DIR, 'tools.json'), 'utf8'));
|
|
1074
|
-
|
|
1075
|
-
const ctx = { tools: {} };
|
|
1076
|
-
for (const name of toolNames) {
|
|
1077
|
-
ctx.tools[name] = (input) => callTool(name, input);
|
|
1078
|
-
}
|
|
1079
|
-
|
|
1080
|
-
const logs = [];
|
|
1081
|
-
const customConsole = {
|
|
1082
|
-
log: (...args) => logs.push(args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
|
|
1083
|
-
error: (...args) => logs.push('[error] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
|
|
1084
|
-
warn: (...args) => logs.push('[warn] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
|
|
1085
|
-
};
|
|
1086
|
-
|
|
1087
|
-
try {
|
|
1088
|
-
const AsyncFunction = Object.getPrototypeOf(async function() {}).constructor;
|
|
1089
|
-
const fn = new AsyncFunction('ctx', 'console', code);
|
|
1090
|
-
const result = await fn(ctx, customConsole);
|
|
1091
|
-
|
|
1092
|
-
const output = { success: true, result: result !== undefined ? result : null };
|
|
1093
|
-
if (logs.length > 0) output.logs = logs;
|
|
1094
|
-
process.stdout.write(JSON.stringify(output));
|
|
1095
|
-
} catch (err) {
|
|
1096
|
-
const output = {
|
|
1097
|
-
success: false,
|
|
1098
|
-
error: err instanceof Error ? err.message : String(err),
|
|
1099
|
-
};
|
|
1100
|
-
if (logs.length > 0) output.logs = logs;
|
|
1101
|
-
process.stdout.write(JSON.stringify(output));
|
|
1102
|
-
}
|
|
1103
|
-
`;
|
|
1104
|
-
function extractJsonSchema(schema) {
|
|
1105
|
-
if (!schema || typeof schema !== "object") {
|
|
1106
|
-
return null;
|
|
1107
|
-
}
|
|
1108
|
-
if ("_def" in schema && typeof schema._def === "object") {
|
|
1109
|
-
try {
|
|
1110
|
-
return z.toJSONSchema(schema);
|
|
1111
|
-
} catch {
|
|
1112
|
-
return null;
|
|
1113
|
-
}
|
|
1114
|
-
}
|
|
1115
|
-
const s = schema;
|
|
1116
|
-
if (/* @__PURE__ */ Symbol.for("vercel.ai.schema") in s && "jsonSchema" in s) {
|
|
1117
|
-
return s.jsonSchema;
|
|
1118
|
-
}
|
|
1119
|
-
if ("type" in s || "properties" in s) {
|
|
1120
|
-
return s;
|
|
1121
|
-
}
|
|
1122
|
-
return null;
|
|
1123
|
-
}
|
|
1124
|
-
function jsonSchemaToTs(schema, indent = 0) {
|
|
1125
|
-
if (!schema) {
|
|
1126
|
-
return "unknown";
|
|
1127
|
-
}
|
|
1128
|
-
if (schema.enum && Array.isArray(schema.enum)) {
|
|
1129
|
-
return schema.enum.map((v) => JSON.stringify(v)).join(" | ");
|
|
1130
|
-
}
|
|
1131
|
-
if (schema.anyOf && Array.isArray(schema.anyOf)) {
|
|
1132
|
-
return schema.anyOf.map((s) => jsonSchemaToTs(s, indent)).join(" | ");
|
|
1133
|
-
}
|
|
1134
|
-
if (schema.oneOf && Array.isArray(schema.oneOf)) {
|
|
1135
|
-
return schema.oneOf.map((s) => jsonSchemaToTs(s, indent)).join(" | ");
|
|
1136
|
-
}
|
|
1137
|
-
const type = schema.type;
|
|
1138
|
-
switch (type) {
|
|
1139
|
-
case "string":
|
|
1140
|
-
return "string";
|
|
1141
|
-
case "number":
|
|
1142
|
-
case "integer":
|
|
1143
|
-
return "number";
|
|
1144
|
-
case "boolean":
|
|
1145
|
-
return "boolean";
|
|
1146
|
-
case "null":
|
|
1147
|
-
return "null";
|
|
1148
|
-
case "array": {
|
|
1149
|
-
const items = schema.items ? jsonSchemaToTs(schema.items, indent) : "unknown";
|
|
1150
|
-
return `Array<${items}>`;
|
|
1151
|
-
}
|
|
1152
|
-
case "object": {
|
|
1153
|
-
const properties = schema.properties;
|
|
1154
|
-
if (!properties) {
|
|
1155
|
-
return "Record<string, unknown>";
|
|
1156
|
-
}
|
|
1157
|
-
const required = new Set(schema.required || []);
|
|
1158
|
-
const pad = " ".repeat(indent + 1);
|
|
1159
|
-
const closePad = " ".repeat(indent);
|
|
1160
|
-
const props = Object.entries(properties).map(([key, value]) => {
|
|
1161
|
-
const opt = required.has(key) ? "" : "?";
|
|
1162
|
-
return `${pad}${key}${opt}: ${jsonSchemaToTs(value, indent + 1)}`;
|
|
1163
|
-
});
|
|
1164
|
-
return `{
|
|
1165
|
-
${props.join(";\n")};
|
|
1166
|
-
${closePad}}`;
|
|
1167
|
-
}
|
|
1168
|
-
default:
|
|
1169
|
-
return "unknown";
|
|
1170
|
-
}
|
|
1171
|
-
}
|
|
1172
|
-
function generateContextTypeString(tools) {
|
|
1173
|
-
const sigs = [];
|
|
1174
|
-
for (const [name, t] of Object.entries(tools)) {
|
|
1175
|
-
const raw = t;
|
|
1176
|
-
const inputSchema = extractJsonSchema(raw.inputSchema ?? raw.parameters);
|
|
1177
|
-
const outputSchema = extractJsonSchema(raw.outputSchema);
|
|
1178
|
-
const inputType = inputSchema ? jsonSchemaToTs(inputSchema, 2) : "{}";
|
|
1179
|
-
const outputType = outputSchema ? jsonSchemaToTs(outputSchema, 2) : "unknown";
|
|
1180
|
-
sigs.push(` ${name}(input: ${inputType}): Promise<${outputType}>`);
|
|
1181
|
-
}
|
|
1182
|
-
return `type Context = {
|
|
1183
|
-
tools: {
|
|
1184
|
-
${sigs.join(";\n")};
|
|
1185
|
-
}
|
|
1186
|
-
}`;
|
|
1187
|
-
}
|
|
1188
|
-
async function executeInSandbox({
|
|
1189
|
-
code,
|
|
1190
|
-
rpcDir,
|
|
1191
|
-
abortController,
|
|
1192
|
-
sandbox,
|
|
1193
|
-
availableTools,
|
|
1194
|
-
onSubToolCall,
|
|
1195
|
-
toolContext
|
|
1196
|
-
}) {
|
|
1197
|
-
const mkdirResult = await sandbox.exec({
|
|
1198
|
-
command: "mkdir",
|
|
1199
|
-
args: ["-p", rpcDir]
|
|
1200
|
-
});
|
|
1201
|
-
if (mkdirResult instanceof Error) {
|
|
1202
|
-
return { success: false, error: mkdirResult.message };
|
|
1203
|
-
}
|
|
1204
|
-
await mkdirResult.result;
|
|
1205
|
-
await sandbox.writeFiles({
|
|
1206
|
-
files: [
|
|
1207
|
-
{ path: "runner.mjs", content: RUNNER_SCRIPT },
|
|
1208
|
-
{ path: "code.js", content: code },
|
|
1209
|
-
{
|
|
1210
|
-
path: "tools.json",
|
|
1211
|
-
content: JSON.stringify(Object.keys(availableTools))
|
|
1212
|
-
}
|
|
1213
|
-
],
|
|
1214
|
-
destPath: rpcDir
|
|
1215
|
-
});
|
|
1216
|
-
const nodeMajor = Number.parseInt(process.versions.node.split(".")[0], 10);
|
|
1217
|
-
const permissionFlag = nodeMajor >= 22 ? "--permission" : "--experimental-permission";
|
|
1218
|
-
const execResult = await sandbox.exec({
|
|
1219
|
-
command: "node",
|
|
1220
|
-
args: [
|
|
1221
|
-
permissionFlag,
|
|
1222
|
-
`--allow-fs-read=${rpcDir}`,
|
|
1223
|
-
`--allow-fs-write=${rpcDir}`,
|
|
1224
|
-
`${rpcDir}/runner.mjs`
|
|
1225
|
-
],
|
|
1226
|
-
signal: abortController.signal
|
|
1227
|
-
});
|
|
1228
|
-
if (execResult instanceof Error) {
|
|
1229
|
-
return { success: false, error: execResult.message };
|
|
1230
|
-
}
|
|
1231
|
-
const fatal = { error: null };
|
|
1232
|
-
const abort = (error) => {
|
|
1233
|
-
if (!fatal.error) {
|
|
1234
|
-
fatal.error = error;
|
|
1235
|
-
abortController.abort();
|
|
1236
|
-
}
|
|
1237
|
-
};
|
|
1238
|
-
const handleToolRequest = async (requestJson) => {
|
|
1239
|
-
let parsed;
|
|
1240
|
-
try {
|
|
1241
|
-
parsed = JSON.parse(requestJson);
|
|
1242
|
-
} catch {
|
|
1243
|
-
return;
|
|
1244
|
-
}
|
|
1245
|
-
const { id, tool: toolName, input } = parsed;
|
|
1246
|
-
let response;
|
|
1247
|
-
try {
|
|
1248
|
-
const t = availableTools[toolName];
|
|
1249
|
-
if (!t?.execute) {
|
|
1250
|
-
throw new Error(
|
|
1251
|
-
`Tool ${toolName} not found or has no execute function`
|
|
1252
|
-
);
|
|
1253
|
-
}
|
|
1254
|
-
const exec2 = t.execute.bind(t);
|
|
1255
|
-
const execOptions = {
|
|
1256
|
-
toolCallId: `js_${toolName}_${Date.now()}`,
|
|
1257
|
-
messages: [],
|
|
1258
|
-
experimental_context: toolContext
|
|
1259
|
-
};
|
|
1260
|
-
if (onSubToolCall) {
|
|
1261
|
-
response = await onSubToolCall(
|
|
1262
|
-
toolName,
|
|
1263
|
-
input,
|
|
1264
|
-
() => exec2(input, execOptions)
|
|
1265
|
-
);
|
|
1266
|
-
} else {
|
|
1267
|
-
const result = await exec2(input, execOptions);
|
|
1268
|
-
response = { result };
|
|
1269
|
-
}
|
|
1270
|
-
} catch (err) {
|
|
1271
|
-
const msg = err instanceof Error ? err.message : String(err);
|
|
1272
|
-
response = { error: msg };
|
|
1273
|
-
}
|
|
1274
|
-
try {
|
|
1275
|
-
await sandbox.writeFiles({
|
|
1276
|
-
files: [
|
|
1277
|
-
{
|
|
1278
|
-
path: `${id}.response.json`,
|
|
1279
|
-
content: JSON.stringify(response)
|
|
1280
|
-
}
|
|
1281
|
-
],
|
|
1282
|
-
destPath: rpcDir
|
|
1283
|
-
});
|
|
1284
|
-
} catch (err) {
|
|
1285
|
-
const msg = err instanceof Error ? err.message : String(err);
|
|
1286
|
-
abort(new Error(`Failed to write RPC response for ${toolName}: ${msg}`));
|
|
1287
|
-
}
|
|
1288
|
-
};
|
|
1289
|
-
const toolCallPromises = [];
|
|
1290
|
-
let stderrBuffer = "";
|
|
1291
|
-
let nonMarkerStderr = "";
|
|
1292
|
-
const timeout = setTimeout(() => {
|
|
1293
|
-
abort(
|
|
1294
|
-
new Error(
|
|
1295
|
-
`JavaScript execution timed out after ${EXECUTION_TIMEOUT_MS / 1e3}s`
|
|
1296
|
-
)
|
|
1297
|
-
);
|
|
1298
|
-
}, EXECUTION_TIMEOUT_MS);
|
|
1299
|
-
try {
|
|
1300
|
-
for await (const entry of execResult.logs()) {
|
|
1301
|
-
if (fatal.error) {
|
|
1302
|
-
break;
|
|
1303
|
-
}
|
|
1304
|
-
if (entry.stream === "stderr") {
|
|
1305
|
-
stderrBuffer += entry.data;
|
|
1306
|
-
while (true) {
|
|
1307
|
-
const startIdx = stderrBuffer.indexOf(REQUEST_MARKER_START);
|
|
1308
|
-
if (startIdx === -1) {
|
|
1309
|
-
break;
|
|
1310
|
-
}
|
|
1311
|
-
const beforeMarker = stderrBuffer.slice(0, startIdx);
|
|
1312
|
-
if (beforeMarker.trim()) {
|
|
1313
|
-
nonMarkerStderr += beforeMarker;
|
|
1314
|
-
}
|
|
1315
|
-
const contentStart = startIdx + REQUEST_MARKER_START.length;
|
|
1316
|
-
const endIdx = stderrBuffer.indexOf(REQUEST_MARKER_END, contentStart);
|
|
1317
|
-
if (endIdx === -1) {
|
|
1318
|
-
break;
|
|
1319
|
-
}
|
|
1320
|
-
const requestJson = stderrBuffer.slice(contentStart, endIdx);
|
|
1321
|
-
stderrBuffer = stderrBuffer.slice(endIdx + REQUEST_MARKER_END.length);
|
|
1322
|
-
toolCallPromises.push(handleToolRequest(requestJson));
|
|
1323
|
-
}
|
|
1324
|
-
}
|
|
1325
|
-
}
|
|
1326
|
-
} finally {
|
|
1327
|
-
clearTimeout(timeout);
|
|
1328
|
-
}
|
|
1329
|
-
if (stderrBuffer.trim()) {
|
|
1330
|
-
nonMarkerStderr += stderrBuffer;
|
|
1331
|
-
}
|
|
1332
|
-
await Promise.allSettled(toolCallPromises);
|
|
1333
|
-
if (fatal.error) {
|
|
1334
|
-
execResult.result.catch(() => void 0);
|
|
1335
|
-
return {
|
|
1336
|
-
success: false,
|
|
1337
|
-
error: fatal.error.message,
|
|
1338
|
-
...nonMarkerStderr ? { stderr: nonMarkerStderr.slice(0, 2e3) } : {}
|
|
1339
|
-
};
|
|
1340
|
-
}
|
|
1341
|
-
const { stdout, stderr, exitCode } = await execResult.result;
|
|
1342
|
-
try {
|
|
1343
|
-
return JSON.parse(stdout);
|
|
1344
|
-
} catch {
|
|
1345
|
-
return {
|
|
1346
|
-
success: false,
|
|
1347
|
-
error: `Runner failed (exit ${exitCode}).`,
|
|
1348
|
-
stderr: (nonMarkerStderr + stderr).slice(0, 2e3),
|
|
1349
|
-
stdout: stdout.slice(0, 1e3)
|
|
1350
|
-
};
|
|
1351
|
-
}
|
|
1352
|
-
}
|
|
1353
|
-
function createJavaScriptTool(opts) {
|
|
1354
|
-
const { sandbox, onSubToolCall } = opts;
|
|
1355
|
-
const activeSet = opts.activeTools ? new Set(opts.activeTools) : null;
|
|
1356
|
-
const availableTools = {};
|
|
1357
|
-
for (const [name, t] of Object.entries(opts.tools)) {
|
|
1358
|
-
if (!activeSet || activeSet.has(name)) {
|
|
1359
|
-
availableTools[name] = t;
|
|
1360
|
-
}
|
|
1361
|
-
}
|
|
1362
|
-
const contextType = generateContextTypeString(availableTools);
|
|
1363
|
-
return tool({
|
|
1364
|
-
...builtInTools.JavaScript,
|
|
1365
|
-
description: `Execute JavaScript to orchestrate multiple tool calls in a single step. Use this when you need to run several tools in sequence, transform intermediate results, or parallelize independent operations with Promise.all.
|
|
1366
|
-
|
|
1367
|
-
The code runs as an async function body with \`ctx\` in scope:
|
|
1368
|
-
|
|
1369
|
-
\`\`\`typescript
|
|
1370
|
-
${contextType}
|
|
1371
|
-
\`\`\`
|
|
1372
|
-
|
|
1373
|
-
Examples:
|
|
1374
|
-
- Sequential: \`const file = await ctx.tools.Read({ path: "package.json" }); return JSON.parse(file.content);\`
|
|
1375
|
-
- Parallel: \`const [a, b] = await Promise.all([ctx.tools.Read({ path: "a.ts" }), ctx.tools.Read({ path: "b.ts" })]); return { a: a.content, b: b.content };\`
|
|
1376
|
-
- Transform: \`const grep = await ctx.tools.Grep({ pattern: "TODO" }); return grep.matches.split("\\n").length;\`
|
|
1377
|
-
- Bash: \`const r = await ctx.tools.Bash({ command: "node -e \\"console.log(JSON.stringify({v:1}))\\"" }); return JSON.parse(r.stdout);\`
|
|
1378
|
-
|
|
1379
|
-
Return a value to pass results back. Use console.log() for debug output.
|
|
1380
|
-
Important: Each tool returns a structured object matching its output schema \u2014 not a raw string. Access the specific field you need (e.g. Bash's result.stdout, Read's result.content) rather than trying to parse the entire result.`,
|
|
1381
|
-
execute: ({ code }) => {
|
|
1382
|
-
const runId = `run_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`;
|
|
1383
|
-
const rpcDir = `.agent/js-rpc/${runId}`;
|
|
1384
|
-
const abortController = new AbortController();
|
|
1385
|
-
return executeInSandbox({
|
|
1386
|
-
code,
|
|
1387
|
-
rpcDir,
|
|
1388
|
-
abortController,
|
|
1389
|
-
sandbox,
|
|
1390
|
-
availableTools,
|
|
1391
|
-
onSubToolCall,
|
|
1392
|
-
toolContext: opts.toolContext
|
|
1393
|
-
});
|
|
1394
|
-
}
|
|
1395
|
-
});
|
|
1396
|
-
}
|
|
1397
|
-
|
|
1398
|
-
// src/tools/needs-approval.ts
|
|
1399
|
-
var builtinNeedsApproval = {};
|
|
1400
|
-
async function checkNeedsApproval({
|
|
1401
|
-
agent: agent2,
|
|
1402
|
-
toolName,
|
|
1403
|
-
input,
|
|
1404
|
-
toolCallId,
|
|
1405
|
-
messages
|
|
1406
|
-
}) {
|
|
1407
|
-
const toolDef = agent2.options.tools?.[toolName];
|
|
1408
|
-
if (toolDef?.needsApproval != null) {
|
|
1409
|
-
if (typeof toolDef.needsApproval === "boolean") {
|
|
1410
|
-
return toolDef.needsApproval;
|
|
1411
|
-
}
|
|
1412
|
-
return await toolDef.needsApproval(input, {
|
|
1413
|
-
toolCallId,
|
|
1414
|
-
messages,
|
|
1415
|
-
experimental_context: void 0
|
|
1416
|
-
});
|
|
1417
|
-
}
|
|
1418
|
-
const agentEntry = agent2.options.needsApproval?.[toolName];
|
|
1419
|
-
if (agentEntry != null) {
|
|
1420
|
-
if (typeof agentEntry === "boolean") {
|
|
1421
|
-
return agentEntry;
|
|
1422
|
-
}
|
|
1423
|
-
return await agentEntry(input, {
|
|
1424
|
-
toolCallId,
|
|
1425
|
-
messages,
|
|
1426
|
-
experimental_context: void 0
|
|
1427
|
-
});
|
|
1428
|
-
}
|
|
1429
|
-
return builtinNeedsApproval[toolName] ?? false;
|
|
1430
|
-
}
|
|
1431
|
-
|
|
1432
|
-
// src/tools/tool.ts
|
|
1433
|
-
import {
|
|
1434
|
-
tool as aiTool
|
|
1435
|
-
} from "ai";
|
|
1436
|
-
function tool2(def) {
|
|
1437
|
-
const { execute, contextSchema, stateSchema, ...rest } = def;
|
|
1438
|
-
const wrappedExecute = execute ? (
|
|
1439
|
-
// biome-ignore lint/suspicious/noExplicitAny: input type erased at runtime
|
|
1440
|
-
(input, options) => {
|
|
1441
|
-
const ctx = options.experimental_context;
|
|
1442
|
-
const context = contextSchema ? contextSchema.parse(ctx.context) : ctx.context;
|
|
1443
|
-
if (stateSchema) {
|
|
1444
|
-
stateSchema.parse(ctx.state);
|
|
1445
|
-
}
|
|
1446
|
-
const state = ctx.state;
|
|
1447
|
-
return execute(input, {
|
|
1448
|
-
toolCallId: options.toolCallId,
|
|
1449
|
-
abortSignal: options.abortSignal,
|
|
1450
|
-
context,
|
|
1451
|
-
state,
|
|
1452
|
-
sessionId: ctx.sessionId,
|
|
1453
|
-
messages: ctx.messages,
|
|
1454
|
-
sandbox: ctx.sandbox
|
|
1455
|
-
});
|
|
1456
|
-
}
|
|
1457
|
-
) : void 0;
|
|
1458
|
-
return wrappedExecute ? aiTool({ ...rest, execute: wrappedExecute }) : (
|
|
1459
|
-
// biome-ignore lint/suspicious/noExplicitAny: no-execute path needs cast
|
|
1460
|
-
aiTool(rest)
|
|
1461
|
-
);
|
|
1462
|
-
}
|
|
1463
|
-
|
|
1464
|
-
// src/tools/index.ts
|
|
1465
|
-
var log = createLogger({ config: { name: "tools" }, subsystem: "tools" });
|
|
1466
|
-
function formatFileSize(bytes) {
|
|
1467
|
-
if (bytes < 1024) {
|
|
1468
|
-
return `${bytes}`;
|
|
1469
|
-
}
|
|
1470
|
-
if (bytes < 1024 * 1024) {
|
|
1471
|
-
return `${(bytes / 1024).toFixed(1)}K`;
|
|
1472
|
-
}
|
|
1473
|
-
if (bytes < 1024 * 1024 * 1024) {
|
|
1474
|
-
return `${(bytes / (1024 * 1024)).toFixed(1)}M`;
|
|
1475
|
-
}
|
|
1476
|
-
return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)}G`;
|
|
1477
|
-
}
|
|
1478
|
-
function isRgNotFoundError(err) {
|
|
1479
|
-
const parts = [err.message];
|
|
1480
|
-
const anyErr = err;
|
|
1481
|
-
if (typeof anyErr.reason === "string") {
|
|
1482
|
-
parts.push(anyErr.reason);
|
|
1483
|
-
}
|
|
1484
|
-
if (anyErr.cause instanceof Error) {
|
|
1485
|
-
parts.push(anyErr.cause.message);
|
|
1486
|
-
const anyCause = anyErr.cause;
|
|
1487
|
-
if (typeof anyCause.text === "string") {
|
|
1488
|
-
parts.push(anyCause.text);
|
|
1489
|
-
}
|
|
1490
|
-
}
|
|
1491
|
-
const msg = parts.join(" ").toLowerCase();
|
|
1492
|
-
return msg.includes("executable file not found") || msg.includes("executable_not_found") || msg.includes("enoent") && msg.includes("rg");
|
|
1493
|
-
}
|
|
1494
|
-
var builtInTools = {
|
|
1495
|
-
Read: tool3({
|
|
1496
|
-
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.",
|
|
1497
|
-
inputSchema: z2.object({
|
|
1498
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1499
|
-
path: z2.string().describe("Absolute path to the file"),
|
|
1500
|
-
startLine: z2.number().optional().describe(
|
|
1501
|
-
"Starting line number (1-indexed). If provided with endLine, reads exact range regardless of file size."
|
|
1502
|
-
),
|
|
1503
|
-
endLine: z2.number().optional().describe(
|
|
1504
|
-
"Ending line number (1-indexed, inclusive). If provided with startLine, reads exact range regardless of file size."
|
|
1505
|
-
)
|
|
1506
|
-
}),
|
|
1507
|
-
outputSchema: z2.object({
|
|
1508
|
-
content: z2.string().describe("File content"),
|
|
1509
|
-
metadata: z2.object({
|
|
1510
|
-
totalLines: z2.number().describe("Total number of lines in the file"),
|
|
1511
|
-
linesShown: z2.number().describe("Number of lines included in this response"),
|
|
1512
|
-
startLine: z2.number().describe("First line number shown (1-indexed)"),
|
|
1513
|
-
endLine: z2.number().describe("Last line number shown (1-indexed)"),
|
|
1514
|
-
isPaginated: z2.boolean().describe("Whether this is a partial view of the file"),
|
|
1515
|
-
fileSize: z2.string().describe("Human-readable file size (e.g., '2.5K', '1.2M')"),
|
|
1516
|
-
path: z2.string().describe("Absolute path to the file")
|
|
1517
|
-
})
|
|
1518
|
-
})
|
|
1519
|
-
}),
|
|
1520
|
-
Grep: tool3({
|
|
1521
|
-
description: "Search for patterns in files using ripgrep. Use this to find code patterns, function definitions, imports, etc.",
|
|
1522
|
-
inputSchema: z2.object({
|
|
1523
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1524
|
-
pattern: z2.string().describe("Regex pattern to search for (ripgrep syntax)"),
|
|
1525
|
-
path: z2.string().optional().describe(
|
|
1526
|
-
"Absolute path to search in (defaults to working directory). Can be a file or directory."
|
|
1527
|
-
),
|
|
1528
|
-
fileType: z2.string().optional().describe(
|
|
1529
|
-
"File type to filter by (e.g., 'ts', 'js', 'py', 'md'). Uses ripgrep's built-in type filters."
|
|
1530
|
-
),
|
|
1531
|
-
glob: z2.string().optional().describe(
|
|
1532
|
-
"Glob pattern to filter files (e.g., '*.tsx', 'src/**/*.ts')"
|
|
1533
|
-
),
|
|
1534
|
-
caseSensitive: z2.boolean().optional().default(true).describe("Whether search is case-sensitive (default: true)"),
|
|
1535
|
-
contextLines: z2.number().optional().describe(
|
|
1536
|
-
"Number of context lines to show before and after each match"
|
|
1537
|
-
),
|
|
1538
|
-
maxCount: z2.number().optional().describe(
|
|
1539
|
-
"Maximum number of matches per file (useful for limiting output)"
|
|
1540
|
-
),
|
|
1541
|
-
filesWithMatches: z2.boolean().optional().default(false).describe(
|
|
1542
|
-
"Only show file paths that contain matches, not the matching lines themselves"
|
|
1543
|
-
)
|
|
1544
|
-
}),
|
|
1545
|
-
outputSchema: z2.object({
|
|
1546
|
-
matches: z2.string().describe(
|
|
1547
|
-
"Search results with file paths, line numbers, and matching content"
|
|
1548
|
-
),
|
|
1549
|
-
summary: z2.object({
|
|
1550
|
-
matchCount: z2.number().describe("Number of matches found"),
|
|
1551
|
-
fileCount: z2.number().describe("Number of files containing matches"),
|
|
1552
|
-
searchPath: z2.string().describe("Path that was searched"),
|
|
1553
|
-
pattern: z2.string().describe("Pattern that was searched for")
|
|
1554
|
-
})
|
|
1555
|
-
})
|
|
1556
|
-
}),
|
|
1557
|
-
List: tool3({
|
|
1558
|
-
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.",
|
|
1559
|
-
inputSchema: z2.object({
|
|
1560
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1561
|
-
path: z2.string().optional().describe("Absolute path to list (defaults to working directory)"),
|
|
1562
|
-
depth: z2.number().optional().describe(
|
|
1563
|
-
"Maximum depth to traverse. Choose based on context: 1-2 for quick overview, 3-4 for detailed exploration, 5+ for comprehensive mapping"
|
|
1564
|
-
),
|
|
1565
|
-
includeHidden: z2.boolean().optional().default(false).describe(
|
|
1566
|
-
"Include hidden files and directories (those starting with '.')"
|
|
1567
|
-
),
|
|
1568
|
-
filesOnly: z2.boolean().optional().default(false).describe("Only show files, not directories"),
|
|
1569
|
-
pattern: z2.string().optional().describe("Glob pattern to filter results (e.g., '*.ts', '*test*')")
|
|
1570
|
-
}),
|
|
1571
|
-
outputSchema: z2.object({
|
|
1572
|
-
listing: z2.string().describe(
|
|
1573
|
-
"Directory tree listing showing paths relative to search root"
|
|
1574
|
-
),
|
|
1575
|
-
summary: z2.object({
|
|
1576
|
-
totalItems: z2.number().describe("Total number of items found"),
|
|
1577
|
-
totalFiles: z2.number().describe("Total number of files found"),
|
|
1578
|
-
totalDirs: z2.number().describe("Total number of directories found"),
|
|
1579
|
-
searchPath: z2.string().describe("Path that was listed"),
|
|
1580
|
-
depth: z2.number().optional().describe("Maximum depth used (if specified)")
|
|
1581
|
-
})
|
|
1582
|
-
})
|
|
1583
|
-
}),
|
|
1584
|
-
Write: tool3({
|
|
1585
|
-
description: "Write content to a file. Creates parent directories automatically. Overwrites existing files.",
|
|
1586
|
-
inputSchema: z2.object({
|
|
1587
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1588
|
-
path: z2.string().describe("Absolute path to the file"),
|
|
1589
|
-
content: z2.string().describe("Content to write to the file")
|
|
1590
|
-
}),
|
|
1591
|
-
outputSchema: z2.object({
|
|
1592
|
-
success: z2.boolean().describe("Whether the write succeeded"),
|
|
1593
|
-
path: z2.string().describe("Path to the written file"),
|
|
1594
|
-
bytesWritten: z2.number().describe("Number of bytes written"),
|
|
1595
|
-
error: z2.string().optional().describe("Error message if write failed")
|
|
1596
|
-
})
|
|
1597
|
-
}),
|
|
1598
|
-
Edit: tool3({
|
|
1599
|
-
description: "Edit a file by replacing an exact string. Fails if old_string is not found or appears multiple times (not unique). For multiple replacements, call this tool multiple times with unique context.",
|
|
1600
|
-
inputSchema: z2.object({
|
|
1601
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1602
|
-
path: z2.string().describe("Absolute path to the file"),
|
|
1603
|
-
old_string: z2.string().describe("Exact string to find and replace (must be unique in file)"),
|
|
1604
|
-
new_string: z2.string().describe("String to replace old_string with")
|
|
1605
|
-
}),
|
|
1606
|
-
outputSchema: z2.object({
|
|
1607
|
-
success: z2.boolean().describe("Whether the edit succeeded"),
|
|
1608
|
-
path: z2.string().describe("Path to the edited file"),
|
|
1609
|
-
error: z2.string().optional().describe("Error message if edit failed")
|
|
1610
|
-
})
|
|
1611
|
-
}),
|
|
1612
|
-
Bash: tool3({
|
|
1613
|
-
description: "Executes a bash command. Returns stdout and stderr separately. Use waitUntil to control how long to wait (0 = return immediately, process keeps running). Large outputs are tail-truncated; full output in outputDir.",
|
|
1614
|
-
inputSchema: z2.object({
|
|
1615
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1616
|
-
command: z2.string().describe("The shell command to execute"),
|
|
1617
|
-
waitUntil: z2.number().optional().describe(
|
|
1618
|
-
`Max ms to wait for completion (default: ${DEFAULT_WAIT_UNTIL}). Use 0 to return immediately while the process keeps running.`
|
|
1619
|
-
)
|
|
1620
|
-
}),
|
|
1621
|
-
outputSchema: z2.object({
|
|
1622
|
-
commandId: z2.string().describe("Command ID. Use to reference or kill running processes."),
|
|
1623
|
-
stdout: z2.string().describe(
|
|
1624
|
-
"Command stdout. Tail-truncated if large; full content in outputDir/stdout.txt."
|
|
1625
|
-
),
|
|
1626
|
-
stderr: z2.string().describe(
|
|
1627
|
-
"Command stderr. Tail-truncated if large; full content in outputDir/stderr.txt."
|
|
1628
|
-
),
|
|
1629
|
-
exitCode: z2.number().describe("Exit code (-1 if still running)"),
|
|
1630
|
-
status: z2.enum(["running", "completed", "failed"]).describe("Process status"),
|
|
1631
|
-
outputDir: z2.string().describe(
|
|
1632
|
-
"Path to output directory containing stdout.txt, stderr.txt, and metadata.json. Empty if still running."
|
|
1633
|
-
)
|
|
1634
|
-
})
|
|
1635
|
-
}),
|
|
1636
|
-
Skill: tool3({
|
|
1637
|
-
description: "Load a skill's full instructions by name. Call this before following a skill.",
|
|
1638
|
-
inputSchema: z2.object({
|
|
1639
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1640
|
-
name: z2.string().describe("Skill name from the Available Skills list")
|
|
1641
|
-
}),
|
|
1642
|
-
outputSchema: z2.object({
|
|
1643
|
-
name: z2.string(),
|
|
1644
|
-
description: z2.string(),
|
|
1645
|
-
content: z2.string().describe("Full SKILL.md content"),
|
|
1646
|
-
path: z2.string().describe("Path to the skill directory in the sandbox")
|
|
1647
|
-
})
|
|
1648
|
-
}),
|
|
1649
|
-
JavaScript: tool3({
|
|
1650
|
-
inputSchema: z2.object({
|
|
1651
|
-
label: z2.string().describe("A label that describes the action being performed"),
|
|
1652
|
-
code: z2.string().describe(
|
|
1653
|
-
"JavaScript async function body. `ctx` is in scope. Must use `return` to produce output."
|
|
1654
|
-
)
|
|
1655
|
-
})
|
|
1656
|
-
})
|
|
1657
|
-
};
|
|
1658
|
-
var builtinToolNames = Object.fromEntries(
|
|
1659
|
-
Object.entries(builtInTools).map(([name]) => [name, name])
|
|
1660
|
-
);
|
|
1661
|
-
var SKILL_MD_SUFFIX = /\/?SKILL\.md$/;
|
|
1662
|
-
function getTools({
|
|
1663
|
-
agent: agent2,
|
|
1664
|
-
sandbox,
|
|
1665
|
-
skillsRef,
|
|
1666
|
-
writer,
|
|
1667
|
-
abortController,
|
|
1668
|
-
nextPartIndexRef,
|
|
1669
|
-
assistantMessageId,
|
|
1670
|
-
sessionId,
|
|
1671
|
-
stepLog,
|
|
1672
|
-
context,
|
|
1673
|
-
state,
|
|
1674
|
-
messages,
|
|
1675
|
-
activeTools
|
|
1676
|
-
}) {
|
|
1677
|
-
const rawTools = {
|
|
1678
|
-
[builtinToolNames.Read]: tool3({
|
|
1679
|
-
...builtInTools.Read,
|
|
1680
|
-
execute: async ({ path, startLine, endLine }) => {
|
|
1681
|
-
const filePath = path;
|
|
1682
|
-
const result = await sandbox.readFile({ path: filePath });
|
|
1683
|
-
if (result instanceof Error) {
|
|
1684
|
-
log.error("Read failed", { error: result.message });
|
|
1685
|
-
throw result;
|
|
1686
|
-
}
|
|
1687
|
-
if (result === null) {
|
|
1688
|
-
return {
|
|
1689
|
-
content: `Error: File not found - ${filePath}`,
|
|
1690
|
-
metadata: {
|
|
1691
|
-
totalLines: 0,
|
|
1692
|
-
linesShown: 0,
|
|
1693
|
-
startLine: 0,
|
|
1694
|
-
endLine: 0,
|
|
1695
|
-
isPaginated: false,
|
|
1696
|
-
fileSize: "0",
|
|
1697
|
-
path: filePath
|
|
1698
|
-
}
|
|
1699
|
-
};
|
|
1700
|
-
}
|
|
1701
|
-
const fullContent = result.toString("utf-8");
|
|
1702
|
-
const lines = fullContent.split("\n");
|
|
1703
|
-
if (lines.length > 0 && lines.at(-1) === "") {
|
|
1704
|
-
lines.pop();
|
|
1705
|
-
}
|
|
1706
|
-
const totalLines = lines.length;
|
|
1707
|
-
const fileBytes = Buffer.byteLength(fullContent);
|
|
1708
|
-
const fileSize = formatFileSize(fileBytes);
|
|
1709
|
-
const PAGE_SIZE = 100;
|
|
1710
|
-
let actualStart;
|
|
1711
|
-
let actualEnd;
|
|
1712
|
-
if (startLine !== void 0 && endLine !== void 0) {
|
|
1713
|
-
actualStart = startLine;
|
|
1714
|
-
actualEnd = endLine;
|
|
1715
|
-
} else if (startLine !== void 0) {
|
|
1716
|
-
actualStart = startLine;
|
|
1717
|
-
actualEnd = Math.min(startLine + PAGE_SIZE - 1, totalLines);
|
|
1718
|
-
} else if (endLine !== void 0) {
|
|
1719
|
-
actualStart = 1;
|
|
1720
|
-
actualEnd = endLine;
|
|
1721
|
-
} else if (totalLines > 200) {
|
|
1722
|
-
actualStart = 1;
|
|
1723
|
-
actualEnd = PAGE_SIZE;
|
|
1724
|
-
} else {
|
|
1725
|
-
actualStart = 1;
|
|
1726
|
-
actualEnd = totalLines;
|
|
1727
|
-
}
|
|
1728
|
-
const slicedLines = lines.slice(actualStart - 1, actualEnd);
|
|
1729
|
-
const content = slicedLines.join("\n");
|
|
1730
|
-
return {
|
|
1731
|
-
metadata: {
|
|
1732
|
-
totalLines,
|
|
1733
|
-
linesShown: Math.max(0, actualEnd - actualStart + 1),
|
|
1734
|
-
startLine: actualStart,
|
|
1735
|
-
endLine: actualEnd,
|
|
1736
|
-
isPaginated: actualEnd < totalLines,
|
|
1737
|
-
fileSize,
|
|
1738
|
-
path: filePath
|
|
1739
|
-
},
|
|
1740
|
-
content
|
|
1741
|
-
};
|
|
1742
|
-
}
|
|
1743
|
-
}),
|
|
1744
|
-
[builtinToolNames.Grep]: tool3({
|
|
1745
|
-
...builtInTools.Grep,
|
|
1746
|
-
execute: async ({
|
|
1747
|
-
pattern,
|
|
1748
|
-
path,
|
|
1749
|
-
fileType,
|
|
1750
|
-
glob,
|
|
1751
|
-
caseSensitive,
|
|
1752
|
-
contextLines,
|
|
1753
|
-
maxCount,
|
|
1754
|
-
filesWithMatches
|
|
1755
|
-
}) => {
|
|
1756
|
-
const searchPath = path ?? ".";
|
|
1757
|
-
const args = [];
|
|
1758
|
-
args.push("--line-number");
|
|
1759
|
-
args.push("--heading");
|
|
1760
|
-
args.push("--color", "never");
|
|
1761
|
-
if (!caseSensitive) {
|
|
1762
|
-
args.push("-i");
|
|
1763
|
-
}
|
|
1764
|
-
if (fileType) {
|
|
1765
|
-
args.push("--type", fileType);
|
|
1766
|
-
}
|
|
1767
|
-
if (glob) {
|
|
1768
|
-
args.push("--glob", glob);
|
|
1769
|
-
}
|
|
1770
|
-
if (contextLines !== void 0) {
|
|
1771
|
-
args.push("-C", String(contextLines));
|
|
1772
|
-
}
|
|
1773
|
-
if (maxCount !== void 0) {
|
|
1774
|
-
args.push("--max-count", String(maxCount));
|
|
1775
|
-
}
|
|
1776
|
-
if (filesWithMatches) {
|
|
1777
|
-
args.push("--files-with-matches");
|
|
1778
|
-
}
|
|
1779
|
-
args.push("--", pattern, searchPath);
|
|
1780
|
-
let result = await sandbox.exec({ command: "rg", args });
|
|
1781
|
-
if (result instanceof Error && isRgNotFoundError(result)) {
|
|
1782
|
-
log.warn("rg not found, installing ripgrep");
|
|
1783
|
-
const installResult = await sandbox.exec({
|
|
1784
|
-
command: "bash",
|
|
1785
|
-
args: [
|
|
1786
|
-
"-c",
|
|
1787
|
-
[
|
|
1788
|
-
"command -v rg >/dev/null 2>&1 && exit 0",
|
|
1789
|
-
"dnf install -y ripgrep 2>/dev/null && exit 0",
|
|
1790
|
-
"(apt-get update -qq && apt-get install -y -qq ripgrep) 2>/dev/null && exit 0",
|
|
1791
|
-
"apk add --no-cache ripgrep 2>/dev/null && exit 0",
|
|
1792
|
-
"curl -sL https://github.com/BurntSushi/ripgrep/releases/download/14.1.1/ripgrep-14.1.1-x86_64-unknown-linux-musl.tar.gz | tar xz -C /tmp && install /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl/rg /usr/local/bin/rg && rm -rf /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl"
|
|
1793
|
-
].join("\n")
|
|
1794
|
-
],
|
|
1795
|
-
sudo: true
|
|
1796
|
-
});
|
|
1797
|
-
if (!(installResult instanceof Error)) {
|
|
1798
|
-
const installOutput = await installResult.result;
|
|
1799
|
-
if (installOutput.exitCode !== 0) {
|
|
1800
|
-
log.warn("ripgrep install failed", {
|
|
1801
|
-
stderr: installOutput.stderr
|
|
1802
|
-
});
|
|
1803
|
-
}
|
|
1804
|
-
}
|
|
1805
|
-
result = await sandbox.exec({ command: "rg", args });
|
|
1806
|
-
}
|
|
1807
|
-
if (result instanceof Error && isRgNotFoundError(result)) {
|
|
1808
|
-
log.warn("rg unavailable, falling back to grep");
|
|
1809
|
-
const grepArgs = ["-rn", "--color=never"];
|
|
1810
|
-
if (!caseSensitive) {
|
|
1811
|
-
grepArgs.push("-i");
|
|
1812
|
-
}
|
|
1813
|
-
if (contextLines !== void 0) {
|
|
1814
|
-
grepArgs.push("-C", String(contextLines));
|
|
1815
|
-
}
|
|
1816
|
-
if (maxCount !== void 0) {
|
|
1817
|
-
grepArgs.push("-m", String(maxCount));
|
|
1818
|
-
}
|
|
1819
|
-
if (filesWithMatches) {
|
|
1820
|
-
grepArgs.push("-l");
|
|
1821
|
-
}
|
|
1822
|
-
if (glob) {
|
|
1823
|
-
grepArgs.push(`--include=${glob}`);
|
|
1824
|
-
}
|
|
1825
|
-
if (fileType) {
|
|
1826
|
-
const includeMap = {
|
|
1827
|
-
ts: ["*.ts", "*.tsx", "*.mts", "*.cts"],
|
|
1828
|
-
js: ["*.js", "*.jsx", "*.mjs", "*.cjs"],
|
|
1829
|
-
py: ["*.py"],
|
|
1830
|
-
rust: ["*.rs"],
|
|
1831
|
-
go: ["*.go"],
|
|
1832
|
-
java: ["*.java"],
|
|
1833
|
-
md: ["*.md"],
|
|
1834
|
-
json: ["*.json"],
|
|
1835
|
-
css: ["*.css"],
|
|
1836
|
-
html: ["*.html"],
|
|
1837
|
-
yaml: ["*.yml", "*.yaml"]
|
|
1838
|
-
};
|
|
1839
|
-
for (const ext of includeMap[fileType] ?? [`*.${fileType}`]) {
|
|
1840
|
-
grepArgs.push(`--include=${ext}`);
|
|
1841
|
-
}
|
|
1842
|
-
}
|
|
1843
|
-
grepArgs.push("--", pattern, searchPath);
|
|
1844
|
-
result = await sandbox.exec({
|
|
1845
|
-
command: "grep",
|
|
1846
|
-
args: grepArgs
|
|
1847
|
-
});
|
|
1848
|
-
}
|
|
1849
|
-
if (result instanceof Error) {
|
|
1850
|
-
log.error("Grep failed", { error: result.message });
|
|
1851
|
-
throw result;
|
|
1852
|
-
}
|
|
1853
|
-
const { stdout, stderr } = await result.result;
|
|
1854
|
-
if (stderr && !stderr.toLowerCase().includes("no matches")) {
|
|
1855
|
-
log.warn("Grep stderr", { stderr });
|
|
1856
|
-
}
|
|
1857
|
-
const MAX_GREP_OUTPUT_CHARS = 5e4;
|
|
1858
|
-
let finalOutput = stdout;
|
|
1859
|
-
let wasTruncated = false;
|
|
1860
|
-
if (finalOutput.length > MAX_GREP_OUTPUT_CHARS) {
|
|
1861
|
-
finalOutput = finalOutput.slice(0, MAX_GREP_OUTPUT_CHARS) + "\n\n[Output truncated - use more specific pattern or path]";
|
|
1862
|
-
wasTruncated = true;
|
|
1863
|
-
}
|
|
1864
|
-
const lines = finalOutput.trim().split("\n").filter((l) => l.length > 0);
|
|
1865
|
-
const fileCount = filesWithMatches ? lines.length : new Set(
|
|
1866
|
-
lines.filter((l) => !l.startsWith(" ") && l.includes(":")).map((l) => l.split(":")[0])
|
|
1867
|
-
).size;
|
|
1868
|
-
return {
|
|
1869
|
-
summary: {
|
|
1870
|
-
matchCount: filesWithMatches ? 0 : lines.filter((l) => l.includes(":")).length,
|
|
1871
|
-
fileCount,
|
|
1872
|
-
searchPath,
|
|
1873
|
-
pattern,
|
|
1874
|
-
wasTruncated
|
|
1875
|
-
},
|
|
1876
|
-
matches: finalOutput || "(no matches found)"
|
|
1877
|
-
};
|
|
1878
|
-
}
|
|
1879
|
-
}),
|
|
1880
|
-
[builtinToolNames.List]: tool3({
|
|
1881
|
-
...builtInTools.List,
|
|
1882
|
-
execute: async ({ path, depth, includeHidden, filesOnly, pattern }) => {
|
|
1883
|
-
const searchPath = path ?? ".";
|
|
1884
|
-
const result = await sandbox.exec({
|
|
1885
|
-
command: "bash",
|
|
1886
|
-
args: [
|
|
1887
|
-
"-c",
|
|
1888
|
-
`
|
|
1889
|
-
set -e
|
|
1890
|
-
SEARCH_PATH="$1"
|
|
1891
|
-
DEPTH="$2"
|
|
1892
|
-
INCLUDE_HIDDEN="$3"
|
|
1893
|
-
FILES_ONLY="$4"
|
|
1894
|
-
PATTERN="$5"
|
|
1895
|
-
|
|
1896
|
-
# Build find command arguments
|
|
1897
|
-
FIND_ARGS=""
|
|
1898
|
-
[ -n "$DEPTH" ] && FIND_ARGS="$FIND_ARGS -maxdepth $DEPTH"
|
|
1899
|
-
[ "$INCLUDE_HIDDEN" != "true" ] && FIND_ARGS="$FIND_ARGS ! -path '*/.*'"
|
|
1900
|
-
[ "$FILES_ONLY" = "true" ] && FIND_ARGS="$FIND_ARGS -type f"
|
|
1901
|
-
[ -n "$PATTERN" ] && FIND_ARGS="$FIND_ARGS -name '$PATTERN'"
|
|
1902
|
-
|
|
1903
|
-
# Get listing
|
|
1904
|
-
LISTING=$(eval "find '$SEARCH_PATH' $FIND_ARGS" 2>/dev/null | sort)
|
|
1905
|
-
|
|
1906
|
-
# Get counts
|
|
1907
|
-
COUNT_ARGS=""
|
|
1908
|
-
[ -n "$DEPTH" ] && COUNT_ARGS="$COUNT_ARGS -maxdepth $DEPTH"
|
|
1909
|
-
[ "$INCLUDE_HIDDEN" != "true" ] && COUNT_ARGS="$COUNT_ARGS ! -path '*/.*'"
|
|
1910
|
-
|
|
1911
|
-
FILE_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type f" 2>/dev/null | wc -l)
|
|
1912
|
-
DIR_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type d" 2>/dev/null | wc -l)
|
|
1913
|
-
|
|
1914
|
-
# Output: counts first, then listing
|
|
1915
|
-
echo "$FILE_COUNT|$DIR_COUNT"
|
|
1916
|
-
echo "|||LISTING|||"
|
|
1917
|
-
echo "$LISTING" | sed "s|^$SEARCH_PATH|.|"
|
|
1918
|
-
`,
|
|
1919
|
-
"--",
|
|
1920
|
-
searchPath,
|
|
1921
|
-
depth?.toString() || "",
|
|
1922
|
-
includeHidden ? "true" : "false",
|
|
1923
|
-
filesOnly ? "true" : "false",
|
|
1924
|
-
pattern || ""
|
|
1925
|
-
]
|
|
1926
|
-
});
|
|
1927
|
-
if (result instanceof Error) {
|
|
1928
|
-
log.error("List failed", { error: result.message });
|
|
1929
|
-
throw result;
|
|
1930
|
-
}
|
|
1931
|
-
const { stdout, stderr } = await result.result;
|
|
1932
|
-
if (stderr) {
|
|
1933
|
-
log.warn("List stderr", { stderr });
|
|
1934
|
-
}
|
|
1935
|
-
const [countsLine, ...rest] = stdout.split("|||LISTING|||");
|
|
1936
|
-
const listing = rest.join("|||LISTING|||").trim();
|
|
1937
|
-
const [fileCountStr, dirCountStr] = countsLine.trim().split("|");
|
|
1938
|
-
const totalFiles = Number.parseInt(fileCountStr, 10) || 0;
|
|
1939
|
-
const totalDirs = Number.parseInt(dirCountStr, 10) || 0;
|
|
1940
|
-
const lines = listing.split("\n").filter((l) => l.length > 0);
|
|
1941
|
-
return {
|
|
1942
|
-
summary: {
|
|
1943
|
-
totalItems: lines.length,
|
|
1944
|
-
totalFiles,
|
|
1945
|
-
totalDirs,
|
|
1946
|
-
searchPath,
|
|
1947
|
-
depth
|
|
1948
|
-
},
|
|
1949
|
-
listing
|
|
1950
|
-
};
|
|
1951
|
-
}
|
|
1952
|
-
}),
|
|
1953
|
-
[builtinToolNames.Write]: tool3({
|
|
1954
|
-
...builtInTools.Write,
|
|
1955
|
-
execute: async ({ path, content }) => {
|
|
1956
|
-
const filePath = path;
|
|
1957
|
-
try {
|
|
1958
|
-
await sandbox.writeFiles({
|
|
1959
|
-
files: [{ path: filePath, content }],
|
|
1960
|
-
destPath: "."
|
|
1961
|
-
});
|
|
1962
|
-
return {
|
|
1963
|
-
success: true,
|
|
1964
|
-
path: filePath,
|
|
1965
|
-
bytesWritten: Buffer.byteLength(content, "utf8")
|
|
1966
|
-
};
|
|
1967
|
-
} catch (err) {
|
|
1968
|
-
const errorMsg = err instanceof Error ? err.message : String(err);
|
|
1969
|
-
return {
|
|
1970
|
-
success: false,
|
|
1971
|
-
path: filePath,
|
|
1972
|
-
bytesWritten: 0,
|
|
1973
|
-
error: errorMsg
|
|
1974
|
-
};
|
|
1975
|
-
}
|
|
1976
|
-
}
|
|
1977
|
-
}),
|
|
1978
|
-
[builtinToolNames.Edit]: tool3({
|
|
1979
|
-
...builtInTools.Edit,
|
|
1980
|
-
execute: async ({ path, old_string, new_string }) => {
|
|
1981
|
-
const filePath = path;
|
|
1982
|
-
const result = await sandbox.readFile({ path: filePath });
|
|
1983
|
-
if (result instanceof Error) {
|
|
1984
|
-
return { success: false, path: filePath, error: result.message };
|
|
1985
|
-
}
|
|
1986
|
-
if (result === null) {
|
|
1987
|
-
return {
|
|
1988
|
-
success: false,
|
|
1989
|
-
path: filePath,
|
|
1990
|
-
error: `File not found: ${filePath}`
|
|
1991
|
-
};
|
|
1992
|
-
}
|
|
1993
|
-
const content = result.toString("utf-8");
|
|
1994
|
-
const occurrences = content.split(old_string).length - 1;
|
|
1995
|
-
if (occurrences === 0) {
|
|
1996
|
-
return {
|
|
1997
|
-
success: false,
|
|
1998
|
-
path: filePath,
|
|
1999
|
-
error: "old_string not found in file"
|
|
2000
|
-
};
|
|
2001
|
-
}
|
|
2002
|
-
if (occurrences > 1) {
|
|
2003
|
-
return {
|
|
2004
|
-
success: false,
|
|
2005
|
-
path: filePath,
|
|
2006
|
-
error: `old_string appears ${occurrences} times in file (must be unique). Include more surrounding context to make the match unique.`
|
|
2007
|
-
};
|
|
2008
|
-
}
|
|
2009
|
-
const newContent = content.replace(old_string, new_string);
|
|
2010
|
-
try {
|
|
2011
|
-
await sandbox.writeFiles({
|
|
2012
|
-
files: [{ path: filePath, content: newContent }],
|
|
2013
|
-
destPath: "."
|
|
2014
|
-
});
|
|
2015
|
-
return { success: true, path: filePath };
|
|
2016
|
-
} catch (err) {
|
|
2017
|
-
const errorMsg = err instanceof Error ? err.message : String(err);
|
|
2018
|
-
return { success: false, path: filePath, error: errorMsg };
|
|
2019
|
-
}
|
|
2020
|
-
}
|
|
2021
|
-
}),
|
|
2022
|
-
[builtinToolNames.Bash]: tool3({
|
|
2023
|
-
...builtInTools.Bash,
|
|
2024
|
-
execute: async ({ command, waitUntil }) => {
|
|
2025
|
-
const { createProcessManager } = await import("./process-manager-WQHAIVRB.mjs");
|
|
2026
|
-
const processManager = createProcessManager({ sandbox });
|
|
2027
|
-
const result = await processManager.run({ command, waitUntil });
|
|
2028
|
-
const MAX_STDOUT = 5e4;
|
|
2029
|
-
const MAX_STDERR = 1e4;
|
|
2030
|
-
let { stdout, stderr } = result;
|
|
2031
|
-
if (stdout.length > MAX_STDOUT) {
|
|
2032
|
-
stdout = `[truncated \u2014 showing last ${MAX_STDOUT} chars. Full: ${result.outputDir}/stdout.txt]
|
|
2033
|
-
|
|
2034
|
-
` + stdout.slice(-MAX_STDOUT);
|
|
2035
|
-
}
|
|
2036
|
-
if (stderr.length > MAX_STDERR) {
|
|
2037
|
-
stderr = `[truncated \u2014 showing last ${MAX_STDERR} chars. Full: ${result.outputDir}/stderr.txt]
|
|
2038
|
-
|
|
2039
|
-
` + stderr.slice(-MAX_STDERR);
|
|
2040
|
-
}
|
|
2041
|
-
return { ...result, stdout, stderr };
|
|
2042
|
-
}
|
|
2043
|
-
})
|
|
2044
|
-
};
|
|
2045
|
-
for (const [name, t] of Object.entries(agent2.options.tools ?? {})) {
|
|
2046
|
-
rawTools[name] = t;
|
|
2047
|
-
}
|
|
2048
|
-
if (skillsRef.current.length > 0) {
|
|
2049
|
-
rawTools[builtinToolNames.Skill] = tool3({
|
|
2050
|
-
...builtInTools.Skill,
|
|
2051
|
-
execute: async ({ name }) => {
|
|
2052
|
-
const skills = skillsRef.current;
|
|
2053
|
-
const skill = skills.find(
|
|
2054
|
-
(s) => s.name.toLowerCase() === name.toLowerCase()
|
|
2055
|
-
);
|
|
2056
|
-
if (!skill) {
|
|
2057
|
-
throw new Error(
|
|
2058
|
-
`Skill not found: "${name}". Available: ${skills.map((s) => s.name).join(", ")}`
|
|
2059
|
-
);
|
|
2060
|
-
}
|
|
2061
|
-
const result = await sandbox.readFile({
|
|
2062
|
-
path: skill.skillMdPath
|
|
2063
|
-
});
|
|
2064
|
-
if (result instanceof Error) {
|
|
2065
|
-
throw result;
|
|
2066
|
-
}
|
|
2067
|
-
if (result === null) {
|
|
2068
|
-
throw new Error(`Skill file not found: ${skill.skillMdPath}`);
|
|
2069
|
-
}
|
|
2070
|
-
const raw = typeof result === "string" ? result : result.toString("utf-8");
|
|
2071
|
-
const endMarker = raw.indexOf("---", 3);
|
|
2072
|
-
const content = endMarker === -1 ? raw : raw.slice(endMarker + 3).trim();
|
|
2073
|
-
const skillDir = skill.skillMdPath.replace(SKILL_MD_SUFFIX, "");
|
|
2074
|
-
return {
|
|
2075
|
-
name: skill.name,
|
|
2076
|
-
description: skill.description,
|
|
2077
|
-
content,
|
|
2078
|
-
path: skillDir
|
|
2079
|
-
};
|
|
2080
|
-
}
|
|
2081
|
-
});
|
|
2082
|
-
}
|
|
2083
|
-
const onSubToolCall = async (toolName, toolInput, execute) => {
|
|
2084
|
-
const needsApproval = await checkNeedsApproval({
|
|
2085
|
-
agent: agent2,
|
|
2086
|
-
toolName,
|
|
2087
|
-
input: toolInput,
|
|
2088
|
-
toolCallId: `js_${toolName}_${Date.now()}`,
|
|
2089
|
-
messages: []
|
|
2090
|
-
});
|
|
2091
|
-
if (!needsApproval) {
|
|
2092
|
-
try {
|
|
2093
|
-
const result = await execute();
|
|
2094
|
-
return { result };
|
|
2095
|
-
} catch (err) {
|
|
2096
|
-
return {
|
|
2097
|
-
error: err instanceof Error ? err.message : String(err)
|
|
2098
|
-
};
|
|
2099
|
-
}
|
|
2100
|
-
}
|
|
2101
|
-
const approvalId = `js_approval_${ulid2()}`;
|
|
2102
|
-
const toolCallId = `js_tc_${ulid2()}`;
|
|
2103
|
-
const partId = approvalPartId(approvalId);
|
|
2104
|
-
await agent2.storage.part.set(partId, {
|
|
2105
|
-
id: partId,
|
|
2106
|
-
index: nextPartIndexRef.current++,
|
|
2107
|
-
messageId: assistantMessageId,
|
|
2108
|
-
sessionId,
|
|
2109
|
-
part: {
|
|
2110
|
-
type: `tool-${toolName}`,
|
|
2111
|
-
toolCallId,
|
|
2112
|
-
state: "approval-requested",
|
|
2113
|
-
input: toolInput,
|
|
2114
|
-
approval: { id: approvalId }
|
|
2115
|
-
}
|
|
2116
|
-
});
|
|
2117
|
-
writer.write({
|
|
2118
|
-
type: "tool-input-start",
|
|
2119
|
-
toolCallId,
|
|
2120
|
-
toolName
|
|
2121
|
-
});
|
|
2122
|
-
writer.write({
|
|
2123
|
-
type: "tool-input-available",
|
|
2124
|
-
toolCallId,
|
|
2125
|
-
toolName,
|
|
2126
|
-
input: toolInput
|
|
2127
|
-
});
|
|
2128
|
-
writer.write({
|
|
2129
|
-
type: "tool-approval-request",
|
|
2130
|
-
approvalId,
|
|
2131
|
-
toolCallId
|
|
2132
|
-
});
|
|
2133
|
-
const POLL_MS = 500;
|
|
2134
|
-
const TIMEOUT_MS = 5 * 60 * 1e3;
|
|
2135
|
-
const start2 = Date.now();
|
|
2136
|
-
while (Date.now() - start2 < TIMEOUT_MS && !abortController.signal.aborted) {
|
|
2137
|
-
const updated = await agent2.storage.part.get(partId);
|
|
2138
|
-
if (updated && "state" in updated.part && (updated.part.state === "approval-responded" || updated.part.state === "output-denied")) {
|
|
2139
|
-
const state2 = updated.part.state;
|
|
2140
|
-
if (state2 === "output-denied") {
|
|
2141
|
-
const reason = updated.part.approval?.reason;
|
|
2142
|
-
return {
|
|
2143
|
-
error: `Tool "${toolName}" denied: ${reason || "user denied"}`
|
|
2144
|
-
};
|
|
2145
|
-
}
|
|
2146
|
-
const approval = updated.part.approval;
|
|
2147
|
-
if (approval?.approved) {
|
|
2148
|
-
try {
|
|
2149
|
-
const result = await execute();
|
|
2150
|
-
writer.write({
|
|
2151
|
-
type: "tool-output-available",
|
|
2152
|
-
toolCallId,
|
|
2153
|
-
output: result
|
|
2154
|
-
});
|
|
2155
|
-
return { result };
|
|
2156
|
-
} catch (err) {
|
|
2157
|
-
const error = err instanceof Error ? err.message : String(err);
|
|
2158
|
-
writer.write({
|
|
2159
|
-
type: "tool-output-error",
|
|
2160
|
-
toolCallId,
|
|
2161
|
-
errorText: error
|
|
2162
|
-
});
|
|
2163
|
-
return { error };
|
|
2164
|
-
}
|
|
2165
|
-
}
|
|
2166
|
-
writer.write({
|
|
2167
|
-
type: "tool-output-denied",
|
|
2168
|
-
toolCallId
|
|
2169
|
-
});
|
|
2170
|
-
return {
|
|
2171
|
-
error: `Tool "${toolName}" denied: ${approval?.reason || "user denied"}`
|
|
2172
|
-
};
|
|
2173
|
-
}
|
|
2174
|
-
await new Promise((r) => setTimeout(r, POLL_MS));
|
|
2175
|
-
}
|
|
2176
|
-
return {
|
|
2177
|
-
error: abortController.signal.aborted ? "interrupted" : "Approval timed out"
|
|
2178
|
-
};
|
|
2179
|
-
};
|
|
2180
|
-
const toolContext = {
|
|
2181
|
-
context,
|
|
2182
|
-
state,
|
|
2183
|
-
sessionId,
|
|
2184
|
-
sandbox,
|
|
2185
|
-
messages
|
|
2186
|
-
};
|
|
2187
|
-
rawTools[builtinToolNames.JavaScript] = createJavaScriptTool({
|
|
2188
|
-
tools: rawTools,
|
|
2189
|
-
sandbox,
|
|
2190
|
-
activeTools,
|
|
2191
|
-
onSubToolCall,
|
|
2192
|
-
toolContext
|
|
2193
|
-
});
|
|
2194
|
-
const toolsWithNeedsApproval = Object.fromEntries(
|
|
2195
|
-
Object.entries(rawTools).map(([name, t]) => {
|
|
2196
|
-
const originalExecute = t.execute;
|
|
2197
|
-
const wrappedExecute = originalExecute ? async (...args) => {
|
|
2198
|
-
const done = stepLog.time(`tool:${name}`);
|
|
2199
|
-
try {
|
|
2200
|
-
let toolInput = args[0];
|
|
2201
|
-
if (agent2.hooks["tool.before"]) {
|
|
2202
|
-
const beforeResult = await agent2.hooks["tool.before"]({
|
|
2203
|
-
name,
|
|
2204
|
-
input: toolInput,
|
|
2205
|
-
context: toolContext
|
|
2206
|
-
});
|
|
2207
|
-
if (beforeResult?.input !== void 0) {
|
|
2208
|
-
toolInput = beforeResult.input;
|
|
2209
|
-
args[0] = toolInput;
|
|
2210
|
-
}
|
|
2211
|
-
}
|
|
2212
|
-
let result = await originalExecute(...args);
|
|
2213
|
-
if (agent2.hooks["tool.after"]) {
|
|
2214
|
-
const afterResult = await agent2.hooks["tool.after"]({
|
|
2215
|
-
name,
|
|
2216
|
-
input: toolInput,
|
|
2217
|
-
result,
|
|
2218
|
-
context: toolContext
|
|
2219
|
-
});
|
|
2220
|
-
if (afterResult?.result !== void 0) {
|
|
2221
|
-
result = afterResult.result;
|
|
2222
|
-
}
|
|
2223
|
-
}
|
|
2224
|
-
done();
|
|
2225
|
-
return result;
|
|
2226
|
-
} catch (e) {
|
|
2227
|
-
done({
|
|
2228
|
-
error: e instanceof Error ? e.message : String(e)
|
|
2229
|
-
});
|
|
2230
|
-
throw e;
|
|
2231
|
-
}
|
|
2232
|
-
} : void 0;
|
|
2233
|
-
return [
|
|
2234
|
-
name,
|
|
2235
|
-
{
|
|
2236
|
-
...t,
|
|
2237
|
-
...wrappedExecute ? { execute: wrappedExecute } : {},
|
|
2238
|
-
needsApproval: async (toolInput, opts) => {
|
|
2239
|
-
return await checkNeedsApproval({
|
|
2240
|
-
agent: agent2,
|
|
2241
|
-
toolName: name,
|
|
2242
|
-
input: toolInput,
|
|
2243
|
-
toolCallId: opts.toolCallId,
|
|
2244
|
-
messages: opts.messages
|
|
2245
|
-
});
|
|
2246
|
-
}
|
|
2247
|
-
}
|
|
2248
|
-
];
|
|
2249
|
-
})
|
|
2250
|
-
);
|
|
2251
|
-
return { rawTools, toolsWithNeedsApproval, toolContext };
|
|
2252
|
-
}
|
|
2253
|
-
|
|
2254
|
-
// src/utils/prompt-cache.ts
|
|
2255
|
-
var ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST = 4;
|
|
2256
|
-
var CLAUDE_LIKE_MODEL_MATCHERS = ["claude", "anthropic"];
|
|
2257
|
-
var CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS = {
|
|
2258
|
-
anthropic: {
|
|
2259
|
-
cacheControl: { type: "ephemeral" }
|
|
2260
|
-
},
|
|
2261
|
-
openrouter: {
|
|
2262
|
-
cacheControl: { type: "ephemeral" }
|
|
2263
|
-
},
|
|
2264
|
-
bedrock: {
|
|
2265
|
-
cachePoint: { type: "default" }
|
|
2266
|
-
},
|
|
2267
|
-
openaiCompatible: {
|
|
2268
|
-
cache_control: { type: "ephemeral" }
|
|
2269
|
-
},
|
|
2270
|
-
copilot: {
|
|
2271
|
-
copilot_cache_control: { type: "ephemeral" }
|
|
2272
|
-
}
|
|
2273
|
-
};
|
|
2274
|
-
function getGatewayProvider(model) {
|
|
2275
|
-
const slashIndex = model.indexOf("/");
|
|
2276
|
-
if (slashIndex === -1) {
|
|
2277
|
-
return null;
|
|
2278
|
-
}
|
|
2279
|
-
return model.slice(0, slashIndex);
|
|
2280
|
-
}
|
|
2281
|
-
function countAnthropicCacheBreakpoints(messages) {
|
|
2282
|
-
let count = 0;
|
|
2283
|
-
for (const message of messages) {
|
|
2284
|
-
const providerOptions = message.providerOptions;
|
|
2285
|
-
if (providerOptions?.anthropic?.cacheControl) {
|
|
2286
|
-
count += 1;
|
|
2287
|
-
}
|
|
2288
|
-
if (Array.isArray(message.content)) {
|
|
2289
|
-
for (const part of message.content) {
|
|
2290
|
-
const partProviderOptions = part.providerOptions;
|
|
2291
|
-
if (partProviderOptions?.anthropic?.cacheControl) {
|
|
2292
|
-
count += 1;
|
|
2293
|
-
}
|
|
2294
|
-
}
|
|
2295
|
-
}
|
|
2296
|
-
}
|
|
2297
|
-
return count;
|
|
2298
|
-
}
|
|
2299
|
-
function isClaudeLikeModel(model) {
|
|
2300
|
-
const lower = model.toLowerCase();
|
|
2301
|
-
return CLAUDE_LIKE_MODEL_MATCHERS.some((m) => lower.includes(m));
|
|
2302
|
-
}
|
|
2303
|
-
function mergeProviderOptions(opts) {
|
|
2304
|
-
const next = { ...opts.current ?? {} };
|
|
2305
|
-
for (const [key, value] of Object.entries(opts.patch)) {
|
|
2306
|
-
next[key] = { ...next[key] ?? {}, ...value };
|
|
2307
|
-
}
|
|
2308
|
-
return next;
|
|
2309
|
-
}
|
|
2310
|
-
function isCacheableClaudePart(part) {
|
|
2311
|
-
if (!part || typeof part !== "object") {
|
|
2312
|
-
return false;
|
|
2313
|
-
}
|
|
2314
|
-
if ("type" in part && part.type === "text") {
|
|
2315
|
-
const text = part.text;
|
|
2316
|
-
if (typeof text === "string") {
|
|
2317
|
-
return text.trim().length > 0;
|
|
2318
|
-
}
|
|
2319
|
-
}
|
|
2320
|
-
if ("type" in part) {
|
|
2321
|
-
const type = part.type;
|
|
2322
|
-
if (type === "thinking" || type === "reasoning") {
|
|
2323
|
-
return false;
|
|
2324
|
-
}
|
|
2325
|
-
}
|
|
2326
|
-
return true;
|
|
2327
|
-
}
|
|
2328
|
-
function hasCacheableContent(message) {
|
|
2329
|
-
if (typeof message.content === "string") {
|
|
2330
|
-
return message.content.trim().length > 0;
|
|
2331
|
-
}
|
|
2332
|
-
if (Array.isArray(message.content)) {
|
|
2333
|
-
return message.content.some((part) => isCacheableClaudePart(part));
|
|
2334
|
-
}
|
|
2335
|
-
return false;
|
|
2336
|
-
}
|
|
2337
|
-
function findLastCacheablePartIndex(content) {
|
|
2338
|
-
for (let i = content.length - 1; i >= 0; i -= 1) {
|
|
2339
|
-
if (isCacheableClaudePart(content[i])) {
|
|
2340
|
-
return i;
|
|
2341
|
-
}
|
|
2342
|
-
}
|
|
2343
|
-
return null;
|
|
2344
|
-
}
|
|
2345
|
-
function selectClaudeCachingTargets(messages) {
|
|
2346
|
-
const systemIndices = [];
|
|
2347
|
-
const finalIndices = [];
|
|
2348
|
-
for (let i = 0; i < messages.length && systemIndices.length < 2; i += 1) {
|
|
2349
|
-
const message = messages[i];
|
|
2350
|
-
if (message.role === "system" && hasCacheableContent(message)) {
|
|
2351
|
-
systemIndices.push(i);
|
|
2352
|
-
}
|
|
2353
|
-
}
|
|
2354
|
-
for (let i = messages.length - 1; i >= 0 && finalIndices.length < 2; i -= 1) {
|
|
2355
|
-
const message = messages[i];
|
|
2356
|
-
if (message.role !== "system" && hasCacheableContent(message)) {
|
|
2357
|
-
finalIndices.push(i);
|
|
2358
|
-
}
|
|
2359
|
-
}
|
|
2360
|
-
finalIndices.reverse();
|
|
2361
|
-
return [...systemIndices, ...finalIndices];
|
|
2362
|
-
}
|
|
2363
|
-
function applyClaudePromptCaching(opts) {
|
|
2364
|
-
const existingBreakpoints = countAnthropicCacheBreakpoints(opts.messages);
|
|
2365
|
-
const remainingBudget = Math.max(
|
|
2366
|
-
0,
|
|
2367
|
-
opts.maxBreakpointsPerRequest - existingBreakpoints
|
|
2368
|
-
);
|
|
2369
|
-
if (remainingBudget === 0) {
|
|
2370
|
-
return opts.messages;
|
|
2371
|
-
}
|
|
2372
|
-
const targetIndices = selectClaudeCachingTargets(opts.messages).slice(
|
|
2373
|
-
0,
|
|
2374
|
-
remainingBudget
|
|
2375
|
-
);
|
|
2376
|
-
if (targetIndices.length === 0) {
|
|
2377
|
-
return opts.messages;
|
|
2378
|
-
}
|
|
2379
|
-
const useMessageLevelOptions = opts.providerId === "anthropic" || Boolean(opts.providerId?.includes("bedrock"));
|
|
2380
|
-
const nextMessages = opts.messages.slice();
|
|
2381
|
-
for (const messageIndex of targetIndices) {
|
|
2382
|
-
const message = nextMessages[messageIndex];
|
|
2383
|
-
const shouldUseContentOptions = !useMessageLevelOptions && Array.isArray(message.content) && message.content.length > 0;
|
|
2384
|
-
if (shouldUseContentOptions && Array.isArray(message.content)) {
|
|
2385
|
-
const partIndex = findLastCacheablePartIndex(message.content);
|
|
2386
|
-
if (partIndex !== null) {
|
|
2387
|
-
const part = message.content[partIndex];
|
|
2388
|
-
if (part && typeof part === "object") {
|
|
2389
|
-
const partProviderOptions = part.providerOptions;
|
|
2390
|
-
const nextContent = message.content.slice();
|
|
2391
|
-
nextContent[partIndex] = {
|
|
2392
|
-
...part,
|
|
2393
|
-
providerOptions: mergeProviderOptions({
|
|
2394
|
-
current: partProviderOptions,
|
|
2395
|
-
patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS
|
|
2396
|
-
})
|
|
2397
|
-
};
|
|
2398
|
-
nextMessages[messageIndex] = {
|
|
2399
|
-
...message,
|
|
2400
|
-
content: nextContent
|
|
2401
|
-
};
|
|
2402
|
-
continue;
|
|
2403
|
-
}
|
|
2404
|
-
}
|
|
2405
|
-
}
|
|
2406
|
-
const messageProviderOptions = message.providerOptions;
|
|
2407
|
-
nextMessages[messageIndex] = {
|
|
2408
|
-
...message,
|
|
2409
|
-
providerOptions: mergeProviderOptions({
|
|
2410
|
-
current: messageProviderOptions,
|
|
2411
|
-
patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS
|
|
2412
|
-
})
|
|
2413
|
-
};
|
|
2414
|
-
}
|
|
2415
|
-
return nextMessages;
|
|
2416
|
-
}
|
|
2417
|
-
function applyPromptCachingToModelRequest(opts) {
|
|
2418
|
-
const provider = getGatewayProvider(opts.model);
|
|
2419
|
-
const providerOptions = {};
|
|
2420
|
-
if ((provider === "openai" || provider === "azure") && opts.openai?.setPromptCacheKey !== false) {
|
|
2421
|
-
providerOptions.openai = { promptCacheKey: opts.sessionId };
|
|
2422
|
-
}
|
|
2423
|
-
const maxBreakpointsPerRequest = opts.anthropic?.maxBreakpointsPerRequest ?? ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST;
|
|
2424
|
-
const cachedMessages = isClaudeLikeModel(opts.model) ? applyClaudePromptCaching({
|
|
2425
|
-
messages: opts.messages,
|
|
2426
|
-
providerId: provider,
|
|
2427
|
-
maxBreakpointsPerRequest
|
|
2428
|
-
}) : opts.messages;
|
|
2429
|
-
return {
|
|
2430
|
-
messages: cachedMessages,
|
|
2431
|
-
providerOptions
|
|
2432
|
-
};
|
|
2433
|
-
}
|
|
2434
|
-
|
|
2435
|
-
// src/session/ai-loop/stream-text.ts
|
|
2436
|
-
var INTERRUPT_POLL_INTERVAL_MS = 250;
|
|
2437
|
-
function pollForInterrupt({
|
|
2438
|
-
agent: agent2,
|
|
2439
|
-
messageId,
|
|
2440
|
-
signal,
|
|
2441
|
-
abortController
|
|
2442
|
-
}) {
|
|
2443
|
-
let resolveFirstCheck;
|
|
2444
|
-
const firstCheck = new Promise((resolve2) => {
|
|
2445
|
-
resolveFirstCheck = resolve2;
|
|
2446
|
-
});
|
|
2447
|
-
const interruptedLastPartRef = {
|
|
2448
|
-
current: null
|
|
2449
|
-
};
|
|
2450
|
-
const log2 = createLogger({
|
|
2451
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
2452
|
-
subsystem: "ai-loop"
|
|
2453
|
-
});
|
|
2454
|
-
const poll = async () => {
|
|
2455
|
-
let isFirst = true;
|
|
2456
|
-
while (!signal.aborted) {
|
|
2457
|
-
const message = await agent2.storage.message.get(messageId);
|
|
2458
|
-
if (!message) {
|
|
2459
|
-
if (isFirst) {
|
|
2460
|
-
resolveFirstCheck();
|
|
2461
|
-
return;
|
|
2462
|
-
}
|
|
2463
|
-
throw new Error(`Message ${messageId} not found`);
|
|
2464
|
-
}
|
|
2465
|
-
if (message.interruptedAt !== null) {
|
|
2466
|
-
log2.info("interrupt detected", { messageId });
|
|
2467
|
-
interruptedLastPartRef.current = message.interruptedLastPart;
|
|
2468
|
-
abortController.abort();
|
|
2469
|
-
if (isFirst) {
|
|
2470
|
-
resolveFirstCheck();
|
|
2471
|
-
}
|
|
2472
|
-
return;
|
|
2473
|
-
}
|
|
2474
|
-
if (isFirst) {
|
|
2475
|
-
isFirst = false;
|
|
2476
|
-
resolveFirstCheck();
|
|
2477
|
-
}
|
|
2478
|
-
await new Promise(
|
|
2479
|
-
(resolve2) => setTimeout(resolve2, INTERRUPT_POLL_INTERVAL_MS)
|
|
2480
|
-
);
|
|
2481
|
-
}
|
|
2482
|
-
};
|
|
2483
|
-
poll();
|
|
2484
|
-
return { firstCheck, interruptedLastPartRef };
|
|
2485
|
-
}
|
|
2486
|
-
async function streamText({
|
|
2487
|
-
stepThroughput,
|
|
2488
|
-
assistantMessageId,
|
|
2489
|
-
agent: agent2,
|
|
2490
|
-
sessionId,
|
|
2491
|
-
context,
|
|
2492
|
-
state,
|
|
2493
|
-
createdAt,
|
|
2494
|
-
writable,
|
|
2495
|
-
lastPartIndex,
|
|
2496
|
-
stepIndexOffset,
|
|
2497
|
-
discoveredSkills
|
|
2498
|
-
}) {
|
|
2499
|
-
const log2 = createLogger({
|
|
2500
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
2501
|
-
subsystem: "ai-loop"
|
|
2502
|
-
});
|
|
2503
|
-
const stepLog = log2.withContext({
|
|
2504
|
-
sessionId,
|
|
2505
|
-
messageId: assistantMessageId
|
|
2506
|
-
});
|
|
2507
|
-
const doneStep = stepLog.time("streamText");
|
|
2508
|
-
const throttledEmitStatus = createThrottledEmitStatus(agent2);
|
|
2509
|
-
const abortController = new AbortController();
|
|
2510
|
-
const pollController = new AbortController();
|
|
2511
|
-
const { firstCheck, interruptedLastPartRef } = pollForInterrupt({
|
|
2512
|
-
agent: agent2,
|
|
2513
|
-
messageId: assistantMessageId,
|
|
2514
|
-
signal: pollController.signal,
|
|
2515
|
-
abortController
|
|
2516
|
-
});
|
|
2517
|
-
const doneLoad = stepLog.time("load session + messages + parts + sandbox");
|
|
2518
|
-
const [, messagesResult, partsResult, { sandbox, sandboxRecord }] = await Promise.all([
|
|
2519
|
-
firstCheck,
|
|
2520
|
-
agent2.storage.message.listBySession(sessionId),
|
|
2521
|
-
agent2.storage.part.listBySession(sessionId),
|
|
2522
|
-
agent2.storage.session.get(sessionId).then(async (session) => {
|
|
2523
|
-
if (!session) {
|
|
2524
|
-
throw new SessionNotFoundError({ id: sessionId });
|
|
2525
|
-
}
|
|
2526
|
-
const sandboxId = session.sandboxId ?? session.id;
|
|
2527
|
-
const sandbox2 = agent2.sandbox(sandboxId);
|
|
2528
|
-
const sandboxRecord2 = await agent2.storage.sandbox.get(sandboxId);
|
|
2529
|
-
return { sandbox: sandbox2, sandboxRecord: sandboxRecord2 };
|
|
2530
|
-
})
|
|
2531
|
-
]);
|
|
2532
|
-
doneLoad();
|
|
2533
|
-
if (abortController.signal.aborted) {
|
|
2534
|
-
pollController.abort();
|
|
2535
|
-
return {
|
|
2536
|
-
finishReason: "stop",
|
|
2537
|
-
lastPartIndex,
|
|
2538
|
-
usageSteps: [],
|
|
2539
|
-
pendingApprovals: [],
|
|
2540
|
-
maxSteps: void 0,
|
|
2541
|
-
discoveredSkills,
|
|
2542
|
-
context
|
|
2543
|
-
};
|
|
2544
|
-
}
|
|
2545
|
-
const setStartedPromise = lastPartIndex === 0 ? (async () => {
|
|
2546
|
-
const now = Date.now();
|
|
2547
|
-
const pendingMessages = messagesResult.filter(
|
|
2548
|
-
(m) => m.createdAt <= createdAt && m.startedAt === null
|
|
2549
|
-
);
|
|
2550
|
-
if (pendingMessages.length > 0) {
|
|
2551
|
-
return await Promise.all(
|
|
2552
|
-
pendingMessages.map(
|
|
2553
|
-
(m) => agent2.storage.message.update(m.id, {
|
|
2554
|
-
startedAt: now,
|
|
2555
|
-
completedAt: m.role === "assistant" ? null : now
|
|
2556
|
-
})
|
|
2557
|
-
)
|
|
2558
|
-
);
|
|
2559
|
-
}
|
|
2560
|
-
})() : null;
|
|
2561
|
-
const allParts = [...partsResult];
|
|
2562
|
-
const nextPartIndexRef = {
|
|
2563
|
-
current: Math.max(
|
|
2564
|
-
lastPartIndex,
|
|
2565
|
-
...allParts.filter((p) => p.messageId === assistantMessageId).map((p) => p.index + 1),
|
|
2566
|
-
0
|
|
2567
|
-
)
|
|
2568
|
-
};
|
|
2569
|
-
const skillsRef = {
|
|
2570
|
-
current: discoveredSkills ?? []
|
|
2571
|
-
};
|
|
2572
|
-
let uiMessages = assembleUIMessages({
|
|
2573
|
-
messages: messagesResult,
|
|
2574
|
-
parts: allParts,
|
|
2575
|
-
until: createdAt,
|
|
2576
|
-
includeQueued: true,
|
|
2577
|
-
excludeSubToolParts: true
|
|
2578
|
-
});
|
|
2579
|
-
const resolvableArgs = {
|
|
2580
|
-
context,
|
|
2581
|
-
state,
|
|
2582
|
-
sessionId,
|
|
2583
|
-
sandbox,
|
|
2584
|
-
get messages() {
|
|
2585
|
-
return uiMessages;
|
|
2586
|
-
}
|
|
2587
|
-
};
|
|
2588
|
-
if (lastPartIndex === 0 && !discoveredSkills) {
|
|
2589
|
-
const resolvedSkills = await resolveOption(
|
|
2590
|
-
agent2.options.skills,
|
|
2591
|
-
resolvableArgs
|
|
2592
|
-
);
|
|
2593
|
-
const skillsDirs = resolvedSkills?.length ? await resolveSkillInputsToSandboxDirs({
|
|
2594
|
-
agent: agent2,
|
|
2595
|
-
sandbox,
|
|
2596
|
-
skills: resolvedSkills,
|
|
2597
|
-
sessionId
|
|
2598
|
-
}) : [];
|
|
2599
|
-
if (skillsDirs.length > 0) {
|
|
2600
|
-
const doneSkills = stepLog.time("discover skills", { skillsDirs });
|
|
2601
|
-
skillsRef.current = await discoverSkillsInSandbox({
|
|
2602
|
-
agent: agent2,
|
|
2603
|
-
sandbox,
|
|
2604
|
-
skillsDirs,
|
|
2605
|
-
sessionId
|
|
2606
|
-
});
|
|
2607
|
-
doneSkills({ count: skillsRef.current.length });
|
|
2608
|
-
}
|
|
2609
|
-
}
|
|
2610
|
-
const usageSteps = [];
|
|
2611
|
-
let internalStepIndex = 0;
|
|
2612
|
-
let loopFinishReason = "unknown";
|
|
2613
|
-
let wasAborted = false;
|
|
2614
|
-
const allPendingApprovals = [];
|
|
2615
|
-
let currentMaxSteps;
|
|
2616
|
-
try {
|
|
2617
|
-
for (let iteration = 0; iteration < stepThroughput; iteration++) {
|
|
2618
|
-
const resolvedModel = await resolveOption(
|
|
2619
|
-
agent2.options.model,
|
|
2620
|
-
resolvableArgs
|
|
2621
|
-
);
|
|
2622
|
-
const resolvedGeneration = await resolveOption(
|
|
2623
|
-
agent2.options.generation,
|
|
2624
|
-
resolvableArgs
|
|
2625
|
-
);
|
|
2626
|
-
const resolvedActiveTools = await resolveOption(
|
|
2627
|
-
agent2.options.activeTools,
|
|
2628
|
-
resolvableArgs
|
|
2629
|
-
);
|
|
2630
|
-
currentMaxSteps = resolvedGeneration?.maxSteps;
|
|
2631
|
-
if (!resolvedModel) {
|
|
2632
|
-
throw new Error("Agent model is not set");
|
|
2633
|
-
}
|
|
2634
|
-
const stepParts = [];
|
|
2635
|
-
let streamTextFinishReason;
|
|
2636
|
-
try {
|
|
2637
|
-
const stream2 = createUIMessageStream({
|
|
2638
|
-
execute: async ({ writer }) => {
|
|
2639
|
-
const { rawTools, toolsWithNeedsApproval, toolContext } = getTools({
|
|
2640
|
-
agent: agent2,
|
|
2641
|
-
sandbox,
|
|
2642
|
-
skillsRef,
|
|
2643
|
-
writer,
|
|
2644
|
-
abortController,
|
|
2645
|
-
nextPartIndexRef,
|
|
2646
|
-
assistantMessageId,
|
|
2647
|
-
sessionId,
|
|
2648
|
-
stepLog,
|
|
2649
|
-
context,
|
|
2650
|
-
state,
|
|
2651
|
-
messages: uiMessages,
|
|
2652
|
-
activeTools: resolvedActiveTools
|
|
2653
|
-
});
|
|
2654
|
-
if (iteration === 0) {
|
|
2655
|
-
await processApprovals({
|
|
2656
|
-
allParts,
|
|
2657
|
-
messagesResult,
|
|
2658
|
-
createdAt,
|
|
2659
|
-
rawTools,
|
|
2660
|
-
abortController,
|
|
2661
|
-
toolContext,
|
|
2662
|
-
agent: agent2,
|
|
2663
|
-
stepLog,
|
|
2664
|
-
throttledEmitStatus,
|
|
2665
|
-
writer
|
|
2666
|
-
});
|
|
2667
|
-
}
|
|
2668
|
-
throttledEmitStatus({
|
|
2669
|
-
writer,
|
|
2670
|
-
status: { type: "thinking" }
|
|
2671
|
-
});
|
|
2672
|
-
const iterResult = streamTextAi({
|
|
2673
|
-
messages: [{ role: "user", content: "." }],
|
|
2674
|
-
tools: toolsWithNeedsApproval,
|
|
2675
|
-
model: resolvedModel,
|
|
2676
|
-
abortSignal: abortController.signal,
|
|
2677
|
-
stopWhen: stepCountIs(1),
|
|
2678
|
-
temperature: resolvedGeneration?.temperature,
|
|
2679
|
-
topK: resolvedGeneration?.topK,
|
|
2680
|
-
topP: resolvedGeneration?.topP,
|
|
2681
|
-
frequencyPenalty: resolvedGeneration?.frequencyPenalty,
|
|
2682
|
-
presencePenalty: resolvedGeneration?.presencePenalty,
|
|
2683
|
-
maxOutputTokens: resolvedGeneration?.maxOutputTokens,
|
|
2684
|
-
headers: resolvedGeneration?.headers,
|
|
2685
|
-
experimental_context: {
|
|
2686
|
-
context,
|
|
2687
|
-
state,
|
|
2688
|
-
sessionId,
|
|
2689
|
-
sandbox,
|
|
2690
|
-
messages: uiMessages
|
|
2691
|
-
},
|
|
2692
|
-
prepareStep: async ({ model }) => {
|
|
2693
|
-
const skillsContext = buildSkillsContext(skillsRef.current);
|
|
2694
|
-
const cwdPrompt = sandboxRecord?.setup?.cwd ? `Your working directory is ${sandboxRecord.setup.cwd}. All file paths should be absolute.` : "All file paths must be absolute.";
|
|
2695
|
-
const resolvedSystem = await resolveSystem(
|
|
2696
|
-
agent2.options.system,
|
|
2697
|
-
resolvableArgs
|
|
2698
|
-
);
|
|
2699
|
-
const systemHeader = joinPromptSections(
|
|
2700
|
-
BASE_SYSTEM_PROMPT,
|
|
2701
|
-
cwdPrompt,
|
|
2702
|
-
resolvedSystem
|
|
2703
|
-
);
|
|
2704
|
-
const systemContext = joinPromptSections(
|
|
2705
|
-
skillsContext,
|
|
2706
|
-
cwdPrompt
|
|
2707
|
-
);
|
|
2708
|
-
const convertedMessages = await convertToModelMessages(
|
|
2709
|
-
uiMessages,
|
|
2710
|
-
{ ignoreIncompleteToolCalls: true }
|
|
2711
|
-
);
|
|
2712
|
-
const modelMessages = [
|
|
2713
|
-
...systemHeader.trim() ? [{ role: "system", content: systemHeader }] : [],
|
|
2714
|
-
...systemContext.trim() ? [{ role: "system", content: systemContext }] : [],
|
|
2715
|
-
...convertedMessages
|
|
2716
|
-
];
|
|
2717
|
-
const promptCaching = applyPromptCachingToModelRequest({
|
|
2718
|
-
model: typeof model === "string" ? model : model.modelId,
|
|
2719
|
-
sessionId,
|
|
2720
|
-
messages: modelMessages
|
|
2721
|
-
});
|
|
2722
|
-
let activeTools = resolvedActiveTools ?? void 0;
|
|
2723
|
-
if (skillsRef.current.length === 0 && activeTools) {
|
|
2724
|
-
activeTools = activeTools.filter(
|
|
2725
|
-
(t) => t !== "Skill"
|
|
2726
|
-
);
|
|
2727
|
-
}
|
|
2728
|
-
return {
|
|
2729
|
-
messages: promptCaching.messages,
|
|
2730
|
-
providerOptions: mergeProviderOptions({
|
|
2731
|
-
current: resolvedGeneration?.providerOptions,
|
|
2732
|
-
patch: promptCaching.providerOptions
|
|
2733
|
-
}),
|
|
2734
|
-
activeTools
|
|
2735
|
-
};
|
|
2736
|
-
},
|
|
2737
|
-
onStepFinish: ({ usage: usage2 }) => {
|
|
2738
|
-
if (usage2) {
|
|
2739
|
-
usageSteps.push({
|
|
2740
|
-
stepIndex: stepIndexOffset + internalStepIndex,
|
|
2741
|
-
model: resolvedModel ?? "unknown",
|
|
2742
|
-
inputTokens: usage2.inputTokens ?? 0,
|
|
2743
|
-
outputTokens: usage2.outputTokens ?? 0,
|
|
2744
|
-
totalTokens: usage2.totalTokens ?? 0,
|
|
2745
|
-
cacheReadTokens: usage2.inputTokenDetails?.cacheReadTokens ?? 0,
|
|
2746
|
-
cacheWriteTokens: usage2.inputTokenDetails?.cacheWriteTokens ?? 0,
|
|
2747
|
-
reasoningTokens: usage2.outputTokenDetails?.reasoningTokens ?? 0
|
|
2748
|
-
});
|
|
2749
|
-
}
|
|
2750
|
-
internalStepIndex++;
|
|
2751
|
-
}
|
|
2752
|
-
});
|
|
2753
|
-
writer.merge(
|
|
2754
|
-
iterResult.toUIMessageStream({
|
|
2755
|
-
generateMessageId: () => assistantMessageId,
|
|
2756
|
-
onFinish: ({ messages }) => {
|
|
2757
|
-
for (const m of messages) {
|
|
2758
|
-
if (m.role === "assistant") {
|
|
2759
|
-
stepParts.push(...m.parts);
|
|
2760
|
-
}
|
|
2761
|
-
}
|
|
2762
|
-
}
|
|
2763
|
-
})
|
|
2764
|
-
);
|
|
2765
|
-
streamTextFinishReason = await iterResult.finishReason;
|
|
2766
|
-
}
|
|
2767
|
-
});
|
|
2768
|
-
await stream2.pipeTo(writable, {
|
|
2769
|
-
preventClose: true,
|
|
2770
|
-
preventAbort: true
|
|
2771
|
-
});
|
|
2772
|
-
} catch (err) {
|
|
2773
|
-
if (abortController.signal.aborted) {
|
|
2774
|
-
wasAborted = true;
|
|
2775
|
-
} else {
|
|
2776
|
-
throw err;
|
|
2777
|
-
}
|
|
2778
|
-
}
|
|
2779
|
-
if (abortController.signal.aborted) {
|
|
2780
|
-
wasAborted = true;
|
|
2781
|
-
}
|
|
2782
|
-
if (wasAborted) {
|
|
2783
|
-
const terminalStates = /* @__PURE__ */ new Set([
|
|
2784
|
-
"output-available",
|
|
2785
|
-
"output-error",
|
|
2786
|
-
"output-denied",
|
|
2787
|
-
"done"
|
|
2788
|
-
]);
|
|
2789
|
-
for (const part of stepParts) {
|
|
2790
|
-
if ("type" in part && typeof part.type === "string" && part.type.startsWith("tool-") && "state" in part && !terminalStates.has(part.state)) {
|
|
2791
|
-
part.state = "output-error";
|
|
2792
|
-
part.errorText = "interrupted";
|
|
2793
|
-
}
|
|
2794
|
-
}
|
|
2795
|
-
}
|
|
2796
|
-
const lastPart = interruptedLastPartRef.current;
|
|
2797
|
-
const newPartRecords = stepParts.map((uiPart, i) => {
|
|
2798
|
-
const index = nextPartIndexRef.current + i;
|
|
2799
|
-
if (lastPart != null && index > lastPart.index) {
|
|
2800
|
-
return null;
|
|
2801
|
-
}
|
|
2802
|
-
const useClientPart = lastPart != null && index === lastPart.index;
|
|
2803
|
-
const content = useClientPart ? lastPart.part : uiPart;
|
|
2804
|
-
const partId = "approval" in content && content.approval?.id ? approvalPartId(content.approval.id) : `part_${assistantMessageId}_${index}`;
|
|
2805
|
-
return {
|
|
2806
|
-
id: partId,
|
|
2807
|
-
index,
|
|
2808
|
-
messageId: assistantMessageId,
|
|
2809
|
-
sessionId,
|
|
2810
|
-
part: content
|
|
2811
|
-
};
|
|
2812
|
-
}).filter((r) => r !== null);
|
|
2813
|
-
const persistPromise = Promise.all([
|
|
2814
|
-
...newPartRecords.map(
|
|
2815
|
-
(record) => agent2.storage.part.set(record.id, record)
|
|
2816
|
-
),
|
|
2817
|
-
agent2.storage.session.update(sessionId, { state })
|
|
2818
|
-
]);
|
|
2819
|
-
nextPartIndexRef.current = lastPart != null ? Math.min(
|
|
2820
|
-
nextPartIndexRef.current + stepParts.length,
|
|
2821
|
-
lastPart.index + 1
|
|
2822
|
-
) : nextPartIndexRef.current + stepParts.length;
|
|
2823
|
-
const iterPendingApprovals = stepParts.filter(
|
|
2824
|
-
(p) => "state" in p && p.state === "approval-requested" && "approval" in p && !!p.approval?.id
|
|
2825
|
-
).map((p) => ({
|
|
2826
|
-
approvalId: p.approval.id,
|
|
2827
|
-
toolName: "type" in p ? String(p.type).replace("tool-", "") : "unknown"
|
|
2828
|
-
}));
|
|
2829
|
-
loopFinishReason = wasAborted ? "stop" : streamTextFinishReason ?? "unknown";
|
|
2830
|
-
const shouldBreak = iterPendingApprovals.length > 0 || loopFinishReason === "stop" || wasAborted || lastPart != null || currentMaxSteps != null && stepIndexOffset + usageSteps.length >= currentMaxSteps || iteration >= stepThroughput - 1;
|
|
2831
|
-
if (shouldBreak) {
|
|
2832
|
-
await persistPromise;
|
|
2833
|
-
allParts.push(...newPartRecords);
|
|
2834
|
-
allPendingApprovals.push(...iterPendingApprovals);
|
|
2835
|
-
break;
|
|
2836
|
-
}
|
|
2837
|
-
await persistPromise;
|
|
2838
|
-
allParts.push(...newPartRecords);
|
|
2839
|
-
uiMessages = assembleUIMessages({
|
|
2840
|
-
messages: messagesResult,
|
|
2841
|
-
parts: allParts,
|
|
2842
|
-
until: createdAt,
|
|
2843
|
-
includeQueued: true,
|
|
2844
|
-
excludeSubToolParts: true
|
|
2845
|
-
});
|
|
2846
|
-
}
|
|
2847
|
-
} finally {
|
|
2848
|
-
pollController.abort();
|
|
2849
|
-
throttledEmitStatus.dispose();
|
|
2850
|
-
}
|
|
2851
|
-
if (setStartedPromise) {
|
|
2852
|
-
await setStartedPromise;
|
|
2853
|
-
}
|
|
2854
|
-
doneStep({
|
|
2855
|
-
finishReason: loopFinishReason,
|
|
2856
|
-
wasAborted,
|
|
2857
|
-
iterations: internalStepIndex,
|
|
2858
|
-
pendingApprovals: allPendingApprovals.length
|
|
2859
|
-
});
|
|
2860
|
-
return {
|
|
2861
|
-
finishReason: loopFinishReason,
|
|
2862
|
-
lastPartIndex: nextPartIndexRef.current,
|
|
2863
|
-
usageSteps,
|
|
2864
|
-
pendingApprovals: allPendingApprovals,
|
|
2865
|
-
maxSteps: currentMaxSteps,
|
|
2866
|
-
discoveredSkills: skillsRef.current,
|
|
2867
|
-
context
|
|
2868
|
-
};
|
|
2869
|
-
}
|
|
2870
|
-
async function streamTextStep(...args) {
|
|
2871
|
-
"use step";
|
|
2872
|
-
return await streamText(...args);
|
|
2873
|
-
}
|
|
2874
|
-
async function processApprovals({
|
|
2875
|
-
allParts,
|
|
2876
|
-
messagesResult,
|
|
2877
|
-
createdAt,
|
|
2878
|
-
rawTools,
|
|
2879
|
-
abortController,
|
|
2880
|
-
toolContext,
|
|
2881
|
-
agent: agent2,
|
|
2882
|
-
stepLog,
|
|
2883
|
-
throttledEmitStatus,
|
|
2884
|
-
writer
|
|
2885
|
-
}) {
|
|
2886
|
-
const approvedParts = allParts.filter(
|
|
2887
|
-
(p) => p.part && "state" in p.part && p.part.state === "approval-responded" && "approval" in p.part && p.part.approval?.approved === true
|
|
2888
|
-
);
|
|
2889
|
-
if (approvedParts.length > 0) {
|
|
2890
|
-
stepLog.info("executing approved tools", {
|
|
2891
|
-
count: approvedParts.length
|
|
2892
|
-
});
|
|
2893
|
-
throttledEmitStatus({
|
|
2894
|
-
writer,
|
|
2895
|
-
status: { type: "processing-approvals" }
|
|
2896
|
-
});
|
|
2897
|
-
const preExecMessages = await convertToModelMessages(
|
|
2898
|
-
assembleUIMessages({
|
|
2899
|
-
messages: messagesResult,
|
|
2900
|
-
parts: allParts,
|
|
2901
|
-
until: createdAt,
|
|
2902
|
-
includeQueued: true,
|
|
2903
|
-
excludeSubToolParts: true
|
|
2904
|
-
}),
|
|
2905
|
-
{ ignoreIncompleteToolCalls: true }
|
|
2906
|
-
);
|
|
2907
|
-
await Promise.all(
|
|
2908
|
-
approvedParts.map(async (ap) => {
|
|
2909
|
-
if (!ap.part.type.startsWith("tool-")) {
|
|
2910
|
-
return;
|
|
2911
|
-
}
|
|
2912
|
-
const part = ap.part;
|
|
2913
|
-
const toolName = part.type.replace("tool-", "");
|
|
2914
|
-
const toolDef = rawTools[toolName];
|
|
2915
|
-
if (toolDef?.execute && part.input !== void 0) {
|
|
2916
|
-
try {
|
|
2917
|
-
const toolOutput = await toolDef.execute(part.input, {
|
|
2918
|
-
toolCallId: part.toolCallId,
|
|
2919
|
-
messages: preExecMessages,
|
|
2920
|
-
abortSignal: abortController.signal,
|
|
2921
|
-
experimental_context: toolContext
|
|
2922
|
-
});
|
|
2923
|
-
part.state = "output-available";
|
|
2924
|
-
part.output = toolOutput;
|
|
2925
|
-
writer.write({
|
|
2926
|
-
type: "tool-output-available",
|
|
2927
|
-
toolCallId: part.toolCallId,
|
|
2928
|
-
output: toolOutput
|
|
2929
|
-
});
|
|
2930
|
-
} catch (err) {
|
|
2931
|
-
part.state = "output-error";
|
|
2932
|
-
part.errorText = err instanceof Error ? err.message : String(err);
|
|
2933
|
-
writer.write({
|
|
2934
|
-
type: "tool-output-error",
|
|
2935
|
-
toolCallId: part.toolCallId,
|
|
2936
|
-
errorText: part.errorText
|
|
2937
|
-
});
|
|
2938
|
-
}
|
|
2939
|
-
await agent2.storage.part.set(ap.id, { ...ap, part });
|
|
2940
|
-
}
|
|
2941
|
-
})
|
|
2942
|
-
);
|
|
2943
|
-
}
|
|
2944
|
-
const pendingApprovalParts = allParts.filter(
|
|
2945
|
-
(p) => p.part.state === "approval-requested"
|
|
2946
|
-
);
|
|
2947
|
-
if (pendingApprovalParts.length > 0) {
|
|
2948
|
-
await Promise.all(
|
|
2949
|
-
pendingApprovalParts.map(async (p) => {
|
|
2950
|
-
const denied = {
|
|
2951
|
-
...p,
|
|
2952
|
-
part: {
|
|
2953
|
-
...p.part,
|
|
2954
|
-
state: "output-denied",
|
|
2955
|
-
approval: {
|
|
2956
|
-
...p.part.approval ?? {},
|
|
2957
|
-
approved: false,
|
|
2958
|
-
reason: "auto-denied: new message sent"
|
|
2959
|
-
}
|
|
2960
|
-
}
|
|
2961
|
-
};
|
|
2962
|
-
Object.assign(p, denied);
|
|
2963
|
-
await agent2.storage.part.set(p.id, denied);
|
|
2964
|
-
})
|
|
2965
|
-
);
|
|
2966
|
-
}
|
|
2967
|
-
}
|
|
2968
|
-
function emitStatus({
|
|
2969
|
-
writer,
|
|
2970
|
-
status,
|
|
2971
|
-
agent: agent2
|
|
2972
|
-
}) {
|
|
2973
|
-
writer.write({ type: "data-status", data: status, transient: true });
|
|
2974
|
-
agent2.hooks?.status?.(status);
|
|
2975
|
-
}
|
|
2976
|
-
var STATUS_THROTTLE_MS = 500;
|
|
2977
|
-
function createThrottledEmitStatus(agent2) {
|
|
2978
|
-
let timer = null;
|
|
2979
|
-
let pending = null;
|
|
2980
|
-
function throttled({
|
|
2981
|
-
writer,
|
|
2982
|
-
status
|
|
2983
|
-
}) {
|
|
2984
|
-
pending = { writer, status };
|
|
2985
|
-
if (timer !== null) {
|
|
2986
|
-
return;
|
|
2987
|
-
}
|
|
2988
|
-
flush();
|
|
2989
|
-
timer = setTimeout(() => {
|
|
2990
|
-
timer = null;
|
|
2991
|
-
if (pending) {
|
|
2992
|
-
flush();
|
|
2993
|
-
}
|
|
2994
|
-
}, STATUS_THROTTLE_MS);
|
|
2995
|
-
}
|
|
2996
|
-
function flush() {
|
|
2997
|
-
if (!pending) {
|
|
2998
|
-
return;
|
|
2999
|
-
}
|
|
3000
|
-
const { writer, status } = pending;
|
|
3001
|
-
pending = null;
|
|
3002
|
-
emitStatus({ writer, status, agent: agent2 });
|
|
3003
|
-
}
|
|
3004
|
-
throttled.dispose = () => {
|
|
3005
|
-
if (timer !== null) {
|
|
3006
|
-
clearTimeout(timer);
|
|
3007
|
-
timer = null;
|
|
3008
|
-
}
|
|
3009
|
-
flush();
|
|
3010
|
-
};
|
|
3011
|
-
return throttled;
|
|
3012
|
-
}
|
|
3013
|
-
async function resolveOption(value, args) {
|
|
3014
|
-
if (value === void 0) {
|
|
3015
|
-
return void 0;
|
|
3016
|
-
}
|
|
3017
|
-
if (typeof value === "function") {
|
|
3018
|
-
return await value(args);
|
|
3019
|
-
}
|
|
3020
|
-
return value;
|
|
3021
|
-
}
|
|
3022
|
-
async function resolveSystem(input, args) {
|
|
3023
|
-
const result = await resolveOption(input, args);
|
|
3024
|
-
if (!result) {
|
|
3025
|
-
return null;
|
|
3026
|
-
}
|
|
3027
|
-
return Array.isArray(result) ? result.filter(Boolean).join("\n") : result;
|
|
3028
|
-
}
|
|
3029
|
-
var BASE_SYSTEM_PROMPT = "You are an AI assistant with basic tools to interact with your environment. Explore and work freely.";
|
|
3030
|
-
function joinPromptSections(...sections) {
|
|
3031
|
-
return sections.filter((s) => s?.trim()).join("\n\n");
|
|
3032
|
-
}
|
|
3033
|
-
var backgroundProcessPrompt = `## Background Processes
|
|
3034
|
-
These background process instructions are for you to manipulate the processes, do not be to verbose to the user about the response details like "how to debug the process output" the user will have an UI.
|
|
3035
|
-
|
|
3036
|
-
Use \`waitUntil: 0\` for dev servers that should run indefinitely.
|
|
3037
|
-
It is a good practice to check the output log after running processes like dev servers to make sure they started correctly.
|
|
3038
|
-
|
|
3039
|
-
To run a background process:
|
|
3040
|
-
Bash({ command: "npm run dev", waitUntil: 0 })
|
|
3041
|
-
Returns immediately with \`commandId\`. The process keeps running in the sandbox.
|
|
3042
|
-
|
|
3043
|
-
To kill a process:
|
|
3044
|
-
Bash({ command: "ps aux | grep node" }) // Find the PID
|
|
3045
|
-
Bash({ command: "kill {pid}" }) // Graceful
|
|
3046
|
-
Bash({ command: "kill -9 {pid}" }) // Force
|
|
3047
|
-
`;
|
|
3048
|
-
function buildSkillsContext(skills) {
|
|
3049
|
-
if (skills.length === 0) {
|
|
3050
|
-
return "";
|
|
3051
|
-
}
|
|
3052
|
-
const skillLines = skills.map((s) => `- ${s.name}: ${s.description}`).join("\n");
|
|
3053
|
-
return `## Available Skills
|
|
3054
|
-
${skillLines}
|
|
3055
|
-
|
|
3056
|
-
${backgroundProcessPrompt}
|
|
3057
|
-
|
|
3058
|
-
Use the Skill tool to load a skill's full instructions before following it.`;
|
|
3059
|
-
}
|
|
3060
|
-
|
|
3061
|
-
// src/session/send.ts
|
|
3062
|
-
var DEFAULT_STEP_THROUGHPUT = 100;
|
|
3063
|
-
var ACTIVE_RUNS_KEY = /* @__PURE__ */ Symbol.for("experimental-agent:activeRuns");
|
|
3064
|
-
function getActiveRuns() {
|
|
3065
|
-
const g = globalThis;
|
|
3066
|
-
if (!g[ACTIVE_RUNS_KEY]) {
|
|
3067
|
-
g[ACTIVE_RUNS_KEY] = /* @__PURE__ */ new Map();
|
|
3068
|
-
}
|
|
3069
|
-
return g[ACTIVE_RUNS_KEY];
|
|
3070
|
-
}
|
|
3071
|
-
async function send(sessionContext, input, opts) {
|
|
3072
|
-
if (opts?.interruptIfStreaming) {
|
|
3073
|
-
const lastPart = typeof opts.interruptIfStreaming === "object" ? opts.interruptIfStreaming.lastPart : void 0;
|
|
3074
|
-
await interrupt(sessionContext, { lastPart });
|
|
3075
|
-
}
|
|
3076
|
-
const {
|
|
3077
|
-
workflowRunId,
|
|
3078
|
-
workflowWritable,
|
|
3079
|
-
existingPartCount,
|
|
3080
|
-
assistantMessageId,
|
|
3081
|
-
assistantMessageCreatedAt,
|
|
3082
|
-
state
|
|
3083
|
-
} = await errore2.tryAsync({
|
|
3084
|
-
try: () => initSessionStep({
|
|
3085
|
-
sessionContext,
|
|
3086
|
-
sessionId: sessionContext.sessionId,
|
|
3087
|
-
sandboxId: opts?.sandboxId,
|
|
3088
|
-
initialState: opts?.initialState,
|
|
3089
|
-
input
|
|
3090
|
-
}),
|
|
3091
|
-
catch: (e) => {
|
|
3092
|
-
if (e instanceof Error) {
|
|
3093
|
-
throw e;
|
|
3094
|
-
}
|
|
3095
|
-
throw new SendError({
|
|
3096
|
-
id: sessionContext.sessionId,
|
|
3097
|
-
reason: String(e),
|
|
3098
|
-
cause: e
|
|
3099
|
-
});
|
|
3100
|
-
}
|
|
3101
|
-
});
|
|
3102
|
-
const loopOpts = {
|
|
3103
|
-
sessionContext,
|
|
3104
|
-
assistantMessageId,
|
|
3105
|
-
createdAt: assistantMessageCreatedAt,
|
|
3106
|
-
context: opts?.context ?? {},
|
|
3107
|
-
state,
|
|
3108
|
-
lastPartIndex: existingPartCount,
|
|
3109
|
-
workflowRunId
|
|
3110
|
-
};
|
|
3111
|
-
if (workflowWritable) {
|
|
3112
|
-
await runLoop({ ...loopOpts, writable: workflowWritable });
|
|
3113
|
-
return { assistantMessageId, done: true };
|
|
3114
|
-
}
|
|
3115
|
-
const chunks = [];
|
|
3116
|
-
let notifyChunk;
|
|
3117
|
-
let onChunk = new Promise((r) => {
|
|
3118
|
-
notifyChunk = r;
|
|
3119
|
-
});
|
|
3120
|
-
const writable = new WritableStream({
|
|
3121
|
-
write(chunk) {
|
|
3122
|
-
chunks.push(chunk);
|
|
3123
|
-
const prev = notifyChunk;
|
|
3124
|
-
onChunk = new Promise((r) => {
|
|
3125
|
-
notifyChunk = r;
|
|
3126
|
-
});
|
|
3127
|
-
prev();
|
|
3128
|
-
}
|
|
3129
|
-
});
|
|
3130
|
-
const done = runLoop({ ...loopOpts, writable });
|
|
3131
|
-
const activeRun = {
|
|
3132
|
-
chunks,
|
|
3133
|
-
writable,
|
|
3134
|
-
done,
|
|
3135
|
-
get onChunk() {
|
|
3136
|
-
return onChunk;
|
|
3137
|
-
}
|
|
3138
|
-
};
|
|
3139
|
-
const activeRuns = getActiveRuns();
|
|
3140
|
-
activeRuns.set(assistantMessageId, activeRun);
|
|
3141
|
-
done.finally(() => activeRuns.delete(assistantMessageId));
|
|
3142
|
-
return { assistantMessageId, done: done.then(() => true) };
|
|
3143
|
-
}
|
|
3144
|
-
async function runLoop({
|
|
3145
|
-
workflowRunId,
|
|
3146
|
-
sessionContext,
|
|
3147
|
-
assistantMessageId,
|
|
3148
|
-
createdAt,
|
|
3149
|
-
context: initialContext,
|
|
3150
|
-
state,
|
|
3151
|
-
writable,
|
|
3152
|
-
lastPartIndex: initialPartIndex
|
|
3153
|
-
}) {
|
|
3154
|
-
const agent2 = sessionContext.agent;
|
|
3155
|
-
const log2 = createLogger({
|
|
3156
|
-
config: { ...agent2.options.logging, name: agent2.name },
|
|
3157
|
-
subsystem: "ai-loop"
|
|
3158
|
-
});
|
|
3159
|
-
const msgLog = log2.withContext({
|
|
3160
|
-
sessionId: sessionContext.sessionId,
|
|
3161
|
-
messageId: assistantMessageId
|
|
3162
|
-
});
|
|
3163
|
-
const doneMessage = msgLog.time("onMessage");
|
|
3164
|
-
let finishReason;
|
|
3165
|
-
let lastPartIndex = initialPartIndex;
|
|
3166
|
-
const usageSteps = [];
|
|
3167
|
-
let discoveredSkills = null;
|
|
3168
|
-
let context = initialContext;
|
|
3169
|
-
try {
|
|
3170
|
-
while (finishReason !== "stop") {
|
|
3171
|
-
const streamTextFn = workflowRunId ? streamTextStep : streamText;
|
|
3172
|
-
const throughput = workflowRunId ? 5 : DEFAULT_STEP_THROUGHPUT;
|
|
3173
|
-
const result = await streamTextFn({
|
|
3174
|
-
stepThroughput: throughput,
|
|
3175
|
-
assistantMessageId,
|
|
3176
|
-
agent: agent2,
|
|
3177
|
-
sessionId: sessionContext.sessionId,
|
|
3178
|
-
context,
|
|
3179
|
-
state,
|
|
3180
|
-
createdAt,
|
|
3181
|
-
writable,
|
|
3182
|
-
lastPartIndex,
|
|
3183
|
-
stepIndexOffset: usageSteps.length,
|
|
3184
|
-
discoveredSkills
|
|
3185
|
-
});
|
|
3186
|
-
finishReason = result.finishReason;
|
|
3187
|
-
lastPartIndex = result.lastPartIndex;
|
|
3188
|
-
usageSteps.push(...result.usageSteps);
|
|
3189
|
-
discoveredSkills = result.discoveredSkills;
|
|
3190
|
-
context = result.context;
|
|
3191
|
-
if (result.maxSteps != null && usageSteps.length >= result.maxSteps) {
|
|
3192
|
-
msgLog.info("reached maxSteps limit", { maxSteps: result.maxSteps });
|
|
3193
|
-
break;
|
|
3194
|
-
}
|
|
3195
|
-
if (result.pendingApprovals.length > 0) {
|
|
3196
|
-
msgLog.info("pending approvals, stopping loop", {
|
|
3197
|
-
count: result.pendingApprovals.length,
|
|
3198
|
-
tools: result.pendingApprovals.map((p) => p.toolName)
|
|
3199
|
-
});
|
|
3200
|
-
break;
|
|
3201
|
-
}
|
|
3202
|
-
}
|
|
3203
|
-
doneMessage({ totalSteps: usageSteps.length });
|
|
3204
|
-
} catch (err) {
|
|
3205
|
-
msgLog.error("AI loop failed", {
|
|
3206
|
-
error: err instanceof Error ? err.message : String(err)
|
|
3207
|
-
});
|
|
3208
|
-
throw err;
|
|
3209
|
-
} finally {
|
|
3210
|
-
await completeMessageStep({
|
|
3211
|
-
assistantMessageId,
|
|
3212
|
-
sessionContext,
|
|
3213
|
-
writable,
|
|
3214
|
-
usageSteps
|
|
3215
|
-
});
|
|
3216
|
-
}
|
|
3217
|
-
}
|
|
3218
|
-
async function initSessionStep({
|
|
3219
|
-
sessionContext,
|
|
3220
|
-
sessionId,
|
|
3221
|
-
sandboxId,
|
|
3222
|
-
initialState,
|
|
3223
|
-
input
|
|
3224
|
-
}) {
|
|
3225
|
-
"use step";
|
|
3226
|
-
let workflowRunId;
|
|
3227
|
-
let workflowWritable;
|
|
3228
|
-
try {
|
|
3229
|
-
const { getWorkflowMetadata, getWritable } = await import("workflow");
|
|
3230
|
-
workflowRunId = getWorkflowMetadata().workflowRunId;
|
|
3231
|
-
workflowWritable = getWritable();
|
|
3232
|
-
} catch {
|
|
3233
|
-
}
|
|
3234
|
-
const newMessages = [];
|
|
3235
|
-
const newParts = [];
|
|
3236
|
-
const newApprovals = [];
|
|
3237
|
-
if (Array.isArray(input)) {
|
|
3238
|
-
for (const i of input) {
|
|
3239
|
-
const { message, parts, approvals } = toMessageAndParts({
|
|
3240
|
-
sessionId: sessionContext.sessionId,
|
|
3241
|
-
input: i,
|
|
3242
|
-
defaultRole: "user"
|
|
3243
|
-
});
|
|
3244
|
-
if (message) {
|
|
3245
|
-
newMessages.push(message);
|
|
3246
|
-
}
|
|
3247
|
-
newParts.push(...parts);
|
|
3248
|
-
newApprovals.push(...approvals);
|
|
3249
|
-
}
|
|
3250
|
-
} else {
|
|
3251
|
-
const { message, parts, approvals } = toMessageAndParts({
|
|
3252
|
-
input,
|
|
3253
|
-
sessionId: sessionContext.sessionId,
|
|
3254
|
-
defaultRole: "user"
|
|
3255
|
-
});
|
|
3256
|
-
if (message) {
|
|
3257
|
-
newMessages.push(message);
|
|
3258
|
-
}
|
|
3259
|
-
newParts.push(...parts);
|
|
3260
|
-
newApprovals.push(...approvals);
|
|
3261
|
-
}
|
|
3262
|
-
const lastNewMessage = newMessages.at(-1);
|
|
3263
|
-
const reuseAssistant = lastNewMessage?.role === "assistant";
|
|
3264
|
-
const assistantMessageId = reuseAssistant ? lastNewMessage.id : `message_${ulid3()}`;
|
|
3265
|
-
const baseTime = Date.now();
|
|
3266
|
-
const assistantMessageCreatedAt = baseTime + newMessages.length;
|
|
3267
|
-
if (!reuseAssistant) {
|
|
3268
|
-
newMessages.push({
|
|
3269
|
-
id: assistantMessageId,
|
|
3270
|
-
sessionId: sessionContext.sessionId,
|
|
3271
|
-
role: "assistant",
|
|
3272
|
-
createdAt: assistantMessageCreatedAt,
|
|
3273
|
-
startedAt: null,
|
|
3274
|
-
completedAt: null,
|
|
3275
|
-
interruptedAt: null,
|
|
3276
|
-
interruptedLastPart: null,
|
|
3277
|
-
usage: null,
|
|
3278
|
-
workflowRunId: null,
|
|
3279
|
-
metadata: null
|
|
3280
|
-
});
|
|
3281
|
-
}
|
|
3282
|
-
const existingPartCount = reuseAssistant ? newParts.filter((p) => p.messageId === assistantMessageId).length : 0;
|
|
3283
|
-
let session = await sessionContext.agent.storage.session.get(sessionId);
|
|
3284
|
-
const now = Date.now();
|
|
3285
|
-
if (!session) {
|
|
3286
|
-
session = {
|
|
3287
|
-
id: sessionId,
|
|
3288
|
-
sandboxId: sandboxId ?? null,
|
|
3289
|
-
lastMessageId: null,
|
|
3290
|
-
createdAt: now,
|
|
3291
|
-
updatedAt: now
|
|
3292
|
-
};
|
|
3293
|
-
await sessionContext.agent.storage.session.set(session.id, session);
|
|
3294
|
-
} else if (sandboxId && sandboxId !== session.sandboxId) {
|
|
3295
|
-
session = { ...session, sandboxId, updatedAt: now };
|
|
3296
|
-
await sessionContext.agent.storage.session.update(session.id, {
|
|
3297
|
-
sandboxId
|
|
3298
|
-
});
|
|
3299
|
-
}
|
|
3300
|
-
await all({
|
|
3301
|
-
async saveMessages() {
|
|
3302
|
-
await Promise.all(
|
|
3303
|
-
newMessages.map(async (m, i) => {
|
|
3304
|
-
const existing = await sessionContext.agent.storage.message.get(m.id);
|
|
3305
|
-
if (existing) {
|
|
3306
|
-
return;
|
|
3307
|
-
}
|
|
3308
|
-
await sessionContext.agent.storage.message.set(m.id, {
|
|
3309
|
-
...m,
|
|
3310
|
-
createdAt: baseTime + i,
|
|
3311
|
-
workflowRunId: m.id === assistantMessageId ? workflowRunId ?? null : m.workflowRunId
|
|
3312
|
-
});
|
|
3313
|
-
})
|
|
3314
|
-
);
|
|
3315
|
-
},
|
|
3316
|
-
async saveParts() {
|
|
3317
|
-
await Promise.all(
|
|
3318
|
-
newParts.map(
|
|
3319
|
-
async (p) => sessionContext.agent.storage.part.set(p.id, p)
|
|
3320
|
-
)
|
|
3321
|
-
);
|
|
3322
|
-
},
|
|
3323
|
-
async resolveApprovals() {
|
|
3324
|
-
await Promise.all(
|
|
3325
|
-
newApprovals.map(async (a) => {
|
|
3326
|
-
const partId = approvalPartId(a.approvalId);
|
|
3327
|
-
const part = await sessionContext.agent.storage.part.get(partId);
|
|
3328
|
-
if (!part) {
|
|
3329
|
-
throw new ApprovalNotFoundError({
|
|
3330
|
-
id: sessionContext.sessionId,
|
|
3331
|
-
approvalId: a.approvalId
|
|
3332
|
-
});
|
|
3333
|
-
}
|
|
3334
|
-
await sessionContext.agent.storage.part.set(partId, {
|
|
3335
|
-
...part,
|
|
3336
|
-
part: {
|
|
3337
|
-
...part.part,
|
|
3338
|
-
state: a.approved ? "approval-responded" : "output-denied",
|
|
3339
|
-
approval: {
|
|
3340
|
-
id: a.approvalId,
|
|
3341
|
-
approved: a.approved,
|
|
3342
|
-
reason: a.reason
|
|
3343
|
-
}
|
|
3344
|
-
}
|
|
3345
|
-
});
|
|
3346
|
-
})
|
|
3347
|
-
);
|
|
3348
|
-
},
|
|
3349
|
-
async updateSession() {
|
|
3350
|
-
await sessionContext.agent.storage.session.set(session.id, {
|
|
3351
|
-
...session,
|
|
3352
|
-
lastMessageId: assistantMessageId,
|
|
3353
|
-
updatedAt: Date.now()
|
|
3354
|
-
});
|
|
3355
|
-
}
|
|
3356
|
-
});
|
|
3357
|
-
const state = session.state ? { ...session.state } : { ...initialState ?? {} };
|
|
3358
|
-
return {
|
|
3359
|
-
workflowRunId,
|
|
3360
|
-
workflowWritable,
|
|
3361
|
-
existingPartCount,
|
|
3362
|
-
assistantMessageId,
|
|
3363
|
-
assistantMessageCreatedAt,
|
|
3364
|
-
state
|
|
3365
|
-
};
|
|
3366
|
-
}
|
|
3367
|
-
async function completeMessageStep({
|
|
3368
|
-
assistantMessageId,
|
|
3369
|
-
sessionContext,
|
|
3370
|
-
writable,
|
|
3371
|
-
usageSteps
|
|
3372
|
-
}) {
|
|
3373
|
-
"use step";
|
|
3374
|
-
const log2 = createLogger({
|
|
3375
|
-
config: {
|
|
3376
|
-
...sessionContext.agent.options.logging,
|
|
3377
|
-
name: sessionContext.agent.name
|
|
3378
|
-
},
|
|
3379
|
-
subsystem: "ai-loop"
|
|
3380
|
-
});
|
|
3381
|
-
const msgLog = log2.withContext({
|
|
3382
|
-
sessionId: sessionContext.sessionId,
|
|
3383
|
-
messageId: assistantMessageId
|
|
3384
|
-
});
|
|
3385
|
-
const usage2 = usageSteps.length > 0 ? {
|
|
3386
|
-
steps: usageSteps,
|
|
3387
|
-
summary: {
|
|
3388
|
-
model: usageSteps[0]?.model ?? "unknown",
|
|
3389
|
-
inputTokens: usageSteps.reduce((a, s) => a + s.inputTokens, 0),
|
|
3390
|
-
outputTokens: usageSteps.reduce((a, s) => a + s.outputTokens, 0),
|
|
3391
|
-
totalTokens: usageSteps.reduce((a, s) => a + s.totalTokens, 0),
|
|
3392
|
-
cacheReadTokens: usageSteps.reduce(
|
|
3393
|
-
(a, s) => a + s.cacheReadTokens,
|
|
3394
|
-
0
|
|
3395
|
-
),
|
|
3396
|
-
cacheWriteTokens: usageSteps.reduce(
|
|
3397
|
-
(a, s) => a + s.cacheWriteTokens,
|
|
3398
|
-
0
|
|
3399
|
-
),
|
|
3400
|
-
reasoningTokens: usageSteps.reduce(
|
|
3401
|
-
(a, s) => a + s.reasoningTokens,
|
|
3402
|
-
0
|
|
3403
|
-
),
|
|
3404
|
-
stepCount: usageSteps.length
|
|
3405
|
-
}
|
|
3406
|
-
} : null;
|
|
3407
|
-
await Promise.all([
|
|
3408
|
-
sessionContext.agent.storage.message.update(assistantMessageId, {
|
|
3409
|
-
completedAt: Date.now(),
|
|
3410
|
-
usage: usage2
|
|
3411
|
-
}),
|
|
3412
|
-
writable.close().catch(() => void 0)
|
|
3413
|
-
]);
|
|
3414
|
-
msgLog.info("message completed", {
|
|
3415
|
-
steps: usageSteps.length,
|
|
3416
|
-
totalTokens: usage2?.summary.totalTokens
|
|
3417
|
-
});
|
|
3418
|
-
}
|
|
3419
|
-
function approvalPartId(approvalId) {
|
|
3420
|
-
return `part_${approvalId}`;
|
|
3421
|
-
}
|
|
3422
|
-
function replayActiveRun(active, startIndex = 0) {
|
|
3423
|
-
let cursor = startIndex;
|
|
3424
|
-
let cancelled = false;
|
|
3425
|
-
return new ReadableStream({
|
|
3426
|
-
async pull(controller) {
|
|
3427
|
-
while (cursor < active.chunks.length) {
|
|
3428
|
-
controller.enqueue(active.chunks[cursor++]);
|
|
3429
|
-
}
|
|
3430
|
-
while (!cancelled && cursor >= active.chunks.length) {
|
|
3431
|
-
const result = await Promise.race([
|
|
3432
|
-
active.onChunk.then(() => "chunk"),
|
|
3433
|
-
active.done.then(() => "done")
|
|
3434
|
-
]);
|
|
3435
|
-
while (cursor < active.chunks.length) {
|
|
3436
|
-
controller.enqueue(active.chunks[cursor++]);
|
|
3437
|
-
}
|
|
3438
|
-
if (result === "done") {
|
|
3439
|
-
controller.close();
|
|
3440
|
-
return;
|
|
3441
|
-
}
|
|
3442
|
-
}
|
|
3443
|
-
},
|
|
3444
|
-
cancel() {
|
|
3445
|
-
cancelled = true;
|
|
3446
|
-
}
|
|
3447
|
-
});
|
|
3448
|
-
}
|
|
3449
|
-
function toMessageAndParts({
|
|
3450
|
-
sessionId,
|
|
3451
|
-
id,
|
|
3452
|
-
input,
|
|
3453
|
-
defaultRole
|
|
3454
|
-
}) {
|
|
3455
|
-
let messageId = id ?? `message_${ulid3()}`;
|
|
3456
|
-
if (typeof input === "string") {
|
|
3457
|
-
const now2 = Date.now();
|
|
3458
|
-
return {
|
|
3459
|
-
approvals: [],
|
|
3460
|
-
message: {
|
|
3461
|
-
id: messageId,
|
|
3462
|
-
sessionId,
|
|
3463
|
-
role: defaultRole,
|
|
3464
|
-
createdAt: now2,
|
|
3465
|
-
startedAt: null,
|
|
3466
|
-
completedAt: null,
|
|
3467
|
-
interruptedAt: null,
|
|
3468
|
-
interruptedLastPart: null,
|
|
3469
|
-
usage: null,
|
|
3470
|
-
workflowRunId: null,
|
|
3471
|
-
metadata: null
|
|
3472
|
-
},
|
|
3473
|
-
parts: [
|
|
3474
|
-
{
|
|
3475
|
-
sessionId,
|
|
3476
|
-
messageId,
|
|
3477
|
-
id: `part_${messageId}_0`,
|
|
3478
|
-
index: 0,
|
|
3479
|
-
part: { type: "text", text: input }
|
|
3480
|
-
}
|
|
3481
|
-
]
|
|
3482
|
-
};
|
|
3483
|
-
}
|
|
3484
|
-
if ("type" in input && input.type === "approval") {
|
|
3485
|
-
return {
|
|
3486
|
-
message: void 0,
|
|
3487
|
-
parts: [],
|
|
3488
|
-
approvals: [input.approval]
|
|
3489
|
-
};
|
|
3490
|
-
}
|
|
3491
|
-
const msg = "type" in input && input.type === "message" ? input.message : input;
|
|
3492
|
-
if ("id" in msg && msg.id) {
|
|
3493
|
-
messageId = msg.id;
|
|
3494
|
-
}
|
|
3495
|
-
const now = Date.now();
|
|
3496
|
-
const metadata = "metadata" in msg && msg.metadata != null ? msg.metadata : null;
|
|
3497
|
-
return {
|
|
3498
|
-
approvals: [],
|
|
3499
|
-
message: {
|
|
3500
|
-
sessionId,
|
|
3501
|
-
id: messageId,
|
|
3502
|
-
createdAt: now,
|
|
3503
|
-
startedAt: null,
|
|
3504
|
-
completedAt: null,
|
|
3505
|
-
interruptedAt: null,
|
|
3506
|
-
interruptedLastPart: null,
|
|
3507
|
-
usage: null,
|
|
3508
|
-
workflowRunId: null,
|
|
3509
|
-
role: msg.role ?? defaultRole,
|
|
3510
|
-
metadata
|
|
3511
|
-
},
|
|
3512
|
-
parts: msg.parts.map((part, index) => {
|
|
3513
|
-
const ap = part;
|
|
3514
|
-
const id2 = ap.approval?.id ? approvalPartId(ap.approval.id) : `part_${messageId}_${index}`;
|
|
3515
|
-
return { sessionId, messageId, id: id2, index, part };
|
|
3516
|
-
})
|
|
3517
|
-
};
|
|
3518
|
-
}
|
|
3519
|
-
var SendError = class extends errore2.createTaggedError({
|
|
3520
|
-
name: "SendError",
|
|
3521
|
-
message: "Send failed for session $id: $reason"
|
|
3522
|
-
}) {
|
|
3523
|
-
};
|
|
3524
|
-
var ApprovalNotFoundError = class extends errore2.createTaggedError({
|
|
3525
|
-
name: "ApprovalNotFoundError",
|
|
3526
|
-
message: "Approval $approvalId not found in session $id"
|
|
3527
|
-
}) {
|
|
3528
|
-
};
|
|
3529
|
-
|
|
3530
|
-
// src/session/stream.ts
|
|
3531
|
-
import * as errore3 from "errore";
|
|
3532
|
-
function isWorkflowRun(arg) {
|
|
3533
|
-
return typeof arg === "object" && arg !== null && "getReadable" in arg && typeof arg.getReadable === "function";
|
|
3534
|
-
}
|
|
3535
|
-
async function stream(sessionContext, runOrOpts) {
|
|
3536
|
-
if (isWorkflowRun(runOrOpts)) {
|
|
3537
|
-
const reader = runOrOpts.getReadable().getReader();
|
|
3538
|
-
await reader.read().catch(() => {
|
|
3539
|
-
throw new StreamError({
|
|
3540
|
-
id: sessionContext.sessionId,
|
|
3541
|
-
reason: "stream() failed to read first chunk from workflow run"
|
|
3542
|
-
});
|
|
3543
|
-
});
|
|
3544
|
-
reader.cancel();
|
|
3545
|
-
return runOrOpts.getReadable();
|
|
3546
|
-
}
|
|
3547
|
-
const opts = runOrOpts;
|
|
3548
|
-
const session = await sessionContext.agent.storage.session.get(
|
|
3549
|
-
sessionContext.sessionId
|
|
3550
|
-
);
|
|
3551
|
-
if (!session) {
|
|
3552
|
-
throw new SessionNotFoundError({ id: sessionContext.sessionId });
|
|
3553
|
-
}
|
|
3554
|
-
const messageId = opts?.messageId ?? session.lastMessageId;
|
|
3555
|
-
if (!messageId) {
|
|
3556
|
-
throw new StreamError({
|
|
3557
|
-
id: sessionContext.sessionId,
|
|
3558
|
-
reason: "No message to stream"
|
|
3559
|
-
});
|
|
3560
|
-
}
|
|
3561
|
-
const activeRuns = getActiveRuns();
|
|
3562
|
-
const active = activeRuns.get(messageId);
|
|
3563
|
-
if (!active) {
|
|
3564
|
-
const message = await sessionContext.agent.storage.message.get(messageId);
|
|
3565
|
-
if (message?.workflowRunId) {
|
|
3566
|
-
const readable = await import("workflow/api").then(({ getRun }) => getRun(message.workflowRunId).getReadable()).catch(() => void 0);
|
|
3567
|
-
if (readable) {
|
|
3568
|
-
return readable;
|
|
3569
|
-
}
|
|
3570
|
-
}
|
|
3571
|
-
throw new StreamError({
|
|
3572
|
-
id: sessionContext.sessionId,
|
|
3573
|
-
reason: `No active run for message ${messageId}`
|
|
3574
|
-
});
|
|
3575
|
-
}
|
|
3576
|
-
return replayActiveRun(active, opts?.startIndex);
|
|
3577
|
-
}
|
|
3578
|
-
var StreamError = class extends errore3.createTaggedError({
|
|
3579
|
-
name: "StreamError",
|
|
3580
|
-
message: "Stream failed for session $id: $reason"
|
|
3581
|
-
}) {
|
|
3582
|
-
};
|
|
3583
|
-
|
|
3584
|
-
// src/session/update.ts
|
|
3585
|
-
async function update(sessionContext, opts) {
|
|
3586
|
-
"use step";
|
|
3587
|
-
const updates = Object.fromEntries(
|
|
3588
|
-
Object.entries(opts).filter(([, value]) => value !== void 0)
|
|
3589
|
-
);
|
|
3590
|
-
if (Object.keys(updates).length === 0) {
|
|
3591
|
-
return;
|
|
3592
|
-
}
|
|
3593
|
-
await sessionContext.agent.storage.session.update(sessionContext.sessionId, {
|
|
3594
|
-
...updates,
|
|
3595
|
-
updatedAt: Date.now()
|
|
3596
|
-
});
|
|
3597
|
-
}
|
|
3598
|
-
|
|
3599
|
-
// src/session/usage.ts
|
|
3600
|
-
async function usage(sessionContext) {
|
|
3601
|
-
"use step";
|
|
3602
|
-
const messages = await sessionContext.agent.storage.message.listBySession(
|
|
3603
|
-
sessionContext.sessionId
|
|
3604
|
-
);
|
|
3605
|
-
return computeSessionUsage(messages);
|
|
3606
|
-
}
|
|
3607
|
-
|
|
3608
|
-
// src/session/handle.ts
|
|
3609
|
-
function sessionHandle(agent2, id) {
|
|
3610
|
-
const sessionId = typeof id === "string" ? id : id?.id ?? `session_${ulid4()}`;
|
|
3611
|
-
const contexts = getSessionContexts(agent2);
|
|
3612
|
-
let sessionContext = contexts.get(sessionId);
|
|
3613
|
-
if (!sessionContext) {
|
|
3614
|
-
sessionContext = { agent: agent2, sessionId };
|
|
3615
|
-
contexts.set(sessionId, sessionContext);
|
|
3616
|
-
}
|
|
3617
|
-
return {
|
|
3618
|
-
id: sessionId,
|
|
3619
|
-
send: send.bind(
|
|
3620
|
-
null,
|
|
3621
|
-
sessionContext
|
|
3622
|
-
),
|
|
3623
|
-
stream: stream.bind(null, sessionContext),
|
|
3624
|
-
history: history.bind(null, sessionContext),
|
|
3625
|
-
interrupt: interrupt.bind(null, sessionContext),
|
|
3626
|
-
usage: usage.bind(null, sessionContext),
|
|
3627
|
-
update: update.bind(null, sessionContext),
|
|
3628
|
-
sandbox: lazySandboxHandle(agent2, sessionId)
|
|
3629
|
-
};
|
|
3630
|
-
}
|
|
3631
|
-
|
|
3632
|
-
// src/agent/registry.ts
|
|
3633
|
-
import * as errore4 from "errore";
|
|
3634
|
-
import { z as z3 } from "zod";
|
|
3635
|
-
var AgentStatus = z3.discriminatedUnion("type", [
|
|
3636
|
-
z3.object({ type: z3.literal("sandbox-setup") }),
|
|
3637
|
-
z3.object({ type: z3.literal("sandbox-setup-cold") }),
|
|
3638
|
-
z3.object({ type: z3.literal("loading-skills") }),
|
|
3639
|
-
z3.object({ type: z3.literal("processing-approvals") }),
|
|
3640
|
-
z3.object({ type: z3.literal("needs-approval") }),
|
|
3641
|
-
z3.object({ type: z3.literal("thinking") }),
|
|
3642
|
-
z3.object({ type: z3.literal("custom"), status: z3.string() })
|
|
3643
|
-
]);
|
|
3644
|
-
var AgentNotFoundError = class extends errore4.createTaggedError({
|
|
3645
|
-
name: "AgentNotFoundError",
|
|
3646
|
-
message: `Agent "$agentName" not found in registry.
|
|
3647
|
-
|
|
3648
|
-
hint: agent() must be called before workflow steps execute.
|
|
3649
|
-
Valid locations: module scope, or inside a "use workflow" body.
|
|
3650
|
-
|
|
3651
|
-
// Module scope (static agents)
|
|
3652
|
-
export const myAgent = agent("$agentName", { ... });
|
|
3653
|
-
|
|
3654
|
-
// Workflow body (dynamic agents)
|
|
3655
|
-
async function myWorkflow({ input }) {
|
|
3656
|
-
"use workflow";
|
|
3657
|
-
const myAgent = agent("$agentName", { ... });
|
|
3658
|
-
...
|
|
3659
|
-
}`
|
|
3660
|
-
}) {
|
|
3661
|
-
};
|
|
3662
|
-
var AGENT_REGISTRY_KEY = /* @__PURE__ */ Symbol.for("experimental-agent-registry");
|
|
3663
|
-
function getAgentRegistry() {
|
|
3664
|
-
const g = globalThis;
|
|
3665
|
-
let registry = g[AGENT_REGISTRY_KEY];
|
|
3666
|
-
if (!registry) {
|
|
3667
|
-
registry = /* @__PURE__ */ new Map();
|
|
3668
|
-
g[AGENT_REGISTRY_KEY] = registry;
|
|
3669
|
-
}
|
|
3670
|
-
return registry;
|
|
3671
|
-
}
|
|
3672
|
-
function registerAgent(name, agent2) {
|
|
3673
|
-
getAgentRegistry().set(name, agent2);
|
|
3674
|
-
}
|
|
3675
|
-
function getAgent(name, required) {
|
|
3676
|
-
const agent2 = getAgentRegistry().get(name);
|
|
3677
|
-
if (required && !agent2) {
|
|
3678
|
-
throw new AgentNotFoundError({ agentName: name });
|
|
3679
|
-
}
|
|
3680
|
-
return agent2;
|
|
3681
|
-
}
|
|
3682
|
-
|
|
3683
|
-
// src/agent/client.ts
|
|
3684
|
-
var sessionContextsMap = /* @__PURE__ */ new WeakMap();
|
|
3685
|
-
var sandboxContextsMap = /* @__PURE__ */ new WeakMap();
|
|
3686
|
-
function getSessionContexts(agent2) {
|
|
3687
|
-
let map = sessionContextsMap.get(agent2);
|
|
3688
|
-
if (!map) {
|
|
3689
|
-
map = /* @__PURE__ */ new Map();
|
|
3690
|
-
sessionContextsMap.set(agent2, map);
|
|
3691
|
-
}
|
|
3692
|
-
return map;
|
|
3693
|
-
}
|
|
3694
|
-
function getSandboxContexts(agent2) {
|
|
3695
|
-
let map = sandboxContextsMap.get(agent2);
|
|
3696
|
-
if (!map) {
|
|
3697
|
-
map = /* @__PURE__ */ new Map();
|
|
3698
|
-
sandboxContextsMap.set(agent2, map);
|
|
3699
|
-
}
|
|
3700
|
-
return map;
|
|
3701
|
-
}
|
|
3702
|
-
var Agent = class {
|
|
3703
|
-
name;
|
|
3704
|
-
options;
|
|
3705
|
-
constructor(name, options) {
|
|
3706
|
-
this.name = name;
|
|
3707
|
-
this.options = options ?? {};
|
|
3708
|
-
this.session = sessionHandle.bind(null, this);
|
|
3709
|
-
this.sandbox = sandboxHandle.bind(
|
|
3710
|
-
null,
|
|
3711
|
-
this
|
|
3712
|
-
);
|
|
3713
|
-
}
|
|
3714
|
-
static [WORKFLOW_SERIALIZE](instance) {
|
|
3715
|
-
return { name: instance.name };
|
|
3716
|
-
}
|
|
3717
|
-
static [WORKFLOW_DESERIALIZE](data) {
|
|
3718
|
-
return getAgent(data.name, true);
|
|
3719
|
-
}
|
|
3720
|
-
_resolvedStorage;
|
|
3721
|
-
get storage() {
|
|
3722
|
-
if (this._resolvedStorage) {
|
|
3723
|
-
return this._resolvedStorage;
|
|
3724
|
-
}
|
|
3725
|
-
if (this.options.storage) {
|
|
3726
|
-
this._resolvedStorage = toStorage(this.options.storage);
|
|
3727
|
-
return this._resolvedStorage;
|
|
3728
|
-
}
|
|
3729
|
-
const log2 = createLogger({
|
|
3730
|
-
config: { ...this.options.logging, name: this.name },
|
|
3731
|
-
subsystem: "storage"
|
|
3732
|
-
});
|
|
3733
|
-
if (shouldFallbackToKV2()) {
|
|
3734
|
-
log2.info(
|
|
3735
|
-
`No storage adapter configured \u2014 using KV2 (namespace: "${this.name}").`
|
|
3736
|
-
);
|
|
3737
|
-
this._resolvedStorage = toStorage(
|
|
3738
|
-
kvStorageFallback({ prefix: this.name })
|
|
3739
|
-
);
|
|
3740
|
-
return this._resolvedStorage;
|
|
3741
|
-
}
|
|
3742
|
-
log2.warn(
|
|
3743
|
-
"No storage adapter configured \u2014 using localStorage(). Data will not persist across restarts."
|
|
3744
|
-
);
|
|
3745
|
-
this._resolvedStorage = toStorage(localStorage());
|
|
3746
|
-
return this._resolvedStorage;
|
|
3747
|
-
}
|
|
3748
|
-
get tools() {
|
|
3749
|
-
return { ...builtInTools, ...this.options.tools };
|
|
3750
|
-
}
|
|
3751
|
-
get hooks() {
|
|
3752
|
-
return this.options.hooks ?? {};
|
|
3753
|
-
}
|
|
3754
|
-
session;
|
|
3755
|
-
sandbox;
|
|
3756
|
-
};
|
|
3757
|
-
function agent(name, options) {
|
|
3758
|
-
const instance = new Agent(name, options);
|
|
3759
|
-
registerAgent(name, instance);
|
|
3760
|
-
return instance;
|
|
3761
|
-
}
|
|
2
|
+
AgentStatus,
|
|
3
|
+
MessageNotFoundError,
|
|
4
|
+
SandboxError,
|
|
5
|
+
SandboxNotFoundError,
|
|
6
|
+
SessionNotFoundError,
|
|
7
|
+
StorageConflictError,
|
|
8
|
+
StorageError,
|
|
9
|
+
agent,
|
|
10
|
+
builtinToolNames,
|
|
11
|
+
tool
|
|
12
|
+
} from "./chunk-EH4L2CPW.mjs";
|
|
13
|
+
import {
|
|
14
|
+
basePathForAgent
|
|
15
|
+
} from "./chunk-PH2FXKOU.mjs";
|
|
16
|
+
import "./chunk-C4VSUEY2.mjs";
|
|
17
|
+
import "./chunk-CS2SEUAA.mjs";
|
|
18
|
+
import "./chunk-MSWINCCM.mjs";
|
|
19
|
+
import "./chunk-BJTO5JO5.mjs";
|
|
3762
20
|
|
|
3763
21
|
// src/rpc/rpc.ts
|
|
22
|
+
import { openapi } from "@elysiajs/openapi";
|
|
3764
23
|
import { createUIMessageStreamResponse } from "ai";
|
|
3765
24
|
import Elysia from "elysia";
|
|
25
|
+
import z3 from "zod";
|
|
3766
26
|
|
|
3767
27
|
// src/rpc/agentContract.ts
|
|
3768
|
-
import
|
|
28
|
+
import z2 from "zod";
|
|
3769
29
|
|
|
3770
30
|
// src/rpc/schemas.ts
|
|
3771
|
-
import { z
|
|
3772
|
-
var TextPart =
|
|
3773
|
-
type:
|
|
3774
|
-
text:
|
|
31
|
+
import { z } from "zod";
|
|
32
|
+
var TextPart = z.object({
|
|
33
|
+
type: z.literal("text"),
|
|
34
|
+
text: z.string()
|
|
3775
35
|
});
|
|
3776
|
-
var ReasoningPart =
|
|
3777
|
-
type:
|
|
3778
|
-
text:
|
|
36
|
+
var ReasoningPart = z.object({
|
|
37
|
+
type: z.literal("reasoning"),
|
|
38
|
+
text: z.string()
|
|
3779
39
|
});
|
|
3780
|
-
var FilePart =
|
|
3781
|
-
type:
|
|
3782
|
-
mediaType:
|
|
3783
|
-
url:
|
|
3784
|
-
filename:
|
|
40
|
+
var FilePart = z.object({
|
|
41
|
+
type: z.literal("file"),
|
|
42
|
+
mediaType: z.string(),
|
|
43
|
+
url: z.string(),
|
|
44
|
+
filename: z.string().optional()
|
|
3785
45
|
});
|
|
3786
|
-
var SourceUrlPart =
|
|
3787
|
-
type:
|
|
3788
|
-
sourceId:
|
|
3789
|
-
url:
|
|
3790
|
-
title:
|
|
46
|
+
var SourceUrlPart = z.object({
|
|
47
|
+
type: z.literal("source-url"),
|
|
48
|
+
sourceId: z.string(),
|
|
49
|
+
url: z.string(),
|
|
50
|
+
title: z.string().optional()
|
|
3791
51
|
});
|
|
3792
|
-
var StepStartPart =
|
|
3793
|
-
type:
|
|
52
|
+
var StepStartPart = z.object({
|
|
53
|
+
type: z.literal("step-start")
|
|
3794
54
|
});
|
|
3795
|
-
function toolPartSchemas(name,
|
|
3796
|
-
const input =
|
|
3797
|
-
const output =
|
|
3798
|
-
const type =
|
|
3799
|
-
const approvalId =
|
|
55
|
+
function toolPartSchemas(name, tool2) {
|
|
56
|
+
const input = tool2.inputSchema ?? z.unknown();
|
|
57
|
+
const output = tool2.outputSchema ?? z.unknown();
|
|
58
|
+
const type = z.literal(`tool-${name}`);
|
|
59
|
+
const approvalId = z.object({ id: z.string() });
|
|
3800
60
|
const approvalApproved = approvalId.extend({
|
|
3801
|
-
approved:
|
|
3802
|
-
reason:
|
|
61
|
+
approved: z.literal(true),
|
|
62
|
+
reason: z.string().optional()
|
|
3803
63
|
});
|
|
3804
64
|
const approvalDenied = approvalId.extend({
|
|
3805
|
-
approved:
|
|
3806
|
-
reason:
|
|
65
|
+
approved: z.literal(false),
|
|
66
|
+
reason: z.string().optional()
|
|
3807
67
|
});
|
|
3808
68
|
const approvalResponse = approvalId.extend({
|
|
3809
|
-
approved:
|
|
3810
|
-
reason:
|
|
69
|
+
approved: z.boolean(),
|
|
70
|
+
reason: z.string().optional()
|
|
3811
71
|
});
|
|
3812
72
|
return [
|
|
3813
|
-
|
|
73
|
+
z.object({
|
|
3814
74
|
type,
|
|
3815
|
-
toolCallId:
|
|
3816
|
-
state:
|
|
3817
|
-
input:
|
|
75
|
+
toolCallId: z.string(),
|
|
76
|
+
state: z.literal("input-streaming"),
|
|
77
|
+
input: z.unknown()
|
|
3818
78
|
}),
|
|
3819
|
-
|
|
79
|
+
z.object({
|
|
3820
80
|
type,
|
|
3821
|
-
toolCallId:
|
|
3822
|
-
state:
|
|
81
|
+
toolCallId: z.string(),
|
|
82
|
+
state: z.literal("input-available"),
|
|
3823
83
|
input
|
|
3824
84
|
}),
|
|
3825
|
-
|
|
85
|
+
z.object({
|
|
3826
86
|
type,
|
|
3827
|
-
toolCallId:
|
|
3828
|
-
state:
|
|
87
|
+
toolCallId: z.string(),
|
|
88
|
+
state: z.literal("approval-requested"),
|
|
3829
89
|
input,
|
|
3830
90
|
approval: approvalId
|
|
3831
91
|
}),
|
|
3832
|
-
|
|
92
|
+
z.object({
|
|
3833
93
|
type,
|
|
3834
|
-
toolCallId:
|
|
3835
|
-
state:
|
|
94
|
+
toolCallId: z.string(),
|
|
95
|
+
state: z.literal("approval-responded"),
|
|
3836
96
|
input,
|
|
3837
97
|
approval: approvalResponse
|
|
3838
98
|
}),
|
|
3839
|
-
|
|
99
|
+
z.object({
|
|
3840
100
|
type,
|
|
3841
|
-
toolCallId:
|
|
3842
|
-
state:
|
|
101
|
+
toolCallId: z.string(),
|
|
102
|
+
state: z.literal("output-available"),
|
|
3843
103
|
input,
|
|
3844
104
|
output,
|
|
3845
105
|
approval: approvalApproved.optional()
|
|
3846
106
|
}),
|
|
3847
|
-
|
|
107
|
+
z.object({
|
|
3848
108
|
type,
|
|
3849
|
-
toolCallId:
|
|
3850
|
-
state:
|
|
109
|
+
toolCallId: z.string(),
|
|
110
|
+
state: z.literal("output-error"),
|
|
3851
111
|
input,
|
|
3852
|
-
errorText:
|
|
112
|
+
errorText: z.string(),
|
|
3853
113
|
approval: approvalApproved.optional()
|
|
3854
114
|
}),
|
|
3855
|
-
|
|
115
|
+
z.object({
|
|
3856
116
|
type,
|
|
3857
|
-
toolCallId:
|
|
3858
|
-
state:
|
|
117
|
+
toolCallId: z.string(),
|
|
118
|
+
state: z.literal("output-denied"),
|
|
3859
119
|
input,
|
|
3860
120
|
approval: approvalDenied
|
|
3861
121
|
})
|
|
3862
122
|
];
|
|
3863
123
|
}
|
|
3864
|
-
var UsageSummary =
|
|
3865
|
-
model:
|
|
3866
|
-
inputTokens:
|
|
3867
|
-
outputTokens:
|
|
3868
|
-
totalTokens:
|
|
3869
|
-
cacheReadTokens:
|
|
3870
|
-
cacheWriteTokens:
|
|
3871
|
-
reasoningTokens:
|
|
3872
|
-
stepCount:
|
|
124
|
+
var UsageSummary = z.object({
|
|
125
|
+
model: z.string(),
|
|
126
|
+
inputTokens: z.number(),
|
|
127
|
+
outputTokens: z.number(),
|
|
128
|
+
totalTokens: z.number(),
|
|
129
|
+
cacheReadTokens: z.number(),
|
|
130
|
+
cacheWriteTokens: z.number(),
|
|
131
|
+
reasoningTokens: z.number(),
|
|
132
|
+
stepCount: z.number()
|
|
3873
133
|
});
|
|
3874
|
-
var SessionUsage =
|
|
3875
|
-
total: UsageSummary.extend({ messageCount:
|
|
3876
|
-
byMessageId:
|
|
134
|
+
var SessionUsage = z.object({
|
|
135
|
+
total: UsageSummary.extend({ messageCount: z.number() }),
|
|
136
|
+
byMessageId: z.record(z.string(), UsageSummary.nullable())
|
|
3877
137
|
});
|
|
3878
138
|
function buildSessionHistorySchema(agent2) {
|
|
3879
|
-
return
|
|
3880
|
-
messages:
|
|
3881
|
-
streamingMessageId:
|
|
139
|
+
return z.object({
|
|
140
|
+
messages: z.array(buildUIMessageSchema(agent2.tools)),
|
|
141
|
+
streamingMessageId: z.string().nullable(),
|
|
3882
142
|
usage: SessionUsage
|
|
3883
143
|
});
|
|
3884
144
|
}
|
|
3885
145
|
function buildUIMessagePartSchema(tools) {
|
|
3886
146
|
const toolParts = Object.entries(tools).flatMap(
|
|
3887
|
-
([name,
|
|
147
|
+
([name, tool2]) => toolPartSchemas(name, tool2)
|
|
3888
148
|
);
|
|
3889
|
-
const
|
|
149
|
+
const all = [
|
|
3890
150
|
TextPart,
|
|
3891
151
|
ReasoningPart,
|
|
3892
152
|
FilePart,
|
|
@@ -3894,13 +154,13 @@ function buildUIMessagePartSchema(tools) {
|
|
|
3894
154
|
StepStartPart,
|
|
3895
155
|
...toolParts
|
|
3896
156
|
];
|
|
3897
|
-
return
|
|
157
|
+
return z.union(all);
|
|
3898
158
|
}
|
|
3899
159
|
function buildUIMessageSchema(tools) {
|
|
3900
|
-
return
|
|
3901
|
-
id:
|
|
3902
|
-
role:
|
|
3903
|
-
parts:
|
|
160
|
+
return z.object({
|
|
161
|
+
id: z.string(),
|
|
162
|
+
role: z.enum(["system", "user", "assistant"]),
|
|
163
|
+
parts: z.array(buildUIMessagePartSchema(tools))
|
|
3904
164
|
});
|
|
3905
165
|
}
|
|
3906
166
|
|
|
@@ -3908,36 +168,36 @@ function buildUIMessageSchema(tools) {
|
|
|
3908
168
|
function agentContract(agent2) {
|
|
3909
169
|
const contract = {
|
|
3910
170
|
"session.get": {
|
|
3911
|
-
params:
|
|
3912
|
-
sessionId:
|
|
171
|
+
params: z2.object({
|
|
172
|
+
sessionId: z2.string()
|
|
3913
173
|
}),
|
|
3914
174
|
response: buildSessionHistorySchema(agent2)
|
|
3915
175
|
},
|
|
3916
176
|
"session.post": {
|
|
3917
|
-
params:
|
|
3918
|
-
sessionId:
|
|
177
|
+
params: z2.object({
|
|
178
|
+
sessionId: z2.string()
|
|
3919
179
|
}),
|
|
3920
|
-
body:
|
|
3921
|
-
response:
|
|
180
|
+
body: z2.union([z2.string(), buildUIMessageSchema(agent2.tools)]),
|
|
181
|
+
response: z2.any()
|
|
3922
182
|
},
|
|
3923
183
|
"reconnect.get": {
|
|
3924
|
-
params:
|
|
3925
|
-
sessionId:
|
|
184
|
+
params: z2.object({
|
|
185
|
+
sessionId: z2.string()
|
|
3926
186
|
}),
|
|
3927
|
-
response:
|
|
187
|
+
response: z2.any()
|
|
3928
188
|
},
|
|
3929
189
|
"interrupt.post": {
|
|
3930
|
-
params:
|
|
3931
|
-
sessionId:
|
|
190
|
+
params: z2.object({
|
|
191
|
+
sessionId: z2.string()
|
|
3932
192
|
}),
|
|
3933
|
-
body:
|
|
3934
|
-
lastPart:
|
|
3935
|
-
index:
|
|
193
|
+
body: z2.object({
|
|
194
|
+
lastPart: z2.object({
|
|
195
|
+
index: z2.number(),
|
|
3936
196
|
part: buildUIMessagePartSchema(agent2.tools)
|
|
3937
197
|
}).optional()
|
|
3938
198
|
}).optional(),
|
|
3939
|
-
response:
|
|
3940
|
-
success:
|
|
199
|
+
response: z2.object({
|
|
200
|
+
success: z2.boolean()
|
|
3941
201
|
})
|
|
3942
202
|
}
|
|
3943
203
|
};
|
|
@@ -3948,7 +208,37 @@ function agentContract(agent2) {
|
|
|
3948
208
|
function handleRequest(agent2, options = {}) {
|
|
3949
209
|
const { basePath = basePathForAgent(agent2.name), workflow } = options;
|
|
3950
210
|
const contract = agentContract(agent2);
|
|
3951
|
-
const router = new Elysia({ prefix: basePath }).
|
|
211
|
+
const router = new Elysia({ prefix: basePath }).use(
|
|
212
|
+
openapi({
|
|
213
|
+
provider: null,
|
|
214
|
+
specPath: "/openapi.json",
|
|
215
|
+
exclude: { staticFile: false },
|
|
216
|
+
documentation: {
|
|
217
|
+
info: {
|
|
218
|
+
title: `${agent2.name} API`,
|
|
219
|
+
version: "1.0.0"
|
|
220
|
+
},
|
|
221
|
+
components: {
|
|
222
|
+
schemas: {
|
|
223
|
+
// @ts-expect-error - elysia is annoying
|
|
224
|
+
AgentStatus: z3.toJSONSchema(AgentStatus, {
|
|
225
|
+
target: "openapi-3.0"
|
|
226
|
+
})
|
|
227
|
+
},
|
|
228
|
+
securitySchemes: {
|
|
229
|
+
bearerAuth: {
|
|
230
|
+
type: "http",
|
|
231
|
+
scheme: "bearer",
|
|
232
|
+
bearerFormat: "JWT"
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
},
|
|
237
|
+
mapJsonSchema: {
|
|
238
|
+
zod: (schema) => z3.toJSONSchema(schema, { target: "openapi-3.0" })
|
|
239
|
+
}
|
|
240
|
+
})
|
|
241
|
+
).get(
|
|
3952
242
|
"/:sessionId",
|
|
3953
243
|
async ({ params }) => {
|
|
3954
244
|
const next = async () => {
|
|
@@ -3975,14 +265,20 @@ function handleRequest(agent2, options = {}) {
|
|
|
3975
265
|
const next = async () => {
|
|
3976
266
|
const session = agent2.session(params.sessionId);
|
|
3977
267
|
if (workflow) {
|
|
3978
|
-
const { start
|
|
3979
|
-
const result = await
|
|
3980
|
-
|
|
3981
|
-
|
|
268
|
+
const { start } = await import("workflow/api");
|
|
269
|
+
const result = await start(
|
|
270
|
+
// for some reason, start() requires SessionSendArgs<AnyAgent>
|
|
271
|
+
// to avoid blowing up types
|
|
272
|
+
// we want to put the typeguard here so consumers of handleRequest still work fine
|
|
273
|
+
workflow,
|
|
274
|
+
[params.sessionId, body]
|
|
275
|
+
);
|
|
276
|
+
const stream2 = await session.stream(result);
|
|
277
|
+
return createUIMessageStreamResponse({ stream: stream2 });
|
|
3982
278
|
}
|
|
3983
279
|
await session.send(body);
|
|
3984
|
-
const
|
|
3985
|
-
return createUIMessageStreamResponse({ stream
|
|
280
|
+
const stream = await session.stream();
|
|
281
|
+
return createUIMessageStreamResponse({ stream });
|
|
3986
282
|
};
|
|
3987
283
|
const override = options.overrides?.["session.post"];
|
|
3988
284
|
if (override) {
|
|
@@ -3996,8 +292,8 @@ function handleRequest(agent2, options = {}) {
|
|
|
3996
292
|
async ({ params }) => {
|
|
3997
293
|
const next = async () => {
|
|
3998
294
|
const session = agent2.session(params.sessionId);
|
|
3999
|
-
const
|
|
4000
|
-
return createUIMessageStreamResponse({ stream
|
|
295
|
+
const stream = await session.stream();
|
|
296
|
+
return createUIMessageStreamResponse({ stream });
|
|
4001
297
|
};
|
|
4002
298
|
const override = options.overrides?.["reconnect.get"];
|
|
4003
299
|
if (override) {
|
|
@@ -4035,6 +331,6 @@ export {
|
|
|
4035
331
|
agent,
|
|
4036
332
|
builtinToolNames,
|
|
4037
333
|
handleRequest,
|
|
4038
|
-
|
|
334
|
+
tool
|
|
4039
335
|
};
|
|
4040
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/agent/client.ts", "../src/sandbox/handle.ts", "../src/agent/is-vercel.ts", "../src/errors.ts", "../src/sandbox/setup.ts", "../src/sandbox/exec.ts", "../src/sandbox/get-domain.ts", "../src/sandbox/get-status.ts", "../src/sandbox/kill.ts", "../src/sandbox/read-file.ts", "../src/sandbox/snapshot.ts", "../src/sandbox/start.ts", "../src/sandbox/stop.ts", "../src/sandbox/update-network-policy.ts", "../src/sandbox/write-files.ts", "../src/session/handle.ts", "../src/utils/ui.ts", "../src/utils/usage.ts", "../src/session/history.ts", "../src/session/interrupt.ts", "../src/session/send.ts", "../src/session/ai-loop/stream-text.ts", "../src/skills/parser.ts", "../src/skills/discover.ts", "../src/skills/resolve-host.ts", "../src/skills/resolve-inline.ts", "../src/skills/resolve-remote.ts", "../src/skills/resolve.ts", "../src/tools/index.ts", "../src/tools/javascript.ts", "../src/tools/needs-approval.ts", "../src/tools/tool.ts", "../src/utils/prompt-cache.ts", "../src/session/stream.ts", "../src/session/update.ts", "../src/session/usage.ts", "../src/agent/registry.ts", "../src/rpc/rpc.ts", "../src/rpc/agentContract.ts", "../src/rpc/schemas.ts"],
  "sourcesContent": ["import { WORKFLOW_DESERIALIZE, WORKFLOW_SERIALIZE } from \"@workflow/serde\";\nimport type { GatewayModelId, InferUITools, ToolSet, UIMessage } from \"ai\";\nimport type { z } from \"zod\";\nimport type { SandboxBinding, SandboxInstance } from \"../sandbox/adapter\";\nimport type { SandboxContext } from \"../sandbox/handle\";\nimport { sandboxHandle } from \"../sandbox/handle\";\nimport { type SessionContext, sessionHandle } from \"../session/handle\";\nimport type { SkillInput } from \"../skills/types\";\nimport { type Storage, type StorageInput, toStorage } from \"../storage/adapter\";\nimport { kvStorageFallback } from \"../storage/bindings/kv\";\nimport { localStorage } from \"../storage/entry\";\nimport { shouldFallbackToKV2 } from \"../storage/should-fallback-kv2\";\nimport type { GenerationOptions } from \"../storage/types\";\nimport { type BuiltInToolName, builtInTools } from \"../tools\";\nimport type { LoggingConfig } from \"../utils/logger\";\nimport { createLogger } from \"../utils/logger\";\nimport {\n  type AgentDataTypes,\n  type AgentHooks,\n  getAgent,\n  type NeedsApprovalMap,\n  registerAgent,\n} from \"./registry\";\n\n// \u2500\u2500\u2500 Resolvable \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport type ResolvableArgs<\n  TContext = Record<string, unknown>,\n  Tools extends ToolSet = ToolSet,\n  TState = Record<string, unknown>,\n> = {\n  context: TContext;\n  sessionId: string;\n  sandbox: SandboxInstance;\n  messages: UIMessage<unknown, AgentDataTypes, InferUITools<Tools>>[];\n  state: Readonly<TState>;\n};\n\nexport type Resolvable<\n  T,\n  TContext = Record<string, unknown>,\n  Tools extends ToolSet = ToolSet,\n  TState = Record<string, unknown>,\n> = T | ((args: ResolvableArgs<TContext, Tools, TState>) => T | Promise<T>);\n\n// \u2500\u2500\u2500 Session Defaults \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport type SessionDefaults<\n  Tools extends ToolSet = ToolSet,\n  TContext extends Record<string, unknown> = Record<string, unknown>,\n  TState extends Record<string, unknown> = Record<string, unknown>,\n> = {\n  model?: Resolvable<GatewayModelId | undefined, TContext, Tools, TState>;\n  system?: Resolvable<string | string[] | undefined, TContext, Tools, TState>;\n  generation?: Resolvable<\n    GenerationOptions | undefined,\n    TContext,\n    Tools,\n    TState\n  >;\n  activeTools?: Resolvable<\n    readonly (Extract<keyof Tools, string> | BuiltInToolName)[] | undefined,\n    TContext,\n    Tools,\n    TState\n  >;\n  skills?: Resolvable<SkillInput[] | undefined, TContext, Tools, TState>;\n};\n\n// \u2500\u2500\u2500 Agent Options \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport type AgentOptions<\n  // biome-ignore lint/suspicious/noExplicitAny: default allows type-erased usage\n  Tools extends ToolSet = any,\n  TSandboxBindings extends SandboxBinding | SandboxBinding[] = SandboxBinding,\n  TContext extends Record<string, unknown> = Record<string, never>,\n  TState extends Record<string, unknown> = Record<string, unknown>,\n  TMessageMetadata = unknown,\n> = SessionDefaults<Tools, TContext, TState> & {\n  storage?: StorageInput;\n  sandbox?: TSandboxBindings;\n  tools?: Tools;\n  hooks?: AgentHooks;\n  needsApproval?: NeedsApprovalMap<Tools>;\n  contextSchema?: z.ZodType<TContext>;\n  stateSchema?: z.ZodType<TState>;\n  messageMetadataSchema?: z.ZodType<TMessageMetadata>;\n  logging?: Omit<LoggingConfig, \"name\">;\n};\n\n// \u2500\u2500\u2500 Internal context caches (hidden from public API) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst sessionContextsMap = new WeakMap<object, Map<string, SessionContext>>();\nconst sandboxContextsMap = new WeakMap<object, Map<string, SandboxContext>>();\n\nexport function getSessionContexts(\n  agent: AnyAgent\n): Map<string, SessionContext> {\n  let map = sessionContextsMap.get(agent);\n  if (!map) {\n    map = new Map();\n    sessionContextsMap.set(agent, map);\n  }\n  return map;\n}\n\nexport function getSandboxContexts(\n  agent: AnyAgent\n): Map<string, SandboxContext> {\n  let map = sandboxContextsMap.get(agent);\n  if (!map) {\n    map = new Map();\n    sandboxContextsMap.set(agent, map);\n  }\n  return map;\n}\n\n// \u2500\u2500\u2500 Agent Class \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n// biome-ignore lint/suspicious/noExplicitAny: intentionally erased \u2013 using `any` avoids contravariance issues with tool-typed methods like `send`\nexport type AnyAgent = Agent<any, any, any, any, any>;\n\nexport type InferSession<Agent extends AnyAgent = AnyAgent> = ReturnType<\n  Agent[\"session\"]\n>;\nexport type SessionSendArgs<\n  Agent extends AnyAgent,\n  Session extends InferSession<Agent> = InferSession<Agent>,\n> = Parameters<Session[\"send\"]>;\n\nexport class Agent<\n  TMessageMetadata = unknown,\n  Tools extends ToolSet = ToolSet,\n  TSandboxBindings extends SandboxBinding | SandboxBinding[] = SandboxBinding,\n  TContext extends Record<string, unknown> = Record<string, never>,\n  TState extends Record<string, unknown> = Record<string, unknown>,\n> {\n  readonly name: string;\n  options: AgentOptions<\n    Tools,\n    TSandboxBindings,\n    TContext,\n    TState,\n    TMessageMetadata\n  >;\n\n  constructor(\n    name: string,\n    options?: AgentOptions<\n      Tools,\n      TSandboxBindings,\n      TContext,\n      TState,\n      TMessageMetadata\n    >\n  ) {\n    this.name = name;\n    this.options = options ?? {};\n    this.session = (sessionHandle<\n      Tools,\n      TContext,\n      TSandboxBindings,\n      TMessageMetadata,\n      TState\n    >).bind(null, this as unknown as AnyAgent);\n    this.sandbox = (sandboxHandle<TSandboxBindings>).bind(\n      null,\n      this as unknown as AnyAgent\n    );\n  }\n\n  static [WORKFLOW_SERIALIZE](instance: AnyAgent) {\n    return { name: instance.name };\n  }\n\n  static [WORKFLOW_DESERIALIZE](data: { name: string }) {\n    return getAgent(data.name, true);\n  }\n\n  private _resolvedStorage: Storage | undefined;\n\n  get storage(): Storage {\n    if (this._resolvedStorage) {\n      return this._resolvedStorage;\n    }\n\n    if (this.options.storage) {\n      this._resolvedStorage = toStorage(this.options.storage);\n      return this._resolvedStorage;\n    }\n\n    const log = createLogger({\n      config: { ...this.options.logging, name: this.name },\n      subsystem: \"storage\",\n    });\n\n    if (shouldFallbackToKV2()) {\n      log.info(\n        `No storage adapter configured \u2014 using KV2 (namespace: \"${this.name}\").`\n      );\n      this._resolvedStorage = toStorage(\n        kvStorageFallback({ prefix: this.name })\n      );\n      return this._resolvedStorage;\n    }\n\n    log.warn(\n      \"No storage adapter configured \u2014 using localStorage(). Data will not persist across restarts.\"\n    );\n    this._resolvedStorage = toStorage(localStorage());\n    return this._resolvedStorage;\n  }\n\n  get tools() {\n    return { ...builtInTools, ...this.options.tools } as typeof builtInTools &\n      Tools;\n  }\n\n  get hooks(): AgentHooks {\n    return this.options.hooks ?? {};\n  }\n\n  readonly session;\n\n  readonly sandbox;\n\n  /** Phantom property for type inference. Use `typeof myAgent.$UIMessage` to get the typed UIMessage. */\n  declare readonly $UIMessage: UIMessage<\n    TMessageMetadata,\n    AgentDataTypes,\n    InferUITools<typeof builtInTools & Tools>\n  >;\n}\n\nexport function agent<\n  Tools extends ToolSet = ToolSet,\n  TSandboxBindings extends SandboxBinding | SandboxBinding[] = SandboxBinding,\n  TContext extends Record<string, unknown> = Record<string, never>,\n  TState extends Record<string, unknown> = Record<string, unknown>,\n  TMessageMetadata = unknown,\n>(\n  name: string,\n  options?: AgentOptions<\n    Tools,\n    TSandboxBindings,\n    TContext,\n    TState,\n    TMessageMetadata\n  >\n) {\n  const instance = new Agent<\n    TMessageMetadata,\n    Tools,\n    TSandboxBindings,\n    TContext,\n    TState\n  >(name, options);\n  registerAgent(name, instance);\n  return instance;\n}\n\n// Re-export everything from registry so existing imports from \"./agent/client\" keep working.\nexport {\n  type AgentDataTypes,\n  type AgentHooks,\n  AgentStatus,\n  getAgent,\n  type InferUIMessage,\n  type NeedsApprovalMap,\n  registerAgent,\n  type TypedUIMessage,\n} from \"./registry\";\n", "import { ulid } from \"ulid\";\nimport type { AnyAgent } from \"../agent/client\";\nimport { getSandboxContexts } from \"../agent/client\";\nimport type { SandboxBinding, SandboxInstance } from \"./adapter\";\nimport { exec } from \"./exec\";\nimport { getDomain } from \"./get-domain\";\nimport { getStatus } from \"./get-status\";\nimport { kill } from \"./kill\";\nimport { readFile } from \"./read-file\";\nimport { setup } from \"./setup\";\nimport { snapshot } from \"./snapshot\";\nimport { start } from \"./start\";\nimport { stop } from \"./stop\";\nimport { updateNetworkPolicy } from \"./update-network-policy\";\nimport { writeFiles } from \"./write-files\";\n\nexport type SandboxContext = {\n  agent: AnyAgent;\n  sandboxId: string | null;\n  _pendingSetup?: Promise<SandboxInstance>;\n  _cwd?: string;\n  _sessionId?: string;\n};\n\nexport type SandboxOptions = {\n  id?: string;\n};\n\nexport type SandboxHandle<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[] = SandboxBinding,\n> = ReturnType<typeof sandboxHandle<TSandboxBindings>>;\n\nexport function sandboxHandle<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[],\n>(agent: AnyAgent, id?: string | SandboxOptions) {\n  const sandboxId =\n    typeof id === \"string\" ? id : (id?.id ?? `sandbox_${ulid()}`);\n\n  const contexts = getSandboxContexts(agent);\n  let sandboxContext = contexts.get(sandboxId);\n  if (!sandboxContext) {\n    sandboxContext = { agent, sandboxId };\n    contexts.set(sandboxId, sandboxContext);\n  }\n\n  return {\n    id: sandboxId,\n    get cwd() {\n      return sandboxContext!._cwd ?? \".\";\n    },\n    setup: (setup<TSandboxBindings>).bind(null, sandboxContext),\n    exec: exec.bind(null, sandboxContext),\n    readFile: readFile.bind(null, sandboxContext),\n    writeFiles: writeFiles.bind(null, sandboxContext),\n    getDomain: getDomain.bind(null, sandboxContext),\n    updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),\n    start: start.bind(null, sandboxContext),\n    stop: stop.bind(null, sandboxContext),\n    kill: kill.bind(null, sandboxContext),\n    snapshot: snapshot.bind(null, sandboxContext),\n    getStatus: getStatus.bind(null, sandboxContext),\n  };\n}\n\ntype SandboxMethods<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[],\n> = Omit<SandboxHandle<TSandboxBindings>, \"id\" | \"cwd\">;\n\nexport function lazySandboxHandle<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[],\n>(\n  agent: AnyAgent,\n  sessionId: string\n): SandboxMethods<TSandboxBindings> {\n  const sandboxContext: SandboxContext = {\n    agent,\n    sandboxId: null,\n    _sessionId: sessionId,\n  };\n\n  return {\n    setup: (setup<TSandboxBindings>).bind(null, sandboxContext),\n    exec: exec.bind(null, sandboxContext),\n    readFile: readFile.bind(null, sandboxContext),\n    writeFiles: writeFiles.bind(null, sandboxContext),\n    getDomain: getDomain.bind(null, sandboxContext),\n    updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),\n    start: start.bind(null, sandboxContext),\n    stop: stop.bind(null, sandboxContext),\n    kill: kill.bind(null, sandboxContext),\n    snapshot: snapshot.bind(null, sandboxContext),\n    getStatus: getStatus.bind(null, sandboxContext),\n  };\n}\n", "export const isVercel = async () => {\n  try {\n    const { getVercelOidcToken } = await import(\"@vercel/oidc\");\n    return Boolean(await getVercelOidcToken());\n  } catch {\n    return false;\n  }\n};\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 StorageConflictError extends errore.createTaggedError({\n  name: \"StorageConflictError\",\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 { isVercel } from \"../agent/is-vercel\";\nimport { SandboxError, SandboxNotFoundError } from \"../errors\";\nimport type { Sandbox } from \"../storage/types\";\nimport { createLogger } from \"../utils/logger\";\nimport type {\n  SandboxBinding,\n  SandboxInstance,\n  SandboxSetupFields,\n  WithConfig,\n} from \"./adapter\";\nimport type { BuiltInBindings } from \"./bindings\";\nimport type { SandboxContext } from \"./handle\";\n\nconst POLL_INTERVAL_MS = 50;\nconst POLL_TIMEOUT_MS = 5 * 60 * 1000;\n\ntype SetupCommonFields = Partial<SandboxSetupFields> & {\n  cwd?: string;\n  run?: (sandbox: SandboxInstance) => Promise<void>;\n  signal?: AbortSignal;\n};\n\ntype ToBindingUnion<T extends SandboxBinding | SandboxBinding[]> =\n  T extends SandboxBinding[] ? T[number] : T;\n\ntype SetupOptsForBinding<B extends SandboxBinding> =\n  // biome-ignore lint/suspicious/noExplicitAny: .\n  B extends SandboxBinding<infer T, infer C, any>\n    ? { type: T } & SetupCommonFields & WithConfig<C>\n    : never;\n\ntype IsDefault<T extends SandboxBinding | SandboxBinding[]> =\n  SandboxBinding extends T ? true : false;\n\nexport type SetupOpts<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[],\n> =\n  | SetupOptsForBinding<\n      IsDefault<TSandboxBindings> extends true\n        ? BuiltInBindings\n        : ToBindingUnion<TSandboxBindings>\n    >\n  | (SetupCommonFields & { type?: never });\n\nexport type SetupResult = {\n  done: Promise<SandboxInstance>;\n};\n\nfunction isSingleBinding(\n  sandbox: SandboxBinding | SandboxBinding[]\n): sandbox is SandboxBinding {\n  return !Array.isArray(sandbox);\n}\n\n/** Auto-detect the default binding: vercel \u2192 docker \u2192 local cascade. */\nasync function getDefaultBinding(): Promise<SandboxBinding> {\n  if (await isVercel()) {\n    const { vercelSandbox } = await import(\"./bindings/vercel\");\n    return vercelSandbox();\n  }\n\n  if (await isDockerAvailable()) {\n    const { dockerSandbox } = await import(\"./bindings/docker\");\n    return dockerSandbox();\n  }\n\n  const { localSandbox } = await import(\"./bindings/local\");\n  return localSandbox();\n}\n\nasync function isDockerAvailable(): Promise<boolean> {\n  try {\n    const { execSync } = await import(\"node:child_process\");\n    execSync(\"docker sandbox --help\", { stdio: \"ignore\", timeout: 5000 });\n    return true;\n  } catch {\n    return false;\n  }\n}\n\nasync function resolveBinding(\n  sandbox: SandboxBinding | SandboxBinding[] | undefined,\n  type?: string\n): Promise<{ binding: SandboxBinding; bindingName: string }> {\n  if (!sandbox) {\n    const binding = await getDefaultBinding();\n    return { binding, bindingName: binding.type };\n  }\n  if (isSingleBinding(sandbox)) {\n    return { binding: sandbox, bindingName: sandbox.type };\n  }\n  if (type) {\n    const match = sandbox.find((b) => b.type === type);\n    if (!match) {\n      throw new SandboxError({\n        reason: `Unknown binding type: \"${type}\". Available: ${sandbox.map((b) => b.type).join(\", \")}`,\n      });\n    }\n    return { binding: match, bindingName: match.type };\n  }\n  if (sandbox.length === 0) {\n    throw new SandboxError({ reason: \"No sandbox bindings configured\" });\n  }\n  return { binding: sandbox[0], bindingName: sandbox[0].type };\n}\n\nasync function resolveSandboxId(ctx: SandboxContext): Promise<string> {\n  if (ctx.sandboxId) {\n    return ctx.sandboxId;\n  }\n  if (!ctx._sessionId) {\n    throw new Error(\n      \"SandboxContext has no sandboxId and no sessionId to resolve from\"\n    );\n  }\n  const record = await ctx.agent.storage.session.get(ctx._sessionId);\n  ctx.sandboxId = record?.sandboxId ?? ctx._sessionId;\n  return ctx.sandboxId;\n}\n\nexport async function setup<\n  TSandboxBindings extends SandboxBinding | SandboxBinding[],\n>(\n  ctx: SandboxContext,\n  opts?: SetupOpts<TSandboxBindings>\n): Promise<SetupResult> {\n  \"use step\";\n  const sandboxId = await resolveSandboxId(ctx);\n  if (ctx._pendingSetup) {\n    return { done: ctx._pendingSetup };\n  }\n\n  const agent = ctx.agent;\n  const log = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"sandbox:setup\",\n    context: { sandboxId },\n  });\n\n  const doneResolveBinding = log.time(\"resolve binding\", {\n    requestedType: opts?.type,\n  });\n  const { binding, bindingName } = await resolveBinding(\n    agent.options.sandbox,\n    opts?.type\n  );\n  doneResolveBinding();\n\n  if (opts?.cwd) {\n    ctx._cwd = opts.cwd;\n  }\n\n  const setupLog = log.withContext({\n    binding: bindingName,\n    version: opts?.version ?? binding.defaults?.version,\n  });\n\n  // biome-ignore lint/suspicious/noConfusingVoidType: .\n  let resolveBlockingPromise = (value: void) => value;\n  // biome-ignore lint/suspicious/noExplicitAny: .\n  // biome-ignore lint/suspicious/noEmptyBlockStatements: .\n  let rejectBlockingPromise = (_reason?: any) => {};\n  const blockingPromise = new Promise<void>((resolve, reject) => {\n    resolveBlockingPromise = resolve;\n    rejectBlockingPromise = reject;\n  });\n\n  ctx._pendingSetup = (async () => {\n    let sleepOldInstance: Promise<void> | undefined;\n    const donePendingSetup = setupLog.time(\"pending setup\");\n\n    try {\n      const doneStorageGet = setupLog.time(\"storage.sandbox.get\");\n      const record = await agent.storage.sandbox.get(sandboxId);\n      doneStorageGet();\n\n      const bindingDefaults = binding.defaults ?? {};\n      const effectiveVersion = opts?.version ?? bindingDefaults.version ?? null;\n      const effectiveRun = opts?.run ?? bindingDefaults.run;\n      const effectiveNetworkPolicy =\n        opts?.networkPolicy ?? bindingDefaults.networkPolicy ?? null;\n      const effectiveCwd =\n        ctx._cwd || opts?.cwd || bindingDefaults.cwd || null;\n      if (effectiveCwd) {\n        ctx._cwd = effectiveCwd;\n      }\n      const effectiveConfig = {\n        ...bindingDefaults.config,\n        // biome-ignore lint/suspicious/noExplicitAny: opts is a union; config exists on binding-specific branches\n        ...(opts as any)?.config,\n      };\n\n      if (record) {\n        const versionMatches =\n          !effectiveVersion || record.setup?.version === effectiveVersion;\n\n        if (versionMatches && record.setup?.completedAt) {\n          resolveBlockingPromise();\n          const doneConnectExisting = setupLog.time(\n            \"connect existing sandbox\",\n            {\n              setupState: \"completed\",\n            }\n          );\n          const instance = await binding.connect({\n            agent,\n            metadata: record.setup.metadata,\n            cwd: record.setup.cwd,\n            signal: opts?.signal,\n            log: setupLog,\n          });\n          doneConnectExisting();\n          ctx._cwd = instance.cwd;\n          return instance;\n        }\n\n        if (versionMatches && !record.setup?.completedAt) {\n          resolveBlockingPromise();\n          const donePollUntilReady = setupLog.time(\"poll until ready\");\n          await pollUntilReady(ctx, opts?.signal);\n          donePollUntilReady();\n\n          const doneStorageGetPostPoll = setupLog.time(\n            \"storage.sandbox.get (post-poll)\"\n          );\n          const ready = await agent.storage.sandbox.get(sandboxId);\n          doneStorageGetPostPoll();\n\n          if (!ready) {\n            throw new SandboxNotFoundError({ id: sandboxId });\n          }\n          const doneConnectExisting = setupLog.time(\n            \"connect existing sandbox\",\n            {\n              setupState: \"post-poll\",\n            }\n          );\n          const instance = await binding.connect({\n            agent,\n            metadata: ready.setup.metadata,\n            cwd: ready.setup.cwd,\n            signal: opts?.signal,\n            log: setupLog,\n          });\n          doneConnectExisting();\n          ctx._cwd = instance.cwd;\n          return instance;\n        }\n\n        if (record.setup.metadata) {\n          const oldInstanceMetadata = record.setup.metadata;\n          sleepOldInstance = (async () => {\n            const doneStopOldInstance = setupLog.time(\n              \"stop old sandbox instance\"\n            );\n            try {\n              const old = await binding.connect({\n                agent,\n                metadata: oldInstanceMetadata,\n                log: setupLog,\n              });\n              await old.stop();\n            } catch {\n              doneStopOldInstance();\n              return;\n            }\n            doneStopOldInstance();\n          })();\n        }\n      }\n\n      const now = Date.now();\n      const setup_: Sandbox[\"setup\"] = {\n        binding: bindingName,\n        version: effectiveVersion ?? record?.setup?.version ?? null,\n        networkPolicy:\n          effectiveNetworkPolicy ?? record?.setup?.networkPolicy ?? null,\n        completedAt: null,\n        metadata: null,\n        cwd: null,\n      };\n\n      const newRecord: Sandbox = {\n        id: sandboxId,\n        setup: setup_,\n        createdAt: record?.createdAt ?? now,\n        lastActiveAt: now,\n      };\n\n      const doneStorageSetPending = setupLog.time(\n        \"storage.sandbox.set (setup pending)\"\n      );\n      await agent.storage.sandbox.set(sandboxId, newRecord);\n      doneStorageSetPending();\n\n      resolveBlockingPromise();\n\n      const doneBindingCreate = setupLog.time(\"binding.create\");\n      const result = await binding.create({\n        cwd: ctx._cwd ?? \".\",\n        agent,\n        setup: {\n          ...setup_,\n          run: effectiveRun,\n          config: effectiveConfig,\n        },\n        record: newRecord,\n        signal: opts?.signal,\n        log: setupLog,\n      });\n      doneBindingCreate();\n\n      setup_.completedAt = Date.now();\n      setup_.metadata = result.metadata;\n      setup_.cwd = result.instance.cwd;\n\n      const doneStorageUpdateComplete = setupLog.time(\n        \"storage.sandbox.update (setup complete)\"\n      );\n      await agent.storage.sandbox.update(sandboxId, {\n        setup: setup_,\n        lastActiveAt: Date.now(),\n      });\n      doneStorageUpdateComplete();\n\n      ctx._cwd = result.instance.cwd;\n      return result.instance;\n    } catch (err) {\n      ctx._pendingSetup = undefined;\n      rejectBlockingPromise(err);\n      throw err;\n    } finally {\n      const doneFinalize = setupLog.time(\"finalize setup\");\n      await sleepOldInstance;\n      doneFinalize();\n      donePendingSetup();\n    }\n  })();\n\n  const doneWaitForUnblock = setupLog.time(\"wait for setup unblock\");\n  await blockingPromise;\n  doneWaitForUnblock();\n  return { done: ctx._pendingSetup };\n}\n\nexport async function resolve(\n  ctx: SandboxContext,\n  signal?: AbortSignal\n): Promise<SandboxInstance> {\n  const sandboxId = await resolveSandboxId(ctx);\n  const agent = ctx.agent;\n  const log = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"sandbox:resolve\",\n    context: { sandboxId },\n  });\n\n  if (ctx._pendingSetup) {\n    return await ctx._pendingSetup;\n  }\n\n  let record = await agent.storage.sandbox.get(sandboxId);\n\n  if (!record) {\n    const { done } = await setup(ctx, { signal });\n    return await done;\n  }\n\n  if (!record.setup?.completedAt) {\n    await pollUntilReady(ctx, signal);\n    record = await agent.storage.sandbox.get(sandboxId);\n  }\n\n  if (!record) {\n    throw new SandboxNotFoundError({ id: sandboxId });\n  }\n\n  const { binding } = await resolveBinding(\n    ctx.agent.options.sandbox,\n    record.setup.binding\n  );\n\n  agent.storage.sandbox\n    .update(sandboxId, { lastActiveAt: Date.now() })\n    .catch((error) => {\n      log.error(\"Failed to update sandbox lastActiveAt\", { error });\n    });\n\n  const instance = await binding.connect({\n    agent,\n    metadata: record.setup.metadata,\n    cwd: record.setup.cwd,\n    signal,\n    log,\n  });\n  ctx._cwd = instance.cwd;\n  return instance;\n}\n\nasync function pollUntilReady(\n  ctx: SandboxContext,\n  signal?: AbortSignal\n): Promise<void> {\n  const id = ctx.sandboxId!;\n  const deadline = Date.now() + POLL_TIMEOUT_MS;\n\n  while (Date.now() < deadline) {\n    signal?.throwIfAborted();\n\n    const record = await ctx.agent.storage.sandbox.get(id);\n    if (record?.setup?.completedAt) {\n      return;\n    }\n\n    await new Promise((r) => setTimeout(r, POLL_INTERVAL_MS));\n  }\n\n  throw new SandboxError({\n    reason: `Timed out waiting for sandbox \"${id}\" setup to complete`,\n  });\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function exec(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"exec\"]>\n) {\n  // note: .exec can't be a step because it returns logs and a result promise which can't be serialized\n  const sbx = await resolve(sandboxContext);\n  return await sbx.exec(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function getDomain(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"getDomain\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.getDomain(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function getStatus(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"getStatus\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.getStatus(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function kill(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"kill\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.kill(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function readFile(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"readFile\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.readFile(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function snapshot(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"snapshot\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.snapshot(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function start(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"start\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.start(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function stop(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"stop\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.stop(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function updateNetworkPolicy(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"updateNetworkPolicy\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.updateNetworkPolicy(...opts);\n}\n", "import type { SandboxInstance } from \"./adapter\";\nimport type { SandboxContext } from \"./handle\";\nimport { resolve } from \"./setup\";\n\nexport async function writeFiles(\n  sandboxContext: SandboxContext,\n  ...opts: Parameters<SandboxInstance[\"writeFiles\"]>\n) {\n  \"use step\";\n  const sbx = await resolve(sandboxContext);\n  return await sbx.writeFiles(...opts);\n}\n", "import type { ToolSet } from \"ai\";\nimport { ulid } from \"ulid\";\nimport type { AnyAgent } from \"../agent/client\";\nimport { getSessionContexts } from \"../agent/client\";\nimport type { SandboxBinding } from \"../sandbox/adapter\";\nimport { lazySandboxHandle } from \"../sandbox/handle\";\nimport { history } from \"./history\";\nimport { interrupt } from \"./interrupt\";\nimport { send } from \"./send\";\nimport { stream } from \"./stream\";\nimport { update } from \"./update\";\nimport { usage } from \"./usage\";\n\ntype SessionOptions = {\n  id?: string;\n};\n\nexport type SessionContext = {\n  agent: AnyAgent;\n  sessionId: string;\n};\n\nexport function sessionHandle<\n  Tools extends ToolSet,\n  TContext,\n  TSandboxBindings extends SandboxBinding | SandboxBinding[] = SandboxBinding,\n  TMessageMetadata = unknown,\n  TState extends Record<string, unknown> = Record<string, unknown>,\n>(agent: AnyAgent, id?: string | SessionOptions) {\n  const sessionId =\n    typeof id === \"string\" ? id : (id?.id ?? `session_${ulid()}`);\n\n  const contexts = getSessionContexts(agent);\n  let sessionContext = contexts.get(sessionId);\n  if (!sessionContext) {\n    sessionContext = { agent, sessionId };\n    contexts.set(sessionId, sessionContext);\n  }\n\n  return {\n    id: sessionId,\n    send: (send<Tools, TContext, TMessageMetadata, TState>).bind(\n      null,\n      sessionContext\n    ),\n    stream: stream.bind(null, sessionContext),\n    history: (history<Tools>).bind(null, sessionContext),\n    interrupt: interrupt.bind(null, sessionContext),\n    usage: usage.bind(null, sessionContext),\n    update: (update<TState>).bind(null, sessionContext),\n    sandbox: lazySandboxHandle<TSandboxBindings>(agent, sessionId),\n  };\n}\n", "import type {\n  InferUITools,\n  ReasoningUIPart,\n  TextUIPart,\n  ToolSet,\n  UIMessage,\n} from \"ai\";\nimport type { AgentDataTypes } from \"../agent/registry\";\nimport type { Message, Part } from \"../storage\";\n\n/** Sub-tool parts (from JavaScript meta-tool) use this toolCallId prefix. */\nfunction isSubToolPart(part: Part): boolean {\n  const p = part.part as { toolCallId?: string };\n  return typeof p?.toolCallId === \"string\" && p.toolCallId.startsWith(\"js_tc_\");\n}\n\nfunction isStreamingContent(\n  part: unknown\n): part is TextUIPart | ReasoningUIPart {\n  return (\n    typeof part === \"object\" &&\n    part !== null &&\n    \"type\" in part &&\n    (part.type === \"text\" || part.type === \"reasoning\")\n  );\n}\n\nexport function applyInterruptCutoff({\n  parts,\n  lastPart,\n}: {\n  parts: Part[];\n  lastPart: { index: number; part: unknown };\n}): Part[] {\n  return parts\n    .filter((p) => p.index <= lastPart.index)\n    .map((p) => {\n      if (p.index !== lastPart.index) {\n        return p;\n      }\n      /**\n       * Only replace content for streaming parts (text, reasoning) where the\n       * client's truncated version is more accurate than the server's full\n       * version. For stateful parts (tools), the server's version is\n       * authoritative \u2014 it may have evolved (e.g. approval-requested \u2192\n       * output-denied) after the client snapshot was taken.\n       */\n      if (isStreamingContent(lastPart.part)) {\n        return { ...p, part: lastPart.part } as Part;\n      }\n      return p;\n    });\n}\n\nexport function assembleUIMessages<Tools extends ToolSet>(opts: {\n  messages: Message[];\n  parts: Part[];\n  until?: number;\n  includeQueued?: boolean;\n  /** When true, filters out sub-tool parts so the LLM doesn't see them. */\n  excludeSubToolParts?: boolean;\n}) {\n  type TypedUIMessage = UIMessage<unknown, AgentDataTypes, InferUITools<Tools>>;\n\n  let filtered = opts.messages;\n\n  if (opts.until !== undefined) {\n    const until = opts.until;\n    filtered = filtered.filter((m) => m.createdAt <= until);\n  }\n\n  if (!opts.includeQueued) {\n    filtered = filtered.filter((m) => m.startedAt !== null);\n  }\n\n  // Sort by createdAt so messages are in chronological order regardless of\n  // how the storage layer returns them (e.g. local FS reads in readdir order).\n  filtered = [...filtered].sort((a, b) => a.createdAt - b.createdAt);\n\n  const parts = opts.excludeSubToolParts\n    ? opts.parts.filter((p) => !isSubToolPart(p))\n    : opts.parts;\n\n  const partsByMessage = new Map<string, Part[]>();\n  for (const part of parts) {\n    const messageId = String(part.messageId);\n    const existing = partsByMessage.get(messageId) ?? [];\n    existing.push(part);\n    partsByMessage.set(messageId, existing);\n  }\n\n  return filtered\n    .map((m) => {\n      const messageId = String(m.id);\n      let messageParts = partsByMessage.get(messageId) ?? [];\n      messageParts.sort((a, b) => a.index - b.index);\n\n      if (m.interruptedLastPart != null) {\n        messageParts = applyInterruptCutoff({\n          parts: messageParts,\n          lastPart: m.interruptedLastPart,\n        });\n      }\n\n      return {\n        id: messageId,\n        role: m.role,\n        parts: messageParts.map((p) => p.part),\n        ...(m.metadata != null ? { metadata: m.metadata } : {}),\n      } as TypedUIMessage;\n    })\n    .filter((m) => m.parts.length > 0);\n}\n", "export type StepUsage = {\n  stepIndex: number;\n  model: string;\n  inputTokens: number;\n  outputTokens: number;\n  totalTokens: number;\n  cacheReadTokens: number;\n  cacheWriteTokens: number;\n  reasoningTokens: number;\n};\n\nexport type UsageSummary = {\n  model: string;\n  inputTokens: number;\n  outputTokens: number;\n  totalTokens: number;\n  cacheReadTokens: number;\n  cacheWriteTokens: number;\n  reasoningTokens: number;\n  stepCount: number;\n};\n\nexport type SessionUsage = {\n  total: UsageSummary & { messageCount: number };\n  byMessageId: Record<string, UsageSummary | null>;\n};\n\nexport function computeSessionUsage(\n  messages: Array<{\n    id: string;\n    usage: { steps: StepUsage[]; summary: UsageSummary } | null;\n  }>\n): SessionUsage {\n  const byMessageId: Record<string, UsageSummary | null> = {};\n  for (const m of messages) {\n    byMessageId[m.id] = m.usage?.summary ?? null;\n  }\n\n  const summaries = messages\n    .map((m) => m.usage?.summary)\n    .filter((s): s is UsageSummary => s !== undefined);\n\n  const total = {\n    model: summaries[0]?.model ?? \"unknown\",\n    inputTokens: summaries.reduce((acc, s) => acc + s.inputTokens, 0),\n    outputTokens: summaries.reduce((acc, s) => acc + s.outputTokens, 0),\n    totalTokens: summaries.reduce((acc, s) => acc + s.totalTokens, 0),\n    cacheReadTokens: summaries.reduce((acc, s) => acc + s.cacheReadTokens, 0),\n    cacheWriteTokens: summaries.reduce((acc, s) => acc + s.cacheWriteTokens, 0),\n    reasoningTokens: summaries.reduce((acc, s) => acc + s.reasoningTokens, 0),\n    stepCount: summaries.reduce((acc, s) => acc + s.stepCount, 0),\n    messageCount: summaries.length,\n  };\n\n  return { total, byMessageId };\n}\n", "import type { ToolSet } from \"ai\";\nimport { assembleUIMessages } from \"../utils/ui\";\nimport { computeSessionUsage, type SessionUsage } from \"../utils/usage\";\nimport type { SessionContext } from \"./handle\";\n\nexport type HistoryResult = {\n  messages: ReturnType<typeof assembleUIMessages>;\n  streamingMessageId: string | null;\n  usage: SessionUsage;\n};\n\nexport async function history<Tools extends ToolSet>(\n  sessionContext: SessionContext\n) {\n  \"use step\";\n  const [session, messages, parts] = await Promise.all([\n    sessionContext.agent.storage.session.get(sessionContext.sessionId),\n    sessionContext.agent.storage.message.listBySession(\n      sessionContext.sessionId\n    ),\n    sessionContext.agent.storage.part.listBySession(sessionContext.sessionId),\n  ]);\n\n  const lastAssistant =\n    messages.find(\n      (m) => m.id === session?.lastMessageId && m.role === \"assistant\"\n    ) ?? messages.filter((m) => m.role === \"assistant\").at(-1);\n\n  const isStreaming =\n    lastAssistant?.completedAt === null &&\n    lastAssistant?.interruptedAt === null;\n  const streamingMessageId = isStreaming ? lastAssistant.id : null;\n  const visibleUntil = lastAssistant?.createdAt;\n\n  const assembled = assembleUIMessages<Tools>({\n    messages,\n    parts,\n    includeQueued: visibleUntil !== undefined,\n    until: visibleUntil,\n  });\n\n  const usage = computeSessionUsage(messages);\n\n  return { messages: assembled, streamingMessageId, usage };\n}\n", "import type { UIMessage } from \"ai\";\nimport type { SessionContext } from \"./handle\";\n\nexport type InterruptOptions = {\n  lastPart?: {\n    index: number;\n    part: unknown;\n  };\n};\n\nexport async function interrupt(\n  sessionContext: SessionContext,\n  opts?: InterruptOptions\n) {\n  \"use step\";\n  const messages = await sessionContext.agent.storage.message.listBySession(\n    sessionContext.sessionId\n  );\n\n  const now = Date.now();\n  let pendingAssistantMessages = messages.filter(\n    (m) =>\n      m.role === \"assistant\" &&\n      m.completedAt === null &&\n      m.interruptedAt === null\n  );\n\n  if (pendingAssistantMessages.length === 0) {\n    const MAX_RETROACTIVE_MS = 5_000;\n    const latest = messages\n      .filter(\n        (m) =>\n          m.role === \"assistant\" &&\n          m.interruptedAt === null &&\n          m.completedAt !== null &&\n          now - m.completedAt < MAX_RETROACTIVE_MS\n      )\n      .sort((a, b) => b.createdAt - a.createdAt)\n      .at(0);\n    if (latest) {\n      pendingAssistantMessages = [latest];\n    }\n  }\n\n  await Promise.all(\n    pendingAssistantMessages.map((m) =>\n      sessionContext.agent.storage.message.set(m.id, {\n        ...m,\n        interruptedAt: now,\n        interruptedLastPart: opts?.lastPart ?? null,\n      })\n    )\n  );\n\n  const messageIds = new Set(pendingAssistantMessages.map((m) => m.id));\n  if (messageIds.size === 0) {\n    return;\n  }\n\n  const allParts = await sessionContext.agent.storage.part.listBySession(\n    sessionContext.sessionId\n  );\n\n  for (const p of allParts) {\n    if (!messageIds.has(p.messageId)) {\n      continue;\n    }\n    if (\n      p.part &&\n      \"state\" in p.part &&\n      (p.part as { state: string }).state === \"approval-requested\" &&\n      \"approval\" in p.part\n    ) {\n      const approval = (p.part as { approval?: { id: string } }).approval;\n      if (approval?.id) {\n        await sessionContext.agent.storage.part.set(p.id, {\n          ...p,\n          part: {\n            ...p.part,\n            state: \"output-denied\",\n            approval: {\n              ...approval,\n              approved: false,\n              reason: \"interrupted\",\n            },\n          } as UIMessage[\"parts\"][number],\n        });\n      }\n    }\n  }\n}\n", "import type { FinishReason, ToolSet, UIMessage, UIMessageChunk } from \"ai\";\nimport { all } from \"better-all\";\nimport * as errore from \"errore\";\nimport { ulid } from \"ulid\";\nimport type { TypedUIMessage } from \"../agent/registry\";\nimport type { SkillSummary } from \"../skills/types\";\nimport type { Message, Part } from \"../storage/types\";\nimport { createLogger } from \"../utils/logger\";\nimport type { StepUsage } from \"../utils/usage\";\nimport {\n  type StreamTextResult,\n  streamText,\n  streamTextStep,\n} from \"./ai-loop/stream-text\";\nimport type { SessionContext } from \"./handle\";\nimport { interrupt } from \"./interrupt\";\n\nconst DEFAULT_STEP_THROUGHPUT = 100;\n\nexport type ApprovalData = {\n  approvalId: string;\n  approved: boolean;\n  reason?: string;\n};\n\ntype MessageData<Tools extends ToolSet, TMessageMetadata = unknown> =\n  | UIMessage<TMessageMetadata>\n  | {\n      role?: TypedUIMessage<unknown, Tools>[\"role\"];\n      parts: TypedUIMessage<unknown, Tools>[\"parts\"];\n      metadata?: TMessageMetadata | null;\n      id?: string;\n    };\n\nexport type SendInput<Tools extends ToolSet, TMessageMetadata = unknown> =\n  | string\n  | MessageData<Tools, TMessageMetadata>\n  | { type: \"message\"; message: MessageData<Tools, TMessageMetadata> }\n  | { type: \"approval\"; approval: ApprovalData };\n\nexport type SendOptions<TContext, TState = unknown> = {\n  interruptIfStreaming?:\n    | boolean\n    | { lastPart: { index: number; part: unknown } };\n  context?: TContext;\n  initialState?: TState;\n  abortSignal?: AbortSignal;\n  sandboxId?: string;\n};\n\nexport type SendResult = {\n  assistantMessageId: string;\n  done: true | Promise<true>;\n};\n\nexport type ActiveRun = {\n  chunks: UIMessageChunk[];\n  writable: WritableStream<UIMessageChunk>;\n  done: Promise<void>;\n  /** Resolves when a new chunk is available. Replaced after each notification. */\n  onChunk: Promise<void>;\n};\n\nconst ACTIVE_RUNS_KEY = Symbol.for(\"experimental-agent:activeRuns\");\n\nexport function getActiveRuns(): Map<string, ActiveRun> {\n  const g = globalThis as Record<symbol, Map<string, ActiveRun> | undefined>;\n  if (!g[ACTIVE_RUNS_KEY]) {\n    g[ACTIVE_RUNS_KEY] = new Map();\n  }\n  return g[ACTIVE_RUNS_KEY];\n}\n\nexport async function send<\n  Tools extends ToolSet,\n  TContext,\n  TMessageMetadata = unknown,\n  TState = unknown,\n>(\n  sessionContext: SessionContext,\n  input:\n    | SendInput<Tools, TMessageMetadata>\n    | SendInput<Tools, TMessageMetadata>[],\n  opts?: SendOptions<TContext, TState>\n): Promise<SendResult> {\n  if (opts?.interruptIfStreaming) {\n    const lastPart =\n      typeof opts.interruptIfStreaming === \"object\"\n        ? opts.interruptIfStreaming.lastPart\n        : undefined;\n    await interrupt(sessionContext, { lastPart });\n  }\n\n  const {\n    workflowRunId,\n    workflowWritable,\n    existingPartCount,\n    assistantMessageId,\n    assistantMessageCreatedAt,\n    state,\n  } = await errore.tryAsync({\n    try: () =>\n      initSessionStep<Tools>({\n        sessionContext,\n        sessionId: sessionContext.sessionId,\n        sandboxId: opts?.sandboxId,\n        initialState: opts?.initialState as Record<string, unknown> | undefined,\n        input,\n      }),\n    catch: (e) => {\n      if (e instanceof Error) {\n        throw e;\n      }\n      throw new SendError({\n        id: sessionContext.sessionId,\n        reason: String(e),\n        cause: e,\n      });\n    },\n  });\n\n  const loopOpts = {\n    sessionContext,\n    assistantMessageId,\n    createdAt: assistantMessageCreatedAt,\n    context: (opts?.context ?? {}) as Record<string, unknown>,\n    state,\n    lastPartIndex: existingPartCount,\n    workflowRunId,\n  };\n\n  if (workflowWritable) {\n    await runLoop<Tools>({ ...loopOpts, writable: workflowWritable });\n    return { assistantMessageId, done: true };\n  }\n\n  const chunks: UIMessageChunk[] = [];\n  let notifyChunk: () => void;\n  let onChunk = new Promise<void>((r) => {\n    notifyChunk = r;\n  });\n\n  const writable = new WritableStream<UIMessageChunk>({\n    write(chunk) {\n      chunks.push(chunk);\n      const prev = notifyChunk;\n      onChunk = new Promise<void>((r) => {\n        notifyChunk = r;\n      });\n      prev();\n    },\n  });\n\n  const done = runLoop<Tools>({ ...loopOpts, writable });\n\n  const activeRun: ActiveRun = {\n    chunks,\n    writable,\n    done,\n    get onChunk() {\n      return onChunk;\n    },\n  };\n  const activeRuns = getActiveRuns();\n  activeRuns.set(assistantMessageId, activeRun);\n  done.finally(() => activeRuns.delete(assistantMessageId));\n\n  return { assistantMessageId, done: done.then(() => true) };\n}\n\n/* -------------------------------------------------------------------------------------------------\n * AI Loop\n * -----------------------------------------------------------------------------------------------*/\n\nasync function runLoop<Tools extends ToolSet>({\n  workflowRunId,\n  sessionContext,\n  assistantMessageId,\n  createdAt,\n  context: initialContext,\n  state,\n  writable,\n  lastPartIndex: initialPartIndex,\n}: {\n  workflowRunId: string | undefined;\n  sessionContext: SessionContext;\n  assistantMessageId: string;\n  createdAt: number;\n  context: Record<string, unknown>;\n  state: Record<string, unknown>;\n  writable: WritableStream;\n  lastPartIndex: number;\n}): Promise<void> {\n  const agent = sessionContext.agent;\n\n  const log = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"ai-loop\",\n  });\n  const msgLog = log.withContext({\n    sessionId: sessionContext.sessionId,\n    messageId: assistantMessageId,\n  });\n  const doneMessage = msgLog.time(\"onMessage\");\n\n  let finishReason: FinishReason | undefined;\n  let lastPartIndex = initialPartIndex;\n  const usageSteps: StepUsage[] = [];\n  let discoveredSkills: SkillSummary[] | null = null;\n  let context = initialContext;\n\n  try {\n    while (finishReason !== \"stop\") {\n      const streamTextFn = workflowRunId\n        ? streamTextStep<Tools>\n        : streamText<Tools>;\n      const throughput = workflowRunId ? 5 : DEFAULT_STEP_THROUGHPUT;\n\n      const result: StreamTextResult = await streamTextFn({\n        stepThroughput: throughput,\n        assistantMessageId,\n        agent,\n        sessionId: sessionContext.sessionId,\n        context,\n        state,\n        createdAt,\n        writable,\n        lastPartIndex,\n        stepIndexOffset: usageSteps.length,\n        discoveredSkills,\n      });\n\n      finishReason = result.finishReason;\n      lastPartIndex = result.lastPartIndex;\n      usageSteps.push(...result.usageSteps);\n      discoveredSkills = result.discoveredSkills;\n      context = result.context;\n\n      if (result.maxSteps != null && usageSteps.length >= result.maxSteps) {\n        msgLog.info(\"reached maxSteps limit\", { maxSteps: result.maxSteps });\n        break;\n      }\n\n      if (result.pendingApprovals.length > 0) {\n        msgLog.info(\"pending approvals, stopping loop\", {\n          count: result.pendingApprovals.length,\n          tools: result.pendingApprovals.map((p) => p.toolName),\n        });\n        break;\n      }\n    }\n\n    doneMessage({ totalSteps: usageSteps.length });\n  } catch (err) {\n    msgLog.error(\"AI loop failed\", {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    throw err;\n  } finally {\n    await completeMessageStep({\n      assistantMessageId,\n      sessionContext,\n      writable,\n      usageSteps,\n    });\n  }\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Steps\n * -----------------------------------------------------------------------------------------------*/\n\nasync function initSessionStep<Tools extends ToolSet>({\n  sessionContext,\n  sessionId,\n  sandboxId,\n  initialState,\n  input,\n}: {\n  sessionContext: SessionContext;\n  sessionId: string;\n  sandboxId?: string;\n  initialState?: Record<string, unknown>;\n  input: SendInput<Tools> | SendInput<Tools>[];\n}) {\n  \"use step\";\n\n  let workflowRunId: string | undefined;\n  let workflowWritable: WritableStream | undefined;\n  try {\n    const { getWorkflowMetadata, getWritable } = await import(\"workflow\");\n    workflowRunId = getWorkflowMetadata().workflowRunId;\n    workflowWritable = getWritable();\n  } catch {\n    // workflow not installed\n  }\n\n  const newMessages: Message[] = [];\n  const newParts: Part[] = [];\n  const newApprovals: ApprovalData[] = [];\n  if (Array.isArray(input)) {\n    for (const i of input) {\n      const { message, parts, approvals } = toMessageAndParts({\n        sessionId: sessionContext.sessionId,\n        input: i,\n        defaultRole: \"user\",\n      });\n      if (message) {\n        newMessages.push(message);\n      }\n      newParts.push(...parts);\n      newApprovals.push(...approvals);\n    }\n  } else {\n    const { message, parts, approvals } = toMessageAndParts({\n      input,\n      sessionId: sessionContext.sessionId,\n      defaultRole: \"user\",\n    });\n    if (message) {\n      newMessages.push(message);\n    }\n    newParts.push(...parts);\n    newApprovals.push(...approvals);\n  }\n\n  const lastNewMessage = newMessages.at(-1);\n  const reuseAssistant = lastNewMessage?.role === \"assistant\";\n\n  const assistantMessageId = reuseAssistant\n    ? lastNewMessage.id\n    : `message_${ulid()}`;\n  const baseTime = Date.now();\n  const assistantMessageCreatedAt = baseTime + newMessages.length;\n\n  if (!reuseAssistant) {\n    newMessages.push({\n      id: assistantMessageId,\n      sessionId: sessionContext.sessionId,\n      role: \"assistant\",\n      createdAt: assistantMessageCreatedAt,\n      startedAt: null,\n      completedAt: null,\n      interruptedAt: null,\n      interruptedLastPart: null,\n      usage: null,\n      workflowRunId: null,\n      metadata: null,\n    });\n  }\n\n  const existingPartCount = reuseAssistant\n    ? newParts.filter((p) => p.messageId === assistantMessageId).length\n    : 0;\n\n  let session = await sessionContext.agent.storage.session.get(sessionId);\n  const now = Date.now();\n  if (!session) {\n    session = {\n      id: sessionId,\n      sandboxId: sandboxId ?? null,\n      lastMessageId: null,\n      createdAt: now,\n      updatedAt: now,\n    };\n    await sessionContext.agent.storage.session.set(session.id, session);\n  } else if (sandboxId && sandboxId !== session.sandboxId) {\n    session = { ...session, sandboxId, updatedAt: now };\n    await sessionContext.agent.storage.session.update(session.id, {\n      sandboxId,\n    });\n  }\n\n  await all({\n    async saveMessages() {\n      await Promise.all(\n        newMessages.map(async (m, i) => {\n          const existing = await sessionContext.agent.storage.message.get(m.id);\n          if (existing) {\n            return;\n          }\n          await sessionContext.agent.storage.message.set(m.id, {\n            ...m,\n            createdAt: baseTime + i,\n            workflowRunId:\n              m.id === assistantMessageId\n                ? (workflowRunId ?? null)\n                : m.workflowRunId,\n          });\n        })\n      );\n    },\n    async saveParts() {\n      await Promise.all(\n        newParts.map(async (p) =>\n          sessionContext.agent.storage.part.set(p.id, p)\n        )\n      );\n    },\n    async resolveApprovals() {\n      await Promise.all(\n        newApprovals.map(async (a) => {\n          const partId = approvalPartId(a.approvalId);\n          const part = await sessionContext.agent.storage.part.get(partId);\n          if (!part) {\n            throw new ApprovalNotFoundError({\n              id: sessionContext.sessionId,\n              approvalId: a.approvalId,\n            });\n          }\n          await sessionContext.agent.storage.part.set(partId, {\n            ...part,\n            part: {\n              ...part.part,\n              state: a.approved ? \"approval-responded\" : \"output-denied\",\n              approval: {\n                id: a.approvalId,\n                approved: a.approved,\n                reason: a.reason,\n              },\n            } as typeof part.part,\n          });\n        })\n      );\n    },\n    async updateSession() {\n      await sessionContext.agent.storage.session.set(session.id, {\n        ...session,\n        lastMessageId: assistantMessageId,\n        updatedAt: Date.now(),\n      });\n    },\n  });\n\n  const state: Record<string, unknown> = session.state\n    ? { ...session.state }\n    : { ...(initialState ?? {}) };\n\n  return {\n    workflowRunId,\n    workflowWritable,\n    existingPartCount,\n    assistantMessageId,\n    assistantMessageCreatedAt,\n    state,\n  };\n}\n\nasync function completeMessageStep({\n  assistantMessageId,\n  sessionContext,\n  writable,\n  usageSteps,\n}: {\n  assistantMessageId: string;\n  sessionContext: SessionContext;\n  writable: WritableStream;\n  usageSteps: StepUsage[];\n}) {\n  \"use step\";\n\n  const log = createLogger({\n    config: {\n      ...sessionContext.agent.options.logging,\n      name: sessionContext.agent.name,\n    },\n    subsystem: \"ai-loop\",\n  });\n  const msgLog = log.withContext({\n    sessionId: sessionContext.sessionId,\n    messageId: assistantMessageId,\n  });\n\n  const usage =\n    usageSteps.length > 0\n      ? {\n          steps: usageSteps,\n          summary: {\n            model: usageSteps[0]?.model ?? \"unknown\",\n            inputTokens: usageSteps.reduce((a, s) => a + s.inputTokens, 0),\n            outputTokens: usageSteps.reduce((a, s) => a + s.outputTokens, 0),\n            totalTokens: usageSteps.reduce((a, s) => a + s.totalTokens, 0),\n            cacheReadTokens: usageSteps.reduce(\n              (a, s) => a + s.cacheReadTokens,\n              0\n            ),\n            cacheWriteTokens: usageSteps.reduce(\n              (a, s) => a + s.cacheWriteTokens,\n              0\n            ),\n            reasoningTokens: usageSteps.reduce(\n              (a, s) => a + s.reasoningTokens,\n              0\n            ),\n            stepCount: usageSteps.length,\n          },\n        }\n      : null;\n\n  await Promise.all([\n    sessionContext.agent.storage.message.update(assistantMessageId, {\n      completedAt: Date.now(),\n      usage,\n    }),\n    writable.close().catch(() => undefined),\n  ]);\n\n  msgLog.info(\"message completed\", {\n    steps: usageSteps.length,\n    totalTokens: usage?.summary.totalTokens,\n  });\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Utils\n * -----------------------------------------------------------------------------------------------*/\n\nexport function approvalPartId(approvalId: string): string {\n  return `part_${approvalId}`;\n}\n\nexport function replayActiveRun(\n  active: ActiveRun,\n  startIndex = 0\n): ReadableStream<UIMessageChunk> {\n  let cursor = startIndex;\n  let cancelled = false;\n\n  return new ReadableStream<UIMessageChunk>({\n    async pull(controller) {\n      // Drain any buffered chunks\n      while (cursor < active.chunks.length) {\n        controller.enqueue(active.chunks[cursor++]);\n      }\n\n      // Wait for either a new chunk or completion\n      while (!cancelled && cursor >= active.chunks.length) {\n        const result = await Promise.race([\n          active.onChunk.then(() => \"chunk\" as const),\n          active.done.then(() => \"done\" as const),\n        ]);\n\n        // Drain new chunks\n        while (cursor < active.chunks.length) {\n          controller.enqueue(active.chunks[cursor++]);\n        }\n\n        if (result === \"done\") {\n          controller.close();\n          return;\n        }\n      }\n    },\n    cancel() {\n      cancelled = true;\n    },\n  });\n}\n\nfunction toMessageAndParts<Tools extends ToolSet>({\n  sessionId,\n  id,\n  input,\n  defaultRole,\n}: {\n  sessionId: string;\n  id?: string;\n  input: SendInput<Tools>;\n  defaultRole: UIMessage[\"role\"];\n}): {\n  message: Message | undefined;\n  parts: Part[];\n  approvals: ApprovalData[];\n} {\n  let messageId = id ?? `message_${ulid()}`;\n  if (typeof input === \"string\") {\n    const now = Date.now();\n    return {\n      approvals: [],\n      message: {\n        id: messageId,\n        sessionId,\n        role: defaultRole,\n        createdAt: now,\n        startedAt: null,\n        completedAt: null,\n        interruptedAt: null,\n        interruptedLastPart: null,\n        usage: null,\n        workflowRunId: null,\n        metadata: null,\n      },\n      parts: [\n        {\n          sessionId,\n          messageId,\n          id: `part_${messageId}_0`,\n          index: 0,\n          part: { type: \"text\", text: input },\n        },\n      ],\n    };\n  }\n  if (\"type\" in input && input.type === \"approval\") {\n    return {\n      message: undefined,\n      parts: [],\n      approvals: [input.approval],\n    };\n  }\n\n  const msg =\n    \"type\" in input && input.type === \"message\" ? input.message : input;\n  if (\"id\" in msg && msg.id) {\n    messageId = msg.id;\n  }\n\n  const now = Date.now();\n  const metadata =\n    \"metadata\" in msg && msg.metadata != null\n      ? (msg.metadata as Record<string, unknown>)\n      : null;\n  return {\n    approvals: [],\n    message: {\n      sessionId,\n      id: messageId,\n      createdAt: now,\n      startedAt: null,\n      completedAt: null,\n      interruptedAt: null,\n      interruptedLastPart: null,\n      usage: null,\n      workflowRunId: null,\n      role: msg.role ?? defaultRole,\n      metadata,\n    },\n    parts: msg.parts.map((part, index) => {\n      const ap = part as { approval?: { id: string } };\n      const id = ap.approval?.id\n        ? approvalPartId(ap.approval.id)\n        : `part_${messageId}_${index}`;\n      return { sessionId, messageId, id, index, part };\n    }),\n  };\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Errors\n * -----------------------------------------------------------------------------------------------*/\n\nexport class SendError extends errore.createTaggedError({\n  name: \"SendError\",\n  message: \"Send failed for session $id: $reason\",\n}) {}\n\nexport class ApprovalNotFoundError extends errore.createTaggedError({\n  name: \"ApprovalNotFoundError\",\n  message: \"Approval $approvalId not found in session $id\",\n}) {}\n", "import type {\n  FinishReason,\n  ToolSet,\n  UIMessage,\n  UIMessageStreamWriter,\n} from \"ai\";\nimport {\n  convertToModelMessages,\n  createUIMessageStream,\n  stepCountIs,\n  streamText as streamTextAi,\n} from \"ai\";\nimport type {\n  AgentStatus,\n  AnyAgent,\n  Resolvable,\n  ResolvableArgs,\n  TypedUIMessage,\n} from \"../../agent/client\";\nimport { SessionNotFoundError } from \"../../errors\";\nimport { discoverSkillsInSandbox } from \"../../skills/discover\";\nimport { resolveSkillInputsToSandboxDirs } from \"../../skills/resolve\";\nimport type { SkillSummary } from \"../../skills/types\";\nimport type { Part } from \"../../storage/types\";\nimport { type BuiltInToolName, getTools, type ToolContext } from \"../../tools\";\nimport { createLogger, type Logger } from \"../../utils/logger\";\nimport {\n  applyPromptCachingToModelRequest,\n  mergeProviderOptions,\n} from \"../../utils/prompt-cache\";\nimport { assembleUIMessages } from \"../../utils/ui\";\nimport type { StepUsage } from \"../../utils/usage\";\nimport { approvalPartId } from \"../send\";\n\nconst INTERRUPT_POLL_INTERVAL_MS = 250;\n\nfunction pollForInterrupt({\n  agent,\n  messageId,\n  signal,\n  abortController,\n}: {\n  agent: AnyAgent;\n  messageId: string;\n  signal: AbortSignal;\n  abortController: AbortController;\n}): {\n  firstCheck: Promise<void>;\n  interruptedLastPartRef: { current: { index: number; part: unknown } | null };\n} {\n  let resolveFirstCheck: () => void;\n  const firstCheck = new Promise<void>((resolve) => {\n    resolveFirstCheck = resolve;\n  });\n  const interruptedLastPartRef = {\n    current: null as { index: number; part: unknown } | null,\n  };\n\n  const log = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"ai-loop\",\n  });\n\n  const poll = async () => {\n    let isFirst = true;\n    while (!signal.aborted) {\n      const message = await agent.storage.message.get(messageId);\n      if (!message) {\n        if (isFirst) {\n          resolveFirstCheck();\n          return;\n        }\n        throw new Error(`Message ${messageId} not found`);\n      }\n      if (message.interruptedAt !== null) {\n        log.info(\"interrupt detected\", { messageId });\n        interruptedLastPartRef.current = message.interruptedLastPart;\n        abortController.abort();\n        if (isFirst) {\n          resolveFirstCheck();\n        }\n        return;\n      }\n      if (isFirst) {\n        isFirst = false;\n        resolveFirstCheck();\n      }\n      await new Promise((resolve) =>\n        setTimeout(resolve, INTERRUPT_POLL_INTERVAL_MS)\n      );\n    }\n  };\n\n  poll();\n  return { firstCheck, interruptedLastPartRef };\n}\n\ntype StreamTextOptions = {\n  stepThroughput: number;\n  assistantMessageId: string;\n  sessionId: string;\n  agent: AnyAgent;\n  createdAt: number;\n  context: Record<string, unknown>;\n  state: Record<string, unknown>;\n  writable: WritableStream;\n  lastPartIndex: number;\n  stepIndexOffset: number;\n  discoveredSkills: SkillSummary[] | null;\n};\n\nexport type StreamTextResult = {\n  finishReason: FinishReason;\n  lastPartIndex: number;\n  usageSteps: StepUsage[];\n  pendingApprovals: { approvalId: string; toolName: string }[];\n  maxSteps: number | undefined;\n  discoveredSkills: SkillSummary[] | null;\n  context: Record<string, unknown>;\n};\n\nexport async function streamText<Tools extends ToolSet>({\n  stepThroughput,\n  assistantMessageId,\n  agent,\n  sessionId,\n  context,\n  state,\n  createdAt,\n  writable,\n  lastPartIndex,\n  stepIndexOffset,\n  discoveredSkills,\n}: StreamTextOptions): Promise<StreamTextResult> {\n  const log = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"ai-loop\",\n  });\n\n  const stepLog = log.withContext({\n    sessionId,\n    messageId: assistantMessageId,\n  });\n  const doneStep = stepLog.time(\"streamText\");\n\n  const throttledEmitStatus = createThrottledEmitStatus(agent);\n\n  const abortController = new AbortController();\n  const pollController = new AbortController();\n\n  const { firstCheck, interruptedLastPartRef } = pollForInterrupt({\n    agent,\n    messageId: assistantMessageId,\n    signal: pollController.signal,\n    abortController,\n  });\n\n  const doneLoad = stepLog.time(\"load session + messages + parts + sandbox\");\n  const [, messagesResult, partsResult, { sandbox, sandboxRecord }] =\n    await Promise.all([\n      firstCheck,\n      agent.storage.message.listBySession(sessionId),\n      agent.storage.part.listBySession(sessionId),\n      agent.storage.session.get(sessionId).then(async (session) => {\n        if (!session) {\n          throw new SessionNotFoundError({ id: sessionId });\n        }\n\n        const sandboxId = session.sandboxId ?? session.id;\n        const sandbox = agent.sandbox(sandboxId);\n        const sandboxRecord = await agent.storage.sandbox.get(sandboxId);\n\n        return { sandbox, sandboxRecord };\n      }),\n    ]);\n\n  doneLoad();\n\n  if (abortController.signal.aborted) {\n    pollController.abort();\n    return {\n      finishReason: \"stop\" as FinishReason,\n      lastPartIndex,\n      usageSteps: [],\n      pendingApprovals: [],\n      maxSteps: undefined,\n      discoveredSkills,\n      context,\n    };\n  }\n\n  const setStartedPromise =\n    lastPartIndex === 0\n      ? (async () => {\n          const now = Date.now();\n          const pendingMessages = messagesResult.filter(\n            (m) => m.createdAt <= createdAt && m.startedAt === null\n          );\n          if (pendingMessages.length > 0) {\n            return await Promise.all(\n              pendingMessages.map((m) =>\n                agent.storage.message.update(m.id, {\n                  startedAt: now,\n                  completedAt: m.role === \"assistant\" ? null : now,\n                })\n              )\n            );\n          }\n        })()\n      : null;\n\n  const allParts = [...partsResult];\n  const nextPartIndexRef = {\n    current: Math.max(\n      lastPartIndex,\n      ...allParts\n        .filter((p) => p.messageId === assistantMessageId)\n        .map((p) => p.index + 1),\n      0\n    ),\n  };\n\n  const skillsRef: { current: SkillSummary[] } = {\n    current: discoveredSkills ?? [],\n  };\n\n  // Build uiMessages from current parts (rebuilt between iterations)\n  let uiMessages = assembleUIMessages({\n    messages: messagesResult,\n    parts: allParts,\n    until: createdAt,\n    includeQueued: true,\n    excludeSubToolParts: true,\n  });\n\n  // Build resolvable args for resolving agent options\n  const resolvableArgs: ResolvableArgs = {\n    context,\n    state,\n    sessionId,\n    sandbox,\n    get messages() {\n      return uiMessages;\n    },\n  };\n\n  // skills discovery \u2014 first iteration only\n  if (lastPartIndex === 0 && !discoveredSkills) {\n    const resolvedSkills = await resolveOption(\n      agent.options.skills,\n      resolvableArgs\n    );\n    const skillsDirs = resolvedSkills?.length\n      ? await resolveSkillInputsToSandboxDirs({\n          agent,\n          sandbox,\n          skills: resolvedSkills,\n          sessionId,\n        })\n      : [];\n    if (skillsDirs.length > 0) {\n      const doneSkills = stepLog.time(\"discover skills\", { skillsDirs });\n      skillsRef.current = await discoverSkillsInSandbox({\n        agent,\n        sandbox,\n        skillsDirs,\n        sessionId,\n      });\n      doneSkills({ count: skillsRef.current.length });\n    }\n  }\n\n  /**\n   * Inner loop: run up to `stepThroughput` sequential LLM calls.\n   * Parts are persisted after each call so progress isn't lost.\n   * Session is re-fetched between iterations (config may change),\n   * parallelized with part persistence on non-final iterations.\n   */\n  const usageSteps: StepUsage[] = [];\n  let internalStepIndex = 0;\n  let loopFinishReason: FinishReason = \"unknown\" as FinishReason;\n  let wasAborted = false;\n  const allPendingApprovals: { approvalId: string; toolName: string }[] = [];\n  let currentMaxSteps: number | undefined;\n\n  try {\n    for (let iteration = 0; iteration < stepThroughput; iteration++) {\n      // Resolve agent options fresh each iteration\n      const resolvedModel = await resolveOption(\n        agent.options.model,\n        resolvableArgs\n      );\n      const resolvedGeneration = await resolveOption(\n        agent.options.generation,\n        resolvableArgs\n      );\n      const resolvedActiveTools = (await resolveOption(\n        agent.options.activeTools,\n        resolvableArgs\n      )) as string[] | undefined;\n\n      currentMaxSteps = resolvedGeneration?.maxSteps;\n\n      if (!resolvedModel) {\n        throw new Error(\"Agent model is not set\");\n      }\n\n      const stepParts: UIMessage[\"parts\"] = [];\n      let streamTextFinishReason: FinishReason | undefined;\n\n      try {\n        const stream = createUIMessageStream({\n          execute: async ({ writer }) => {\n            const { rawTools, toolsWithNeedsApproval, toolContext } = getTools({\n              agent,\n              sandbox,\n              skillsRef,\n              writer,\n              abortController,\n              nextPartIndexRef,\n              assistantMessageId,\n              sessionId,\n              stepLog,\n              context,\n              state,\n              messages: uiMessages,\n              activeTools: resolvedActiveTools,\n            });\n\n            // process approvals once (first iteration only)\n            if (iteration === 0) {\n              await processApprovals({\n                allParts,\n                messagesResult,\n                createdAt,\n                rawTools,\n                abortController,\n                toolContext,\n                agent,\n                stepLog,\n                throttledEmitStatus,\n                writer,\n              });\n            }\n\n            throttledEmitStatus({\n              writer,\n              status: { type: \"thinking\" },\n            });\n\n            const iterResult = streamTextAi({\n              messages: [{ role: \"user\" as const, content: \".\" }],\n              tools: toolsWithNeedsApproval,\n              model: resolvedModel,\n              abortSignal: abortController.signal,\n              stopWhen: stepCountIs(1),\n              temperature: resolvedGeneration?.temperature,\n              topK: resolvedGeneration?.topK,\n              topP: resolvedGeneration?.topP,\n              frequencyPenalty: resolvedGeneration?.frequencyPenalty,\n              presencePenalty: resolvedGeneration?.presencePenalty,\n              maxOutputTokens: resolvedGeneration?.maxOutputTokens,\n              headers: resolvedGeneration?.headers,\n              experimental_context: {\n                context,\n                state,\n                sessionId,\n                sandbox,\n                messages: uiMessages,\n              } satisfies ToolContext,\n              prepareStep: async ({ model }) => {\n                const skillsContext = buildSkillsContext(skillsRef.current);\n                const cwdPrompt = sandboxRecord?.setup?.cwd\n                  ? `Your working directory is ${sandboxRecord.setup.cwd}. All file paths should be absolute.`\n                  : \"All file paths must be absolute.\";\n\n                const resolvedSystem = await resolveSystem(\n                  agent.options.system,\n                  resolvableArgs\n                );\n\n                const systemHeader = joinPromptSections(\n                  BASE_SYSTEM_PROMPT,\n                  cwdPrompt,\n                  resolvedSystem\n                );\n                const systemContext = joinPromptSections(\n                  skillsContext,\n                  cwdPrompt\n                );\n\n                const convertedMessages = await convertToModelMessages(\n                  uiMessages,\n                  { ignoreIncompleteToolCalls: true }\n                );\n\n                const modelMessages = [\n                  ...(systemHeader.trim()\n                    ? [{ role: \"system\" as const, content: systemHeader }]\n                    : []),\n                  ...(systemContext.trim()\n                    ? [{ role: \"system\" as const, content: systemContext }]\n                    : []),\n                  ...convertedMessages,\n                ];\n\n                const promptCaching = applyPromptCachingToModelRequest({\n                  model: typeof model === \"string\" ? model : model.modelId,\n                  sessionId,\n                  messages: modelMessages,\n                });\n\n                let activeTools = resolvedActiveTools ?? undefined;\n                if (skillsRef.current.length === 0 && activeTools) {\n                  activeTools = activeTools.filter(\n                    (t) => t !== (\"Skill\" satisfies BuiltInToolName)\n                  );\n                }\n\n                return {\n                  messages: promptCaching.messages,\n                  providerOptions: mergeProviderOptions({\n                    current: resolvedGeneration?.providerOptions,\n                    patch: promptCaching.providerOptions,\n                  }),\n                  activeTools,\n                };\n              },\n              onStepFinish: ({ usage }) => {\n                if (usage) {\n                  usageSteps.push({\n                    stepIndex: stepIndexOffset + internalStepIndex,\n                    model: resolvedModel ?? \"unknown\",\n                    inputTokens: usage.inputTokens ?? 0,\n                    outputTokens: usage.outputTokens ?? 0,\n                    totalTokens: usage.totalTokens ?? 0,\n                    cacheReadTokens:\n                      usage.inputTokenDetails?.cacheReadTokens ?? 0,\n                    cacheWriteTokens:\n                      usage.inputTokenDetails?.cacheWriteTokens ?? 0,\n                    reasoningTokens:\n                      usage.outputTokenDetails?.reasoningTokens ?? 0,\n                  });\n                }\n                internalStepIndex++;\n              },\n            });\n\n            writer.merge(\n              iterResult.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            );\n\n            streamTextFinishReason = await iterResult.finishReason;\n          },\n        });\n\n        await stream.pipeTo(writable, {\n          preventClose: true,\n          preventAbort: true,\n        });\n      } catch (err) {\n        if (abortController.signal.aborted) {\n          wasAborted = true;\n        } else {\n          throw err;\n        }\n      }\n\n      if (abortController.signal.aborted) {\n        wasAborted = true;\n      }\n\n      // Mark interrupted tool parts\n      if (wasAborted) {\n        const terminalStates = new Set([\n          \"output-available\",\n          \"output-error\",\n          \"output-denied\",\n          \"done\",\n        ]);\n        for (const part of stepParts) {\n          if (\n            \"type\" in part &&\n            typeof part.type === \"string\" &&\n            part.type.startsWith(\"tool-\") &&\n            \"state\" in part &&\n            !terminalStates.has(part.state as string)\n          ) {\n            (part as { state: string }).state = \"output-error\";\n            (part as { errorText?: string }).errorText = \"interrupted\";\n          }\n        }\n      }\n\n      // Handle interrupted lastPart truncation\n      const lastPart = interruptedLastPartRef.current;\n\n      const newPartRecords = stepParts\n        .map((uiPart, i) => {\n          const index = nextPartIndexRef.current + i;\n          if (lastPart != null && index > lastPart.index) {\n            return null;\n          }\n          const useClientPart = lastPart != null && index === lastPart.index;\n          const content = (\n            useClientPart ? lastPart.part : uiPart\n          ) as TypedUIMessage<Tools>[\"parts\"][number];\n          const partId =\n            \"approval\" in content && content.approval?.id\n              ? approvalPartId(content.approval.id)\n              : `part_${assistantMessageId}_${index}`;\n          return {\n            id: partId,\n            index,\n            messageId: assistantMessageId,\n            sessionId,\n            part: content,\n          };\n        })\n        .filter((r): r is NonNullable<typeof r> => r !== null);\n\n      const persistPromise = Promise.all([\n        ...newPartRecords.map((record) =>\n          agent.storage.part.set(record.id, record)\n        ),\n        agent.storage.session.update(sessionId, { state }),\n      ]);\n\n      nextPartIndexRef.current =\n        lastPart != null\n          ? Math.min(\n              nextPartIndexRef.current + stepParts.length,\n              lastPart.index + 1\n            )\n          : nextPartIndexRef.current + stepParts.length;\n\n      const iterPendingApprovals = stepParts\n        .filter(\n          (p): p is typeof p & { approval: { id: string } } =>\n            \"state\" in p &&\n            p.state === \"approval-requested\" &&\n            \"approval\" in p &&\n            !!(p as { approval?: { id: string } }).approval?.id\n        )\n        .map((p) => ({\n          approvalId: p.approval.id,\n          toolName:\n            \"type\" in p ? String(p.type).replace(\"tool-\", \"\") : \"unknown\",\n        }));\n\n      loopFinishReason = wasAborted\n        ? (\"stop\" as FinishReason)\n        : (streamTextFinishReason ?? (\"unknown\" as FinishReason));\n\n      // Decide whether to break or continue\n      const shouldBreak =\n        iterPendingApprovals.length > 0 ||\n        loopFinishReason === \"stop\" ||\n        wasAborted ||\n        lastPart != null ||\n        (currentMaxSteps != null &&\n          stepIndexOffset + usageSteps.length >= currentMaxSteps) ||\n        iteration >= stepThroughput - 1;\n\n      if (shouldBreak) {\n        await persistPromise;\n        allParts.push(...newPartRecords);\n        allPendingApprovals.push(...iterPendingApprovals);\n        break;\n      }\n\n      // Non-final iteration: persist parts and rebuild messages\n      await persistPromise;\n      allParts.push(...newPartRecords);\n      uiMessages = assembleUIMessages({\n        messages: messagesResult,\n        parts: allParts,\n        until: createdAt,\n        includeQueued: true,\n        excludeSubToolParts: true,\n      });\n    }\n  } finally {\n    pollController.abort();\n    throttledEmitStatus.dispose();\n  }\n\n  if (setStartedPromise) {\n    await setStartedPromise;\n  }\n\n  doneStep({\n    finishReason: loopFinishReason,\n    wasAborted,\n    iterations: internalStepIndex,\n    pendingApprovals: allPendingApprovals.length,\n  });\n\n  return {\n    finishReason: loopFinishReason,\n    lastPartIndex: nextPartIndexRef.current,\n    usageSteps,\n    pendingApprovals: allPendingApprovals,\n    maxSteps: currentMaxSteps,\n    discoveredSkills: skillsRef.current,\n    context,\n  };\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Steps\n * -----------------------------------------------------------------------------------------------*/\n\nexport async function streamTextStep<Tools extends ToolSet>(\n  ...args: Parameters<typeof streamText<Tools>>\n) {\n  \"use step\";\n  return await streamText(...args);\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Utils\n * -----------------------------------------------------------------------------------------------*/\n\nasync function processApprovals({\n  allParts,\n  messagesResult,\n  createdAt,\n  rawTools,\n  abortController,\n  toolContext,\n  agent,\n  stepLog,\n  throttledEmitStatus,\n  writer,\n}: {\n  allParts: Part[];\n  messagesResult: Parameters<typeof assembleUIMessages>[0][\"messages\"];\n  createdAt: number;\n  rawTools: ToolSet;\n  abortController: AbortController;\n  toolContext: ToolContext;\n  agent: AnyAgent;\n  stepLog: Logger;\n  throttledEmitStatus: ReturnType<typeof createThrottledEmitStatus>;\n  writer: StreamWriter;\n}) {\n  // Execute approved tools so the model sees complete tool call + result pairs\n  const approvedParts = allParts.filter(\n    (p) =>\n      p.part &&\n      \"state\" in p.part &&\n      (p.part as { state: string }).state === \"approval-responded\" &&\n      \"approval\" in p.part &&\n      (p.part as { approval?: { approved: boolean } }).approval?.approved ===\n        true\n  );\n\n  if (approvedParts.length > 0) {\n    stepLog.info(\"executing approved tools\", {\n      count: approvedParts.length,\n    });\n    throttledEmitStatus({\n      writer,\n      status: { type: \"processing-approvals\" },\n    });\n\n    const preExecMessages = await convertToModelMessages(\n      assembleUIMessages({\n        messages: messagesResult,\n        parts: allParts,\n        until: createdAt,\n        includeQueued: true,\n        excludeSubToolParts: true,\n      }),\n      { ignoreIncompleteToolCalls: true }\n    );\n\n    await Promise.all(\n      approvedParts.map(async (ap) => {\n        if (!ap.part.type.startsWith(\"tool-\")) {\n          return;\n        }\n        const part = ap.part as Extract<\n          UIMessage[\"parts\"][number],\n          { type: `tool-${string}` }\n        >;\n        const toolName = part.type.replace(\"tool-\", \"\");\n        const toolDef = rawTools[toolName];\n\n        if (toolDef?.execute && part.input !== undefined) {\n          try {\n            const toolOutput = await toolDef.execute(part.input, {\n              toolCallId: part.toolCallId,\n              messages: preExecMessages,\n              abortSignal: abortController.signal,\n              experimental_context: toolContext,\n            });\n            part.state = \"output-available\";\n            part.output = toolOutput;\n            writer.write({\n              type: \"tool-output-available\",\n              toolCallId: part.toolCallId,\n              output: toolOutput,\n            });\n          } catch (err) {\n            part.state = \"output-error\";\n            part.errorText = err instanceof Error ? err.message : String(err);\n            writer.write({\n              type: \"tool-output-error\",\n              toolCallId: part.toolCallId,\n              errorText: part.errorText,\n            });\n          }\n          await agent.storage.part.set(ap.id, { ...ap, part });\n        }\n      })\n    );\n  }\n\n  // Auto-deny any remaining approval-requested parts so the model\n  // sees complete tool call + denied result pairs (not missing results).\n  const pendingApprovalParts = allParts.filter(\n    (p) => (p.part as { state?: string }).state === \"approval-requested\"\n  );\n  if (pendingApprovalParts.length > 0) {\n    await Promise.all(\n      pendingApprovalParts.map(async (p) => {\n        const denied = {\n          ...p,\n          part: {\n            ...p.part,\n            state: \"output-denied\",\n            approval: {\n              ...((p.part as { approval?: object }).approval ?? {}),\n              approved: false,\n              reason: \"auto-denied: new message sent\",\n            },\n          } as typeof p.part,\n        };\n        Object.assign(p, denied);\n        await agent.storage.part.set(p.id, denied);\n      })\n    );\n  }\n}\n\ntype StreamWriter = UIMessageStreamWriter;\n\nfunction emitStatus({\n  writer,\n  status,\n  agent,\n}: {\n  writer: StreamWriter;\n  status: AgentStatus;\n  agent: AnyAgent;\n}) {\n  writer.write({ type: \"data-status\", data: status, transient: true });\n  agent.hooks?.status?.(status);\n}\n\nconst STATUS_THROTTLE_MS = 500;\n\nfunction createThrottledEmitStatus(agent: AnyAgent) {\n  let timer: ReturnType<typeof setTimeout> | null = null;\n  let pending: {\n    writer: StreamWriter;\n    status: AgentStatus;\n  } | null = null;\n\n  function throttled({\n    writer,\n    status,\n  }: {\n    writer: StreamWriter;\n    status: AgentStatus;\n  }) {\n    pending = { writer, status };\n    if (timer !== null) {\n      return;\n    }\n    flush();\n    timer = setTimeout(() => {\n      timer = null;\n      if (pending) {\n        flush();\n      }\n    }, STATUS_THROTTLE_MS);\n  }\n\n  function flush() {\n    if (!pending) {\n      return;\n    }\n    const { writer, status } = pending;\n    pending = null;\n    emitStatus({ writer, status, agent });\n  }\n\n  throttled.dispose = () => {\n    if (timer !== null) {\n      clearTimeout(timer);\n      timer = null;\n    }\n    flush();\n  };\n\n  return throttled;\n}\n\nasync function resolveOption<T>(\n  value: Resolvable<T> | undefined,\n  args: ResolvableArgs\n): Promise<T | undefined> {\n  if (value === undefined) {\n    return undefined;\n  }\n  if (typeof value === \"function\") {\n    return await (value as (args: ResolvableArgs) => T | Promise<T>)(args);\n  }\n  return value;\n}\n\nasync function resolveSystem(\n  input: Resolvable<string | string[] | undefined> | undefined,\n  args: ResolvableArgs\n): Promise<string | null> {\n  const result = await resolveOption(input, args);\n  if (!result) {\n    return null;\n  }\n  return Array.isArray(result) ? result.filter(Boolean).join(\"\\n\") : result;\n}\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\nconst backgroundProcessPrompt = `## Background Processes\nThese background process instructions are for you to manipulate the processes, do not be to verbose to the user about the response details like \"how to debug the process output\" the user will have an UI.\n\nUse \\`waitUntil: 0\\` for dev servers that should run indefinitely.\nIt is a good practice to check the output log after running processes like dev servers to make sure they started correctly.\n\nTo run a background process:\nBash({ command: \"npm run dev\", waitUntil: 0 })\nReturns immediately with \\`commandId\\`. The process keeps running in the sandbox.\n\nTo kill a process:\nBash({ command: \"ps aux | grep node\" })   // Find the PID\nBash({ command: \"kill {pid}\" })            // Graceful\nBash({ command: \"kill -9 {pid}\" })         // Force\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    .join(\"\\n\");\n\n  return `## Available Skills\n${skillLines}\n\n${backgroundProcessPrompt}\n\nUse the Skill tool to load a skill's full instructions before following it.`;\n}\n", "import type { SkillEntry, SkillInput } 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\nexport function normalizeSkills(\n  skills: SkillInput[] | null | undefined\n): SkillEntry[] {\n  if (!skills || skills.length === 0) {\n    return [];\n  }\n\n  return skills.map((skill) => {\n    if (typeof skill !== \"object\" || skill === null) {\n      throw new Error(\n        \"Invalid skills entry: expected an object with a `type` field.\"\n      );\n    }\n    if (\"type\" in skill && skill.type) {\n      return skill as SkillEntry;\n    }\n    throw new Error(\"Invalid skills entry: expected `type`.\");\n  });\n}\n", "import type { AnyAgent } from \"../agent/client\";\nimport type { SandboxInstance } from \"../sandbox/adapter\";\nimport { createLogger, type Logger } from \"../utils/logger\";\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  agent: AnyAgent;\n  sandbox: SandboxInstance;\n  skillsDirs: string[];\n  sessionId?: string;\n}): Promise<SkillSummary[]> {\n  const { agent, sandbox, skillsDirs, sessionId } = opts;\n  const baseLog = createLogger({\n    config: { ...agent.options.logging, name: agent.name },\n    subsystem: \"skills\",\n  });\n  const log = sessionId ? baseLog.withContext({ sessionId }) : baseLog;\n  const done = log.time(\"discoverSkillsInSandbox\");\n\n  const summaries: SkillSummary[] = [];\n  const seenNames = new Set<string>();\n\n  for (const skillsDir of skillsDirs) {\n    const dirDone = log.time(\"scan directory\", { dir: skillsDir });\n    const dirSummaries = await discoverSkillsInDirectory({\n      sandbox,\n      skillsDir,\n      log,\n    });\n    dirDone({ count: dirSummaries.length });\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  done({ totalSkills: summaries.length });\n\n  return summaries;\n}\n\nasync function discoverSkillsInDirectory(opts: {\n  sandbox: SandboxInstance;\n  skillsDir: string;\n  log: Logger;\n}): Promise<SkillSummary[]> {\n  const { sandbox, skillsDir, log } = opts;\n  const skillPaths = await findSkillFiles({ sandbox, skillsDir, log });\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, log });\n    if (summary) {\n      summaries.push(summary);\n    }\n  }\n\n  return summaries;\n}\n\nasync function findSkillFiles(opts: {\n  sandbox: SandboxInstance;\n  skillsDir: string;\n  log: Logger;\n}): Promise<string[]> {\n  const { sandbox, skillsDir, log } = opts;\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    log.warn(\"failed to scan skills directory\", {\n      dir: skillsDir,\n      error: execResult.message,\n    });\n    return [];\n  }\n\n  const { stdout, exitCode } = await execResult.result;\n\n  if (exitCode !== 0) {\n    log.warn(\"skills directory not found or inaccessible\", { dir: skillsDir });\n    return [];\n  }\n\n  return stdout\n    .trim()\n    .split(\"\\n\")\n    .filter((p) => p.length > 0);\n}\n\nasync function parseSkillFile(opts: {\n  sandbox: SandboxInstance;\n  skillMdPath: string;\n  log: Logger;\n}): Promise<SkillSummary | null> {\n  const { sandbox, skillMdPath, log } = opts;\n  const execResult = await sandbox.exec({\n    command: \"cat\",\n    args: [skillMdPath],\n  });\n\n  if (execResult instanceof Error) {\n    log.warn(\"failed to read skill file\", {\n      path: skillMdPath,\n      error: execResult.message,\n    });\n    return null;\n  }\n\n  const { stdout, exitCode } = await execResult.result;\n\n  if (exitCode !== 0) {\n    log.warn(\"could not read skill file\", { path: skillMdPath });\n    return null;\n  }\n\n  const parsed = parseSkillFrontmatter(stdout);\n\n  if (!parsed) {\n    log.warn(\"invalid or missing frontmatter\", { path: skillMdPath });\n    return null;\n  }\n\n  return {\n    name: parsed.name,\n    description: parsed.description,\n    skillMdPath,\n  };\n}\n", "import type { SandboxInstance } from \"../sandbox/adapter\";\nimport type { HostSkillInput } from \"./types\";\n\nasync function collectFiles(opts: {\n  absolutePath: string;\n  relativePrefix?: string;\n}): Promise<{ path: string; content: Buffer }[]> {\n  const { readdir, readFile, stat } = await import(\"node:fs/promises\");\n  const { resolve, posix: pathPosix } = await import(\"node:path\");\n\n  const fileStat = await stat(opts.absolutePath);\n  if (fileStat.isFile()) {\n    const content = await readFile(opts.absolutePath);\n    return [{ path: opts.relativePrefix ?? \"SKILL.md\", content }];\n  }\n\n  if (!fileStat.isDirectory()) {\n    return [];\n  }\n\n  const entries = await readdir(opts.absolutePath, { withFileTypes: true });\n  const files: { path: string; content: Buffer }[] = [];\n\n  for (const entry of entries) {\n    const nextAbs = resolve(opts.absolutePath, entry.name);\n    const nextRel = opts.relativePrefix\n      ? pathPosix.join(opts.relativePrefix, entry.name)\n      : entry.name;\n\n    if (entry.isDirectory()) {\n      files.push(\n        ...(await collectFiles({\n          absolutePath: nextAbs,\n          relativePrefix: nextRel,\n        }))\n      );\n      continue;\n    }\n\n    if (entry.isFile()) {\n      files.push({\n        path: nextRel,\n        content: await readFile(nextAbs),\n      });\n    }\n  }\n\n  return files;\n}\n\nasync function stableDirName(inputPath: string): Promise<string> {\n  const { createHash } = await import(\"node:crypto\");\n  const hash = createHash(\"sha1\").update(inputPath).digest(\"hex\").slice(0, 10);\n  return `host-${hash}`;\n}\n\nexport async function resolveHostSkill(opts: {\n  sandbox: SandboxInstance;\n  skill: HostSkillInput;\n}): Promise<string> {\n  const { resolve, posix: pathPosix } = await import(\"node:path\");\n\n  const hostPath = resolve(process.cwd(), opts.skill.path);\n  const files = await collectFiles({ absolutePath: hostPath });\n  const dirName = await stableDirName(hostPath);\n  const destPath = pathPosix.join(\".agent\", \"skills\", \".materialized\", dirName);\n\n  if (files.length === 0) {\n    return destPath;\n  }\n\n  await opts.sandbox.writeFiles({\n    files,\n    destPath,\n  });\n\n  return destPath;\n}\n", "import type { SandboxInstance } from \"../sandbox/adapter\";\nimport type { InlineSkillInput } from \"./types\";\n\nfunction slugify(value: string): string {\n  return value\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, \"-\")\n    .replace(/(^-|-$)/g, \"\");\n}\n\nasync function stableInlineDirName(name: string): Promise<string> {\n  const { createHash } = await import(\"node:crypto\");\n  const slug = slugify(name) || \"inline-skill\";\n  const hash = createHash(\"sha1\").update(name).digest(\"hex\").slice(0, 8);\n  return `${slug}-${hash}`;\n}\n\nfunction buildSkillMarkdown(skill: InlineSkillInput): string {\n  return `---\nname: ${skill.name}\ndescription: ${skill.description}\n---\n\n${skill.instructions.trim()}\n`;\n}\n\nexport async function resolveInlineSkill(opts: {\n  sandbox: SandboxInstance;\n  skill: InlineSkillInput;\n}): Promise<string> {\n  const { posix: pathPosix } = await import(\"node:path\");\n\n  const dirName = await stableInlineDirName(opts.skill.name);\n  const destPath = pathPosix.join(\".agent\", \"skills\", \".materialized\", dirName);\n\n  await opts.sandbox.writeFiles({\n    destPath,\n    files: [\n      {\n        path: \"SKILL.md\",\n        content: buildSkillMarkdown(opts.skill),\n      },\n    ],\n  });\n\n  return destPath;\n}\n", "import type { SandboxInstance } from \"../sandbox/adapter\";\nimport type { GitSkillInput } from \"./types\";\n\nfunction shellQuote(value: string): string {\n  return `'${value.replace(/'/g, `'\\\\''`)}'`;\n}\n\nasync function stableRepoDirName(skill: GitSkillInput): Promise<string> {\n  const { createHash } = await import(\"node:crypto\");\n  const hashInput = `${skill.repo}::${skill.ref ?? \"HEAD\"}`;\n  const hash = createHash(\"sha1\").update(hashInput).digest(\"hex\").slice(0, 12);\n  return `git-${hash}`;\n}\n\nexport async function resolveRemoteSkill(opts: {\n  sandbox: SandboxInstance;\n  skill: GitSkillInput;\n}): Promise<string> {\n  const { posix: pathPosix } = await import(\"node:path\");\n\n  const dirName = await stableRepoDirName(opts.skill);\n  const cloneDir = pathPosix.join(\".agent\", \"skills\", \".materialized\", dirName);\n  const quotedDir = shellQuote(cloneDir);\n  const quotedRepo = shellQuote(opts.skill.repo);\n  const quotedRef = shellQuote(opts.skill.ref ?? \"HEAD\");\n\n  const syncCommand = [\n    \"set -euo pipefail\",\n    `if [ ! -d ${quotedDir}/.git ]; then`,\n    `  rm -rf ${quotedDir}`,\n    `  mkdir -p ${quotedDir}`,\n    `  git clone --depth 1 ${quotedRepo} ${quotedDir}`,\n    \"fi\",\n    `if [ ${quotedRef} != 'HEAD' ]; then`,\n    `  git -C ${quotedDir} fetch --depth 1 origin ${quotedRef}`,\n    `  git -C ${quotedDir} checkout FETCH_HEAD`,\n    \"fi\",\n  ].join(\"\\n\");\n\n  const execResult = await opts.sandbox.exec({\n    command: \"bash\",\n    args: [\"-lc\", syncCommand],\n  });\n  const { exitCode, stderr } = await execResult.result;\n  if (exitCode !== 0) {\n    throw new Error(\n      `Failed to materialize git skill from ${opts.skill.repo}: ${stderr || \"unknown error\"}`\n    );\n  }\n\n  let resolvedDir = cloneDir;\n  if (opts.skill.path) {\n    resolvedDir = pathPosix.join(resolvedDir, opts.skill.path);\n  }\n  if (opts.skill.name) {\n    resolvedDir = pathPosix.join(resolvedDir, opts.skill.name);\n  }\n  return resolvedDir;\n}\n", "import type { AnyAgent } from \"../agent/client\";\nimport type { SandboxInstance } from \"../sandbox/adapter\";\nimport { createLogger } from \"../utils/logger\";\nimport { normalizeSkills } from \"./parser\";\nimport { resolveHostSkill } from \"./resolve-host\";\nimport { resolveInlineSkill } from \"./resolve-inline\";\nimport { resolveRemoteSkill } from \"./resolve-remote\";\nimport type { SkillInput } from \"./types\";\n\nexport async function resolveSkillInputsToSandboxDirs(opts: {\n  agent: AnyAgent;\n  sandbox: SandboxInstance;\n  skills: SkillInput[];\n  sessionId?: string;\n}): Promise<string[]> {\n  const baseLog = createLogger({\n    config: { ...opts.agent.options.logging, name: opts.agent.name },\n    subsystem: \"skills\",\n  });\n  const log = opts.sessionId\n    ? baseLog.withContext({ sessionId: opts.sessionId })\n    : baseLog;\n  const entries = normalizeSkills(opts.skills);\n  const resolved: string[] = [];\n\n  for (const entry of entries) {\n    if (entry.type === \"sandbox\") {\n      resolved.push(entry.path);\n      continue;\n    }\n    if (entry.type === \"host\") {\n      resolved.push(\n        await resolveHostSkill({\n          sandbox: opts.sandbox,\n          skill: entry,\n        })\n      );\n      continue;\n    }\n    if (entry.type === \"git\") {\n      resolved.push(\n        await resolveRemoteSkill({\n          sandbox: opts.sandbox,\n          skill: entry,\n        })\n      );\n      continue;\n    }\n    if (entry.type === \"inline\") {\n      resolved.push(\n        await resolveInlineSkill({\n          sandbox: opts.sandbox,\n          skill: entry,\n        })\n      );\n      continue;\n    }\n    log.warn(\"unknown skill source entry encountered\", { entry });\n  }\n\n  return resolved;\n}\n", "import {\n  type ToolSet,\n  tool,\n  type UIMessage,\n  type UIMessageStreamWriter,\n} from \"ai\";\nimport { ulid } from \"ulid\";\nimport { z } from \"zod\";\nimport type { AnyAgent } from \"../agent/client\";\nimport type { SandboxInstance } from \"../sandbox/adapter\";\nimport { DEFAULT_WAIT_UNTIL } from \"../sandbox/process-manager\";\nimport { approvalPartId } from \"../session/send\";\nimport type { SkillSummary } from \"../skills/types\";\nimport { createLogger, type Logger } from \"../utils/logger\";\nimport { createJavaScriptTool, type OnSubToolCall } from \"./javascript\";\nimport { checkNeedsApproval } from \"./needs-approval\";\n\nexport { tool as agentTool } from \"./tool\";\n\nconst log = createLogger({ config: { name: \"tools\" }, subsystem: \"tools\" });\n\nfunction formatFileSize(bytes: number): string {\n  if (bytes < 1024) {\n    return `${bytes}`;\n  }\n  if (bytes < 1024 * 1024) {\n    return `${(bytes / 1024).toFixed(1)}K`;\n  }\n  if (bytes < 1024 * 1024 * 1024) {\n    return `${(bytes / (1024 * 1024)).toFixed(1)}M`;\n  }\n  return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)}G`;\n}\n\ntype StreamWriter = UIMessageStreamWriter;\n\nexport type ToolContext<\n  TContext = Record<string, unknown>,\n  TState = Record<string, unknown>,\n> = {\n  context: TContext;\n  state: TState;\n  sessionId: string;\n  sandbox: SandboxInstance;\n  messages: UIMessage[];\n};\n\nfunction isRgNotFoundError(err: Error): boolean {\n  const parts = [err.message];\n\n  const anyErr = err as unknown as Record<string, unknown>;\n  if (typeof anyErr.reason === \"string\") {\n    parts.push(anyErr.reason);\n  }\n  if (anyErr.cause instanceof Error) {\n    parts.push(anyErr.cause.message);\n    const anyCause = anyErr.cause as unknown as Record<string, unknown>;\n    if (typeof anyCause.text === \"string\") {\n      parts.push(anyCause.text);\n    }\n  }\n\n  const msg = parts.join(\" \").toLowerCase();\n  return (\n    msg.includes(\"executable file not found\") ||\n    msg.includes(\"executable_not_found\") ||\n    (msg.includes(\"enoent\") && msg.includes(\"rg\"))\n  );\n}\n\nexport const builtInTools = {\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      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      path: z.string().describe(\"Absolute path to the file\"),\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.string().describe(\"Absolute path to the file\"),\n      }),\n    }),\n  }),\n  Grep: tool({\n    description:\n      \"Search for patterns in files using ripgrep. Use this to find code patterns, function definitions, imports, etc.\",\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      pattern: z\n        .string()\n        .describe(\"Regex pattern to search for (ripgrep syntax)\"),\n      path: z\n        .string()\n        .optional()\n        .describe(\n          \"Absolute path to search in (defaults to working directory). 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  }),\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      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      path: z\n        .string()\n        .optional()\n        .describe(\"Absolute path to list (defaults to working directory)\"),\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  }),\n  Write: tool({\n    description:\n      \"Write content to a file. Creates parent directories automatically. Overwrites existing files.\",\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      path: z.string().describe(\"Absolute path to the file\"),\n      content: z.string().describe(\"Content to write to the file\"),\n    }),\n    outputSchema: z.object({\n      success: z.boolean().describe(\"Whether the write succeeded\"),\n      path: z.string().describe(\"Path to the written file\"),\n      bytesWritten: z.number().describe(\"Number of bytes written\"),\n      error: z.string().optional().describe(\"Error message if write failed\"),\n    }),\n  }),\n  Edit: tool({\n    description:\n      \"Edit a file by replacing an exact string. Fails if old_string is not found or appears multiple times (not unique). For multiple replacements, call this tool multiple times with unique context.\",\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      path: z.string().describe(\"Absolute path to the file\"),\n      old_string: z\n        .string()\n        .describe(\"Exact string to find and replace (must be unique in file)\"),\n      new_string: z.string().describe(\"String to replace old_string with\"),\n    }),\n    outputSchema: z.object({\n      success: z.boolean().describe(\"Whether the edit succeeded\"),\n      path: z.string().describe(\"Path to the edited file\"),\n      error: z.string().optional().describe(\"Error message if edit failed\"),\n    }),\n  }),\n  Bash: tool({\n    description:\n      \"Executes a bash command. Returns stdout and stderr separately. Use waitUntil to control how long to wait (0 = return immediately, process keeps running). Large outputs are tail-truncated; full output in outputDir.\",\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\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: ${DEFAULT_WAIT_UNTIL}). Use 0 to return immediately while the process keeps running.`\n        ),\n    }),\n    outputSchema: z.object({\n      commandId: z\n        .string()\n        .describe(\"Command ID. Use to reference or kill running processes.\"),\n      stdout: z\n        .string()\n        .describe(\n          \"Command stdout. Tail-truncated if large; full content in outputDir/stdout.txt.\"\n        ),\n      stderr: z\n        .string()\n        .describe(\n          \"Command stderr. Tail-truncated if large; full content in outputDir/stderr.txt.\"\n        ),\n      exitCode: z.number().describe(\"Exit code (-1 if still running)\"),\n      status: z\n        .enum([\"running\", \"completed\", \"failed\"])\n        .describe(\"Process status\"),\n      outputDir: z\n        .string()\n        .describe(\n          \"Path to output directory containing stdout.txt, stderr.txt, and metadata.json. Empty if still running.\"\n        ),\n    }),\n  }),\n  Skill: tool({\n    description:\n      \"Load a skill's full instructions by name. Call this before following a skill.\",\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      name: z.string().describe(\"Skill name from the Available Skills list\"),\n    }),\n    outputSchema: z.object({\n      name: z.string(),\n      description: z.string(),\n      content: z.string().describe(\"Full SKILL.md content\"),\n      path: z.string().describe(\"Path to the skill directory in the sandbox\"),\n    }),\n  }),\n  JavaScript: tool({\n    inputSchema: z.object({\n      label: z\n        .string()\n        .describe(\"A label that describes the action being performed\"),\n      code: z\n        .string()\n        .describe(\n          \"JavaScript async function body. `ctx` is in scope. Must use `return` to produce output.\"\n        ),\n    }),\n  }),\n} satisfies ToolSet;\n\nexport type BuiltInToolName = keyof typeof builtInTools;\nexport const builtinToolNames = Object.fromEntries(\n  Object.entries(builtInTools).map(([name]) => [name, name])\n) as { [K in BuiltInToolName]: K };\n\nconst SKILL_MD_SUFFIX = /\\/?SKILL\\.md$/;\n\nexport function getTools({\n  agent,\n  sandbox,\n  skillsRef,\n  writer,\n  abortController,\n  nextPartIndexRef,\n  assistantMessageId,\n  sessionId,\n  stepLog,\n  context,\n  state,\n  messages,\n  activeTools,\n}: {\n  agent: AnyAgent;\n  sandbox: SandboxInstance;\n  skillsRef: { current: SkillSummary[] };\n  writer: StreamWriter;\n  abortController: AbortController;\n  nextPartIndexRef: { current: number };\n  assistantMessageId: string;\n  sessionId: string;\n  stepLog: Logger;\n  context: Record<string, unknown>;\n  state: Record<string, unknown>;\n  messages: UIMessage[];\n  activeTools?: string[];\n}): {\n  rawTools: ToolSet;\n  toolsWithNeedsApproval: ToolSet;\n  toolContext: ToolContext;\n} {\n  const rawTools: ToolSet = {\n    [builtinToolNames.Read]: tool({\n      ...builtInTools.Read,\n      execute: async ({ path, startLine, endLine }) => {\n        const filePath = path;\n\n        const result = await sandbox.readFile({ path: filePath });\n\n        if (result instanceof Error) {\n          log.error(\"Read failed\", { error: result.message });\n          throw result;\n        }\n\n        if (result === null) {\n          return {\n            content: `Error: File not found - ${filePath}`,\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 fullContent = result.toString(\"utf-8\");\n        const lines = fullContent.split(\"\\n\");\n        // A trailing newline produces an empty last element \u2014 remove it\n        // so line count matches `wc -l` / `awk 'END{print NR}'` behavior.\n        if (lines.length > 0 && lines.at(-1) === \"\") {\n          lines.pop();\n        }\n        const totalLines = lines.length;\n        const fileBytes = Buffer.byteLength(fullContent);\n        const fileSize = formatFileSize(fileBytes);\n\n        // Determine range\n        const PAGE_SIZE = 100;\n        let actualStart: number;\n        let actualEnd: number;\n\n        if (startLine !== undefined && endLine !== undefined) {\n          actualStart = startLine;\n          actualEnd = endLine;\n        } else if (startLine !== undefined) {\n          actualStart = startLine;\n          actualEnd = Math.min(startLine + PAGE_SIZE - 1, totalLines);\n        } else if (endLine !== undefined) {\n          actualStart = 1;\n          actualEnd = endLine;\n        } else if (totalLines > 200) {\n          actualStart = 1;\n          actualEnd = PAGE_SIZE;\n        } else {\n          actualStart = 1;\n          actualEnd = totalLines;\n        }\n\n        // Extract the line range (1-indexed)\n        const slicedLines = lines.slice(actualStart - 1, actualEnd);\n        const content = slicedLines.join(\"\\n\");\n\n        return {\n          metadata: {\n            totalLines,\n            linesShown: Math.max(0, actualEnd - actualStart + 1),\n            startLine: actualStart,\n            endLine: actualEnd,\n            isPaginated: actualEnd < totalLines,\n            fileSize,\n            path: filePath,\n          },\n          content,\n        };\n      },\n    }),\n    [builtinToolNames.Grep]: tool({\n      ...builtInTools.Grep,\n      execute: async ({\n        pattern,\n        path,\n        fileType,\n        glob,\n        caseSensitive,\n        contextLines,\n        maxCount,\n        filesWithMatches,\n      }) => {\n        const searchPath = path ?? \".\";\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        let result = await sandbox.exec({ command: \"rg\", args });\n\n        if (result instanceof Error && isRgNotFoundError(result)) {\n          log.warn(\"rg not found, installing ripgrep\");\n          const installResult = await sandbox.exec({\n            command: \"bash\",\n            args: [\n              \"-c\",\n              [\n                \"command -v rg >/dev/null 2>&1 && exit 0\",\n                \"dnf install -y ripgrep 2>/dev/null && exit 0\",\n                \"(apt-get update -qq && apt-get install -y -qq ripgrep) 2>/dev/null && exit 0\",\n                \"apk add --no-cache ripgrep 2>/dev/null && exit 0\",\n                \"curl -sL https://github.com/BurntSushi/ripgrep/releases/download/14.1.1/ripgrep-14.1.1-x86_64-unknown-linux-musl.tar.gz | tar xz -C /tmp && install /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl/rg /usr/local/bin/rg && rm -rf /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl\",\n              ].join(\"\\n\"),\n            ],\n            sudo: true,\n          });\n          if (!(installResult instanceof Error)) {\n            const installOutput = await installResult.result;\n            if (installOutput.exitCode !== 0) {\n              log.warn(\"ripgrep install failed\", {\n                stderr: installOutput.stderr,\n              });\n            }\n          }\n          result = await sandbox.exec({ command: \"rg\", args });\n        }\n\n        if (result instanceof Error && isRgNotFoundError(result)) {\n          log.warn(\"rg unavailable, falling back to grep\");\n          const grepArgs = [\"-rn\", \"--color=never\"];\n          if (!caseSensitive) {\n            grepArgs.push(\"-i\");\n          }\n          if (contextLines !== undefined) {\n            grepArgs.push(\"-C\", String(contextLines));\n          }\n          if (maxCount !== undefined) {\n            grepArgs.push(\"-m\", String(maxCount));\n          }\n          if (filesWithMatches) {\n            grepArgs.push(\"-l\");\n          }\n          if (glob) {\n            grepArgs.push(`--include=${glob}`);\n          }\n          if (fileType) {\n            const includeMap: Record<string, string[]> = {\n              ts: [\"*.ts\", \"*.tsx\", \"*.mts\", \"*.cts\"],\n              js: [\"*.js\", \"*.jsx\", \"*.mjs\", \"*.cjs\"],\n              py: [\"*.py\"],\n              rust: [\"*.rs\"],\n              go: [\"*.go\"],\n              java: [\"*.java\"],\n              md: [\"*.md\"],\n              json: [\"*.json\"],\n              css: [\"*.css\"],\n              html: [\"*.html\"],\n              yaml: [\"*.yml\", \"*.yaml\"],\n            };\n            for (const ext of includeMap[fileType] ?? [`*.${fileType}`]) {\n              grepArgs.push(`--include=${ext}`);\n            }\n          }\n          grepArgs.push(\"--\", pattern, searchPath);\n          result = await sandbox.exec({\n            command: \"grep\",\n            args: grepArgs,\n          });\n        }\n\n        if (result instanceof Error) {\n          log.error(\"Grep failed\", { error: result.message });\n          throw result;\n        }\n\n        const { stdout, stderr } = await result.result;\n\n        if (stderr && !stderr.toLowerCase().includes(\"no matches\")) {\n          log.warn(\"Grep stderr\", { 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          summary: {\n            matchCount: filesWithMatches\n              ? 0\n              : lines.filter((l) => l.includes(\":\")).length,\n            fileCount,\n            searchPath,\n            pattern,\n            wasTruncated,\n          },\n          matches: finalOutput || \"(no matches found)\",\n        };\n      },\n    }),\n    [builtinToolNames.List]: tool({\n      ...builtInTools.List,\n      execute: async ({ path, depth, includeHidden, filesOnly, pattern }) => {\n        const searchPath = path ?? \".\";\n\n        const result = await 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          log.error(\"List failed\", { error: result.message });\n          throw result;\n        }\n\n        const { stdout, stderr } = await result.result;\n\n        if (stderr) {\n          log.warn(\"List 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          summary: {\n            totalItems: lines.length,\n            totalFiles,\n            totalDirs,\n            searchPath,\n            depth,\n          },\n          listing,\n        };\n      },\n    }),\n    [builtinToolNames.Write]: tool({\n      ...builtInTools.Write,\n      execute: async ({ path, content }) => {\n        const filePath = path;\n\n        try {\n          await sandbox.writeFiles({\n            files: [{ path: filePath, content }],\n            destPath: \".\",\n          });\n\n          return {\n            success: true,\n            path: filePath,\n            bytesWritten: Buffer.byteLength(content, \"utf8\"),\n          };\n        } catch (err) {\n          const errorMsg = err instanceof Error ? err.message : String(err);\n          return {\n            success: false,\n            path: filePath,\n            bytesWritten: 0,\n            error: errorMsg,\n          };\n        }\n      },\n    }),\n    [builtinToolNames.Edit]: tool({\n      ...builtInTools.Edit,\n      execute: async ({ path, old_string, new_string }) => {\n        const filePath = path;\n\n        const result = await sandbox.readFile({ path: filePath });\n\n        if (result instanceof Error) {\n          return { success: false, path: filePath, error: result.message };\n        }\n\n        if (result === null) {\n          return {\n            success: false,\n            path: filePath,\n            error: `File not found: ${filePath}`,\n          };\n        }\n\n        const content = result.toString(\"utf-8\");\n        const occurrences = content.split(old_string).length - 1;\n\n        if (occurrences === 0) {\n          return {\n            success: false,\n            path: filePath,\n            error: \"old_string not found in file\",\n          };\n        }\n\n        if (occurrences > 1) {\n          return {\n            success: false,\n            path: filePath,\n            error: `old_string appears ${occurrences} times in file (must be unique). Include more surrounding context to make the match unique.`,\n          };\n        }\n\n        const newContent = content.replace(old_string, new_string);\n\n        try {\n          await sandbox.writeFiles({\n            files: [{ path: filePath, content: newContent }],\n            destPath: \".\",\n          });\n\n          return { success: true, path: filePath };\n        } catch (err) {\n          const errorMsg = err instanceof Error ? err.message : String(err);\n          return { success: false, path: filePath, error: errorMsg };\n        }\n      },\n    }),\n    [builtinToolNames.Bash]: tool({\n      ...builtInTools.Bash,\n      execute: async ({ command, waitUntil }) => {\n        const { createProcessManager } = await import(\n          \"../sandbox/process-manager\"\n        );\n\n        const processManager = createProcessManager({ sandbox });\n\n        const result = await processManager.run({ command, waitUntil });\n\n        const MAX_STDOUT = 50_000;\n        const MAX_STDERR = 10_000;\n\n        let { stdout, stderr } = result;\n        if (stdout.length > MAX_STDOUT) {\n          stdout =\n            `[truncated \u2014 showing last ${MAX_STDOUT} chars. Full: ${result.outputDir}/stdout.txt]\\n\\n` +\n            stdout.slice(-MAX_STDOUT);\n        }\n        if (stderr.length > MAX_STDERR) {\n          stderr =\n            `[truncated \u2014 showing last ${MAX_STDERR} chars. Full: ${result.outputDir}/stderr.txt]\\n\\n` +\n            stderr.slice(-MAX_STDERR);\n        }\n\n        return { ...result, stdout, stderr };\n      },\n    }),\n  };\n\n  // Add user-defined tools\n  for (const [name, t] of Object.entries(agent.options.tools ?? {})) {\n    rawTools[name] = t as ToolSet[string];\n  }\n\n  if (skillsRef.current.length > 0) {\n    rawTools[builtinToolNames.Skill] = tool({\n      ...builtInTools.Skill,\n      execute: async ({ name }) => {\n        const skills = skillsRef.current;\n        const skill = skills.find(\n          (s) => s.name.toLowerCase() === name.toLowerCase()\n        );\n        if (!skill) {\n          throw new Error(\n            `Skill not found: \"${name}\". Available: ${skills.map((s) => s.name).join(\", \")}`\n          );\n        }\n        const result = await sandbox.readFile({\n          path: skill.skillMdPath,\n        });\n        if (result instanceof Error) {\n          throw result;\n        }\n        if (result === null) {\n          throw new Error(`Skill file not found: ${skill.skillMdPath}`);\n        }\n        const raw =\n          typeof result === \"string\" ? result : result.toString(\"utf-8\");\n        const endMarker = raw.indexOf(\"---\", 3);\n        const content =\n          endMarker === -1 ? raw : raw.slice(endMarker + 3).trim();\n        const skillDir = skill.skillMdPath.replace(SKILL_MD_SUFFIX, \"\");\n        return {\n          name: skill.name,\n          description: skill.description,\n          content,\n          path: skillDir,\n        };\n      },\n    });\n  }\n\n  const onSubToolCall: OnSubToolCall = async (toolName, toolInput, execute) => {\n    const needsApproval = await checkNeedsApproval({\n      agent,\n      toolName,\n      input: toolInput,\n      toolCallId: `js_${toolName}_${Date.now()}`,\n      messages: [],\n    });\n\n    if (!needsApproval) {\n      try {\n        const result = await execute();\n        return { result };\n      } catch (err) {\n        return {\n          error: err instanceof Error ? err.message : String(err),\n        };\n      }\n    }\n\n    const approvalId = `js_approval_${ulid()}`;\n    const toolCallId = `js_tc_${ulid()}`;\n    const partId = approvalPartId(approvalId);\n\n    await agent.storage.part.set(partId, {\n      id: partId,\n      index: nextPartIndexRef.current++,\n      messageId: assistantMessageId,\n      sessionId,\n      part: {\n        type: `tool-${toolName}`,\n        toolCallId,\n        state: \"approval-requested\",\n        input: toolInput,\n        approval: { id: approvalId },\n      } as unknown as UIMessage[\"parts\"][number],\n    });\n\n    writer.write({\n      type: \"tool-input-start\",\n      toolCallId,\n      toolName,\n    });\n    writer.write({\n      type: \"tool-input-available\",\n      toolCallId,\n      toolName,\n      input: toolInput,\n    });\n    writer.write({\n      type: \"tool-approval-request\",\n      approvalId,\n      toolCallId,\n    });\n\n    const POLL_MS = 500;\n    const TIMEOUT_MS = 5 * 60 * 1000;\n    const start = Date.now();\n\n    while (Date.now() - start < TIMEOUT_MS && !abortController.signal.aborted) {\n      const updated = await agent.storage.part.get(partId);\n      if (\n        updated &&\n        \"state\" in updated.part &&\n        ((updated.part as { state: string }).state === \"approval-responded\" ||\n          (updated.part as { state: string }).state === \"output-denied\")\n      ) {\n        const state = (updated.part as { state: string }).state;\n\n        if (state === \"output-denied\") {\n          const reason = (updated.part as { approval?: { reason?: string } })\n            .approval?.reason;\n          return {\n            error: `Tool \"${toolName}\" denied: ${reason || \"user denied\"}`,\n          };\n        }\n\n        const approval = (\n          updated.part as {\n            approval?: { approved: boolean; reason?: string };\n          }\n        ).approval;\n        if (approval?.approved) {\n          try {\n            const result = await execute();\n            writer.write({\n              type: \"tool-output-available\",\n              toolCallId,\n              output: result,\n            });\n            return { result };\n          } catch (err) {\n            const error = err instanceof Error ? err.message : String(err);\n            writer.write({\n              type: \"tool-output-error\",\n              toolCallId,\n              errorText: error,\n            });\n            return { error };\n          }\n        }\n        writer.write({\n          type: \"tool-output-denied\",\n          toolCallId,\n        });\n        return {\n          error: `Tool \"${toolName}\" denied: ${approval?.reason || \"user denied\"}`,\n        };\n      }\n      await new Promise((r) => setTimeout(r, POLL_MS));\n    }\n\n    return {\n      error: abortController.signal.aborted\n        ? \"interrupted\"\n        : \"Approval timed out\",\n    };\n  };\n\n  const toolContext: ToolContext = {\n    context,\n    state,\n    sessionId,\n    sandbox,\n    messages: messages as ToolContext[\"messages\"],\n  };\n\n  rawTools[builtinToolNames.JavaScript] = createJavaScriptTool({\n    tools: rawTools,\n    sandbox,\n    activeTools,\n    onSubToolCall,\n    toolContext,\n  });\n\n  const toolsWithNeedsApproval = Object.fromEntries(\n    Object.entries(rawTools).map(([name, t]) => {\n      const originalExecute = t.execute;\n      const wrappedExecute = originalExecute\n        ? async (...args: Parameters<typeof originalExecute>) => {\n            const done = stepLog.time(`tool:${name}`);\n            try {\n              let toolInput = args[0];\n              if (agent.hooks[\"tool.before\"]) {\n                const beforeResult = await agent.hooks[\"tool.before\"]({\n                  name,\n                  input: toolInput,\n                  context: toolContext,\n                });\n                if (beforeResult?.input !== undefined) {\n                  toolInput = beforeResult.input;\n                  args[0] = toolInput as (typeof args)[0];\n                }\n              }\n\n              let result = await originalExecute(...args);\n\n              if (agent.hooks[\"tool.after\"]) {\n                const afterResult = await agent.hooks[\"tool.after\"]({\n                  name,\n                  input: toolInput,\n                  result,\n                  context: toolContext,\n                });\n                if (afterResult?.result !== undefined) {\n                  result = afterResult.result as typeof result;\n                }\n              }\n\n              done();\n              return result;\n            } catch (e) {\n              done({\n                error: e instanceof Error ? e.message : String(e),\n              });\n              throw e;\n            }\n          }\n        : undefined;\n      return [\n        name,\n        {\n          ...t,\n          ...(wrappedExecute ? { execute: wrappedExecute } : {}),\n          needsApproval: async (\n            toolInput: unknown,\n            opts: { toolCallId: string; messages: unknown[] }\n          ) => {\n            return await checkNeedsApproval({\n              agent,\n              toolName: name,\n              input: toolInput,\n              toolCallId: opts.toolCallId,\n              messages: opts.messages,\n            });\n          },\n        },\n      ];\n    })\n  );\n\n  return { rawTools, toolsWithNeedsApproval, toolContext };\n}\n", "import { type ToolSet, tool } from \"ai\";\nimport { z } from \"zod\";\nimport type { SandboxInstance } from \"../sandbox/adapter\";\nimport { builtInTools } from \".\";\n\nexport const JAVASCRIPT_TOOL_NAME = \"JavaScript\";\n\nexport type SubToolCallResult = { result?: unknown; error?: string };\nexport type OnSubToolCall = (\n  toolName: string,\n  input: unknown,\n  execute: () => Promise<unknown>\n) => Promise<SubToolCallResult>;\n\nconst REQUEST_MARKER_START = \"__TOOL_REQUEST__\";\nconst REQUEST_MARKER_END = \"__TOOL_REQUEST_END__\";\nconst EXECUTION_TIMEOUT_MS = 5 * 60 * 1000;\n\n/**\n * The runner script is written to the sandbox and executed via Node.js.\n * It reads the model's code from code.js, sets up ctx.tools as RPC proxies\n * (requests sent via stderr markers, responses polled from files), executes\n * the code, and writes the result to stdout as JSON.\n */\nconst RUNNER_SCRIPT = `\nimport { readFileSync, existsSync, unlinkSync, mkdirSync } from 'node:fs';\nimport { join, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nconst SCRIPT_DIR = dirname(fileURLToPath(import.meta.url));\nconst TOOL_CALL_TIMEOUT_MS = 300_000;\n\nlet reqCounter = 0;\n\nasync function callTool(name, input) {\n  const id = 'req_' + (++reqCounter) + '_' + Date.now();\n  const responseFile = join(SCRIPT_DIR, id + '.response.json');\n\n  const request = JSON.stringify({ id, tool: name, input });\n  process.stderr.write('${REQUEST_MARKER_START}' + request + '${REQUEST_MARKER_END}' + '\\\\n');\n\n  const start = Date.now();\n  while (!existsSync(responseFile)) {\n    if (Date.now() - start > TOOL_CALL_TIMEOUT_MS) {\n      throw new Error('Tool call ' + name + ' timed out waiting for response');\n    }\n    await new Promise(r => setTimeout(r, 30));\n  }\n\n  let raw;\n  try {\n    raw = readFileSync(responseFile, 'utf8');\n  } catch {\n    await new Promise(r => setTimeout(r, 50));\n    raw = readFileSync(responseFile, 'utf8');\n  }\n\n  const response = JSON.parse(raw);\n  try { unlinkSync(responseFile); } catch {}\n\n  if (response.error) {\n    throw new Error(response.error);\n  }\n  return response.result;\n}\n\nconst code = readFileSync(join(SCRIPT_DIR, 'code.js'), 'utf8');\nconst toolNames = JSON.parse(readFileSync(join(SCRIPT_DIR, 'tools.json'), 'utf8'));\n\nconst ctx = { tools: {} };\nfor (const name of toolNames) {\n  ctx.tools[name] = (input) => callTool(name, input);\n}\n\nconst logs = [];\nconst customConsole = {\n  log: (...args) => logs.push(args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),\n  error: (...args) => logs.push('[error] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),\n  warn: (...args) => logs.push('[warn] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),\n};\n\ntry {\n  const AsyncFunction = Object.getPrototypeOf(async function() {}).constructor;\n  const fn = new AsyncFunction('ctx', 'console', code);\n  const result = await fn(ctx, customConsole);\n\n  const output = { success: true, result: result !== undefined ? result : null };\n  if (logs.length > 0) output.logs = logs;\n  process.stdout.write(JSON.stringify(output));\n} catch (err) {\n  const output = {\n    success: false,\n    error: err instanceof Error ? err.message : String(err),\n  };\n  if (logs.length > 0) output.logs = logs;\n  process.stdout.write(JSON.stringify(output));\n}\n`;\n\n/**\n * Extracts a plain JSON Schema from whatever format the AI SDK tool uses\n * (Zod schema, AI SDK wrapped schema, or raw JSON Schema).\n */\nexport function extractJsonSchema(\n  schema: unknown\n): Record<string, unknown> | null {\n  if (!schema || typeof schema !== \"object\") {\n    return null;\n  }\n\n  if (\n    \"_def\" in schema &&\n    typeof (schema as { _def: unknown })._def === \"object\"\n  ) {\n    try {\n      return z.toJSONSchema(schema as z.ZodType) as Record<string, unknown>;\n    } catch {\n      return null;\n    }\n  }\n\n  const s = schema as Record<string | symbol, unknown>;\n  if (Symbol.for(\"vercel.ai.schema\") in s && \"jsonSchema\" in s) {\n    return s.jsonSchema as Record<string, unknown>;\n  }\n\n  if (\"type\" in s || \"properties\" in s) {\n    return s as Record<string, unknown>;\n  }\n\n  return null;\n}\n\nexport function jsonSchemaToTs(\n  schema: Record<string, unknown>,\n  indent = 0\n): string {\n  if (!schema) {\n    return \"unknown\";\n  }\n\n  if (schema.enum && Array.isArray(schema.enum)) {\n    return schema.enum.map((v) => JSON.stringify(v)).join(\" | \");\n  }\n\n  if (schema.anyOf && Array.isArray(schema.anyOf)) {\n    return schema.anyOf\n      .map((s: Record<string, unknown>) => jsonSchemaToTs(s, indent))\n      .join(\" | \");\n  }\n  if (schema.oneOf && Array.isArray(schema.oneOf)) {\n    return schema.oneOf\n      .map((s: Record<string, unknown>) => jsonSchemaToTs(s, indent))\n      .join(\" | \");\n  }\n\n  const type = schema.type as string;\n\n  switch (type) {\n    case \"string\":\n      return \"string\";\n    case \"number\":\n    case \"integer\":\n      return \"number\";\n    case \"boolean\":\n      return \"boolean\";\n    case \"null\":\n      return \"null\";\n    case \"array\": {\n      const items = schema.items\n        ? jsonSchemaToTs(schema.items as Record<string, unknown>, indent)\n        : \"unknown\";\n      return `Array<${items}>`;\n    }\n    case \"object\": {\n      const properties = schema.properties as\n        | Record<string, Record<string, unknown>>\n        | undefined;\n      if (!properties) {\n        return \"Record<string, unknown>\";\n      }\n      const required = new Set((schema.required as string[]) || []);\n      const pad = \"  \".repeat(indent + 1);\n      const closePad = \"  \".repeat(indent);\n      const props = Object.entries(properties).map(([key, value]) => {\n        const opt = required.has(key) ? \"\" : \"?\";\n        return `${pad}${key}${opt}: ${jsonSchemaToTs(value, indent + 1)}`;\n      });\n      return `{\\n${props.join(\";\\n\")};\\n${closePad}}`;\n    }\n    default:\n      return \"unknown\";\n  }\n}\n\nexport function generateContextTypeString(tools: ToolSet): string {\n  const sigs: string[] = [];\n\n  for (const [name, t] of Object.entries(tools)) {\n    const raw = t as Record<string, unknown>;\n    const inputSchema = extractJsonSchema(raw.inputSchema ?? raw.parameters);\n    const outputSchema = extractJsonSchema(raw.outputSchema);\n\n    const inputType = inputSchema ? jsonSchemaToTs(inputSchema, 2) : \"{}\";\n    const outputType = outputSchema\n      ? jsonSchemaToTs(outputSchema, 2)\n      : \"unknown\";\n\n    sigs.push(`    ${name}(input: ${inputType}): Promise<${outputType}>`);\n  }\n\n  return `type Context = {\\n  tools: {\\n${sigs.join(\";\\n\")};\\n  }\\n}`;\n}\n\nasync function executeInSandbox({\n  code,\n  rpcDir,\n  abortController,\n  sandbox,\n  availableTools,\n  onSubToolCall,\n  toolContext,\n}: {\n  code: string;\n  rpcDir: string;\n  abortController: AbortController;\n  sandbox: SandboxInstance;\n  availableTools: ToolSet;\n  onSubToolCall?: OnSubToolCall;\n  toolContext?: Record<string, unknown>;\n}): Promise<Record<string, unknown>> {\n  const mkdirResult = await sandbox.exec({\n    command: \"mkdir\",\n    args: [\"-p\", rpcDir],\n  });\n  if (mkdirResult instanceof Error) {\n    return { success: false, error: mkdirResult.message };\n  }\n  await mkdirResult.result;\n\n  await sandbox.writeFiles({\n    files: [\n      { path: \"runner.mjs\", content: RUNNER_SCRIPT },\n      { path: \"code.js\", content: code },\n      {\n        path: \"tools.json\",\n        content: JSON.stringify(Object.keys(availableTools)),\n      },\n    ],\n    destPath: rpcDir,\n  });\n\n  const nodeMajor = Number.parseInt(process.versions.node.split(\".\")[0], 10);\n  const permissionFlag =\n    nodeMajor >= 22 ? \"--permission\" : \"--experimental-permission\";\n\n  const execResult = await sandbox.exec({\n    command: \"node\",\n    args: [\n      permissionFlag,\n      `--allow-fs-read=${rpcDir}`,\n      `--allow-fs-write=${rpcDir}`,\n      `${rpcDir}/runner.mjs`,\n    ],\n    signal: abortController.signal,\n  });\n\n  if (execResult instanceof Error) {\n    return { success: false, error: execResult.message };\n  }\n\n  const fatal: { error: Error | null } = { error: null };\n\n  const abort = (error: Error) => {\n    if (!fatal.error) {\n      fatal.error = error;\n      abortController.abort();\n    }\n  };\n\n  const handleToolRequest = async (requestJson: string) => {\n    let parsed: { id: string; tool: string; input: unknown };\n    try {\n      parsed = JSON.parse(requestJson);\n    } catch {\n      return;\n    }\n\n    const { id, tool: toolName, input } = parsed;\n\n    let response: { result?: unknown; error?: string };\n    try {\n      const t = availableTools[toolName];\n      if (!t?.execute) {\n        throw new Error(\n          `Tool ${toolName} not found or has no execute function`\n        );\n      }\n      const exec = t.execute.bind(t);\n      const execOptions = {\n        toolCallId: `js_${toolName}_${Date.now()}`,\n        messages: [],\n        experimental_context: toolContext,\n      };\n      if (onSubToolCall) {\n        response = await onSubToolCall(toolName, input, () =>\n          exec(input, execOptions)\n        );\n      } else {\n        const result = await exec(input, execOptions);\n        response = { result };\n      }\n    } catch (err) {\n      const msg = err instanceof Error ? err.message : String(err);\n      response = { error: msg };\n    }\n\n    try {\n      await sandbox.writeFiles({\n        files: [\n          {\n            path: `${id}.response.json`,\n            content: JSON.stringify(response),\n          },\n        ],\n        destPath: rpcDir,\n      });\n    } catch (err) {\n      /**\n       * If writing the response file fails, the runner will poll forever\n       * waiting for it. Abort the runner process so we don't deadlock.\n       */\n      const msg = err instanceof Error ? err.message : String(err);\n      abort(new Error(`Failed to write RPC response for ${toolName}: ${msg}`));\n    }\n  };\n\n  const toolCallPromises: Promise<void>[] = [];\n  let stderrBuffer = \"\";\n  let nonMarkerStderr = \"\";\n\n  const timeout = setTimeout(() => {\n    abort(\n      new Error(\n        `JavaScript execution timed out after ${EXECUTION_TIMEOUT_MS / 1000}s`\n      )\n    );\n  }, EXECUTION_TIMEOUT_MS);\n\n  try {\n    for await (const entry of execResult.logs()) {\n      if (fatal.error) {\n        break;\n      }\n\n      if (entry.stream === \"stderr\") {\n        stderrBuffer += entry.data;\n\n        while (true) {\n          const startIdx = stderrBuffer.indexOf(REQUEST_MARKER_START);\n          if (startIdx === -1) {\n            break;\n          }\n\n          const beforeMarker = stderrBuffer.slice(0, startIdx);\n          if (beforeMarker.trim()) {\n            nonMarkerStderr += beforeMarker;\n          }\n\n          const contentStart = startIdx + REQUEST_MARKER_START.length;\n          const endIdx = stderrBuffer.indexOf(REQUEST_MARKER_END, contentStart);\n          if (endIdx === -1) {\n            break;\n          }\n\n          const requestJson = stderrBuffer.slice(contentStart, endIdx);\n          stderrBuffer = stderrBuffer.slice(endIdx + REQUEST_MARKER_END.length);\n\n          toolCallPromises.push(handleToolRequest(requestJson));\n        }\n      }\n    }\n  } finally {\n    clearTimeout(timeout);\n  }\n\n  if (stderrBuffer.trim()) {\n    nonMarkerStderr += stderrBuffer;\n  }\n\n  await Promise.allSettled(toolCallPromises);\n\n  if (fatal.error) {\n    execResult.result.catch(() => undefined);\n    return {\n      success: false,\n      error: fatal.error.message,\n      ...(nonMarkerStderr ? { stderr: nonMarkerStderr.slice(0, 2000) } : {}),\n    };\n  }\n\n  const { stdout, stderr, exitCode } = await execResult.result;\n\n  try {\n    return JSON.parse(stdout);\n  } catch {\n    return {\n      success: false,\n      error: `Runner failed (exit ${exitCode}).`,\n      stderr: (nonMarkerStderr + stderr).slice(0, 2000),\n      stdout: stdout.slice(0, 1000),\n    };\n  }\n}\n\nexport function createJavaScriptTool(opts: {\n  tools: ToolSet;\n  sandbox: SandboxInstance;\n  activeTools?: string[];\n  onSubToolCall?: OnSubToolCall;\n  toolContext?: Record<string, unknown>;\n}) {\n  const { sandbox, onSubToolCall } = opts;\n  const activeSet = opts.activeTools ? new Set(opts.activeTools) : null;\n  const availableTools: ToolSet = {};\n  for (const [name, t] of Object.entries(opts.tools)) {\n    if (!activeSet || activeSet.has(name)) {\n      availableTools[name] = t;\n    }\n  }\n  const contextType = generateContextTypeString(availableTools);\n\n  return tool({\n    ...builtInTools.JavaScript,\n    description: `Execute JavaScript to orchestrate multiple tool calls in a single step. Use this when you need to run several tools in sequence, transform intermediate results, or parallelize independent operations with Promise.all.\n\nThe code runs as an async function body with \\`ctx\\` in scope:\n\n\\`\\`\\`typescript\n${contextType}\n\\`\\`\\`\n\nExamples:\n- Sequential: \\`const file = await ctx.tools.Read({ path: \"package.json\" }); return JSON.parse(file.content);\\`\n- Parallel: \\`const [a, b] = await Promise.all([ctx.tools.Read({ path: \"a.ts\" }), ctx.tools.Read({ path: \"b.ts\" })]); return { a: a.content, b: b.content };\\`\n- Transform: \\`const grep = await ctx.tools.Grep({ pattern: \"TODO\" }); return grep.matches.split(\"\\\\n\").length;\\`\n- Bash: \\`const r = await ctx.tools.Bash({ command: \"node -e \\\\\"console.log(JSON.stringify({v:1}))\\\\\"\" }); return JSON.parse(r.stdout);\\`\n\nReturn a value to pass results back. Use console.log() for debug output.\nImportant: Each tool returns a structured object matching its output schema \u2014 not a raw string. Access the specific field you need (e.g. Bash's result.stdout, Read's result.content) rather than trying to parse the entire result.`,\n    execute: ({ code }) => {\n      const runId = `run_${Date.now()}_${Math.random()\n        .toString(36)\n        .slice(2, 8)}`;\n      const rpcDir = `.agent/js-rpc/${runId}`;\n      const abortController = new AbortController();\n\n      return executeInSandbox({\n        code,\n        rpcDir,\n        abortController,\n        sandbox,\n        availableTools,\n        onSubToolCall,\n        toolContext: opts.toolContext,\n      });\n    },\n  });\n}\n", "import type { Tool } from \"ai\";\nimport type { AnyAgent } from \"../agent/client\";\nimport type { BuiltInToolName } from \"./index\";\n\n/** Built-in defaults for needsApproval (user can override via agent config). */\nconst builtinNeedsApproval: Partial<Record<BuiltInToolName, boolean>> = {};\n\nexport async function checkNeedsApproval({\n  agent,\n  toolName,\n  input,\n  toolCallId,\n  messages,\n}: {\n  agent: AnyAgent;\n  toolName: string;\n  input: unknown;\n  toolCallId: string;\n  messages: unknown[];\n}): Promise<boolean> {\n  // 1. Per-tool needsApproval on the Tool definition takes priority\n  const toolDef = agent.options.tools?.[toolName] as Tool | undefined;\n  if (toolDef?.needsApproval != null) {\n    if (typeof toolDef.needsApproval === \"boolean\") {\n      return toolDef.needsApproval;\n    }\n    return await toolDef.needsApproval(input, {\n      toolCallId,\n      messages: messages as never[],\n      experimental_context: undefined,\n    });\n  }\n\n  // 2. Agent-level needsApproval map\n  const agentEntry = (\n    agent.options.needsApproval as\n      | Record<\n          string,\n          boolean | ((...args: unknown[]) => boolean | Promise<boolean>)\n        >\n      | undefined\n  )?.[toolName];\n\n  if (agentEntry != null) {\n    if (typeof agentEntry === \"boolean\") {\n      return agentEntry;\n    }\n    return await agentEntry(input, {\n      toolCallId,\n      messages,\n      experimental_context: undefined,\n    });\n  }\n\n  // 3. Built-in defaults\n  return builtinNeedsApproval[toolName as BuiltInToolName] ?? false;\n}\n", "import {\n  tool as aiTool,\n  type Tool,\n  type ToolExecutionOptions,\n  type UIMessage,\n} from \"ai\";\nimport type { z } from \"zod\";\nimport type { SandboxInstance } from \"../sandbox/adapter\";\nimport type { ToolContext } from \"./index\";\n\n/**\n * Define an agent tool with typed context and state.\n *\n * Pass the same `contextSchema` and/or `stateSchema` (Zod objects) used in\n * your agent definition to get typed `context` and `state` inside `execute`.\n *\n * ```ts\n * import { tool } from \"experimental-agent\";\n * import { contextSchema, stateSchema } from \"./context\";\n *\n * export const MyTool = tool({\n *   contextSchema,\n *   stateSchema,\n *   inputSchema: z.object({ query: z.string() }),\n *   execute: async ({ query }, { context, state, sandbox }) => {\n *     context.channelId // \u2713 typed from contextSchema\n *     state.counter++   // \u2713 typed from stateSchema, mutable\n *   },\n * });\n * ```\n */\nexport function tool<\n  TInputSchema extends z.ZodType,\n  TContextSchema extends z.ZodType = z.ZodType<Record<string, unknown>>,\n  TStateSchema extends z.ZodType = z.ZodType<Record<string, unknown>>,\n  // biome-ignore lint/suspicious/noExplicitAny: default erased\n  TOutputSchema extends z.ZodType = z.ZodType<any>,\n>(def: {\n  description?: string;\n  inputSchema: TInputSchema;\n  outputSchema?: TOutputSchema;\n  contextSchema?: TContextSchema;\n  stateSchema?: TStateSchema;\n  execute?: (\n    input: z.infer<TInputSchema>,\n    options: ToolContext<z.infer<TContextSchema>, z.infer<TStateSchema>> & {\n      toolCallId: string;\n      abortSignal: AbortSignal;\n    }\n    // biome-ignore lint/suspicious/noExplicitAny: execute can return anything\n  ) => any;\n  // biome-ignore lint/suspicious/noExplicitAny: erased tool generics\n}): Tool<any, any> {\n  const { execute, contextSchema, stateSchema, ...rest } = def;\n\n  const wrappedExecute = execute\n    ? // biome-ignore lint/suspicious/noExplicitAny: input type erased at runtime\n      (input: any, options: ToolExecutionOptions) => {\n        const ctx = options.experimental_context as {\n          context: unknown;\n          state: unknown;\n          sessionId: string;\n          messages: UIMessage[];\n          sandbox: SandboxInstance;\n        };\n\n        const context = (\n          contextSchema ? contextSchema.parse(ctx.context) : ctx.context\n        ) as z.infer<TContextSchema>;\n\n        if (stateSchema) {\n          stateSchema.parse(ctx.state);\n        }\n        const state = ctx.state as z.infer<TStateSchema>;\n\n        return execute(input, {\n          toolCallId: options.toolCallId,\n          abortSignal: options.abortSignal!,\n          context,\n          state,\n          sessionId: ctx.sessionId,\n          messages: ctx.messages,\n          sandbox: ctx.sandbox,\n        });\n      }\n    : undefined;\n\n  return wrappedExecute\n    ? aiTool({ ...rest, execute: wrappedExecute })\n    : // biome-ignore lint/suspicious/noExplicitAny: no-execute path needs cast\n      aiTool(rest as any);\n}\n", "import type { JSONValue, ModelMessage } from \"ai\";\n\ntype ProviderOptions = Record<string, Record<string, JSONValue>>;\n\nconst ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST = 4;\n\nconst CLAUDE_LIKE_MODEL_MATCHERS = [\"claude\", \"anthropic\"];\n\nconst CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS: ProviderOptions = {\n  anthropic: {\n    cacheControl: { type: \"ephemeral\" },\n  },\n  openrouter: {\n    cacheControl: { type: \"ephemeral\" },\n  },\n  bedrock: {\n    cachePoint: { type: \"default\" },\n  },\n  openaiCompatible: {\n    cache_control: { type: \"ephemeral\" },\n  },\n  copilot: {\n    copilot_cache_control: { type: \"ephemeral\" },\n  },\n};\n\nfunction getGatewayProvider(model: string): string | null {\n  const slashIndex = model.indexOf(\"/\");\n  if (slashIndex === -1) {\n    return null;\n  }\n  return model.slice(0, slashIndex);\n}\n\nfunction countAnthropicCacheBreakpoints(messages: ModelMessage[]): number {\n  let count = 0;\n  for (const message of messages) {\n    const providerOptions = (message as { providerOptions?: ProviderOptions })\n      .providerOptions;\n    if (providerOptions?.anthropic?.cacheControl) {\n      count += 1;\n    }\n    if (Array.isArray(message.content)) {\n      for (const part of message.content) {\n        const partProviderOptions = (\n          part as { providerOptions?: ProviderOptions }\n        ).providerOptions;\n        if (partProviderOptions?.anthropic?.cacheControl) {\n          count += 1;\n        }\n      }\n    }\n  }\n  return count;\n}\n\nfunction isClaudeLikeModel(model: string): boolean {\n  const lower = model.toLowerCase();\n  return CLAUDE_LIKE_MODEL_MATCHERS.some((m) => lower.includes(m));\n}\n\nexport function mergeProviderOptions(opts: {\n  current: ProviderOptions | undefined;\n  patch: ProviderOptions;\n}): ProviderOptions {\n  const next: ProviderOptions = { ...(opts.current ?? {}) };\n  for (const [key, value] of Object.entries(opts.patch)) {\n    next[key] = { ...(next[key] ?? {}), ...value };\n  }\n  return next;\n}\n\nfunction isCacheableClaudePart(part: unknown): boolean {\n  if (!part || typeof part !== \"object\") {\n    return false;\n  }\n\n  if (\"type\" in part && (part as { type?: unknown }).type === \"text\") {\n    const text = (part as { text?: unknown }).text;\n    if (typeof text === \"string\") {\n      return text.trim().length > 0;\n    }\n  }\n\n  if (\"type\" in part) {\n    const type = (part as { type?: unknown }).type;\n    if (type === \"thinking\" || type === \"reasoning\") {\n      return false;\n    }\n  }\n\n  return true;\n}\n\nfunction hasCacheableContent(message: ModelMessage): boolean {\n  if (typeof message.content === \"string\") {\n    return message.content.trim().length > 0;\n  }\n\n  if (Array.isArray(message.content)) {\n    return message.content.some((part) => isCacheableClaudePart(part));\n  }\n\n  return false;\n}\n\nfunction findLastCacheablePartIndex(content: unknown[]): number | null {\n  for (let i = content.length - 1; i >= 0; i -= 1) {\n    if (isCacheableClaudePart(content[i])) {\n      return i;\n    }\n  }\n  return null;\n}\n\nfunction selectClaudeCachingTargets(messages: ModelMessage[]): number[] {\n  const systemIndices: number[] = [];\n  const finalIndices: number[] = [];\n\n  for (let i = 0; i < messages.length && systemIndices.length < 2; i += 1) {\n    const message = messages[i];\n    if (message.role === \"system\" && hasCacheableContent(message)) {\n      systemIndices.push(i);\n    }\n  }\n\n  for (let i = messages.length - 1; i >= 0 && finalIndices.length < 2; i -= 1) {\n    const message = messages[i];\n    if (message.role !== \"system\" && hasCacheableContent(message)) {\n      finalIndices.push(i);\n    }\n  }\n\n  finalIndices.reverse();\n\n  return [...systemIndices, ...finalIndices];\n}\n\nfunction applyClaudePromptCaching(opts: {\n  messages: ModelMessage[];\n  providerId: string | null;\n  maxBreakpointsPerRequest: number;\n}): ModelMessage[] {\n  const existingBreakpoints = countAnthropicCacheBreakpoints(opts.messages);\n  const remainingBudget = Math.max(\n    0,\n    opts.maxBreakpointsPerRequest - existingBreakpoints\n  );\n\n  if (remainingBudget === 0) {\n    return opts.messages;\n  }\n\n  const targetIndices = selectClaudeCachingTargets(opts.messages).slice(\n    0,\n    remainingBudget\n  );\n  if (targetIndices.length === 0) {\n    return opts.messages;\n  }\n\n  const useMessageLevelOptions =\n    opts.providerId === \"anthropic\" ||\n    Boolean(opts.providerId?.includes(\"bedrock\"));\n\n  const nextMessages = opts.messages.slice();\n\n  for (const messageIndex of targetIndices) {\n    const message = nextMessages[messageIndex];\n\n    const shouldUseContentOptions =\n      !useMessageLevelOptions &&\n      Array.isArray(message.content) &&\n      message.content.length > 0;\n\n    if (shouldUseContentOptions && Array.isArray(message.content)) {\n      const partIndex = findLastCacheablePartIndex(message.content);\n      if (partIndex !== null) {\n        const part = message.content[partIndex];\n        if (part && typeof part === \"object\") {\n          const partProviderOptions = (\n            part as { providerOptions?: ProviderOptions }\n          ).providerOptions;\n\n          const nextContent = message.content.slice();\n          nextContent[partIndex] = {\n            ...(part as Record<string, unknown>),\n            providerOptions: mergeProviderOptions({\n              current: partProviderOptions,\n              patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS,\n            }),\n          } as (typeof message.content)[number];\n\n          nextMessages[messageIndex] = {\n            ...message,\n            content: nextContent,\n          } as ModelMessage;\n          continue;\n        }\n      }\n    }\n\n    const messageProviderOptions = (\n      message as { providerOptions?: ProviderOptions }\n    ).providerOptions;\n\n    nextMessages[messageIndex] = {\n      ...message,\n      providerOptions: mergeProviderOptions({\n        current: messageProviderOptions,\n        patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS,\n      }),\n    };\n  }\n\n  return nextMessages;\n}\n\nexport type PromptCachingResult = {\n  messages: ModelMessage[];\n  providerOptions: ProviderOptions;\n};\n\nexport function applyPromptCachingToModelRequest(opts: {\n  model: string;\n  sessionId: string;\n  messages: ModelMessage[];\n  anthropic?: { maxBreakpointsPerRequest?: number };\n  openai?: { setPromptCacheKey?: boolean };\n}): PromptCachingResult {\n  const provider = getGatewayProvider(opts.model);\n\n  const providerOptions: ProviderOptions = {};\n\n  if (\n    (provider === \"openai\" || provider === \"azure\") &&\n    opts.openai?.setPromptCacheKey !== false\n  ) {\n    providerOptions.openai = { promptCacheKey: opts.sessionId };\n  }\n\n  const maxBreakpointsPerRequest =\n    opts.anthropic?.maxBreakpointsPerRequest ??\n    ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST;\n\n  const cachedMessages = isClaudeLikeModel(opts.model)\n    ? applyClaudePromptCaching({\n        messages: opts.messages,\n        providerId: provider,\n        maxBreakpointsPerRequest,\n      })\n    : opts.messages;\n\n  return {\n    messages: cachedMessages,\n    providerOptions,\n  };\n}\n", "import type { UIMessageChunk } from \"ai\";\nimport * as errore from \"errore\";\nimport { SessionNotFoundError } from \"../errors\";\nimport type { SessionContext } from \"./handle\";\nimport { getActiveRuns, replayActiveRun } from \"./send\";\n\nexport type StreamOptions = {\n  messageId?: string;\n  startIndex?: number;\n};\n\nexport type WorkflowRunLike = {\n  getReadable(options?: { startIndex?: number }): ReadableStream;\n};\n\nfunction isWorkflowRun(arg: unknown): arg is WorkflowRunLike {\n  return (\n    typeof arg === \"object\" &&\n    arg !== null &&\n    \"getReadable\" in arg &&\n    typeof (arg as WorkflowRunLike).getReadable === \"function\"\n  );\n}\n\nexport async function stream(\n  sessionContext: SessionContext,\n  runOrOpts?: WorkflowRunLike | StreamOptions\n): Promise<ReadableStream<UIMessageChunk>> {\n  if (isWorkflowRun(runOrOpts)) {\n    const reader = runOrOpts.getReadable().getReader();\n    // await for the first chunk to be ready before finishing the workflow to avoid race conditions\n    await reader.read().catch(() => {\n      throw new StreamError({\n        id: sessionContext.sessionId,\n        reason: \"stream() failed to read first chunk from workflow run\",\n      });\n    });\n    reader.cancel();\n\n    return runOrOpts.getReadable();\n  }\n\n  const opts = runOrOpts;\n  const session = await sessionContext.agent.storage.session.get(\n    sessionContext.sessionId\n  );\n  if (!session) {\n    throw new SessionNotFoundError({ id: sessionContext.sessionId });\n  }\n  const messageId = opts?.messageId ?? session.lastMessageId;\n  if (!messageId) {\n    throw new StreamError({\n      id: sessionContext.sessionId,\n      reason: \"No message to stream\",\n    });\n  }\n\n  const activeRuns = getActiveRuns();\n  const active = activeRuns.get(messageId);\n  if (!active) {\n    const message = await sessionContext.agent.storage.message.get(messageId);\n    if (message?.workflowRunId) {\n      const readable = await import(\"workflow/api\")\n        .then(({ getRun }) => getRun(message.workflowRunId!).getReadable())\n        .catch(() => undefined);\n      if (readable) {\n        return readable;\n      }\n    }\n\n    throw new StreamError({\n      id: sessionContext.sessionId,\n      reason: `No active run for message ${messageId}`,\n    });\n  }\n\n  return replayActiveRun(active, opts?.startIndex);\n}\n\n/* -------------------------------------------------------------------------------------------------\n * Errors\n * -----------------------------------------------------------------------------------------------*/\n\nexport class StreamError extends errore.createTaggedError({\n  name: \"StreamError\",\n  message: \"Stream failed for session $id: $reason\",\n}) {}\n", "import type { Session } from \"../storage/types\";\nimport type { SessionContext } from \"./handle\";\n\nexport type UpdateOptions<\n  TState extends Record<string, unknown> = Record<string, unknown>,\n> = Partial<Omit<Session, \"id\" | \"createdAt\" | \"updatedAt\" | \"state\">> & {\n  state?: TState;\n};\n\nexport async function update<\n  TState extends Record<string, unknown> = Record<string, unknown>,\n>(\n  sessionContext: SessionContext,\n  opts: UpdateOptions<TState>\n) {\n  \"use step\";\n  const updates = Object.fromEntries(\n    Object.entries(opts).filter(([, value]) => value !== undefined)\n  );\n\n  if (Object.keys(updates).length === 0) {\n    return;\n  }\n\n  await sessionContext.agent.storage.session.update(sessionContext.sessionId, {\n    ...updates,\n    updatedAt: Date.now(),\n  });\n}\n", "import { computeSessionUsage, type SessionUsage } from \"../utils/usage\";\nimport type { SessionContext } from \"./handle\";\n\nexport async function usage(\n  sessionContext: SessionContext\n): Promise<SessionUsage> {\n  \"use step\";\n  const messages = await sessionContext.agent.storage.message.listBySession(\n    sessionContext.sessionId\n  );\n  return computeSessionUsage(messages);\n}\n", "import type {\n  InferToolInput,\n  InferUITools,\n  ModelMessage,\n  ToolSet,\n  UIMessage,\n} from \"ai\";\nimport * as errore from \"errore\";\nimport { z } from \"zod\";\nimport type { BuiltInToolName, builtInTools, ToolContext } from \"../tools\";\nimport type { AnyAgent } from \"./client\";\n\n// \u2500\u2500\u2500 Agent Status \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n/**\n * Transient status indicators emitted over the stream during long-running phases.\n * Delivered to the client as `data-status` chunks with `transient: true` \u2014 not persisted.\n */\nexport const AgentStatus = z.discriminatedUnion(\"type\", [\n  z.object({ type: z.literal(\"sandbox-setup\") }),\n  z.object({ type: z.literal(\"sandbox-setup-cold\") }),\n  z.object({ type: z.literal(\"loading-skills\") }),\n  z.object({ type: z.literal(\"processing-approvals\") }),\n  z.object({ type: z.literal(\"needs-approval\") }),\n  z.object({ type: z.literal(\"thinking\") }),\n  z.object({ type: z.literal(\"custom\"), status: z.string() }),\n]);\n\nexport type AgentStatus = z.infer<typeof AgentStatus>;\n\nexport type AgentDataTypes = { status: AgentStatus };\n\n// \u2500\u2500\u2500 Hooks \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport type AgentHooks = {\n  \"tool.before\"?: (opts: {\n    name: string;\n    input: unknown;\n    context: ToolContext;\n  }) => Promise<undefined | { input: unknown }>;\n  \"tool.after\"?: (opts: {\n    name: string;\n    input: unknown;\n    result: unknown;\n    context: ToolContext;\n  }) => Promise<undefined | { result: unknown }>;\n  status?: (status: AgentStatus) => void | Promise<void>;\n};\n\n// \u2500\u2500\u2500 Needs Approval \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\ntype ToolName<Tools> = Extract<keyof Tools, string> | BuiltInToolName;\n\ntype ToolInput<Tools, K> = K extends BuiltInToolName\n  ? InferToolInput<(typeof builtInTools)[K]>\n  : K extends keyof Tools\n    ? Tools[K] extends import(\"ai\").Tool\n      ? InferToolInput<Tools[K]>\n      : unknown\n    : unknown;\n\nexport type NeedsApprovalMap<Tools> = {\n  [K in ToolName<Tools>]?:\n    | boolean\n    | ((\n        input: ToolInput<Tools, K>,\n        options: {\n          toolCallId: string;\n          messages: ModelMessage[];\n          experimental_context: unknown;\n        }\n      ) => boolean | Promise<boolean>);\n};\n\n// \u2500\u2500\u2500 Infer UI Message \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport type InferUIMessage<A> = A extends {\n  tools: infer T extends ToolSet;\n  $UIMessage: UIMessage<infer M>;\n}\n  ? UIMessage<M, AgentDataTypes, InferUITools<T>>\n  : A extends { tools: infer T extends ToolSet }\n    ? UIMessage<unknown, AgentDataTypes, InferUITools<T>>\n    : UIMessage<unknown, AgentDataTypes>;\n\nexport type TypedUIMessage<\n  TMessageMetadata = unknown,\n  Tools extends ToolSet = ToolSet,\n> = UIMessage<TMessageMetadata, AgentDataTypes, InferUITools<Tools>>;\n\n// \u2500\u2500\u2500 Registry \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nclass AgentNotFoundError extends errore.createTaggedError({\n  name: \"AgentNotFoundError\",\n  message: `Agent \"$agentName\" not found in registry.\n\nhint: agent() must be called before workflow steps execute.\n      Valid locations: module scope, or inside a \"use workflow\" body.\n\n  // Module scope (static agents)\n  export const myAgent = agent(\"$agentName\", { ... });\n\n  // Workflow body (dynamic agents)\n  async function myWorkflow({ input }) {\n    \"use workflow\";\n    const myAgent = agent(\"$agentName\", { ... });\n    ...\n  }`,\n}) {}\n\nconst AGENT_REGISTRY_KEY = Symbol.for(\"experimental-agent-registry\");\n\nfunction getAgentRegistry(): Map<string, AnyAgent> {\n  const g = globalThis as Record<symbol, unknown>;\n  let registry = g[AGENT_REGISTRY_KEY] as Map<string, AnyAgent> | undefined;\n  if (!registry) {\n    registry = new Map();\n    g[AGENT_REGISTRY_KEY] = registry;\n  }\n  return registry;\n}\n\nexport function registerAgent(name: string, agent: AnyAgent) {\n  getAgentRegistry().set(name, agent);\n}\nexport function getAgent<Required extends boolean>(\n  name: string,\n  required?: Required\n): Required extends true ? AnyAgent : AnyAgent | undefined {\n  const agent = getAgentRegistry().get(name);\n  if (required && !agent) {\n    throw new AgentNotFoundError({ agentName: name });\n  }\n  return agent as Required extends true ? AnyAgent : AnyAgent | undefined;\n}\n", "import { createUIMessageStreamResponse } from \"ai\";\nimport Elysia from \"elysia\";\nimport type { AnyAgent, SessionSendArgs } from \"../agent/client\";\nimport type { SendResult } from \"../session/send\";\nimport {\n  type AgentContract,\n  agentContract,\n  type InferContract,\n  type InferResponse,\n} from \"./agentContract\";\nimport { type BasePath, basePathForAgent } from \"./base-path\";\n\ntype Without<O, K extends keyof O> = Omit<O, K>;\n\nexport function handleRequest<\n  Agent extends AnyAgent,\n  AgentBasePath extends string = BasePath,\n>(\n  agent: Agent,\n  options: {\n    basePath?: AgentBasePath;\n    workflow?: (\n      sessionId: string,\n      ...args: SessionSendArgs<AnyAgent>\n    ) => Promise<SendResult>;\n    overrides?: {\n      [K in keyof AgentContract<Agent>]?: (\n        contract: Without<InferContract<Agent>[K], \"response\">,\n        next: () => Promise<InferResponse<Agent, K> | Response>\n      ) => Promise<InferResponse<Agent, K> | Response>;\n    };\n  } = {}\n) {\n  const { basePath = basePathForAgent(agent.name), workflow } = options;\n\n  const contract = agentContract(agent);\n\n  const router = new Elysia({ prefix: basePath })\n    .get(\n      \"/:sessionId\",\n      async ({ params }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          const ui = await session.history();\n          return ui;\n        };\n\n        const override = options.overrides?.[\"session.get\"];\n        if (override) {\n          return Response.json(\n            await override(\n              { params },\n              // TODO make the mismatched types with AI SDK less weird\n              next as () => Promise<\n                InferResponse<Agent, \"session.get\"> | Response\n              >\n            )\n          );\n        }\n\n        return Response.json(await next());\n      },\n      contract[\"session.get\"]\n    )\n    .post(\n      \"/:sessionId\",\n      async ({ params, body }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n\n          if (workflow) {\n            const { start } = await import(\"workflow/api\");\n            const result = await start(workflow, [params.sessionId!, body]);\n            const stream = await session.stream(result);\n            return createUIMessageStreamResponse({ stream });\n          }\n\n          await session.send(body);\n          const stream = await session.stream();\n          return createUIMessageStreamResponse({ stream });\n        };\n\n        const override = options.overrides?.[\"session.post\"];\n        if (override) {\n          return await override({ params, body }, next);\n        }\n\n        return await next();\n      },\n      contract[\"session.post\"]\n    )\n    .get(\n      \"/:sessionId/reconnect\",\n      async ({ params }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          const stream = await session.stream();\n          return createUIMessageStreamResponse({ stream });\n        };\n\n        const override = options.overrides?.[\"reconnect.get\"];\n        if (override) {\n          return await override({ params }, next);\n        }\n\n        return await next();\n      },\n      contract[\"reconnect.get\"]\n    )\n    .post(\n      \"/:sessionId/interrupt\",\n\n      async ({ params, body }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          await session.interrupt(body);\n          return { success: true };\n        };\n\n        const override = options.overrides?.[\"interrupt.post\"];\n        if (override) {\n          return Response.json(await override({ params, body }, next));\n        }\n\n        return Response.json(await next());\n      },\n      contract[\"interrupt.post\"]\n    );\n\n  return router;\n}\n", "import z from \"zod\";\nimport type { AnyAgent } from \"../agent/client\";\nimport {\n  buildSessionHistorySchema,\n  buildUIMessagePartSchema,\n  buildUIMessageSchema,\n} from \"./schemas\";\n\nexport function agentContract<Agent extends AnyAgent>(agent: Agent) {\n  const contract = {\n    \"session.get\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      response: buildSessionHistorySchema(agent),\n    },\n    \"session.post\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      body: z.union([z.string(), buildUIMessageSchema(agent.tools)]),\n      response: z.any(),\n    },\n    \"reconnect.get\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      response: z.any(),\n    },\n    \"interrupt.post\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      body: z\n        .object({\n          lastPart: z\n            .object({\n              index: z.number(),\n              part: buildUIMessagePartSchema(agent.tools),\n            })\n            .optional(),\n        })\n        .optional(),\n      response: z.object({\n        success: z.boolean(),\n      }),\n    },\n  } satisfies Record<\n    string,\n    {\n      params: z.ZodType;\n      body?: z.ZodType;\n      response: z.ZodType;\n    }\n  >;\n\n  return contract;\n}\n\nexport type AgentContract<Agent extends AnyAgent> = ReturnType<\n  typeof agentContract<Agent>\n>;\n\nexport type InferContract<Agent extends AnyAgent> = {\n  [K in keyof AgentContract<Agent>]: {\n    [K2 in keyof AgentContract<Agent>[K]]: z.infer<AgentContract<Agent>[K][K2]>;\n  };\n};\n\nexport type InferResponse<\n  Agent extends AnyAgent,\n  Key extends keyof AgentContract<Agent>,\n> = InferContract<Agent>[Key][\"response\"];\n", "import type {\n  FileUIPart,\n  InferUITools,\n  ReasoningUIPart,\n  SourceUrlUIPart,\n  StepStartUIPart,\n  TextUIPart,\n  ToolSet,\n  ToolUIPart,\n} from \"ai\";\nimport { z } from \"zod\";\nimport type { AnyAgent } from \"../agent/client\";\n\ntype UIMessagePart<T extends ToolSet> =\n  | TextUIPart\n  | ReasoningUIPart\n  | FileUIPart\n  | SourceUrlUIPart\n  | StepStartUIPart\n  | ToolUIPart<InferUITools<T>>;\n\nconst TextPart = z.object({\n  type: z.literal(\"text\"),\n  text: z.string(),\n});\n\nconst ReasoningPart = z.object({\n  type: z.literal(\"reasoning\"),\n  text: z.string(),\n});\n\nconst FilePart = z.object({\n  type: z.literal(\"file\"),\n  mediaType: z.string(),\n  url: z.string(),\n  filename: z.string().optional(),\n});\n\nconst SourceUrlPart = z.object({\n  type: z.literal(\"source-url\"),\n  sourceId: z.string(),\n  url: z.string(),\n  title: z.string().optional(),\n});\n\nconst StepStartPart = z.object({\n  type: z.literal(\"step-start\"),\n});\n\nfunction toolPartSchemas(name: string, tool: ToolSet[string]) {\n  const input = tool.inputSchema ?? z.unknown();\n  const output = tool.outputSchema ?? z.unknown();\n  const type = z.literal(`tool-${name}`);\n\n  const approvalId = z.object({ id: z.string() });\n  const approvalApproved = approvalId.extend({\n    approved: z.literal(true),\n    reason: z.string().optional(),\n  });\n  const approvalDenied = approvalId.extend({\n    approved: z.literal(false),\n    reason: z.string().optional(),\n  });\n  const approvalResponse = approvalId.extend({\n    approved: z.boolean(),\n    reason: z.string().optional(),\n  });\n\n  return [\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"input-streaming\"),\n      input: z.unknown(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"input-available\"),\n      input,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"approval-requested\"),\n      input,\n      approval: approvalId,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"approval-responded\"),\n      input,\n      approval: approvalResponse,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-available\"),\n      input,\n      output,\n      approval: approvalApproved.optional(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-error\"),\n      input,\n      errorText: z.string(),\n      approval: approvalApproved.optional(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-denied\"),\n      input,\n      approval: approvalDenied,\n    }),\n  ];\n}\n\nconst UsageSummary = z.object({\n  model: z.string(),\n  inputTokens: z.number(),\n  outputTokens: z.number(),\n  totalTokens: z.number(),\n  cacheReadTokens: z.number(),\n  cacheWriteTokens: z.number(),\n  reasoningTokens: z.number(),\n  stepCount: z.number(),\n});\n\nconst SessionUsage = z.object({\n  total: UsageSummary.extend({ messageCount: z.number() }),\n  byMessageId: z.record(z.string(), UsageSummary.nullable()),\n});\n\nexport function buildSessionHistorySchema<Agent extends AnyAgent>(\n  agent: Agent\n) {\n  return z.object({\n    messages: z.array(buildUIMessageSchema(agent.tools)),\n    streamingMessageId: z.string().nullable(),\n    usage: SessionUsage,\n  });\n}\n\nexport function buildUIMessagePartSchema<T extends ToolSet>(\n  tools: T\n): z.ZodType<UIMessagePart<T>> {\n  const toolParts = Object.entries(tools).flatMap(([name, tool]) =>\n    toolPartSchemas(name, tool)\n  );\n\n  const all = [\n    TextPart,\n    ReasoningPart,\n    FilePart,\n    SourceUrlPart,\n    StepStartPart,\n    ...toolParts,\n  ];\n\n  return z.union(all) as z.ZodType<UIMessagePart<T>>;\n}\n\nexport function buildUIMessageSchema<T extends ToolSet>(tools: T) {\n  return z.object({\n    id: z.string(),\n    role: z.enum([\"system\", \"user\", \"assistant\"]),\n    parts: z.array(buildUIMessagePartSchema(tools)),\n  });\n}\n\nexport function buildSessionUISchema<T extends ToolSet>(tools: T) {\n  const UIMessage = buildUIMessageSchema(tools);\n\n  return z.object({\n    title: z.string().nullable(),\n    messages: z.array(UIMessage),\n    streamingMessageId: z.string().nullable(),\n    usage: SessionUsage,\n  });\n}\n"],
  "mappings": "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",
  "names": ["agent", "log", "resolve", "agent", "ulid", "usage", "errore", "ulid", "agent", "log", "readFile", "resolve", "log", "tool", "ulid", "z", "exec", "agent", "tool", "tool", "z", "agent", "ulid", "start", "state", "agent", "resolve", "log", "sandbox", "sandboxRecord", "stream", "usage", "agent", "log", "ulid", "usage", "now", "id", "errore", "agent", "ulid", "errore", "z", "agent", "agent", "log", "z", "z", "tool", "agent", "all", "agent", "z", "agent", "start", "stream"]
}

|
|
336
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/rpc/rpc.ts", "../src/rpc/agentContract.ts", "../src/rpc/schemas.ts"],
  "sourcesContent": ["import { openapi } from \"@elysiajs/openapi\";\nimport { createUIMessageStreamResponse } from \"ai\";\nimport Elysia from \"elysia\";\nimport z from \"zod\";\nimport {\n  AgentStatus,\n  type AnyAgent,\n  type SessionSendArgs,\n} from \"../agent/client\";\nimport type { SendResult } from \"../session/send\";\nimport {\n  type AgentContract,\n  agentContract,\n  type InferContract,\n  type InferResponse,\n} from \"./agentContract\";\nimport { type BasePath, basePathForAgent } from \"./base-path\";\n\ntype Without<O, K extends keyof O> = Omit<O, K>;\n\nexport function handleRequest<\n  Agent extends AnyAgent,\n  AgentBasePath extends string = BasePath,\n>(\n  agent: Agent,\n  options: {\n    basePath?: AgentBasePath;\n    workflow?: (\n      sessionId: string,\n      ...args: SessionSendArgs<Agent>\n    ) => Promise<SendResult>;\n    overrides?: {\n      [K in keyof AgentContract<Agent>]?: (\n        contract: Without<InferContract<Agent>[K], \"response\">,\n        next: () => Promise<InferResponse<Agent, K> | Response>\n      ) => Promise<InferResponse<Agent, K> | Response>;\n    };\n  } = {}\n) {\n  const { basePath = basePathForAgent(agent.name), workflow } = options;\n\n  const contract = agentContract(agent);\n\n  const router = new Elysia({ prefix: basePath })\n    .use(\n      openapi({\n        provider: null,\n        specPath: \"/openapi.json\",\n        exclude: { staticFile: false },\n        documentation: {\n          info: {\n            title: `${agent.name} API`,\n            version: \"1.0.0\",\n          },\n          components: {\n            schemas: {\n              // @ts-expect-error - elysia is annoying\n              AgentStatus: z.toJSONSchema(AgentStatus, {\n                target: \"openapi-3.0\",\n              }),\n            },\n            securitySchemes: {\n              bearerAuth: {\n                type: \"http\",\n                scheme: \"bearer\",\n                bearerFormat: \"JWT\",\n              },\n            },\n          },\n        },\n        mapJsonSchema: {\n          zod: (schema: z.ZodType) =>\n            z.toJSONSchema(schema, { target: \"openapi-3.0\" }),\n        },\n      })\n    )\n    .get(\n      \"/:sessionId\",\n      async ({ params }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          const ui = await session.history();\n          return ui;\n        };\n\n        const override = options.overrides?.[\"session.get\"];\n        if (override) {\n          return Response.json(\n            await override(\n              { params },\n              // TODO make the mismatched types with AI SDK less weird\n              next as () => Promise<\n                InferResponse<Agent, \"session.get\"> | Response\n              >\n            )\n          );\n        }\n\n        return Response.json(await next());\n      },\n      contract[\"session.get\"]\n    )\n    .post(\n      \"/:sessionId\",\n      async ({ params, body }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n\n          if (workflow) {\n            const { start } = await import(\"workflow/api\");\n\n            const result = await start(\n              // for some reason, start() requires SessionSendArgs<AnyAgent>\n              // to avoid blowing up types\n              // we want to put the typeguard here so consumers of handleRequest still work fine\n              workflow as (\n                sessionId: string,\n                ...args: SessionSendArgs<AnyAgent>\n              ) => Promise<SendResult>,\n              [params.sessionId!, body]\n            );\n            const stream = await session.stream(result);\n            return createUIMessageStreamResponse({ stream });\n          }\n\n          await session.send(body);\n          const stream = await session.stream();\n          return createUIMessageStreamResponse({ stream });\n        };\n\n        const override = options.overrides?.[\"session.post\"];\n        if (override) {\n          return await override({ params, body }, next);\n        }\n\n        return await next();\n      },\n      contract[\"session.post\"]\n    )\n    .get(\n      \"/:sessionId/reconnect\",\n      async ({ params }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          const stream = await session.stream();\n          return createUIMessageStreamResponse({ stream });\n        };\n\n        const override = options.overrides?.[\"reconnect.get\"];\n        if (override) {\n          return await override({ params }, next);\n        }\n\n        return await next();\n      },\n      contract[\"reconnect.get\"]\n    )\n    .post(\n      \"/:sessionId/interrupt\",\n\n      async ({ params, body }) => {\n        const next = async () => {\n          const session = agent.session(params.sessionId!);\n          await session.interrupt(body);\n          return { success: true };\n        };\n\n        const override = options.overrides?.[\"interrupt.post\"];\n        if (override) {\n          return Response.json(await override({ params, body }, next));\n        }\n\n        return Response.json(await next());\n      },\n      contract[\"interrupt.post\"]\n    );\n\n  return router;\n}\n", "import z from \"zod\";\nimport type { AnyAgent } from \"../agent/client\";\nimport {\n  buildSessionHistorySchema,\n  buildUIMessagePartSchema,\n  buildUIMessageSchema,\n} from \"./schemas\";\n\nexport function agentContract<Agent extends AnyAgent>(agent: Agent) {\n  const contract = {\n    \"session.get\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      response: buildSessionHistorySchema(agent),\n    },\n    \"session.post\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      body: z.union([z.string(), buildUIMessageSchema(agent.tools)]),\n      response: z.any(),\n    },\n    \"reconnect.get\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      response: z.any(),\n    },\n    \"interrupt.post\": {\n      params: z.object({\n        sessionId: z.string(),\n      }),\n      body: z\n        .object({\n          lastPart: z\n            .object({\n              index: z.number(),\n              part: buildUIMessagePartSchema(agent.tools),\n            })\n            .optional(),\n        })\n        .optional(),\n      response: z.object({\n        success: z.boolean(),\n      }),\n    },\n  } satisfies Record<\n    string,\n    {\n      params: z.ZodType;\n      body?: z.ZodType;\n      response: z.ZodType;\n    }\n  >;\n\n  return contract;\n}\n\nexport type AgentContract<Agent extends AnyAgent> = ReturnType<\n  typeof agentContract<Agent>\n>;\n\nexport type InferContract<Agent extends AnyAgent> = {\n  [K in keyof AgentContract<Agent>]: {\n    [K2 in keyof AgentContract<Agent>[K]]: z.infer<AgentContract<Agent>[K][K2]>;\n  };\n};\n\nexport type InferResponse<\n  Agent extends AnyAgent,\n  Key extends keyof AgentContract<Agent>,\n> = InferContract<Agent>[Key][\"response\"];\n", "import type {\n  FileUIPart,\n  InferUITools,\n  ReasoningUIPart,\n  SourceUrlUIPart,\n  StepStartUIPart,\n  TextUIPart,\n  ToolSet,\n  ToolUIPart,\n} from \"ai\";\nimport { z } from \"zod\";\nimport type { AnyAgent } from \"../agent/client\";\n\ntype UIMessagePart<T extends ToolSet> =\n  | TextUIPart\n  | ReasoningUIPart\n  | FileUIPart\n  | SourceUrlUIPart\n  | StepStartUIPart\n  | ToolUIPart<InferUITools<T>>;\n\nconst TextPart = z.object({\n  type: z.literal(\"text\"),\n  text: z.string(),\n});\n\nconst ReasoningPart = z.object({\n  type: z.literal(\"reasoning\"),\n  text: z.string(),\n});\n\nconst FilePart = z.object({\n  type: z.literal(\"file\"),\n  mediaType: z.string(),\n  url: z.string(),\n  filename: z.string().optional(),\n});\n\nconst SourceUrlPart = z.object({\n  type: z.literal(\"source-url\"),\n  sourceId: z.string(),\n  url: z.string(),\n  title: z.string().optional(),\n});\n\nconst StepStartPart = z.object({\n  type: z.literal(\"step-start\"),\n});\n\nfunction toolPartSchemas(name: string, tool: ToolSet[string]) {\n  const input = tool.inputSchema ?? z.unknown();\n  const output = tool.outputSchema ?? z.unknown();\n  const type = z.literal(`tool-${name}`);\n\n  const approvalId = z.object({ id: z.string() });\n  const approvalApproved = approvalId.extend({\n    approved: z.literal(true),\n    reason: z.string().optional(),\n  });\n  const approvalDenied = approvalId.extend({\n    approved: z.literal(false),\n    reason: z.string().optional(),\n  });\n  const approvalResponse = approvalId.extend({\n    approved: z.boolean(),\n    reason: z.string().optional(),\n  });\n\n  return [\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"input-streaming\"),\n      input: z.unknown(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"input-available\"),\n      input,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"approval-requested\"),\n      input,\n      approval: approvalId,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"approval-responded\"),\n      input,\n      approval: approvalResponse,\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-available\"),\n      input,\n      output,\n      approval: approvalApproved.optional(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-error\"),\n      input,\n      errorText: z.string(),\n      approval: approvalApproved.optional(),\n    }),\n    z.object({\n      type,\n      toolCallId: z.string(),\n      state: z.literal(\"output-denied\"),\n      input,\n      approval: approvalDenied,\n    }),\n  ];\n}\n\nconst UsageSummary = z.object({\n  model: z.string(),\n  inputTokens: z.number(),\n  outputTokens: z.number(),\n  totalTokens: z.number(),\n  cacheReadTokens: z.number(),\n  cacheWriteTokens: z.number(),\n  reasoningTokens: z.number(),\n  stepCount: z.number(),\n});\n\nconst SessionUsage = z.object({\n  total: UsageSummary.extend({ messageCount: z.number() }),\n  byMessageId: z.record(z.string(), UsageSummary.nullable()),\n});\n\nexport function buildSessionHistorySchema<Agent extends AnyAgent>(\n  agent: Agent\n) {\n  return z.object({\n    messages: z.array(buildUIMessageSchema(agent.tools)),\n    streamingMessageId: z.string().nullable(),\n    usage: SessionUsage,\n  });\n}\n\nexport function buildUIMessagePartSchema<T extends ToolSet>(\n  tools: T\n): z.ZodType<UIMessagePart<T>> {\n  const toolParts = Object.entries(tools).flatMap(([name, tool]) =>\n    toolPartSchemas(name, tool)\n  );\n\n  const all = [\n    TextPart,\n    ReasoningPart,\n    FilePart,\n    SourceUrlPart,\n    StepStartPart,\n    ...toolParts,\n  ];\n\n  return z.union(all) as z.ZodType<UIMessagePart<T>>;\n}\n\nexport function buildUIMessageSchema<T extends ToolSet>(tools: T) {\n  return z.object({\n    id: z.string(),\n    role: z.enum([\"system\", \"user\", \"assistant\"]),\n    parts: z.array(buildUIMessagePartSchema(tools)),\n  });\n}\n\nexport function buildSessionUISchema<T extends ToolSet>(tools: T) {\n  const UIMessage = buildUIMessageSchema(tools);\n\n  return z.object({\n    title: z.string().nullable(),\n    messages: z.array(UIMessage),\n    streamingMessageId: z.string().nullable(),\n    usage: SessionUsage,\n  });\n}\n"],
  "mappings": "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",
  "names": ["z", "z", "tool", "agent", "agent", "z", "agent", "z", "stream"]
}

|