experimental-agent 0.2.1 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +55 -254
- package/dist/adapter-BigchkkI.d.mts +201 -0
- package/dist/adapter-BigchkkI.d.ts +201 -0
- package/dist/chunk-BFFNCESS.mjs +302 -0
- package/dist/chunk-C4VSUEY2.mjs +72 -0
- package/dist/chunk-DOD4MC5D.mjs +196 -0
- package/dist/chunk-ELWIUJUK.mjs +96 -0
- package/dist/chunk-GKASMIBR.mjs +50 -0
- package/dist/chunk-JO3JDCH5.mjs +107 -0
- package/dist/chunk-MSWINCCM.mjs +128 -0
- package/dist/chunk-RT72C52I.mjs +324 -0
- package/dist/chunk-ZUFJJYC4.mjs +150 -0
- package/dist/{handler-FRUPZ4LX.mjs → docker-QPCLWLYR.mjs} +3 -4
- package/dist/entry-BmQ8FO-5.d.ts +36 -0
- package/dist/entry-CZd9aAwn.d.mts +36 -0
- package/dist/index.d.mts +415 -18
- package/dist/index.d.ts +415 -18
- package/dist/index.js +3036 -5494
- package/dist/index.mjs +3264 -1142
- package/dist/lifecycle-workflow-steps.d.mts +5 -0
- package/dist/lifecycle-workflow-steps.d.ts +5 -0
- package/dist/lifecycle-workflow-steps.js +263 -0
- package/dist/lifecycle-workflow-steps.mjs +9 -0
- package/dist/lifecycle-workflow.d.mts +6 -6
- package/dist/lifecycle-workflow.d.ts +6 -6
- package/dist/lifecycle-workflow.js +192 -905
- package/dist/lifecycle-workflow.mjs +3 -1
- package/dist/local-KJ3BSIFJ.mjs +8 -0
- package/dist/next/loader.d.mts +1 -0
- package/dist/next/loader.d.ts +1 -0
- package/dist/next/loader.js +44 -18
- package/dist/next/loader.mjs +18 -13
- package/dist/next.js +32 -9
- package/dist/next.mjs +6 -4
- package/dist/{process-manager-JDUJDYGU.mjs → process-manager-WQHAIVRB.mjs} +1 -1
- package/dist/sandbox.d.mts +6 -0
- package/dist/sandbox.d.ts +6 -0
- package/dist/sandbox.js +1070 -0
- package/dist/sandbox.mjs +19 -0
- package/dist/steps-BnkRQKlc.d.ts +173 -0
- package/dist/steps-u-mGDbP_.d.mts +173 -0
- package/dist/storage.d.mts +11 -0
- package/dist/storage.d.ts +11 -0
- package/dist/storage.js +234 -0
- package/dist/storage.mjs +12 -0
- package/dist/vercel-QZ6INPMV.mjs +11 -0
- package/package.json +26 -5
- package/dist/agent-workflow.d.mts +0 -30
- package/dist/agent-workflow.d.ts +0 -30
- package/dist/agent-workflow.js +0 -5433
- package/dist/agent-workflow.mjs +0 -14
- package/dist/chunk-AML2VCQS.mjs +0 -1287
- package/dist/chunk-FQ67QZOI.mjs +0 -75
- package/dist/chunk-NO7RHGTH.mjs +0 -2367
- package/dist/chunk-NXDVNJRS.mjs +0 -106
- package/dist/chunk-OZZVS6L5.mjs +0 -139
- package/dist/chunk-QRWGDFFY.mjs +0 -75
- package/dist/chunk-SJVFFE5D.mjs +0 -402
- package/dist/chunk-TAXLUVIC.mjs +0 -1
- package/dist/chunk-TGNVXSMX.mjs +0 -399
- package/dist/chunk-YRYXN7W4.mjs +0 -48
- package/dist/chunk-ZIAHPXOJ.mjs +0 -595
- package/dist/client-BKA7XBGW.mjs +0 -15
- package/dist/client-CEeSFGva.d.mts +0 -2376
- package/dist/client-CEeSFGva.d.ts +0 -2376
- package/dist/docker-FB2MJTHJ.mjs +0 -12
- package/dist/local-fs-handlers-SYOCKTPN.mjs +0 -447
- package/dist/sandbox-UENKQV3T.mjs +0 -21
- package/dist/storage-LSDMRW73.mjs +0 -20
- package/dist/vercel-SD3JTECG.mjs +0 -20
- package/dist/vercel-sdk-I6A4MVAN.mjs +0 -8
package/dist/sandbox.js
ADDED
|
@@ -0,0 +1,1070 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __export = (target, all) => {
|
|
9
|
+
for (var name in all)
|
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
+
};
|
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (let key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
+
}
|
|
18
|
+
return to;
|
|
19
|
+
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
+
mod
|
|
27
|
+
));
|
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
+
|
|
30
|
+
// src/sandbox/entry.ts
|
|
31
|
+
var entry_exports = {};
|
|
32
|
+
__export(entry_exports, {
|
|
33
|
+
dockerSandbox: () => dockerSandbox,
|
|
34
|
+
localSandbox: () => localSandbox,
|
|
35
|
+
vercelSandbox: () => vercelSandbox
|
|
36
|
+
});
|
|
37
|
+
module.exports = __toCommonJS(entry_exports);
|
|
38
|
+
|
|
39
|
+
// src/sandbox/bindings/docker.ts
|
|
40
|
+
var import_ulid = require("ulid");
|
|
41
|
+
async function execDocker(args, opts) {
|
|
42
|
+
const { spawn } = await import("child_process");
|
|
43
|
+
return new Promise((resolve, reject) => {
|
|
44
|
+
const child = spawn("docker", ["sandbox", ...args], {
|
|
45
|
+
signal: opts?.signal
|
|
46
|
+
});
|
|
47
|
+
let stdout = "";
|
|
48
|
+
let stderr = "";
|
|
49
|
+
child.stdout.on("data", (data) => {
|
|
50
|
+
stdout += data.toString();
|
|
51
|
+
});
|
|
52
|
+
child.stderr.on("data", (data) => {
|
|
53
|
+
stderr += data.toString();
|
|
54
|
+
});
|
|
55
|
+
const timeoutId = opts?.timeoutMs ? setTimeout(() => {
|
|
56
|
+
child.kill("SIGTERM");
|
|
57
|
+
reject(new Error(`docker sandbox ${args[0]} timed out`));
|
|
58
|
+
}, opts.timeoutMs) : void 0;
|
|
59
|
+
child.on("error", (err) => {
|
|
60
|
+
if (timeoutId) {
|
|
61
|
+
clearTimeout(timeoutId);
|
|
62
|
+
}
|
|
63
|
+
reject(err);
|
|
64
|
+
});
|
|
65
|
+
child.on("close", (code) => {
|
|
66
|
+
if (timeoutId) {
|
|
67
|
+
clearTimeout(timeoutId);
|
|
68
|
+
}
|
|
69
|
+
resolve({ stdout, stderr, exitCode: code ?? 0 });
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
var ensurePromises = /* @__PURE__ */ new Map();
|
|
74
|
+
var activeSandboxes = /* @__PURE__ */ new Set();
|
|
75
|
+
var cleanupRegistered = false;
|
|
76
|
+
var _execSync = null;
|
|
77
|
+
async function registerCleanup() {
|
|
78
|
+
if (cleanupRegistered) {
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
cleanupRegistered = true;
|
|
82
|
+
const cp = await import("child_process");
|
|
83
|
+
_execSync = cp.execSync;
|
|
84
|
+
const cleanup = () => {
|
|
85
|
+
if (!_execSync) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
for (const name of Array.from(activeSandboxes)) {
|
|
89
|
+
try {
|
|
90
|
+
_execSync(`docker sandbox stop ${name}`, {
|
|
91
|
+
timeout: 1e4,
|
|
92
|
+
stdio: "pipe"
|
|
93
|
+
});
|
|
94
|
+
} catch {
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
};
|
|
98
|
+
process.on("exit", cleanup);
|
|
99
|
+
process.on("SIGINT", () => {
|
|
100
|
+
cleanup();
|
|
101
|
+
process.exit(130);
|
|
102
|
+
});
|
|
103
|
+
process.on("SIGTERM", () => {
|
|
104
|
+
cleanup();
|
|
105
|
+
process.exit(143);
|
|
106
|
+
});
|
|
107
|
+
}
|
|
108
|
+
async function ensureSandbox(sandboxName) {
|
|
109
|
+
const existing = ensurePromises.get(sandboxName);
|
|
110
|
+
if (existing) {
|
|
111
|
+
return existing;
|
|
112
|
+
}
|
|
113
|
+
const promise = (async () => {
|
|
114
|
+
const ls = await execDocker(["ls", "-q"], { timeoutMs: 1e4 });
|
|
115
|
+
const existingNames = ls.exitCode === 0 ? ls.stdout.split("\n").map((s) => s.trim()).filter(Boolean) : [];
|
|
116
|
+
if (existingNames.includes(sandboxName)) {
|
|
117
|
+
activeSandboxes.add(sandboxName);
|
|
118
|
+
registerCleanup();
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
const path = await import("path");
|
|
122
|
+
const os = await import("os");
|
|
123
|
+
const fs = await import("fs/promises");
|
|
124
|
+
const workspaceDir = path.join(
|
|
125
|
+
os.tmpdir(),
|
|
126
|
+
"agent-docker-sandbox",
|
|
127
|
+
sandboxName
|
|
128
|
+
);
|
|
129
|
+
await fs.mkdir(workspaceDir, { recursive: true });
|
|
130
|
+
const create = await execDocker(
|
|
131
|
+
["create", "--name", sandboxName, "shell", workspaceDir],
|
|
132
|
+
{ timeoutMs: 6e4 }
|
|
133
|
+
);
|
|
134
|
+
if (create.exitCode !== 0) {
|
|
135
|
+
if (create.stderr.includes("already exists")) {
|
|
136
|
+
activeSandboxes.add(sandboxName);
|
|
137
|
+
registerCleanup();
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
throw new Error(
|
|
141
|
+
`Failed to create docker sandbox "${sandboxName}": ${create.stderr}`
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
activeSandboxes.add(sandboxName);
|
|
145
|
+
registerCleanup();
|
|
146
|
+
})();
|
|
147
|
+
ensurePromises.set(sandboxName, promise);
|
|
148
|
+
try {
|
|
149
|
+
await promise;
|
|
150
|
+
} catch (e) {
|
|
151
|
+
ensurePromises.delete(sandboxName);
|
|
152
|
+
throw e;
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
var DEFAULT_DOCKER_CWD = "/home/agent/workspace";
|
|
156
|
+
var DockerSandboxInstance = class {
|
|
157
|
+
cwd;
|
|
158
|
+
sandboxName;
|
|
159
|
+
processes = /* @__PURE__ */ new Map();
|
|
160
|
+
constructor(sandboxName, cwd) {
|
|
161
|
+
this.sandboxName = sandboxName;
|
|
162
|
+
this.cwd = cwd ?? DEFAULT_DOCKER_CWD;
|
|
163
|
+
}
|
|
164
|
+
async exec(opts) {
|
|
165
|
+
await ensureSandbox(this.sandboxName);
|
|
166
|
+
const { spawn } = await import("child_process");
|
|
167
|
+
const commandId = `command_${(0, import_ulid.ulid)()}`;
|
|
168
|
+
const envFlags = opts.env ? Object.entries(opts.env).flatMap(([k, v]) => ["-e", `${k}=${v}`]) : [];
|
|
169
|
+
const cwdFlags = opts.cwd ? ["-w", opts.cwd] : [];
|
|
170
|
+
const baseCmd = opts.sudo ? ["sudo", opts.command, ...opts.args ?? []] : opts.args ? [opts.command, ...opts.args] : [opts.command];
|
|
171
|
+
const fullCmd = baseCmd;
|
|
172
|
+
const child = spawn(
|
|
173
|
+
"docker",
|
|
174
|
+
[
|
|
175
|
+
"sandbox",
|
|
176
|
+
"exec",
|
|
177
|
+
...envFlags,
|
|
178
|
+
...cwdFlags,
|
|
179
|
+
this.sandboxName,
|
|
180
|
+
...fullCmd
|
|
181
|
+
],
|
|
182
|
+
{ signal: opts.signal }
|
|
183
|
+
);
|
|
184
|
+
this.processes.set(commandId, child);
|
|
185
|
+
let stdout = "";
|
|
186
|
+
let stderr = "";
|
|
187
|
+
const logQueue = [];
|
|
188
|
+
let logResolve = null;
|
|
189
|
+
let closed = false;
|
|
190
|
+
child.stdout.on("data", (data) => {
|
|
191
|
+
const str = String(data);
|
|
192
|
+
stdout += str;
|
|
193
|
+
logQueue.push({ stream: "stdout", data: str });
|
|
194
|
+
logResolve?.();
|
|
195
|
+
});
|
|
196
|
+
child.stderr.on("data", (data) => {
|
|
197
|
+
const str = String(data);
|
|
198
|
+
stderr += str;
|
|
199
|
+
logQueue.push({ stream: "stderr", data: str });
|
|
200
|
+
logResolve?.();
|
|
201
|
+
});
|
|
202
|
+
const result = new Promise((resolve, reject) => {
|
|
203
|
+
child.on("error", (err) => {
|
|
204
|
+
this.processes.delete(commandId);
|
|
205
|
+
closed = true;
|
|
206
|
+
logResolve?.();
|
|
207
|
+
reject(err);
|
|
208
|
+
});
|
|
209
|
+
child.on("close", (code) => {
|
|
210
|
+
this.processes.delete(commandId);
|
|
211
|
+
closed = true;
|
|
212
|
+
logResolve?.();
|
|
213
|
+
resolve({ stdout, stderr, exitCode: code ?? 0 });
|
|
214
|
+
});
|
|
215
|
+
});
|
|
216
|
+
async function* logs() {
|
|
217
|
+
while (!closed || logQueue.length > 0) {
|
|
218
|
+
const entry = logQueue.shift();
|
|
219
|
+
if (entry) {
|
|
220
|
+
yield entry;
|
|
221
|
+
} else if (!closed) {
|
|
222
|
+
await new Promise((r) => {
|
|
223
|
+
logResolve = r;
|
|
224
|
+
});
|
|
225
|
+
logResolve = null;
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
return { commandId, logs, result };
|
|
230
|
+
}
|
|
231
|
+
async readFile(opts) {
|
|
232
|
+
await ensureSandbox(this.sandboxName);
|
|
233
|
+
const result = await execDocker(
|
|
234
|
+
[
|
|
235
|
+
"exec",
|
|
236
|
+
this.sandboxName,
|
|
237
|
+
"bash",
|
|
238
|
+
"-c",
|
|
239
|
+
`base64 '${opts.path.replace(/'/g, "'\\''")}'`
|
|
240
|
+
],
|
|
241
|
+
{ timeoutMs: 3e4, signal: opts.signal }
|
|
242
|
+
);
|
|
243
|
+
if (result.exitCode !== 0) {
|
|
244
|
+
if (result.stderr.includes("No such file") || result.stderr.includes("ENOENT")) {
|
|
245
|
+
return null;
|
|
246
|
+
}
|
|
247
|
+
throw new Error(`readFile failed: ${result.stderr}`);
|
|
248
|
+
}
|
|
249
|
+
return Buffer.from(result.stdout.trim(), "base64");
|
|
250
|
+
}
|
|
251
|
+
async writeFiles(opts) {
|
|
252
|
+
await ensureSandbox(this.sandboxName);
|
|
253
|
+
for (const file of opts.files) {
|
|
254
|
+
const fullPath = opts.destPath ? `${opts.destPath}/${file.path}` : file.path;
|
|
255
|
+
const parentDir = fullPath.substring(0, fullPath.lastIndexOf("/"));
|
|
256
|
+
if (parentDir) {
|
|
257
|
+
await execDocker(["exec", this.sandboxName, "mkdir", "-p", parentDir], {
|
|
258
|
+
signal: opts.signal
|
|
259
|
+
});
|
|
260
|
+
}
|
|
261
|
+
const b64 = typeof file.content === "string" ? Buffer.from(file.content).toString("base64") : file.content.toString("base64");
|
|
262
|
+
await execDocker(
|
|
263
|
+
[
|
|
264
|
+
"exec",
|
|
265
|
+
this.sandboxName,
|
|
266
|
+
"bash",
|
|
267
|
+
"-c",
|
|
268
|
+
`echo '${b64}' | base64 -d > '${fullPath.replace(/'/g, "'\\''")}'`
|
|
269
|
+
],
|
|
270
|
+
{ signal: opts.signal }
|
|
271
|
+
);
|
|
272
|
+
if (file.path.endsWith(".sh")) {
|
|
273
|
+
await execDocker(["exec", this.sandboxName, "chmod", "+x", fullPath], {
|
|
274
|
+
signal: opts.signal
|
|
275
|
+
});
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
280
|
+
async getDomain(opts) {
|
|
281
|
+
return `http://localhost:${opts.port}`;
|
|
282
|
+
}
|
|
283
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
284
|
+
async kill(opts) {
|
|
285
|
+
const child = this.processes.get(opts.commandId);
|
|
286
|
+
if (child) {
|
|
287
|
+
child.kill("SIGTERM");
|
|
288
|
+
this.processes.delete(opts.commandId);
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
292
|
+
async getStatus() {
|
|
293
|
+
return "running";
|
|
294
|
+
}
|
|
295
|
+
async start() {
|
|
296
|
+
await ensureSandbox(this.sandboxName);
|
|
297
|
+
}
|
|
298
|
+
async stop() {
|
|
299
|
+
await execDocker(["stop", this.sandboxName], { timeoutMs: 3e4 });
|
|
300
|
+
activeSandboxes.delete(this.sandboxName);
|
|
301
|
+
ensurePromises.delete(this.sandboxName);
|
|
302
|
+
}
|
|
303
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
304
|
+
async snapshot() {
|
|
305
|
+
throw new Error("snapshot is not supported for docker sandboxes");
|
|
306
|
+
}
|
|
307
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
308
|
+
async updateNetworkPolicy() {
|
|
309
|
+
throw new Error(
|
|
310
|
+
"updateNetworkPolicy is not supported for docker sandboxes"
|
|
311
|
+
);
|
|
312
|
+
}
|
|
313
|
+
};
|
|
314
|
+
function dockerSandbox() {
|
|
315
|
+
return {
|
|
316
|
+
type: "docker",
|
|
317
|
+
async create(opts) {
|
|
318
|
+
const sandboxName = `agent-${(0, import_ulid.ulid)()}`;
|
|
319
|
+
const cwd = opts.setup?.config?.cwd;
|
|
320
|
+
await ensureSandbox(sandboxName);
|
|
321
|
+
const instance = new DockerSandboxInstance(sandboxName, cwd);
|
|
322
|
+
if (opts.setup?.run) {
|
|
323
|
+
await opts.setup.run(instance);
|
|
324
|
+
}
|
|
325
|
+
return {
|
|
326
|
+
instance,
|
|
327
|
+
metadata: { sandboxName }
|
|
328
|
+
};
|
|
329
|
+
},
|
|
330
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
331
|
+
async connect(opts) {
|
|
332
|
+
return new DockerSandboxInstance(opts.metadata.sandboxName);
|
|
333
|
+
}
|
|
334
|
+
};
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// src/sandbox/bindings/local.ts
|
|
338
|
+
var import_ulid2 = require("ulid");
|
|
339
|
+
var LocalSandboxInstance = class {
|
|
340
|
+
cwd;
|
|
341
|
+
basePath;
|
|
342
|
+
processes = /* @__PURE__ */ new Map();
|
|
343
|
+
constructor(basePath) {
|
|
344
|
+
this.basePath = basePath;
|
|
345
|
+
this.cwd = basePath;
|
|
346
|
+
}
|
|
347
|
+
async exec(opts) {
|
|
348
|
+
const { spawn } = await import("child_process");
|
|
349
|
+
const path = await import("path");
|
|
350
|
+
const commandId = `command_${(0, import_ulid2.ulid)()}`;
|
|
351
|
+
const finalCmd = opts.sudo ? "sudo" : opts.command;
|
|
352
|
+
const finalArgs = opts.sudo ? [opts.command, ...opts.args ?? []] : opts.args ?? [];
|
|
353
|
+
const child = spawn(finalCmd, finalArgs, {
|
|
354
|
+
cwd: opts.cwd ? path.resolve(this.basePath, opts.cwd) : this.basePath,
|
|
355
|
+
env: opts.env ? { ...process.env, ...opts.env } : void 0,
|
|
356
|
+
signal: opts.signal
|
|
357
|
+
});
|
|
358
|
+
this.processes.set(commandId, child);
|
|
359
|
+
let stdout = "";
|
|
360
|
+
let stderr = "";
|
|
361
|
+
const logQueue = [];
|
|
362
|
+
let logResolve = null;
|
|
363
|
+
let closed = false;
|
|
364
|
+
child.stdout.on("data", (data) => {
|
|
365
|
+
const str = String(data);
|
|
366
|
+
stdout += str;
|
|
367
|
+
logQueue.push({ stream: "stdout", data: str });
|
|
368
|
+
logResolve?.();
|
|
369
|
+
});
|
|
370
|
+
child.stderr.on("data", (data) => {
|
|
371
|
+
const str = String(data);
|
|
372
|
+
stderr += str;
|
|
373
|
+
logQueue.push({ stream: "stderr", data: str });
|
|
374
|
+
logResolve?.();
|
|
375
|
+
});
|
|
376
|
+
const result = new Promise((resolve, reject) => {
|
|
377
|
+
child.on("error", (err) => {
|
|
378
|
+
this.processes.delete(commandId);
|
|
379
|
+
closed = true;
|
|
380
|
+
logResolve?.();
|
|
381
|
+
reject(err);
|
|
382
|
+
});
|
|
383
|
+
child.on("close", (code) => {
|
|
384
|
+
this.processes.delete(commandId);
|
|
385
|
+
closed = true;
|
|
386
|
+
logResolve?.();
|
|
387
|
+
resolve({ stdout, stderr, exitCode: code ?? 0 });
|
|
388
|
+
});
|
|
389
|
+
});
|
|
390
|
+
async function* logs() {
|
|
391
|
+
while (!closed || logQueue.length > 0) {
|
|
392
|
+
const entry = logQueue.shift();
|
|
393
|
+
if (entry) {
|
|
394
|
+
yield entry;
|
|
395
|
+
} else if (!closed) {
|
|
396
|
+
await new Promise((r) => {
|
|
397
|
+
logResolve = r;
|
|
398
|
+
});
|
|
399
|
+
logResolve = null;
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
return { commandId, logs, result };
|
|
404
|
+
}
|
|
405
|
+
async readFile(opts) {
|
|
406
|
+
const fs = await import("fs/promises");
|
|
407
|
+
const path = await import("path");
|
|
408
|
+
const fullPath = path.resolve(this.basePath, opts.path);
|
|
409
|
+
try {
|
|
410
|
+
return await fs.readFile(fullPath);
|
|
411
|
+
} catch (e) {
|
|
412
|
+
if (e instanceof Error && "code" in e && e.code === "ENOENT") {
|
|
413
|
+
return null;
|
|
414
|
+
}
|
|
415
|
+
throw e;
|
|
416
|
+
}
|
|
417
|
+
}
|
|
418
|
+
async writeFiles(opts) {
|
|
419
|
+
const fs = await import("fs/promises");
|
|
420
|
+
const pathMod = await import("path");
|
|
421
|
+
for (const file of opts.files) {
|
|
422
|
+
const fullPath = pathMod.resolve(this.basePath, opts.destPath, file.path);
|
|
423
|
+
await fs.mkdir(pathMod.dirname(fullPath), { recursive: true });
|
|
424
|
+
await fs.writeFile(fullPath, file.content);
|
|
425
|
+
if (file.path.endsWith(".sh")) {
|
|
426
|
+
await fs.chmod(fullPath, 493);
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
}
|
|
430
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
431
|
+
async getDomain(opts) {
|
|
432
|
+
return `http://localhost:${opts.port}`;
|
|
433
|
+
}
|
|
434
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
435
|
+
async kill(opts) {
|
|
436
|
+
const child = this.processes.get(opts.commandId);
|
|
437
|
+
if (child) {
|
|
438
|
+
child.kill("SIGTERM");
|
|
439
|
+
this.processes.delete(opts.commandId);
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
443
|
+
async getStatus() {
|
|
444
|
+
return "running";
|
|
445
|
+
}
|
|
446
|
+
async start() {
|
|
447
|
+
}
|
|
448
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
449
|
+
async stop() {
|
|
450
|
+
throw new Error("stop is not supported for local sandboxes");
|
|
451
|
+
}
|
|
452
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
453
|
+
async snapshot() {
|
|
454
|
+
throw new Error("snapshot is not supported for local sandboxes");
|
|
455
|
+
}
|
|
456
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
457
|
+
async updateNetworkPolicy() {
|
|
458
|
+
throw new Error("updateNetworkPolicy is not supported for local sandboxes");
|
|
459
|
+
}
|
|
460
|
+
};
|
|
461
|
+
function localSandbox() {
|
|
462
|
+
return {
|
|
463
|
+
type: "local",
|
|
464
|
+
async create(opts) {
|
|
465
|
+
const basePath = opts.setup.config?.cwd ?? process.cwd();
|
|
466
|
+
const instance = new LocalSandboxInstance(basePath);
|
|
467
|
+
if (opts.setup?.run) {
|
|
468
|
+
await opts.setup.run(instance);
|
|
469
|
+
}
|
|
470
|
+
return {
|
|
471
|
+
instance,
|
|
472
|
+
metadata: { basePath, pid: process.pid }
|
|
473
|
+
};
|
|
474
|
+
},
|
|
475
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
476
|
+
async connect(opts) {
|
|
477
|
+
const basePath = opts.metadata.basePath;
|
|
478
|
+
return new LocalSandboxInstance(basePath);
|
|
479
|
+
}
|
|
480
|
+
};
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
// src/sandbox/bindings/vercel.ts
|
|
484
|
+
var import_api = require("workflow/api");
|
|
485
|
+
|
|
486
|
+
// src/sandbox/bindings/vercel-lifecycle/workflow.ts
|
|
487
|
+
var import_workflow = require("workflow");
|
|
488
|
+
|
|
489
|
+
// src/utils/logger.ts
|
|
490
|
+
var LOG_LEVEL_PRIORITY = {
|
|
491
|
+
info: 0,
|
|
492
|
+
warn: 1,
|
|
493
|
+
error: 2,
|
|
494
|
+
silent: 3
|
|
495
|
+
};
|
|
496
|
+
function getLevel(config) {
|
|
497
|
+
return config.level ?? "warn";
|
|
498
|
+
}
|
|
499
|
+
function getPrefix(config) {
|
|
500
|
+
return config.prefix ?? "agent";
|
|
501
|
+
}
|
|
502
|
+
function shouldLog(config, level) {
|
|
503
|
+
return LOG_LEVEL_PRIORITY[level] >= LOG_LEVEL_PRIORITY[getLevel(config)];
|
|
504
|
+
}
|
|
505
|
+
function formatTag(config, subsystem) {
|
|
506
|
+
return `[${getPrefix(config)}:${config.name}:${subsystem}]`;
|
|
507
|
+
}
|
|
508
|
+
function formatData(context, extra) {
|
|
509
|
+
const merged = { ...context, ...extra };
|
|
510
|
+
const entries = Object.entries(merged).filter(
|
|
511
|
+
([, v]) => v !== void 0 && v !== null
|
|
512
|
+
);
|
|
513
|
+
if (entries.length === 0) {
|
|
514
|
+
return void 0;
|
|
515
|
+
}
|
|
516
|
+
return Object.fromEntries(entries);
|
|
517
|
+
}
|
|
518
|
+
var Logger = class _Logger {
|
|
519
|
+
subsystem;
|
|
520
|
+
config;
|
|
521
|
+
context;
|
|
522
|
+
constructor({
|
|
523
|
+
subsystem,
|
|
524
|
+
config,
|
|
525
|
+
context
|
|
526
|
+
}) {
|
|
527
|
+
this.subsystem = subsystem;
|
|
528
|
+
this.config = config;
|
|
529
|
+
this.context = context ?? {};
|
|
530
|
+
}
|
|
531
|
+
child({
|
|
532
|
+
subsystem,
|
|
533
|
+
context
|
|
534
|
+
}) {
|
|
535
|
+
return new _Logger({
|
|
536
|
+
config: this.config,
|
|
537
|
+
subsystem: subsystem ? `${this.subsystem}:${subsystem}` : this.subsystem,
|
|
538
|
+
context: { ...this.context, ...context }
|
|
539
|
+
});
|
|
540
|
+
}
|
|
541
|
+
withContext(context) {
|
|
542
|
+
return new _Logger({
|
|
543
|
+
config: this.config,
|
|
544
|
+
subsystem: this.subsystem,
|
|
545
|
+
context: { ...this.context, ...context }
|
|
546
|
+
});
|
|
547
|
+
}
|
|
548
|
+
info(message, data) {
|
|
549
|
+
if (!shouldLog(this.config, "info")) {
|
|
550
|
+
return;
|
|
551
|
+
}
|
|
552
|
+
const d = formatData(this.context, data);
|
|
553
|
+
if (d) {
|
|
554
|
+
console.info(formatTag(this.config, this.subsystem), message, d);
|
|
555
|
+
} else {
|
|
556
|
+
console.info(formatTag(this.config, this.subsystem), message);
|
|
557
|
+
}
|
|
558
|
+
}
|
|
559
|
+
warn(message, data) {
|
|
560
|
+
if (!shouldLog(this.config, "warn")) {
|
|
561
|
+
return;
|
|
562
|
+
}
|
|
563
|
+
const d = formatData(this.context, data);
|
|
564
|
+
if (d) {
|
|
565
|
+
console.warn(formatTag(this.config, this.subsystem), message, d);
|
|
566
|
+
} else {
|
|
567
|
+
console.warn(formatTag(this.config, this.subsystem), message);
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
error(message, data) {
|
|
571
|
+
if (!shouldLog(this.config, "error")) {
|
|
572
|
+
return;
|
|
573
|
+
}
|
|
574
|
+
const d = formatData(this.context, data);
|
|
575
|
+
if (d) {
|
|
576
|
+
console.error(formatTag(this.config, this.subsystem), message, d);
|
|
577
|
+
} else {
|
|
578
|
+
console.error(formatTag(this.config, this.subsystem), message);
|
|
579
|
+
}
|
|
580
|
+
}
|
|
581
|
+
/**
|
|
582
|
+
* Returns a function that, when called, logs the elapsed time at info level.
|
|
583
|
+
* Pass `logOnStart: true` to also log when the timer begins.
|
|
584
|
+
*
|
|
585
|
+
* @example
|
|
586
|
+
* const done = log.time("sandbox setup", { sandboxId }, { logOnStart: true });
|
|
587
|
+
* await setup();
|
|
588
|
+
* done(); // start: [agent:default:sandbox] sandbox setup { sandboxId: '...' }
|
|
589
|
+
* // end: [agent:default:sandbox] sandbox setup { sandboxId: '...', durationMs: 123 }
|
|
590
|
+
*/
|
|
591
|
+
time(message, data, opts) {
|
|
592
|
+
if (opts?.logOnStart) {
|
|
593
|
+
this.info(`[start] ${message}`, data);
|
|
594
|
+
}
|
|
595
|
+
const t0 = Date.now();
|
|
596
|
+
return (endData) => {
|
|
597
|
+
this.info(opts?.logOnStart ? `[end] ${message}` : message, {
|
|
598
|
+
...data,
|
|
599
|
+
...endData,
|
|
600
|
+
durationMs: Date.now() - t0
|
|
601
|
+
});
|
|
602
|
+
};
|
|
603
|
+
}
|
|
604
|
+
};
|
|
605
|
+
function createLogger({
|
|
606
|
+
config,
|
|
607
|
+
subsystem,
|
|
608
|
+
context
|
|
609
|
+
}) {
|
|
610
|
+
return new Logger({ config, subsystem, context });
|
|
611
|
+
}
|
|
612
|
+
|
|
613
|
+
// src/sandbox/bindings/vercel-lifecycle/steps.ts
|
|
614
|
+
async function loadSandboxSDK() {
|
|
615
|
+
return (await import("@vercel/sandbox")).Sandbox;
|
|
616
|
+
}
|
|
617
|
+
var DEFAULT_POLL_INTERVAL_MS = 2 * 60 * 1e3;
|
|
618
|
+
var DEFAULT_IDLE_TIMEOUT_MS = 5 * 60 * 1e3;
|
|
619
|
+
var SNAPSHOT_BEFORE_TIMEOUT_MS = 10 * 60 * 1e3;
|
|
620
|
+
var getTestCredentials = () => process.env.NODE_ENV === "test" ? {
|
|
621
|
+
token: process.env.TEST_VERCEL_TOKEN,
|
|
622
|
+
teamId: process.env.TEST_VERCEL_TEAM_ID,
|
|
623
|
+
projectId: process.env.TEST_VERCEL_PROJECT_ID
|
|
624
|
+
} : {};
|
|
625
|
+
async function checkAndSnapshotStep(input) {
|
|
626
|
+
"use step";
|
|
627
|
+
const storage = input.agent.storage;
|
|
628
|
+
const log = createLogger({
|
|
629
|
+
config: { ...input.agent.options.logging, name: input.agent.name },
|
|
630
|
+
subsystem: "sandbox:lifecycle",
|
|
631
|
+
context: { sandboxId: input.sandboxId }
|
|
632
|
+
});
|
|
633
|
+
log.info("lifecycle check started", {
|
|
634
|
+
vercelSandboxId: input.vercelSandboxId
|
|
635
|
+
});
|
|
636
|
+
const record = await storage.sandbox.get(input.sandboxId);
|
|
637
|
+
if (!record) {
|
|
638
|
+
log.warn("sandbox record not found, exiting lifecycle");
|
|
639
|
+
return { action: "exit", reason: "not_found" };
|
|
640
|
+
}
|
|
641
|
+
const metadata = record.setup?.metadata;
|
|
642
|
+
const currentSandboxId = metadata?.sandboxId ?? null;
|
|
643
|
+
if (currentSandboxId !== input.vercelSandboxId) {
|
|
644
|
+
log.info("sandbox ID changed, exiting lifecycle", {
|
|
645
|
+
expected: input.vercelSandboxId,
|
|
646
|
+
actual: currentSandboxId
|
|
647
|
+
});
|
|
648
|
+
return { action: "exit", reason: "sandboxId_changed" };
|
|
649
|
+
}
|
|
650
|
+
if (!currentSandboxId) {
|
|
651
|
+
log.warn("no current sandbox ID in metadata, exiting lifecycle");
|
|
652
|
+
return { action: "exit", reason: "not_found" };
|
|
653
|
+
}
|
|
654
|
+
const pollIntervalMs = input.config?.pollIntervalMs ?? DEFAULT_POLL_INTERVAL_MS;
|
|
655
|
+
const idleTimeoutMs = input.config?.stopAfterInactiveMs ?? DEFAULT_IDLE_TIMEOUT_MS;
|
|
656
|
+
const snapshotBeforeTimeoutMs = input.config?.snapshotBeforeTimeoutMs ?? SNAPSHOT_BEFORE_TIMEOUT_MS;
|
|
657
|
+
const now = Date.now();
|
|
658
|
+
const lastActivity = record.lastActiveAt ?? record.createdAt ?? now;
|
|
659
|
+
const idleDuration = now - lastActivity;
|
|
660
|
+
const shouldSnapshotDueToIdle = idleDuration > idleTimeoutMs;
|
|
661
|
+
let shouldSnapshotDueToTimeout = false;
|
|
662
|
+
try {
|
|
663
|
+
const SandboxSDK = await loadSandboxSDK();
|
|
664
|
+
const sandbox = await SandboxSDK.get({
|
|
665
|
+
sandboxId: currentSandboxId,
|
|
666
|
+
...getTestCredentials()
|
|
667
|
+
});
|
|
668
|
+
if (sandbox.timeout < snapshotBeforeTimeoutMs) {
|
|
669
|
+
shouldSnapshotDueToTimeout = true;
|
|
670
|
+
}
|
|
671
|
+
} catch {
|
|
672
|
+
log.warn("failed to get sandbox from SDK, exiting lifecycle", {
|
|
673
|
+
vercelSandboxId: currentSandboxId
|
|
674
|
+
});
|
|
675
|
+
return { action: "exit", reason: "not_found" };
|
|
676
|
+
}
|
|
677
|
+
if (shouldSnapshotDueToIdle || shouldSnapshotDueToTimeout) {
|
|
678
|
+
const reason = shouldSnapshotDueToIdle ? "idle" : "timeout";
|
|
679
|
+
log.info("snapshotting sandbox before exit", {
|
|
680
|
+
reason,
|
|
681
|
+
idleDurationMs: idleDuration,
|
|
682
|
+
vercelSandboxId: currentSandboxId
|
|
683
|
+
});
|
|
684
|
+
try {
|
|
685
|
+
const SandboxSDK = await loadSandboxSDK();
|
|
686
|
+
const sandbox = await SandboxSDK.get({
|
|
687
|
+
sandboxId: currentSandboxId,
|
|
688
|
+
...getTestCredentials()
|
|
689
|
+
});
|
|
690
|
+
const snapshot = await sandbox.snapshot();
|
|
691
|
+
log.info("snapshot created", { snapshotId: snapshot.snapshotId });
|
|
692
|
+
await storage.sandbox.update(input.sandboxId, {
|
|
693
|
+
setup: {
|
|
694
|
+
...record.setup,
|
|
695
|
+
metadata: { sandboxId: null, snapshotId: snapshot.snapshotId }
|
|
696
|
+
}
|
|
697
|
+
});
|
|
698
|
+
log.info("sandbox record updated with snapshot");
|
|
699
|
+
} catch (e) {
|
|
700
|
+
log.error("failed to snapshot sandbox", { cause: e });
|
|
701
|
+
return e instanceof Error ? e : new Error(String(e));
|
|
702
|
+
}
|
|
703
|
+
return { action: "exit", reason };
|
|
704
|
+
}
|
|
705
|
+
log.info("sandbox still active, continuing lifecycle", {
|
|
706
|
+
idleDurationMs: idleDuration,
|
|
707
|
+
nextPollMs: pollIntervalMs
|
|
708
|
+
});
|
|
709
|
+
return { action: "continue", nextPollMs: pollIntervalMs };
|
|
710
|
+
}
|
|
711
|
+
|
|
712
|
+
// src/sandbox/bindings/vercel-lifecycle/workflow.ts
|
|
713
|
+
var DEFAULT_POLL_MS = 2 * 60 * 1e3;
|
|
714
|
+
async function sandboxLifecycleWorkflow({
|
|
715
|
+
input
|
|
716
|
+
}) {
|
|
717
|
+
"use workflow";
|
|
718
|
+
let nextPollMs = input.config?.pollIntervalMs ?? DEFAULT_POLL_MS;
|
|
719
|
+
const log = createLogger({
|
|
720
|
+
config: { ...input.agent.options.logging, name: input.agent.name },
|
|
721
|
+
subsystem: "sandbox:lifecycle",
|
|
722
|
+
context: { sandboxId: input.sandboxId }
|
|
723
|
+
});
|
|
724
|
+
log.info("lifecycle workflow started", {
|
|
725
|
+
sandboxId: input.sandboxId,
|
|
726
|
+
vercelSandboxId: input.vercelSandboxId,
|
|
727
|
+
pollMs: nextPollMs
|
|
728
|
+
});
|
|
729
|
+
while (true) {
|
|
730
|
+
await (0, import_workflow.sleep)(nextPollMs);
|
|
731
|
+
const result = await checkAndSnapshotStep(input);
|
|
732
|
+
if (result instanceof Error) {
|
|
733
|
+
log.error("lifecycle step failed", {
|
|
734
|
+
sandboxId: input.sandboxId,
|
|
735
|
+
error: result.message
|
|
736
|
+
});
|
|
737
|
+
break;
|
|
738
|
+
}
|
|
739
|
+
if (result.action === "exit") {
|
|
740
|
+
log.info("lifecycle workflow exiting", {
|
|
741
|
+
sandboxId: input.sandboxId,
|
|
742
|
+
reason: result.reason
|
|
743
|
+
});
|
|
744
|
+
break;
|
|
745
|
+
}
|
|
746
|
+
nextPollMs = result.nextPollMs;
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
// src/sandbox/bindings/vercel.ts
|
|
751
|
+
async function loadSandboxSDK2() {
|
|
752
|
+
return (await import("@vercel/sandbox")).Sandbox;
|
|
753
|
+
}
|
|
754
|
+
var MAX_TIMEOUT_MS = 5 * 60 * 60 * 1e3;
|
|
755
|
+
var HOME_DIR = "/home/vercel-sandbox";
|
|
756
|
+
var DEFAULT_VCPUS = 2;
|
|
757
|
+
function isSandboxGoneError(e) {
|
|
758
|
+
if (!(e instanceof Error)) {
|
|
759
|
+
return false;
|
|
760
|
+
}
|
|
761
|
+
const status = e.response?.status ?? e.cause?.response?.status;
|
|
762
|
+
if (status === 410 || status === 422) {
|
|
763
|
+
return true;
|
|
764
|
+
}
|
|
765
|
+
const msg = e.message || String(e);
|
|
766
|
+
return msg.includes("Expected a stream of command data") || msg.includes("Expected a stream of logs");
|
|
767
|
+
}
|
|
768
|
+
var getTestCredentials2 = () => process.env.NODE_ENV === "test" ? {
|
|
769
|
+
token: process.env.TEST_VERCEL_TOKEN,
|
|
770
|
+
teamId: process.env.TEST_VERCEL_TEAM_ID,
|
|
771
|
+
projectId: process.env.TEST_VERCEL_PROJECT_ID
|
|
772
|
+
} : {};
|
|
773
|
+
async function createSdk({
|
|
774
|
+
resources,
|
|
775
|
+
ports,
|
|
776
|
+
networkPolicy,
|
|
777
|
+
snapshotId
|
|
778
|
+
}) {
|
|
779
|
+
const SandboxSDK = await loadSandboxSDK2();
|
|
780
|
+
const base = {
|
|
781
|
+
resources,
|
|
782
|
+
timeout: MAX_TIMEOUT_MS,
|
|
783
|
+
...ports ? { ports } : {},
|
|
784
|
+
...networkPolicy ? { networkPolicy } : {},
|
|
785
|
+
...getTestCredentials2()
|
|
786
|
+
};
|
|
787
|
+
if (snapshotId) {
|
|
788
|
+
return SandboxSDK.create({
|
|
789
|
+
...base,
|
|
790
|
+
source: { type: "snapshot", snapshotId }
|
|
791
|
+
});
|
|
792
|
+
}
|
|
793
|
+
return SandboxSDK.create(base);
|
|
794
|
+
}
|
|
795
|
+
async function createSetupSnapshot({
|
|
796
|
+
sandboxOpts,
|
|
797
|
+
run,
|
|
798
|
+
storage,
|
|
799
|
+
snapshotVersion,
|
|
800
|
+
log
|
|
801
|
+
}) {
|
|
802
|
+
let tempSdk = null;
|
|
803
|
+
try {
|
|
804
|
+
tempSdk = await createSdk(sandboxOpts);
|
|
805
|
+
const tempInstance = new VercelSandboxInstance(tempSdk);
|
|
806
|
+
await run(tempInstance);
|
|
807
|
+
const snapshot = await tempSdk.snapshot();
|
|
808
|
+
await storage.setup.set(snapshotVersion, {
|
|
809
|
+
version: snapshotVersion,
|
|
810
|
+
snapshotId: snapshot.snapshotId,
|
|
811
|
+
createdAt: Date.now(),
|
|
812
|
+
lastUsedAt: null
|
|
813
|
+
});
|
|
814
|
+
await tempSdk.stop().catch(() => void 0);
|
|
815
|
+
} catch (error) {
|
|
816
|
+
log.error("failed to create setup snapshot", {
|
|
817
|
+
snapshotVersion,
|
|
818
|
+
cause: error
|
|
819
|
+
});
|
|
820
|
+
if (tempSdk) {
|
|
821
|
+
await tempSdk.stop().catch(() => void 0);
|
|
822
|
+
}
|
|
823
|
+
}
|
|
824
|
+
}
|
|
825
|
+
var VercelSandboxInstance = class {
|
|
826
|
+
sdk;
|
|
827
|
+
cwd;
|
|
828
|
+
constructor(sdk, cwd) {
|
|
829
|
+
this.sdk = sdk;
|
|
830
|
+
this.cwd = cwd ?? HOME_DIR;
|
|
831
|
+
}
|
|
832
|
+
async exec({
|
|
833
|
+
command,
|
|
834
|
+
args,
|
|
835
|
+
cwd,
|
|
836
|
+
env,
|
|
837
|
+
sudo,
|
|
838
|
+
signal
|
|
839
|
+
}) {
|
|
840
|
+
const output = await this.sdk.runCommand({
|
|
841
|
+
cmd: command,
|
|
842
|
+
args,
|
|
843
|
+
cwd: cwd ?? this.cwd,
|
|
844
|
+
env,
|
|
845
|
+
sudo,
|
|
846
|
+
signal,
|
|
847
|
+
detached: true
|
|
848
|
+
});
|
|
849
|
+
let stdout = "";
|
|
850
|
+
let stderr = "";
|
|
851
|
+
const logBuffer = [];
|
|
852
|
+
const state = {
|
|
853
|
+
resolve: null,
|
|
854
|
+
consumed: false
|
|
855
|
+
};
|
|
856
|
+
const consumeLogs = (async () => {
|
|
857
|
+
try {
|
|
858
|
+
for await (const entry of output.logs()) {
|
|
859
|
+
if (entry.stream === "stdout") {
|
|
860
|
+
stdout += entry.data;
|
|
861
|
+
} else {
|
|
862
|
+
stderr += entry.data;
|
|
863
|
+
}
|
|
864
|
+
logBuffer.push(entry);
|
|
865
|
+
state.resolve?.();
|
|
866
|
+
}
|
|
867
|
+
} catch {
|
|
868
|
+
}
|
|
869
|
+
state.consumed = true;
|
|
870
|
+
state.resolve?.();
|
|
871
|
+
})();
|
|
872
|
+
async function* logs() {
|
|
873
|
+
let index = 0;
|
|
874
|
+
while (!state.consumed || index < logBuffer.length) {
|
|
875
|
+
if (index < logBuffer.length) {
|
|
876
|
+
yield logBuffer[index++];
|
|
877
|
+
} else {
|
|
878
|
+
await new Promise((r) => {
|
|
879
|
+
state.resolve = r;
|
|
880
|
+
});
|
|
881
|
+
state.resolve = null;
|
|
882
|
+
}
|
|
883
|
+
}
|
|
884
|
+
}
|
|
885
|
+
const result = consumeLogs.then(async () => {
|
|
886
|
+
try {
|
|
887
|
+
const finished = await output.wait();
|
|
888
|
+
return { stdout, stderr, exitCode: finished.exitCode };
|
|
889
|
+
} catch (e) {
|
|
890
|
+
if (isSandboxGoneError(e)) {
|
|
891
|
+
return { stdout, stderr, exitCode: 1 };
|
|
892
|
+
}
|
|
893
|
+
throw e;
|
|
894
|
+
}
|
|
895
|
+
});
|
|
896
|
+
return { commandId: output.cmdId, logs, result };
|
|
897
|
+
}
|
|
898
|
+
async readFile({
|
|
899
|
+
path: filePath
|
|
900
|
+
}) {
|
|
901
|
+
try {
|
|
902
|
+
return await this.sdk.readFileToBuffer({ path: filePath, cwd: this.cwd });
|
|
903
|
+
} catch {
|
|
904
|
+
return null;
|
|
905
|
+
}
|
|
906
|
+
}
|
|
907
|
+
async writeFiles({
|
|
908
|
+
files,
|
|
909
|
+
destPath
|
|
910
|
+
}) {
|
|
911
|
+
if (files.length === 0) {
|
|
912
|
+
return;
|
|
913
|
+
}
|
|
914
|
+
const path = await import("path");
|
|
915
|
+
const nativeFiles = files.map((file) => {
|
|
916
|
+
const filePath = path.posix.join(destPath, file.path);
|
|
917
|
+
const absolutePath = path.posix.isAbsolute(filePath) ? filePath : path.posix.join(this.cwd, filePath);
|
|
918
|
+
return {
|
|
919
|
+
path: absolutePath,
|
|
920
|
+
content: typeof file.content === "string" ? Buffer.from(file.content) : file.content
|
|
921
|
+
};
|
|
922
|
+
});
|
|
923
|
+
await this.sdk.writeFiles(nativeFiles);
|
|
924
|
+
const shellScripts = nativeFiles.filter((f) => f.path.endsWith(".sh"));
|
|
925
|
+
if (shellScripts.length > 0) {
|
|
926
|
+
const result = await this.exec({
|
|
927
|
+
command: "chmod",
|
|
928
|
+
args: ["+x", ...shellScripts.map((f) => f.path)]
|
|
929
|
+
});
|
|
930
|
+
await result.result;
|
|
931
|
+
}
|
|
932
|
+
}
|
|
933
|
+
// biome-ignore lint/suspicious/useAwait: synchronous SDK call
|
|
934
|
+
async getDomain({
|
|
935
|
+
port
|
|
936
|
+
}) {
|
|
937
|
+
return this.sdk.domain(port);
|
|
938
|
+
}
|
|
939
|
+
async kill({
|
|
940
|
+
commandId
|
|
941
|
+
}) {
|
|
942
|
+
const cmd = await this.sdk.getCommand(commandId);
|
|
943
|
+
await cmd.kill();
|
|
944
|
+
}
|
|
945
|
+
// biome-ignore lint/suspicious/useAwait: synchronous SDK getter
|
|
946
|
+
async getStatus() {
|
|
947
|
+
const status = this.sdk.status;
|
|
948
|
+
if (status === "snapshotting") {
|
|
949
|
+
return "stopping";
|
|
950
|
+
}
|
|
951
|
+
if (status === "aborted") {
|
|
952
|
+
return "failed";
|
|
953
|
+
}
|
|
954
|
+
return status;
|
|
955
|
+
}
|
|
956
|
+
async start() {
|
|
957
|
+
const result = await this.exec({ command: "true" });
|
|
958
|
+
await result.result;
|
|
959
|
+
}
|
|
960
|
+
async stop() {
|
|
961
|
+
await this.sdk.stop();
|
|
962
|
+
}
|
|
963
|
+
async snapshot() {
|
|
964
|
+
const snap = await this.sdk.snapshot();
|
|
965
|
+
return { snapshotId: snap.snapshotId };
|
|
966
|
+
}
|
|
967
|
+
async updateNetworkPolicy({
|
|
968
|
+
policy
|
|
969
|
+
}) {
|
|
970
|
+
return await this.sdk.updateNetworkPolicy(policy);
|
|
971
|
+
}
|
|
972
|
+
};
|
|
973
|
+
function vercelSandbox() {
|
|
974
|
+
return {
|
|
975
|
+
type: "vercel",
|
|
976
|
+
async create({ agent, setup, record, signal: _signal, log }) {
|
|
977
|
+
const storage = agent.storage;
|
|
978
|
+
const config = setup.config ?? {};
|
|
979
|
+
const cwd = config.cwd;
|
|
980
|
+
const resources = config.resources ?? { vcpus: DEFAULT_VCPUS };
|
|
981
|
+
const ports = config.ports;
|
|
982
|
+
const networkPolicy = config.networkPolicy ?? setup.networkPolicy ?? void 0;
|
|
983
|
+
const sandboxOpts = {
|
|
984
|
+
resources,
|
|
985
|
+
ports,
|
|
986
|
+
networkPolicy,
|
|
987
|
+
snapshotId: config.snapshotId
|
|
988
|
+
};
|
|
989
|
+
const _startLifecycle = (vercelSandboxId) => {
|
|
990
|
+
if (!record) {
|
|
991
|
+
return;
|
|
992
|
+
}
|
|
993
|
+
(0, import_api.start)(sandboxLifecycleWorkflow, [
|
|
994
|
+
{
|
|
995
|
+
input: {
|
|
996
|
+
agent,
|
|
997
|
+
sandboxId: record.id,
|
|
998
|
+
vercelSandboxId,
|
|
999
|
+
config: config.lifecycle
|
|
1000
|
+
}
|
|
1001
|
+
}
|
|
1002
|
+
]).catch((e) => {
|
|
1003
|
+
log.error("failed to start lifecycle workflow", { cause: e });
|
|
1004
|
+
});
|
|
1005
|
+
};
|
|
1006
|
+
if (setup.version) {
|
|
1007
|
+
const existing = await storage.setup.get(setup.version);
|
|
1008
|
+
if (existing?.snapshotId) {
|
|
1009
|
+
const sdk2 = await createSdk({
|
|
1010
|
+
...sandboxOpts,
|
|
1011
|
+
snapshotId: existing.snapshotId
|
|
1012
|
+
});
|
|
1013
|
+
const instance2 = new VercelSandboxInstance(sdk2, cwd);
|
|
1014
|
+
storage.setup.set(setup.version, {
|
|
1015
|
+
...existing,
|
|
1016
|
+
version: setup.version,
|
|
1017
|
+
lastUsedAt: Date.now()
|
|
1018
|
+
}).catch(() => void 0);
|
|
1019
|
+
return {
|
|
1020
|
+
instance: instance2,
|
|
1021
|
+
metadata: { sandboxId: sdk2.sandboxId }
|
|
1022
|
+
};
|
|
1023
|
+
}
|
|
1024
|
+
}
|
|
1025
|
+
if (setup.run) {
|
|
1026
|
+
if (!setup.version) {
|
|
1027
|
+
log.error("setup.run provided without setup.version", { setup });
|
|
1028
|
+
throw new Error("setup.run provided without setup.version");
|
|
1029
|
+
}
|
|
1030
|
+
createSetupSnapshot({
|
|
1031
|
+
sandboxOpts,
|
|
1032
|
+
run: setup.run,
|
|
1033
|
+
storage,
|
|
1034
|
+
snapshotVersion: setup.version,
|
|
1035
|
+
log
|
|
1036
|
+
}).catch((error) => {
|
|
1037
|
+
log.error("background setup snapshot failed", { cause: error });
|
|
1038
|
+
});
|
|
1039
|
+
const sdk2 = await createSdk(sandboxOpts);
|
|
1040
|
+
const instance2 = new VercelSandboxInstance(sdk2, cwd);
|
|
1041
|
+
await setup.run(instance2);
|
|
1042
|
+
return {
|
|
1043
|
+
instance: instance2,
|
|
1044
|
+
metadata: { sandboxId: sdk2.sandboxId }
|
|
1045
|
+
};
|
|
1046
|
+
}
|
|
1047
|
+
const sdk = await createSdk(sandboxOpts);
|
|
1048
|
+
const instance = new VercelSandboxInstance(sdk, cwd);
|
|
1049
|
+
return {
|
|
1050
|
+
instance,
|
|
1051
|
+
metadata: { sandboxId: sdk.sandboxId }
|
|
1052
|
+
};
|
|
1053
|
+
},
|
|
1054
|
+
async connect({ metadata }) {
|
|
1055
|
+
const SandboxSDK = await loadSandboxSDK2();
|
|
1056
|
+
const sdk = await SandboxSDK.get({
|
|
1057
|
+
sandboxId: metadata.sandboxId,
|
|
1058
|
+
...getTestCredentials2()
|
|
1059
|
+
});
|
|
1060
|
+
return new VercelSandboxInstance(sdk);
|
|
1061
|
+
}
|
|
1062
|
+
};
|
|
1063
|
+
}
|
|
1064
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
1065
|
+
0 && (module.exports = {
|
|
1066
|
+
dockerSandbox,
|
|
1067
|
+
localSandbox,
|
|
1068
|
+
vercelSandbox
|
|
1069
|
+
});
|
|
1070
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/sandbox/entry.ts", "../src/sandbox/bindings/docker.ts", "../src/sandbox/bindings/local.ts", "../src/sandbox/bindings/vercel.ts", "../src/sandbox/bindings/vercel-lifecycle/workflow.ts", "../src/utils/logger.ts", "../src/sandbox/bindings/vercel-lifecycle/steps.ts"],
  "sourcesContent": ["export type {\n  ExecResult,\n  SandboxAgentRef,\n  SandboxBinding,\n  SandboxInstance,\n  SandboxSetupFields,\n  UploadableFile,\n} from \"./adapter\";\nexport { dockerSandbox } from \"./bindings/docker\";\nexport { localSandbox } from \"./bindings/local\";\nexport { vercelSandbox } from \"./bindings/vercel\";\n", "import { ulid } from \"ulid\";\nimport type { ExecResult, SandboxBinding, SandboxInstance } from \"../adapter\";\n\nexport type DockerBindingConfig = {\n  cwd?: string;\n};\n\nexport type DockerBindingMetadata = {\n  sandboxName: string;\n};\n\n/**\n * Run a `docker sandbox` CLI command and return its output.\n */\nasync function execDocker(\n  args: string[],\n  opts?: { timeoutMs?: number; signal?: AbortSignal }\n): Promise<{ stdout: string; stderr: string; exitCode: number }> {\n  const { spawn } = await import(\"node:child_process\");\n  return new Promise((resolve, reject) => {\n    const child = spawn(\"docker\", [\"sandbox\", ...args], {\n      signal: opts?.signal,\n    });\n\n    let stdout = \"\";\n    let stderr = \"\";\n\n    child.stdout.on(\"data\", (data: Buffer) => {\n      stdout += data.toString();\n    });\n    child.stderr.on(\"data\", (data: Buffer) => {\n      stderr += data.toString();\n    });\n\n    const timeoutId = opts?.timeoutMs\n      ? setTimeout(() => {\n          child.kill(\"SIGTERM\");\n          reject(new Error(`docker sandbox ${args[0]} timed out`));\n        }, opts.timeoutMs)\n      : undefined;\n\n    child.on(\"error\", (err) => {\n      if (timeoutId) {\n        clearTimeout(timeoutId);\n      }\n      reject(err);\n    });\n\n    child.on(\"close\", (code) => {\n      if (timeoutId) {\n        clearTimeout(timeoutId);\n      }\n      resolve({ stdout, stderr, exitCode: code ?? 0 });\n    });\n  });\n}\n\n/**\n * Track which sandboxes have been verified to exist in this process.\n * Maps sandbox name -> Promise so concurrent callers wait on the same check.\n */\nconst ensurePromises = new Map<string, Promise<void>>();\n\n/**\n * Sandboxes that this process has used. Stopped on process exit.\n */\nconst activeSandboxes = new Set<string>();\n\nlet cleanupRegistered = false;\nlet _execSync: typeof import(\"node:child_process\").execSync | null = null;\n\nasync function registerCleanup() {\n  if (cleanupRegistered) {\n    return;\n  }\n  cleanupRegistered = true;\n\n  // Pre-load execSync so the synchronous exit handler can use it.\n  const cp = await import(\"node:child_process\");\n  _execSync = cp.execSync;\n\n  const cleanup = () => {\n    if (!_execSync) {\n      return;\n    }\n    for (const name of Array.from(activeSandboxes)) {\n      try {\n        _execSync(`docker sandbox stop ${name}`, {\n          timeout: 10_000,\n          stdio: \"pipe\",\n        });\n      } catch {\n        // Best-effort \u2014 sandbox may already be stopped\n      }\n    }\n  };\n\n  process.on(\"exit\", cleanup);\n  process.on(\"SIGINT\", () => {\n    cleanup();\n    process.exit(130);\n  });\n  process.on(\"SIGTERM\", () => {\n    cleanup();\n    process.exit(143);\n  });\n}\n\n/**\n * Ensure a Docker sandbox exists for the given name.\n * If it already exists (from a previous process), reuses it.\n * If it doesn't exist, creates it.\n */\nasync function ensureSandbox(sandboxName: string): Promise<void> {\n  const existing = ensurePromises.get(sandboxName);\n  if (existing) {\n    return existing;\n  }\n\n  const promise = (async () => {\n    const ls = await execDocker([\"ls\", \"-q\"], { timeoutMs: 10_000 });\n    const existingNames =\n      ls.exitCode === 0\n        ? ls.stdout\n            .split(\"\\n\")\n            .map((s) => s.trim())\n            .filter(Boolean)\n        : [];\n\n    if (existingNames.includes(sandboxName)) {\n      activeSandboxes.add(sandboxName);\n      registerCleanup();\n      return;\n    }\n\n    const path = await import(\"node:path\");\n    const os = await import(\"node:os\");\n    const fs = await import(\"node:fs/promises\");\n\n    const workspaceDir = path.join(\n      os.tmpdir(),\n      \"agent-docker-sandbox\",\n      sandboxName\n    );\n    await fs.mkdir(workspaceDir, { recursive: true });\n\n    const create = await execDocker(\n      [\"create\", \"--name\", sandboxName, \"shell\", workspaceDir],\n      { timeoutMs: 60_000 }\n    );\n\n    if (create.exitCode !== 0) {\n      if (create.stderr.includes(\"already exists\")) {\n        activeSandboxes.add(sandboxName);\n        registerCleanup();\n        return;\n      }\n      throw new Error(\n        `Failed to create docker sandbox \"${sandboxName}\": ${create.stderr}`\n      );\n    }\n\n    activeSandboxes.add(sandboxName);\n    registerCleanup();\n  })();\n\n  ensurePromises.set(sandboxName, promise);\n\n  try {\n    await promise;\n  } catch (e) {\n    ensurePromises.delete(sandboxName);\n    throw e;\n  }\n}\n\nconst DEFAULT_DOCKER_CWD = \"/home/agent/workspace\";\n\nclass DockerSandboxInstance implements SandboxInstance {\n  readonly cwd: string;\n  private readonly sandboxName: string;\n  private readonly processes = new Map<\n    string,\n    import(\"node:child_process\").ChildProcess\n  >();\n\n  constructor(sandboxName: string, cwd?: string) {\n    this.sandboxName = sandboxName;\n    this.cwd = cwd ?? DEFAULT_DOCKER_CWD;\n  }\n\n  async exec(opts: {\n    command: string;\n    args?: string[];\n    cwd?: string;\n    env?: Record<string, string>;\n    sudo?: boolean;\n    signal?: AbortSignal;\n  }): Promise<ExecResult> {\n    await ensureSandbox(this.sandboxName);\n\n    const { spawn } = await import(\"node:child_process\");\n    const commandId = `command_${ulid()}`;\n    const envFlags = opts.env\n      ? Object.entries(opts.env).flatMap(([k, v]) => [\"-e\", `${k}=${v}`])\n      : [];\n    const cwdFlags = opts.cwd ? [\"-w\", opts.cwd] : [];\n    const baseCmd = opts.sudo\n      ? [\"sudo\", opts.command, ...(opts.args ?? [])]\n      : opts.args\n        ? [opts.command, ...opts.args]\n        : [opts.command];\n    const fullCmd = baseCmd;\n\n    const child = spawn(\n      \"docker\",\n      [\n        \"sandbox\",\n        \"exec\",\n        ...envFlags,\n        ...cwdFlags,\n        this.sandboxName,\n        ...fullCmd,\n      ],\n      { signal: opts.signal }\n    );\n\n    this.processes.set(commandId, child);\n\n    let stdout = \"\";\n    let stderr = \"\";\n    const logQueue: { stream: \"stdout\" | \"stderr\"; data: string }[] = [];\n    let logResolve: (() => void) | null = null;\n    let closed = false;\n\n    child.stdout.on(\"data\", (data: string | Buffer) => {\n      const str = String(data);\n      stdout += str;\n      logQueue.push({ stream: \"stdout\", data: str });\n      logResolve?.();\n    });\n\n    child.stderr.on(\"data\", (data: string | Buffer) => {\n      const str = String(data);\n      stderr += str;\n      logQueue.push({ stream: \"stderr\", data: str });\n      logResolve?.();\n    });\n\n    const result = new Promise<{\n      stdout: string;\n      stderr: string;\n      exitCode: number;\n    }>((resolve, reject) => {\n      child.on(\"error\", (err) => {\n        this.processes.delete(commandId);\n        closed = true;\n        logResolve?.();\n        reject(err);\n      });\n\n      child.on(\"close\", (code: number | null) => {\n        this.processes.delete(commandId);\n        closed = true;\n        logResolve?.();\n        resolve({ stdout, stderr, exitCode: code ?? 0 });\n      });\n    });\n\n    async function* logs(): AsyncIterable<{\n      stream: \"stdout\" | \"stderr\";\n      data: string;\n    }> {\n      while (!closed || logQueue.length > 0) {\n        const entry = logQueue.shift();\n        if (entry) {\n          yield entry;\n        } else if (!closed) {\n          await new Promise<void>((r) => {\n            logResolve = r;\n          });\n          logResolve = null;\n        }\n      }\n    }\n\n    return { commandId, logs, result };\n  }\n\n  async readFile(opts: {\n    path: string;\n    signal?: AbortSignal;\n  }): Promise<Buffer | null> {\n    await ensureSandbox(this.sandboxName);\n\n    const result = await execDocker(\n      [\n        \"exec\",\n        this.sandboxName,\n        \"bash\",\n        \"-c\",\n        `base64 '${opts.path.replace(/'/g, \"'\\\\''\")}'`,\n      ],\n      { timeoutMs: 30_000, signal: opts.signal }\n    );\n\n    if (result.exitCode !== 0) {\n      if (\n        result.stderr.includes(\"No such file\") ||\n        result.stderr.includes(\"ENOENT\")\n      ) {\n        return null;\n      }\n      throw new Error(`readFile failed: ${result.stderr}`);\n    }\n\n    return Buffer.from(result.stdout.trim(), \"base64\");\n  }\n\n  async writeFiles(opts: {\n    files: { path: string; content: string | Buffer }[];\n    destPath: string;\n    signal?: AbortSignal;\n  }): Promise<void> {\n    await ensureSandbox(this.sandboxName);\n\n    for (const file of opts.files) {\n      const fullPath = opts.destPath\n        ? `${opts.destPath}/${file.path}`\n        : file.path;\n\n      // Ensure parent directory exists\n      const parentDir = fullPath.substring(0, fullPath.lastIndexOf(\"/\"));\n      if (parentDir) {\n        await execDocker([\"exec\", this.sandboxName, \"mkdir\", \"-p\", parentDir], {\n          signal: opts.signal,\n        });\n      }\n\n      // Write via base64 to handle both text and binary content\n      const b64 =\n        typeof file.content === \"string\"\n          ? Buffer.from(file.content).toString(\"base64\")\n          : file.content.toString(\"base64\");\n\n      await execDocker(\n        [\n          \"exec\",\n          this.sandboxName,\n          \"bash\",\n          \"-c\",\n          `echo '${b64}' | base64 -d > '${fullPath.replace(/'/g, \"'\\\\''\")}'`,\n        ],\n        { signal: opts.signal }\n      );\n\n      if (file.path.endsWith(\".sh\")) {\n        await execDocker([\"exec\", this.sandboxName, \"chmod\", \"+x\", fullPath], {\n          signal: opts.signal,\n        });\n      }\n    }\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async getDomain(opts: {\n    port: number;\n    signal?: AbortSignal;\n  }): Promise<string> {\n    return `http://localhost:${opts.port}`;\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async kill(opts: { commandId: string; signal?: AbortSignal }): Promise<void> {\n    const child = this.processes.get(opts.commandId);\n    if (child) {\n      child.kill(\"SIGTERM\");\n      this.processes.delete(opts.commandId);\n    }\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async getStatus(): Promise<\n    \"pending\" | \"running\" | \"stopping\" | \"stopped\" | \"failed\"\n  > {\n    // docker sandbox CLI doesn't expose status \u2014 assume running if ensured\n    return \"running\";\n  }\n\n  async start(): Promise<void> {\n    await ensureSandbox(this.sandboxName);\n  }\n\n  async stop(): Promise<void> {\n    await execDocker([\"stop\", this.sandboxName], { timeoutMs: 30_000 });\n    activeSandboxes.delete(this.sandboxName);\n    ensurePromises.delete(this.sandboxName);\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async snapshot(): Promise<{ snapshotId: string }> {\n    throw new Error(\"snapshot is not supported for docker sandboxes\");\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async updateNetworkPolicy(): Promise<never> {\n    throw new Error(\n      \"updateNetworkPolicy is not supported for docker sandboxes\"\n    );\n  }\n}\n\nexport function dockerSandbox(): SandboxBinding<\n  \"docker\",\n  DockerBindingConfig,\n  DockerBindingMetadata\n> {\n  return {\n    type: \"docker\",\n    async create(opts) {\n      const sandboxName = `agent-${ulid()}`;\n      const cwd = opts.setup?.config?.cwd;\n\n      await ensureSandbox(sandboxName);\n\n      const instance = new DockerSandboxInstance(sandboxName, cwd);\n\n      if (opts.setup?.run) {\n        await opts.setup.run(instance);\n      }\n\n      return {\n        instance,\n        metadata: { sandboxName },\n      };\n    },\n\n    // biome-ignore lint/suspicious/useAwait: .\n    async connect(opts) {\n      return new DockerSandboxInstance(opts.metadata.sandboxName);\n    },\n  };\n}\n", "import { ulid } from \"ulid\";\nimport type { ExecResult, SandboxBinding, SandboxInstance } from \"../adapter\";\n\nexport type LocalBindingConfig = {\n  cwd?: string;\n};\n\nexport type LocalBindingMetadata = {\n  basePath: string;\n  pid: number;\n};\n\nclass LocalSandboxInstance implements SandboxInstance {\n  readonly cwd: string;\n  private readonly basePath: string;\n  private readonly processes = new Map<\n    string,\n    import(\"node:child_process\").ChildProcess\n  >();\n\n  constructor(basePath: string) {\n    this.basePath = basePath;\n    this.cwd = basePath;\n  }\n\n  async exec(opts: {\n    command: string;\n    args?: string[];\n    cwd?: string;\n    env?: Record<string, string>;\n    sudo?: boolean;\n    signal?: AbortSignal;\n  }): Promise<ExecResult> {\n    const { spawn } = await import(\"node:child_process\");\n    const path = await import(\"node:path\");\n    const commandId = `command_${ulid()}`;\n\n    const finalCmd = opts.sudo ? \"sudo\" : opts.command;\n    const finalArgs = opts.sudo\n      ? [opts.command, ...(opts.args ?? [])]\n      : (opts.args ?? []);\n\n    const child = spawn(finalCmd, finalArgs, {\n      cwd: opts.cwd ? path.resolve(this.basePath, opts.cwd) : this.basePath,\n      env: opts.env ? { ...process.env, ...opts.env } : undefined,\n      signal: opts.signal,\n    });\n\n    this.processes.set(commandId, child);\n\n    let stdout = \"\";\n    let stderr = \"\";\n    const logQueue: { stream: \"stdout\" | \"stderr\"; data: string }[] = [];\n    let logResolve: (() => void) | null = null;\n    let closed = false;\n\n    child.stdout.on(\"data\", (data: string | Buffer) => {\n      const str = String(data);\n      stdout += str;\n      logQueue.push({ stream: \"stdout\", data: str });\n      logResolve?.();\n    });\n\n    child.stderr.on(\"data\", (data: string | Buffer) => {\n      const str = String(data);\n      stderr += str;\n      logQueue.push({ stream: \"stderr\", data: str });\n      logResolve?.();\n    });\n\n    const result = new Promise<{\n      stdout: string;\n      stderr: string;\n      exitCode: number;\n    }>((resolve, reject) => {\n      child.on(\"error\", (err) => {\n        this.processes.delete(commandId);\n        closed = true;\n        logResolve?.();\n        reject(err);\n      });\n\n      child.on(\"close\", (code: number | null) => {\n        this.processes.delete(commandId);\n        closed = true;\n        logResolve?.();\n        resolve({ stdout, stderr, exitCode: code ?? 0 });\n      });\n    });\n\n    async function* logs(): AsyncIterable<{\n      stream: \"stdout\" | \"stderr\";\n      data: string;\n    }> {\n      while (!closed || logQueue.length > 0) {\n        const entry = logQueue.shift();\n        if (entry) {\n          yield entry;\n        } else if (!closed) {\n          await new Promise<void>((r) => {\n            logResolve = r;\n          });\n          logResolve = null;\n        }\n      }\n    }\n\n    return { commandId, logs, result };\n  }\n\n  async readFile(opts: {\n    path: string;\n    signal?: AbortSignal;\n  }): Promise<Buffer | null> {\n    const fs = await import(\"node:fs/promises\");\n    const path = await import(\"node:path\");\n    const fullPath = path.resolve(this.basePath, opts.path);\n    try {\n      return await fs.readFile(fullPath);\n    } catch (e: unknown) {\n      if (\n        e instanceof Error &&\n        \"code\" in e &&\n        (e as NodeJS.ErrnoException).code === \"ENOENT\"\n      ) {\n        return null;\n      }\n      throw e;\n    }\n  }\n\n  async writeFiles(opts: {\n    files: { path: string; content: string | Buffer }[];\n    destPath: string;\n    signal?: AbortSignal;\n  }): Promise<void> {\n    const fs = await import(\"node:fs/promises\");\n    const pathMod = await import(\"node:path\");\n    for (const file of opts.files) {\n      const fullPath = pathMod.resolve(this.basePath, opts.destPath, file.path);\n      await fs.mkdir(pathMod.dirname(fullPath), { recursive: true });\n      await fs.writeFile(fullPath, file.content);\n      if (file.path.endsWith(\".sh\")) {\n        await fs.chmod(fullPath, 0o755);\n      }\n    }\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async getDomain(opts: {\n    port: number;\n    signal?: AbortSignal;\n  }): Promise<string> {\n    return `http://localhost:${opts.port}`;\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async kill(opts: { commandId: string; signal?: AbortSignal }): Promise<void> {\n    const child = this.processes.get(opts.commandId);\n    if (child) {\n      child.kill(\"SIGTERM\");\n      this.processes.delete(opts.commandId);\n    }\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async getStatus(): Promise<\"running\"> {\n    return \"running\";\n  }\n\n  async start(): Promise<void> {\n    // no-op: local sandbox is always started\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async stop(): Promise<void> {\n    throw new Error(\"stop is not supported for local sandboxes\");\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async snapshot(): Promise<{ snapshotId: string }> {\n    throw new Error(\"snapshot is not supported for local sandboxes\");\n  }\n\n  // biome-ignore lint/suspicious/useAwait: .\n  async updateNetworkPolicy(): Promise<never> {\n    throw new Error(\"updateNetworkPolicy is not supported for local sandboxes\");\n  }\n}\n\nexport function localSandbox(): SandboxBinding<\n  \"local\",\n  LocalBindingConfig,\n  LocalBindingMetadata\n> {\n  return {\n    type: \"local\",\n    async create(opts) {\n      const basePath = opts.setup.config?.cwd ?? process.cwd();\n      const instance = new LocalSandboxInstance(basePath);\n\n      if (opts.setup?.run) {\n        await opts.setup.run(instance);\n      }\n\n      return {\n        instance,\n        metadata: { basePath, pid: process.pid },\n      };\n    },\n\n    // biome-ignore lint/suspicious/useAwait: .\n    async connect(opts) {\n      const basePath = opts.metadata.basePath;\n      return new LocalSandboxInstance(basePath);\n    },\n  };\n}\n", "import { start as startWorkflow } from \"workflow/api\";\nimport type { Storage } from \"../../storage\";\nimport type { Logger } from \"../../utils/logger\";\nimport type { ExecResult, SandboxBinding, SandboxInstance } from \"../adapter\";\nimport type { LifecycleConfig } from \"./vercel-lifecycle/steps\";\nimport { sandboxLifecycleWorkflow } from \"./vercel-lifecycle/workflow\";\n\nasync function loadSandboxSDK() {\n  return (await import(\"@vercel/sandbox\")).Sandbox;\n}\n\nconst MAX_TIMEOUT_MS = 5 * 60 * 60 * 1000; // 5 hours\nconst HOME_DIR = \"/home/vercel-sandbox\";\nconst DEFAULT_VCPUS = 2;\n\nexport type VercelBindingConfig = {\n  cwd?: string;\n  resources?: { vcpus: number };\n  ports?: number[];\n  networkPolicy?: import(\"@vercel/sandbox\").NetworkPolicy;\n  snapshotId?: string;\n  lifecycle?: LifecycleConfig;\n};\n\nexport type VercelBindingMetadata = {\n  sandboxId: string;\n};\n\ntype CreateSandboxOpts = {\n  resources: { vcpus: number };\n  ports?: number[];\n  networkPolicy?: import(\"@vercel/sandbox\").NetworkPolicy;\n  snapshotId?: string;\n};\n\nfunction isSandboxGoneError(e: unknown): boolean {\n  if (!(e instanceof Error)) {\n    return false;\n  }\n\n  const status =\n    (e as { response?: { status?: number } }).response?.status ??\n    (e as { cause?: { response?: { status?: number } } }).cause?.response\n      ?.status;\n\n  if (status === 410 || status === 422) {\n    return true;\n  }\n\n  const msg = e.message || String(e);\n  return (\n    msg.includes(\"Expected a stream of command data\") ||\n    msg.includes(\"Expected a stream of logs\")\n  );\n}\n\nconst getTestCredentials = () =>\n  process.env.NODE_ENV === \"test\"\n    ? {\n        token: process.env.TEST_VERCEL_TOKEN,\n        teamId: process.env.TEST_VERCEL_TEAM_ID,\n        projectId: process.env.TEST_VERCEL_PROJECT_ID,\n      }\n    : {};\n\nasync function createSdk({\n  resources,\n  ports,\n  networkPolicy,\n  snapshotId,\n}: CreateSandboxOpts) {\n  const SandboxSDK = await loadSandboxSDK();\n  const base = {\n    resources,\n    timeout: MAX_TIMEOUT_MS,\n    ...(ports ? { ports } : {}),\n    ...(networkPolicy ? { networkPolicy } : {}),\n    ...getTestCredentials(),\n  };\n\n  if (snapshotId) {\n    return SandboxSDK.create({\n      ...base,\n      source: { type: \"snapshot\" as const, snapshotId },\n    });\n  }\n\n  return SandboxSDK.create(base);\n}\n\nasync function createSetupSnapshot({\n  sandboxOpts,\n  run,\n  storage,\n  snapshotVersion,\n  log,\n}: {\n  sandboxOpts: CreateSandboxOpts;\n  run: (sandbox: SandboxInstance) => Promise<void>;\n  storage: Storage;\n  snapshotVersion: string;\n  log: Logger;\n}): Promise<void> {\n  let tempSdk: Awaited<ReturnType<typeof createSdk>> | null = null;\n  try {\n    tempSdk = await createSdk(sandboxOpts);\n    const tempInstance = new VercelSandboxInstance(tempSdk);\n    await run(tempInstance);\n    const snapshot = await tempSdk.snapshot();\n    await storage.setup.set(snapshotVersion, {\n      version: snapshotVersion,\n      snapshotId: snapshot.snapshotId,\n      createdAt: Date.now(),\n      lastUsedAt: null,\n    });\n    await tempSdk.stop().catch(() => undefined);\n  } catch (error) {\n    log.error(\"failed to create setup snapshot\", {\n      snapshotVersion,\n      cause: error,\n    });\n    if (tempSdk) {\n      await tempSdk.stop().catch(() => undefined);\n    }\n  }\n}\n\nclass VercelSandboxInstance implements SandboxInstance {\n  private readonly sdk: import(\"@vercel/sandbox\").Sandbox;\n  readonly cwd: string;\n\n  constructor(sdk: import(\"@vercel/sandbox\").Sandbox, cwd?: string) {\n    this.sdk = sdk;\n    this.cwd = cwd ?? HOME_DIR;\n  }\n\n  async exec({\n    command,\n    args,\n    cwd,\n    env,\n    sudo,\n    signal,\n  }: {\n    command: string;\n    args?: string[];\n    cwd?: string;\n    env?: Record<string, string>;\n    sudo?: boolean;\n    signal?: AbortSignal;\n  }): Promise<ExecResult> {\n    const output = await this.sdk.runCommand({\n      cmd: command,\n      args,\n      cwd: cwd ?? this.cwd,\n      env,\n      sudo,\n      signal,\n      detached: true,\n    });\n\n    let stdout = \"\";\n    let stderr = \"\";\n    const logBuffer: { stream: \"stdout\" | \"stderr\"; data: string }[] = [];\n    const state = {\n      resolve: null as (() => void) | null,\n      consumed: false,\n    };\n\n    const consumeLogs = (async () => {\n      try {\n        for await (const entry of output.logs()) {\n          if (entry.stream === \"stdout\") {\n            stdout += entry.data;\n          } else {\n            stderr += entry.data;\n          }\n\n          logBuffer.push(entry);\n          state.resolve?.();\n        }\n      } catch {\n        // Sandbox may have been stopped \u2014 logs endpoint returns 422\n      }\n      state.consumed = true;\n      state.resolve?.();\n    })();\n\n    async function* logs(): AsyncIterable<{\n      stream: \"stdout\" | \"stderr\";\n      data: string;\n    }> {\n      let index = 0;\n      while (!state.consumed || index < logBuffer.length) {\n        if (index < logBuffer.length) {\n          yield logBuffer[index++];\n        } else {\n          await new Promise<void>((r) => {\n            state.resolve = r;\n          });\n          state.resolve = null;\n        }\n      }\n    }\n\n    const result = consumeLogs.then(async () => {\n      try {\n        const finished = await output.wait();\n        return { stdout, stderr, exitCode: finished.exitCode };\n      } catch (e) {\n        if (isSandboxGoneError(e)) {\n          return { stdout, stderr, exitCode: 1 };\n        }\n        throw e;\n      }\n    });\n\n    return { commandId: output.cmdId, logs, result };\n  }\n\n  async readFile({\n    path: filePath,\n  }: {\n    path: string;\n    signal?: AbortSignal;\n  }): Promise<Buffer | null> {\n    try {\n      return await this.sdk.readFileToBuffer({ path: filePath, cwd: this.cwd });\n    } catch {\n      return null;\n    }\n  }\n\n  async writeFiles({\n    files,\n    destPath,\n  }: {\n    files: { path: string; content: string | Buffer }[];\n    destPath: string;\n    signal?: AbortSignal;\n  }): Promise<void> {\n    if (files.length === 0) {\n      return;\n    }\n\n    const path = await import(\"node:path\");\n    const nativeFiles = files.map((file) => {\n      const filePath = path.posix.join(destPath, file.path);\n      const absolutePath = path.posix.isAbsolute(filePath)\n        ? filePath\n        : path.posix.join(this.cwd, filePath);\n      return {\n        path: absolutePath,\n        content:\n          typeof file.content === \"string\"\n            ? Buffer.from(file.content)\n            : file.content,\n      };\n    });\n\n    await this.sdk.writeFiles(nativeFiles);\n\n    const shellScripts = nativeFiles.filter((f) => f.path.endsWith(\".sh\"));\n    if (shellScripts.length > 0) {\n      const result = await this.exec({\n        command: \"chmod\",\n        args: [\"+x\", ...shellScripts.map((f) => f.path)],\n      });\n      await result.result;\n    }\n  }\n\n  // biome-ignore lint/suspicious/useAwait: synchronous SDK call\n  async getDomain({\n    port,\n  }: {\n    port: number;\n    signal?: AbortSignal;\n  }): Promise<string> {\n    return this.sdk.domain(port);\n  }\n\n  async kill({\n    commandId,\n  }: {\n    commandId: string;\n    signal?: AbortSignal;\n  }): Promise<void> {\n    const cmd = await this.sdk.getCommand(commandId);\n    await cmd.kill();\n  }\n\n  // biome-ignore lint/suspicious/useAwait: synchronous SDK getter\n  async getStatus(): Promise<\n    \"pending\" | \"running\" | \"stopping\" | \"stopped\" | \"failed\"\n  > {\n    const status = this.sdk.status;\n    // Map SDK-only statuses to our interface's union\n    if (status === \"snapshotting\") {\n      return \"stopping\";\n    }\n    if (status === \"aborted\") {\n      return \"failed\";\n    }\n    return status;\n  }\n\n  async start(): Promise<void> {\n    // The Vercel SDK doesn't have a resume method.\n    // Run a no-op command to verify the sandbox is reachable.\n    const result = await this.exec({ command: \"true\" });\n    await result.result;\n  }\n\n  async stop(): Promise<void> {\n    await this.sdk.stop();\n  }\n\n  async snapshot(): Promise<{ snapshotId: string }> {\n    const snap = await this.sdk.snapshot();\n    return { snapshotId: snap.snapshotId };\n  }\n\n  async updateNetworkPolicy({\n    policy,\n  }: {\n    policy: import(\"@vercel/sandbox\").NetworkPolicy;\n    signal?: AbortSignal;\n  }): Promise<import(\"@vercel/sandbox\").NetworkPolicy> {\n    return await this.sdk.updateNetworkPolicy(policy);\n  }\n}\n\nexport function vercelSandbox(): SandboxBinding<\n  \"vercel\",\n  VercelBindingConfig,\n  VercelBindingMetadata\n> {\n  return {\n    type: \"vercel\",\n    async create({ agent, setup, record, signal: _signal, log }) {\n      const storage = agent.storage;\n      const config = setup.config ?? {};\n      const cwd = config.cwd;\n      const resources = config.resources ?? { vcpus: DEFAULT_VCPUS };\n      const ports = config.ports;\n      const networkPolicy =\n        config.networkPolicy ?? setup.networkPolicy ?? undefined;\n\n      const sandboxOpts: CreateSandboxOpts = {\n        resources,\n        ports,\n        networkPolicy,\n        snapshotId: config.snapshotId,\n      };\n\n      const _startLifecycle = (vercelSandboxId: string) => {\n        if (!record) {\n          return;\n        }\n        startWorkflow(sandboxLifecycleWorkflow, [\n          {\n            input: {\n              agent,\n              sandboxId: record.id,\n              vercelSandboxId,\n              config: config.lifecycle,\n            },\n          },\n        ]).catch((e) => {\n          log.error(\"failed to start lifecycle workflow\", { cause: e });\n        });\n      };\n\n      if (setup.version) {\n        const existing = await storage.setup.get(setup.version);\n        if (existing?.snapshotId) {\n          const sdk = await createSdk({\n            ...sandboxOpts,\n            snapshotId: existing.snapshotId,\n          });\n          const instance = new VercelSandboxInstance(sdk, cwd);\n\n          storage.setup\n            .set(setup.version, {\n              ...existing,\n              version: setup.version,\n              lastUsedAt: Date.now(),\n            })\n            .catch(() => undefined);\n\n          return {\n            instance,\n            metadata: { sandboxId: sdk.sandboxId },\n          };\n        }\n      }\n\n      if (setup.run) {\n        if (!setup.version) {\n          log.error(\"setup.run provided without setup.version\", { setup });\n          throw new Error(\"setup.run provided without setup.version\");\n        }\n        createSetupSnapshot({\n          sandboxOpts,\n          run: setup.run,\n          storage,\n          snapshotVersion: setup.version,\n          log,\n        }).catch((error) => {\n          log.error(\"background setup snapshot failed\", { cause: error });\n        });\n\n        const sdk = await createSdk(sandboxOpts);\n        const instance = new VercelSandboxInstance(sdk, cwd);\n        await setup.run(instance);\n\n        return {\n          instance,\n          metadata: { sandboxId: sdk.sandboxId },\n        };\n      }\n\n      const sdk = await createSdk(sandboxOpts);\n      const instance = new VercelSandboxInstance(sdk, cwd);\n\n      return {\n        instance,\n        metadata: { sandboxId: sdk.sandboxId },\n      };\n    },\n\n    async connect({ metadata }) {\n      const SandboxSDK = await loadSandboxSDK();\n      const sdk = await SandboxSDK.get({\n        sandboxId: metadata.sandboxId,\n        ...getTestCredentials(),\n      });\n      return new VercelSandboxInstance(sdk);\n    },\n  };\n}\n", "import { sleep } from \"workflow\";\nimport { createLogger } from \"../../../utils/logger\";\nimport { checkAndSnapshotStep, type LifecycleInput } from \"./steps\";\n\nexport type { LifecycleInput };\n\nconst DEFAULT_POLL_MS = 2 * 60 * 1000;\n\nexport async function sandboxLifecycleWorkflow({\n  input,\n}: {\n  input: LifecycleInput;\n}) {\n  \"use workflow\";\n\n  let nextPollMs = input.config?.pollIntervalMs ?? DEFAULT_POLL_MS;\n  const log = createLogger({\n    config: { ...input.agent.options.logging, name: input.agent.name },\n    subsystem: \"sandbox:lifecycle\",\n    context: { sandboxId: input.sandboxId },\n  });\n\n  log.info(\"lifecycle workflow started\", {\n    sandboxId: input.sandboxId,\n    vercelSandboxId: input.vercelSandboxId,\n    pollMs: nextPollMs,\n  });\n\n  while (true) {\n    await sleep(nextPollMs);\n\n    const result = await checkAndSnapshotStep(input);\n\n    if (result instanceof Error) {\n      log.error(\"lifecycle step failed\", {\n        sandboxId: input.sandboxId,\n        error: result.message,\n      });\n      break;\n    }\n\n    if (result.action === \"exit\") {\n      log.info(\"lifecycle workflow exiting\", {\n        sandboxId: input.sandboxId,\n        reason: result.reason,\n      });\n      break;\n    }\n\n    nextPollMs = result.nextPollMs;\n  }\n}\n", "export type LogLevel = \"info\" | \"warn\" | \"error\" | \"silent\";\n\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\n  info: 0,\n  warn: 1,\n  error: 2,\n  silent: 3,\n};\n\nexport type LogContext = {\n  sessionId?: string;\n  messageId?: string;\n  sandboxId?: string;\n  [key: string]: unknown;\n};\n\nexport type LoggingConfig = {\n  level?: LogLevel;\n  /**\n   * Top-level label for all log lines.\n   * @default \"agent\"\n   * @example \"my-app\" \u2192 [my-app:default:workflow]\n   */\n  prefix?: string;\n  name?: string;\n};\n\nfunction getLevel(config: LoggingConfig): LogLevel {\n  return config.level ?? \"warn\";\n}\n\nfunction getPrefix(config: LoggingConfig): string {\n  return config.prefix ?? \"agent\";\n}\n\nfunction shouldLog(config: LoggingConfig, level: LogLevel): boolean {\n  return LOG_LEVEL_PRIORITY[level] >= LOG_LEVEL_PRIORITY[getLevel(config)];\n}\n\nfunction formatTag(config: LoggingConfig, subsystem: string): string {\n  return `[${getPrefix(config)}:${config.name}:${subsystem}]`;\n}\n\nfunction formatData(\n  context: LogContext,\n  extra?: Record<string, unknown>\n): Record<string, unknown> | undefined {\n  const merged = { ...context, ...extra };\n  const entries = Object.entries(merged).filter(\n    ([, v]) => v !== undefined && v !== null\n  );\n  if (entries.length === 0) {\n    return undefined;\n  }\n  return Object.fromEntries(entries);\n}\n\nexport class Logger {\n  readonly subsystem: string;\n  private readonly config: LoggingConfig;\n  private readonly context: LogContext;\n\n  constructor({\n    subsystem,\n    config,\n    context,\n  }: {\n    subsystem: string;\n    config: LoggingConfig;\n    context?: LogContext;\n  }) {\n    this.subsystem = subsystem;\n    this.config = config;\n    this.context = context ?? {};\n  }\n\n  child({\n    subsystem,\n    context,\n  }: {\n    subsystem?: string;\n    context?: LogContext;\n  }): Logger {\n    return new Logger({\n      config: this.config,\n      subsystem: subsystem ? `${this.subsystem}:${subsystem}` : this.subsystem,\n      context: { ...this.context, ...context },\n    });\n  }\n\n  withContext(context: LogContext): Logger {\n    return new Logger({\n      config: this.config,\n      subsystem: this.subsystem,\n      context: { ...this.context, ...context },\n    });\n  }\n\n  info(message: string, data?: Record<string, unknown>): void {\n    if (!shouldLog(this.config, \"info\")) {\n      return;\n    }\n    const d = formatData(this.context, data);\n    if (d) {\n      console.info(formatTag(this.config, this.subsystem), message, d);\n    } else {\n      console.info(formatTag(this.config, this.subsystem), message);\n    }\n  }\n\n  warn(message: string, data?: Record<string, unknown>): void {\n    if (!shouldLog(this.config, \"warn\")) {\n      return;\n    }\n    const d = formatData(this.context, data);\n    if (d) {\n      console.warn(formatTag(this.config, this.subsystem), message, d);\n    } else {\n      console.warn(formatTag(this.config, this.subsystem), message);\n    }\n  }\n\n  error(message: string, data?: Record<string, unknown>): void {\n    if (!shouldLog(this.config, \"error\")) {\n      return;\n    }\n    const d = formatData(this.context, data);\n    if (d) {\n      console.error(formatTag(this.config, this.subsystem), message, d);\n    } else {\n      console.error(formatTag(this.config, this.subsystem), message);\n    }\n  }\n\n  /**\n   * Returns a function that, when called, logs the elapsed time at info level.\n   * Pass `logOnStart: true` to also log when the timer begins.\n   *\n   * @example\n   * const done = log.time(\"sandbox setup\", { sandboxId }, { logOnStart: true });\n   * await setup();\n   * done(); // start: [agent:default:sandbox] sandbox setup { sandboxId: '...' }\n   *         // end:   [agent:default:sandbox] sandbox setup { sandboxId: '...', durationMs: 123 }\n   */\n  time(\n    message: string,\n    data?: Record<string, unknown>,\n    opts?: { logOnStart?: boolean }\n  ): (endData?: Record<string, unknown>) => void {\n    if (opts?.logOnStart) {\n      this.info(`[start] ${message}`, data);\n    }\n    const t0 = Date.now();\n    return (endData?: Record<string, unknown>) => {\n      this.info(opts?.logOnStart ? `[end] ${message}` : message, {\n        ...data,\n        ...endData,\n        durationMs: Date.now() - t0,\n      });\n    };\n  }\n}\n\nexport function createLogger({\n  config,\n  subsystem,\n  context,\n}: {\n  config: LoggingConfig;\n  subsystem: string;\n  context?: LogContext;\n}): Logger {\n  return new Logger({ config, subsystem, context });\n}\n", "import type { Sandbox as VercelSandboxSDK } from \"@vercel/sandbox\";\nimport { createLogger } from \"../../../utils/logger\";\nimport type { SandboxAgentRef } from \"../../adapter\";\nimport type { VercelBindingMetadata } from \"../vercel\";\n\nasync function loadSandboxSDK(): Promise<typeof VercelSandboxSDK> {\n  return (await import(\"@vercel/sandbox\")).Sandbox;\n}\n\nconst DEFAULT_POLL_INTERVAL_MS = 2 * 60 * 1000;\nconst DEFAULT_IDLE_TIMEOUT_MS = 5 * 60 * 1000;\nconst SNAPSHOT_BEFORE_TIMEOUT_MS = 10 * 60 * 1000;\n\nconst getTestCredentials = () =>\n  process.env.NODE_ENV === \"test\"\n    ? {\n        token: process.env.TEST_VERCEL_TOKEN,\n        teamId: process.env.TEST_VERCEL_TEAM_ID,\n        projectId: process.env.TEST_VERCEL_PROJECT_ID,\n      }\n    : {};\n\nexport type LifecycleStepResult =\n  | { action: \"continue\"; nextPollMs: number }\n  | {\n      action: \"exit\";\n      reason: \"sandboxId_changed\" | \"idle\" | \"timeout\" | \"not_found\";\n    };\n\nexport type LifecycleConfig = {\n  pollIntervalMs?: number;\n  stopAfterInactiveMs?: number;\n  snapshotBeforeTimeoutMs?: number;\n};\n\nexport type LifecycleInput = {\n  agent: SandboxAgentRef;\n  sandboxId: string;\n  vercelSandboxId: string;\n  config?: LifecycleConfig;\n};\n\nexport async function checkAndSnapshotStep(\n  input: LifecycleInput\n): Promise<Error | LifecycleStepResult> {\n  \"use step\";\n\n  const storage = input.agent.storage;\n  const log = createLogger({\n    config: { ...input.agent.options.logging, name: input.agent.name },\n    subsystem: \"sandbox:lifecycle\",\n    context: { sandboxId: input.sandboxId },\n  });\n\n  log.info(\"lifecycle check started\", {\n    vercelSandboxId: input.vercelSandboxId,\n  });\n\n  const record = await storage.sandbox.get(input.sandboxId);\n  if (!record) {\n    log.warn(\"sandbox record not found, exiting lifecycle\");\n    return { action: \"exit\", reason: \"not_found\" };\n  }\n\n  const metadata = record.setup?.metadata as VercelBindingMetadata | null;\n  const currentSandboxId = metadata?.sandboxId ?? null;\n\n  if (currentSandboxId !== input.vercelSandboxId) {\n    log.info(\"sandbox ID changed, exiting lifecycle\", {\n      expected: input.vercelSandboxId,\n      actual: currentSandboxId,\n    });\n    return { action: \"exit\", reason: \"sandboxId_changed\" };\n  }\n\n  if (!currentSandboxId) {\n    log.warn(\"no current sandbox ID in metadata, exiting lifecycle\");\n    return { action: \"exit\", reason: \"not_found\" };\n  }\n\n  const pollIntervalMs =\n    input.config?.pollIntervalMs ?? DEFAULT_POLL_INTERVAL_MS;\n  const idleTimeoutMs =\n    input.config?.stopAfterInactiveMs ?? DEFAULT_IDLE_TIMEOUT_MS;\n  const snapshotBeforeTimeoutMs =\n    input.config?.snapshotBeforeTimeoutMs ?? SNAPSHOT_BEFORE_TIMEOUT_MS;\n\n  const now = Date.now();\n  const lastActivity = record.lastActiveAt ?? record.createdAt ?? now;\n  const idleDuration = now - lastActivity;\n  const shouldSnapshotDueToIdle = idleDuration > idleTimeoutMs;\n\n  let shouldSnapshotDueToTimeout = false;\n  try {\n    const SandboxSDK = await loadSandboxSDK();\n    const sandbox = await SandboxSDK.get({\n      sandboxId: currentSandboxId,\n      ...getTestCredentials(),\n    });\n    if (sandbox.timeout < snapshotBeforeTimeoutMs) {\n      shouldSnapshotDueToTimeout = true;\n    }\n  } catch {\n    log.warn(\"failed to get sandbox from SDK, exiting lifecycle\", {\n      vercelSandboxId: currentSandboxId,\n    });\n    return { action: \"exit\", reason: \"not_found\" };\n  }\n\n  if (shouldSnapshotDueToIdle || shouldSnapshotDueToTimeout) {\n    const reason = shouldSnapshotDueToIdle ? \"idle\" : \"timeout\";\n    log.info(\"snapshotting sandbox before exit\", {\n      reason,\n      idleDurationMs: idleDuration,\n      vercelSandboxId: currentSandboxId,\n    });\n\n    try {\n      const SandboxSDK = await loadSandboxSDK();\n      const sandbox = await SandboxSDK.get({\n        sandboxId: currentSandboxId,\n        ...getTestCredentials(),\n      });\n      const snapshot = await sandbox.snapshot();\n      log.info(\"snapshot created\", { snapshotId: snapshot.snapshotId });\n\n      await storage.sandbox.update(input.sandboxId, {\n        setup: {\n          ...record.setup,\n          metadata: { sandboxId: null, snapshotId: snapshot.snapshotId },\n        },\n      });\n      log.info(\"sandbox record updated with snapshot\");\n    } catch (e) {\n      log.error(\"failed to snapshot sandbox\", { cause: e });\n      return e instanceof Error ? e : new Error(String(e));\n    }\n    return { action: \"exit\", reason };\n  }\n\n  log.info(\"sandbox still active, continuing lifecycle\", {\n    idleDurationMs: idleDuration,\n    nextPollMs: pollIntervalMs,\n  });\n  return { action: \"continue\", nextPollMs: pollIntervalMs };\n}\n"],
  "mappings": "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",
  "names": ["import_ulid", "loadSandboxSDK", "getTestCredentials", "startWorkflow", "sdk", "instance"]
}

|