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
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
import { WORKFLOW_SERIALIZE, WORKFLOW_DESERIALIZE } from '@workflow/serde';
|
|
2
|
+
import { NetworkPolicy } from '@vercel/sandbox';
|
|
3
|
+
import { UIMessage } from 'ai';
|
|
4
|
+
|
|
5
|
+
type Session = {
|
|
6
|
+
id: string;
|
|
7
|
+
createdAt: number;
|
|
8
|
+
updatedAt: number;
|
|
9
|
+
model: string | null;
|
|
10
|
+
system: string | null;
|
|
11
|
+
sandboxId: string | null;
|
|
12
|
+
lastMessageId: string | null;
|
|
13
|
+
activeTools: string[] | null;
|
|
14
|
+
skillsDir: string[] | null;
|
|
15
|
+
generation: {
|
|
16
|
+
maxSteps?: number;
|
|
17
|
+
temperature?: number;
|
|
18
|
+
topK?: number;
|
|
19
|
+
topP?: number;
|
|
20
|
+
frequencyPenalty?: number;
|
|
21
|
+
presencePenalty?: number;
|
|
22
|
+
maxOutputTokens?: number;
|
|
23
|
+
headers?: Record<string, string>;
|
|
24
|
+
} | null;
|
|
25
|
+
};
|
|
26
|
+
type Message = {
|
|
27
|
+
id: string;
|
|
28
|
+
sessionId: string;
|
|
29
|
+
role: "user" | "assistant" | "system";
|
|
30
|
+
createdAt: number;
|
|
31
|
+
startedAt: number | null;
|
|
32
|
+
completedAt: number | null;
|
|
33
|
+
interruptedAt: number | null;
|
|
34
|
+
interruptedLastPart: {
|
|
35
|
+
index: number;
|
|
36
|
+
part: unknown;
|
|
37
|
+
} | null;
|
|
38
|
+
usage: {
|
|
39
|
+
steps: {
|
|
40
|
+
stepIndex: number;
|
|
41
|
+
model: string;
|
|
42
|
+
inputTokens: number;
|
|
43
|
+
outputTokens: number;
|
|
44
|
+
totalTokens: number;
|
|
45
|
+
cacheReadTokens: number;
|
|
46
|
+
cacheWriteTokens: number;
|
|
47
|
+
reasoningTokens: number;
|
|
48
|
+
}[];
|
|
49
|
+
summary: {
|
|
50
|
+
model: string;
|
|
51
|
+
inputTokens: number;
|
|
52
|
+
outputTokens: number;
|
|
53
|
+
totalTokens: number;
|
|
54
|
+
cacheReadTokens: number;
|
|
55
|
+
cacheWriteTokens: number;
|
|
56
|
+
reasoningTokens: number;
|
|
57
|
+
stepCount: number;
|
|
58
|
+
};
|
|
59
|
+
} | null;
|
|
60
|
+
workflowRunId?: string | null;
|
|
61
|
+
};
|
|
62
|
+
type Part = {
|
|
63
|
+
id: string;
|
|
64
|
+
messageId: string;
|
|
65
|
+
sessionId: string;
|
|
66
|
+
index: number;
|
|
67
|
+
part: UIMessage["parts"][number];
|
|
68
|
+
};
|
|
69
|
+
type Sandbox = {
|
|
70
|
+
id: string;
|
|
71
|
+
setup: {
|
|
72
|
+
binding: string;
|
|
73
|
+
version: string | null;
|
|
74
|
+
completedAt: number | null;
|
|
75
|
+
metadata: Record<string, unknown> | null;
|
|
76
|
+
networkPolicy: NetworkPolicy | null;
|
|
77
|
+
};
|
|
78
|
+
createdAt: number | null;
|
|
79
|
+
lastActiveAt: number | null;
|
|
80
|
+
};
|
|
81
|
+
type Setup = {
|
|
82
|
+
version: string;
|
|
83
|
+
snapshotId: string | null;
|
|
84
|
+
createdAt: number;
|
|
85
|
+
lastUsedAt: number | null;
|
|
86
|
+
};
|
|
87
|
+
type SetupSnapshot = Setup;
|
|
88
|
+
|
|
89
|
+
type StorageHandlers = {
|
|
90
|
+
"session.get"(p: {
|
|
91
|
+
id: string;
|
|
92
|
+
}): Promise<Session | null>;
|
|
93
|
+
"session.set"(p: {
|
|
94
|
+
id: string;
|
|
95
|
+
value: Session;
|
|
96
|
+
}): Promise<void>;
|
|
97
|
+
"session.update"(p: {
|
|
98
|
+
id: string;
|
|
99
|
+
updates: Partial<Session>;
|
|
100
|
+
}): Promise<Session>;
|
|
101
|
+
"message.get"(p: {
|
|
102
|
+
id: string;
|
|
103
|
+
}): Promise<Message | null>;
|
|
104
|
+
"message.set"(p: {
|
|
105
|
+
id: string;
|
|
106
|
+
value: Message;
|
|
107
|
+
}): Promise<void>;
|
|
108
|
+
"message.update"(p: {
|
|
109
|
+
id: string;
|
|
110
|
+
updates: Partial<Message>;
|
|
111
|
+
}): Promise<Message>;
|
|
112
|
+
"message.listBySession"(p: {
|
|
113
|
+
sessionId: string;
|
|
114
|
+
}): Promise<Message[]>;
|
|
115
|
+
"part.get"(p: {
|
|
116
|
+
id: string;
|
|
117
|
+
}): Promise<Part | null>;
|
|
118
|
+
"part.set"(p: {
|
|
119
|
+
id: string;
|
|
120
|
+
value: Part;
|
|
121
|
+
}): Promise<void>;
|
|
122
|
+
"part.listBySession"(p: {
|
|
123
|
+
sessionId: string;
|
|
124
|
+
}): Promise<Part[]>;
|
|
125
|
+
"sandbox.get"(p: {
|
|
126
|
+
id: string;
|
|
127
|
+
}): Promise<Sandbox | null>;
|
|
128
|
+
"sandbox.set"(p: {
|
|
129
|
+
id: string;
|
|
130
|
+
value: Sandbox;
|
|
131
|
+
}): Promise<void>;
|
|
132
|
+
"sandbox.update"(p: {
|
|
133
|
+
id: string;
|
|
134
|
+
updates: Partial<Sandbox>;
|
|
135
|
+
}): Promise<Sandbox>;
|
|
136
|
+
"setup.get"(p: {
|
|
137
|
+
id: string;
|
|
138
|
+
}): Promise<Setup | null>;
|
|
139
|
+
"setup.set"(p: {
|
|
140
|
+
id: string;
|
|
141
|
+
value: Setup;
|
|
142
|
+
}): Promise<void>;
|
|
143
|
+
};
|
|
144
|
+
type StorageCall = {
|
|
145
|
+
[K in keyof StorageHandlers]: {
|
|
146
|
+
method: K;
|
|
147
|
+
} & Parameters<StorageHandlers[K]>[0];
|
|
148
|
+
}[keyof StorageHandlers];
|
|
149
|
+
declare class StorageStep {
|
|
150
|
+
event: StorageCall;
|
|
151
|
+
constructor(event: StorageCall);
|
|
152
|
+
static [WORKFLOW_SERIALIZE](instance: StorageStep): {
|
|
153
|
+
event: StorageCall;
|
|
154
|
+
};
|
|
155
|
+
static [WORKFLOW_DESERIALIZE](data: {
|
|
156
|
+
event: StorageCall;
|
|
157
|
+
}): StorageStep;
|
|
158
|
+
on(handlers: StorageHandlers): Promise<any>;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* A single function that handles all storage operations. Intended to be
|
|
162
|
+
* marked with `"use step"` so that its body (and Node.js-dependent
|
|
163
|
+
* imports like database clients) is extracted by the workflow bundler
|
|
164
|
+
* and runs server-side.
|
|
165
|
+
*
|
|
166
|
+
* The step receives a `StorageStep` instance (serializable via
|
|
167
|
+
* `@workflow/serde`) and should call `step.handle({ ... })` with a
|
|
168
|
+
* `StorageHandlers` map.
|
|
169
|
+
*/
|
|
170
|
+
type StorageStepFunction = (store: StorageStep) => Promise<any>;
|
|
171
|
+
interface Storage {
|
|
172
|
+
session: {
|
|
173
|
+
get(key: string): Promise<Session | null>;
|
|
174
|
+
set(key: string, value: Session): Promise<void>;
|
|
175
|
+
update(key: string, updates: Partial<Session>): Promise<Session>;
|
|
176
|
+
};
|
|
177
|
+
message: {
|
|
178
|
+
get(key: string): Promise<Message | null>;
|
|
179
|
+
set(key: string, value: Message): Promise<void>;
|
|
180
|
+
update(key: string, updates: Partial<Message>): Promise<Message>;
|
|
181
|
+
listBySession(sessionId: string): Promise<Message[]>;
|
|
182
|
+
};
|
|
183
|
+
part: {
|
|
184
|
+
get(key: string): Promise<Part | null>;
|
|
185
|
+
set(key: string, value: Part): Promise<void>;
|
|
186
|
+
listBySession(sessionId: string): Promise<Part[]>;
|
|
187
|
+
};
|
|
188
|
+
sandbox: {
|
|
189
|
+
get(key: string): Promise<Sandbox | null>;
|
|
190
|
+
set(key: string, value: Sandbox): Promise<void>;
|
|
191
|
+
update(key: string, updates: Partial<Sandbox>): Promise<Sandbox>;
|
|
192
|
+
};
|
|
193
|
+
setup: {
|
|
194
|
+
get(key: string): Promise<Setup | null>;
|
|
195
|
+
set(key: string, value: Setup): Promise<void>;
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
type StorageInput = StorageHandlers | StorageStepFunction;
|
|
199
|
+
declare function toStorage(h: StorageInput): Storage;
|
|
200
|
+
|
|
201
|
+
export { type Message as M, type Part as P, type StorageHandlers as S, type Sandbox as a, type Session as b, type Setup as c, type SetupSnapshot as d, type Storage as e, type StorageCall as f, StorageStep as g, type StorageStepFunction as h, type StorageInput as i, toStorage as t };
|
|
@@ -0,0 +1,302 @@
|
|
|
1
|
+
// src/sandbox/bindings/docker.ts
|
|
2
|
+
import { ulid } from "ulid";
|
|
3
|
+
async function execDocker(args, opts) {
|
|
4
|
+
const { spawn } = await import("child_process");
|
|
5
|
+
return new Promise((resolve, reject) => {
|
|
6
|
+
const child = spawn("docker", ["sandbox", ...args], {
|
|
7
|
+
signal: opts?.signal
|
|
8
|
+
});
|
|
9
|
+
let stdout = "";
|
|
10
|
+
let stderr = "";
|
|
11
|
+
child.stdout.on("data", (data) => {
|
|
12
|
+
stdout += data.toString();
|
|
13
|
+
});
|
|
14
|
+
child.stderr.on("data", (data) => {
|
|
15
|
+
stderr += data.toString();
|
|
16
|
+
});
|
|
17
|
+
const timeoutId = opts?.timeoutMs ? setTimeout(() => {
|
|
18
|
+
child.kill("SIGTERM");
|
|
19
|
+
reject(new Error(`docker sandbox ${args[0]} timed out`));
|
|
20
|
+
}, opts.timeoutMs) : void 0;
|
|
21
|
+
child.on("error", (err) => {
|
|
22
|
+
if (timeoutId) {
|
|
23
|
+
clearTimeout(timeoutId);
|
|
24
|
+
}
|
|
25
|
+
reject(err);
|
|
26
|
+
});
|
|
27
|
+
child.on("close", (code) => {
|
|
28
|
+
if (timeoutId) {
|
|
29
|
+
clearTimeout(timeoutId);
|
|
30
|
+
}
|
|
31
|
+
resolve({ stdout, stderr, exitCode: code ?? 0 });
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
var ensurePromises = /* @__PURE__ */ new Map();
|
|
36
|
+
var activeSandboxes = /* @__PURE__ */ new Set();
|
|
37
|
+
var cleanupRegistered = false;
|
|
38
|
+
var _execSync = null;
|
|
39
|
+
async function registerCleanup() {
|
|
40
|
+
if (cleanupRegistered) {
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
cleanupRegistered = true;
|
|
44
|
+
const cp = await import("child_process");
|
|
45
|
+
_execSync = cp.execSync;
|
|
46
|
+
const cleanup = () => {
|
|
47
|
+
if (!_execSync) {
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
for (const name of Array.from(activeSandboxes)) {
|
|
51
|
+
try {
|
|
52
|
+
_execSync(`docker sandbox stop ${name}`, {
|
|
53
|
+
timeout: 1e4,
|
|
54
|
+
stdio: "pipe"
|
|
55
|
+
});
|
|
56
|
+
} catch {
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
};
|
|
60
|
+
process.on("exit", cleanup);
|
|
61
|
+
process.on("SIGINT", () => {
|
|
62
|
+
cleanup();
|
|
63
|
+
process.exit(130);
|
|
64
|
+
});
|
|
65
|
+
process.on("SIGTERM", () => {
|
|
66
|
+
cleanup();
|
|
67
|
+
process.exit(143);
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
async function ensureSandbox(sandboxName) {
|
|
71
|
+
const existing = ensurePromises.get(sandboxName);
|
|
72
|
+
if (existing) {
|
|
73
|
+
return existing;
|
|
74
|
+
}
|
|
75
|
+
const promise = (async () => {
|
|
76
|
+
const ls = await execDocker(["ls", "-q"], { timeoutMs: 1e4 });
|
|
77
|
+
const existingNames = ls.exitCode === 0 ? ls.stdout.split("\n").map((s) => s.trim()).filter(Boolean) : [];
|
|
78
|
+
if (existingNames.includes(sandboxName)) {
|
|
79
|
+
activeSandboxes.add(sandboxName);
|
|
80
|
+
registerCleanup();
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
const path = await import("path");
|
|
84
|
+
const os = await import("os");
|
|
85
|
+
const fs = await import("fs/promises");
|
|
86
|
+
const workspaceDir = path.join(
|
|
87
|
+
os.tmpdir(),
|
|
88
|
+
"agent-docker-sandbox",
|
|
89
|
+
sandboxName
|
|
90
|
+
);
|
|
91
|
+
await fs.mkdir(workspaceDir, { recursive: true });
|
|
92
|
+
const create = await execDocker(
|
|
93
|
+
["create", "--name", sandboxName, "shell", workspaceDir],
|
|
94
|
+
{ timeoutMs: 6e4 }
|
|
95
|
+
);
|
|
96
|
+
if (create.exitCode !== 0) {
|
|
97
|
+
if (create.stderr.includes("already exists")) {
|
|
98
|
+
activeSandboxes.add(sandboxName);
|
|
99
|
+
registerCleanup();
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
throw new Error(
|
|
103
|
+
`Failed to create docker sandbox "${sandboxName}": ${create.stderr}`
|
|
104
|
+
);
|
|
105
|
+
}
|
|
106
|
+
activeSandboxes.add(sandboxName);
|
|
107
|
+
registerCleanup();
|
|
108
|
+
})();
|
|
109
|
+
ensurePromises.set(sandboxName, promise);
|
|
110
|
+
try {
|
|
111
|
+
await promise;
|
|
112
|
+
} catch (e) {
|
|
113
|
+
ensurePromises.delete(sandboxName);
|
|
114
|
+
throw e;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
var DEFAULT_DOCKER_CWD = "/home/agent/workspace";
|
|
118
|
+
var DockerSandboxInstance = class {
|
|
119
|
+
cwd;
|
|
120
|
+
sandboxName;
|
|
121
|
+
processes = /* @__PURE__ */ new Map();
|
|
122
|
+
constructor(sandboxName, cwd) {
|
|
123
|
+
this.sandboxName = sandboxName;
|
|
124
|
+
this.cwd = cwd ?? DEFAULT_DOCKER_CWD;
|
|
125
|
+
}
|
|
126
|
+
async exec(opts) {
|
|
127
|
+
await ensureSandbox(this.sandboxName);
|
|
128
|
+
const { spawn } = await import("child_process");
|
|
129
|
+
const commandId = `command_${ulid()}`;
|
|
130
|
+
const envFlags = opts.env ? Object.entries(opts.env).flatMap(([k, v]) => ["-e", `${k}=${v}`]) : [];
|
|
131
|
+
const cwdFlags = opts.cwd ? ["-w", opts.cwd] : [];
|
|
132
|
+
const baseCmd = opts.sudo ? ["sudo", opts.command, ...opts.args ?? []] : opts.args ? [opts.command, ...opts.args] : [opts.command];
|
|
133
|
+
const fullCmd = baseCmd;
|
|
134
|
+
const child = spawn(
|
|
135
|
+
"docker",
|
|
136
|
+
[
|
|
137
|
+
"sandbox",
|
|
138
|
+
"exec",
|
|
139
|
+
...envFlags,
|
|
140
|
+
...cwdFlags,
|
|
141
|
+
this.sandboxName,
|
|
142
|
+
...fullCmd
|
|
143
|
+
],
|
|
144
|
+
{ signal: opts.signal }
|
|
145
|
+
);
|
|
146
|
+
this.processes.set(commandId, child);
|
|
147
|
+
let stdout = "";
|
|
148
|
+
let stderr = "";
|
|
149
|
+
const logQueue = [];
|
|
150
|
+
let logResolve = null;
|
|
151
|
+
let closed = false;
|
|
152
|
+
child.stdout.on("data", (data) => {
|
|
153
|
+
const str = String(data);
|
|
154
|
+
stdout += str;
|
|
155
|
+
logQueue.push({ stream: "stdout", data: str });
|
|
156
|
+
logResolve?.();
|
|
157
|
+
});
|
|
158
|
+
child.stderr.on("data", (data) => {
|
|
159
|
+
const str = String(data);
|
|
160
|
+
stderr += str;
|
|
161
|
+
logQueue.push({ stream: "stderr", data: str });
|
|
162
|
+
logResolve?.();
|
|
163
|
+
});
|
|
164
|
+
const result = new Promise((resolve, reject) => {
|
|
165
|
+
child.on("error", (err) => {
|
|
166
|
+
this.processes.delete(commandId);
|
|
167
|
+
closed = true;
|
|
168
|
+
logResolve?.();
|
|
169
|
+
reject(err);
|
|
170
|
+
});
|
|
171
|
+
child.on("close", (code) => {
|
|
172
|
+
this.processes.delete(commandId);
|
|
173
|
+
closed = true;
|
|
174
|
+
logResolve?.();
|
|
175
|
+
resolve({ stdout, stderr, exitCode: code ?? 0 });
|
|
176
|
+
});
|
|
177
|
+
});
|
|
178
|
+
async function* logs() {
|
|
179
|
+
while (!closed || logQueue.length > 0) {
|
|
180
|
+
const entry = logQueue.shift();
|
|
181
|
+
if (entry) {
|
|
182
|
+
yield entry;
|
|
183
|
+
} else if (!closed) {
|
|
184
|
+
await new Promise((r) => {
|
|
185
|
+
logResolve = r;
|
|
186
|
+
});
|
|
187
|
+
logResolve = null;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
return { commandId, logs, result };
|
|
192
|
+
}
|
|
193
|
+
async readFile(opts) {
|
|
194
|
+
await ensureSandbox(this.sandboxName);
|
|
195
|
+
const result = await execDocker(
|
|
196
|
+
[
|
|
197
|
+
"exec",
|
|
198
|
+
this.sandboxName,
|
|
199
|
+
"bash",
|
|
200
|
+
"-c",
|
|
201
|
+
`base64 '${opts.path.replace(/'/g, "'\\''")}'`
|
|
202
|
+
],
|
|
203
|
+
{ timeoutMs: 3e4, signal: opts.signal }
|
|
204
|
+
);
|
|
205
|
+
if (result.exitCode !== 0) {
|
|
206
|
+
if (result.stderr.includes("No such file") || result.stderr.includes("ENOENT")) {
|
|
207
|
+
return null;
|
|
208
|
+
}
|
|
209
|
+
throw new Error(`readFile failed: ${result.stderr}`);
|
|
210
|
+
}
|
|
211
|
+
return Buffer.from(result.stdout.trim(), "base64");
|
|
212
|
+
}
|
|
213
|
+
async writeFiles(opts) {
|
|
214
|
+
await ensureSandbox(this.sandboxName);
|
|
215
|
+
for (const file of opts.files) {
|
|
216
|
+
const fullPath = opts.destPath ? `${opts.destPath}/${file.path}` : file.path;
|
|
217
|
+
const parentDir = fullPath.substring(0, fullPath.lastIndexOf("/"));
|
|
218
|
+
if (parentDir) {
|
|
219
|
+
await execDocker(["exec", this.sandboxName, "mkdir", "-p", parentDir], {
|
|
220
|
+
signal: opts.signal
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
const b64 = typeof file.content === "string" ? Buffer.from(file.content).toString("base64") : file.content.toString("base64");
|
|
224
|
+
await execDocker(
|
|
225
|
+
[
|
|
226
|
+
"exec",
|
|
227
|
+
this.sandboxName,
|
|
228
|
+
"bash",
|
|
229
|
+
"-c",
|
|
230
|
+
`echo '${b64}' | base64 -d > '${fullPath.replace(/'/g, "'\\''")}'`
|
|
231
|
+
],
|
|
232
|
+
{ signal: opts.signal }
|
|
233
|
+
);
|
|
234
|
+
if (file.path.endsWith(".sh")) {
|
|
235
|
+
await execDocker(["exec", this.sandboxName, "chmod", "+x", fullPath], {
|
|
236
|
+
signal: opts.signal
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
242
|
+
async getDomain(opts) {
|
|
243
|
+
return `http://localhost:${opts.port}`;
|
|
244
|
+
}
|
|
245
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
246
|
+
async kill(opts) {
|
|
247
|
+
const child = this.processes.get(opts.commandId);
|
|
248
|
+
if (child) {
|
|
249
|
+
child.kill("SIGTERM");
|
|
250
|
+
this.processes.delete(opts.commandId);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
254
|
+
async getStatus() {
|
|
255
|
+
return "running";
|
|
256
|
+
}
|
|
257
|
+
async start() {
|
|
258
|
+
await ensureSandbox(this.sandboxName);
|
|
259
|
+
}
|
|
260
|
+
async stop() {
|
|
261
|
+
await execDocker(["stop", this.sandboxName], { timeoutMs: 3e4 });
|
|
262
|
+
activeSandboxes.delete(this.sandboxName);
|
|
263
|
+
ensurePromises.delete(this.sandboxName);
|
|
264
|
+
}
|
|
265
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
266
|
+
async snapshot() {
|
|
267
|
+
throw new Error("snapshot is not supported for docker sandboxes");
|
|
268
|
+
}
|
|
269
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
270
|
+
async updateNetworkPolicy() {
|
|
271
|
+
throw new Error(
|
|
272
|
+
"updateNetworkPolicy is not supported for docker sandboxes"
|
|
273
|
+
);
|
|
274
|
+
}
|
|
275
|
+
};
|
|
276
|
+
function dockerSandbox() {
|
|
277
|
+
return {
|
|
278
|
+
type: "docker",
|
|
279
|
+
async create(opts) {
|
|
280
|
+
const sandboxName = `agent-${ulid()}`;
|
|
281
|
+
const cwd = opts.setup?.config?.cwd;
|
|
282
|
+
await ensureSandbox(sandboxName);
|
|
283
|
+
const instance = new DockerSandboxInstance(sandboxName, cwd);
|
|
284
|
+
if (opts.setup?.run) {
|
|
285
|
+
await opts.setup.run(instance);
|
|
286
|
+
}
|
|
287
|
+
return {
|
|
288
|
+
instance,
|
|
289
|
+
metadata: { sandboxName }
|
|
290
|
+
};
|
|
291
|
+
},
|
|
292
|
+
// biome-ignore lint/suspicious/useAwait: .
|
|
293
|
+
async connect(opts) {
|
|
294
|
+
return new DockerSandboxInstance(opts.metadata.sandboxName);
|
|
295
|
+
}
|
|
296
|
+
};
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
export {
|
|
300
|
+
dockerSandbox
|
|
301
|
+
};
|
|
302
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/sandbox/bindings/docker.ts"],
  "sourcesContent": ["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"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
// src/sandbox/process-manager.ts
|
|
2
|
+
var DEFAULT_WAIT_UNTIL = 2e3;
|
|
3
|
+
var createProcessManager = (opts) => {
|
|
4
|
+
const { sandbox } = opts;
|
|
5
|
+
const run = async (runOpts) => {
|
|
6
|
+
const { command, waitUntil = DEFAULT_WAIT_UNTIL } = runOpts;
|
|
7
|
+
const execResult = await sandbox.exec({
|
|
8
|
+
command: "sh",
|
|
9
|
+
args: ["-c", command]
|
|
10
|
+
});
|
|
11
|
+
if (waitUntil === 0) {
|
|
12
|
+
return {
|
|
13
|
+
commandId: execResult.commandId,
|
|
14
|
+
stdout: "",
|
|
15
|
+
stderr: "",
|
|
16
|
+
exitCode: -1,
|
|
17
|
+
status: "running",
|
|
18
|
+
outputDir: ""
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
const completed = await raceTimeout(execResult, waitUntil);
|
|
22
|
+
if (!completed) {
|
|
23
|
+
return {
|
|
24
|
+
commandId: execResult.commandId,
|
|
25
|
+
stdout: "",
|
|
26
|
+
stderr: "",
|
|
27
|
+
exitCode: -1,
|
|
28
|
+
status: "running",
|
|
29
|
+
outputDir: ""
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
const outputDir = `.agent/bash/${Date.now()}-${execResult.commandId}`;
|
|
33
|
+
writeOutputDir(sandbox, outputDir, completed, command);
|
|
34
|
+
return {
|
|
35
|
+
commandId: execResult.commandId,
|
|
36
|
+
stdout: completed.stdout,
|
|
37
|
+
stderr: completed.stderr,
|
|
38
|
+
exitCode: completed.exitCode,
|
|
39
|
+
status: completed.exitCode === 0 ? "completed" : "failed",
|
|
40
|
+
outputDir
|
|
41
|
+
};
|
|
42
|
+
};
|
|
43
|
+
return { run };
|
|
44
|
+
};
|
|
45
|
+
async function raceTimeout(execResult, timeoutMs) {
|
|
46
|
+
const TIMED_OUT = /* @__PURE__ */ Symbol("timeout");
|
|
47
|
+
const timer = new Promise(
|
|
48
|
+
(resolve) => setTimeout(() => resolve(TIMED_OUT), timeoutMs)
|
|
49
|
+
);
|
|
50
|
+
const winner = await Promise.race([execResult.result, timer]);
|
|
51
|
+
return winner === TIMED_OUT ? null : winner;
|
|
52
|
+
}
|
|
53
|
+
function writeOutputDir(sandbox, dir, result, command) {
|
|
54
|
+
const files = [
|
|
55
|
+
{ path: "stdout.txt", content: result.stdout },
|
|
56
|
+
{
|
|
57
|
+
path: "metadata.json",
|
|
58
|
+
content: JSON.stringify({ exitCode: result.exitCode, command }, null, 2)
|
|
59
|
+
}
|
|
60
|
+
];
|
|
61
|
+
if (result.stderr) {
|
|
62
|
+
files.push({ path: "stderr.txt", content: result.stderr });
|
|
63
|
+
}
|
|
64
|
+
sandbox.writeFiles({ files, destPath: dir }).catch(() => {
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export {
|
|
69
|
+
DEFAULT_WAIT_UNTIL,
|
|
70
|
+
createProcessManager
|
|
71
|
+
};
|
|
72
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL3NhbmRib3gvcHJvY2Vzcy1tYW5hZ2VyLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyJpbXBvcnQgdHlwZSB7IEV4ZWNSZXN1bHQgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5cbmV4cG9ydCBjb25zdCBERUZBVUxUX1dBSVRfVU5USUwgPSAyMDAwO1xuXG50eXBlIFNhbmRib3hIYW5kbGUgPSB7XG4gIGV4ZWM6IChvcHRzOiB7IGNvbW1hbmQ6IHN0cmluZzsgYXJncz86IHN0cmluZ1tdIH0pID0+IFByb21pc2U8RXhlY1Jlc3VsdD47XG4gIHdyaXRlRmlsZXM6IChvcHRzOiB7XG4gICAgZmlsZXM6IHsgcGF0aDogc3RyaW5nOyBjb250ZW50OiBzdHJpbmcgfVtdO1xuICAgIGRlc3RQYXRoOiBzdHJpbmc7XG4gIH0pID0+IFByb21pc2U8dm9pZD47XG59O1xuXG50eXBlIEJhc2hSZXN1bHQgPSB7XG4gIGNvbW1hbmRJZDogc3RyaW5nO1xuICBzdGRvdXQ6IHN0cmluZztcbiAgc3RkZXJyOiBzdHJpbmc7XG4gIGV4aXRDb2RlOiBudW1iZXI7XG4gIHN0YXR1czogXCJydW5uaW5nXCIgfCBcImNvbXBsZXRlZFwiIHwgXCJmYWlsZWRcIjtcbiAgb3V0cHV0RGlyOiBzdHJpbmc7XG59O1xuXG50eXBlIFJ1bk9wdGlvbnMgPSB7XG4gIGNvbW1hbmQ6IHN0cmluZztcbiAgd2FpdFVudGlsPzogbnVtYmVyO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVByb2Nlc3NNYW5hZ2VyID0gKG9wdHM6IHsgc2FuZGJveDogU2FuZGJveEhhbmRsZSB9KSA9PiB7XG4gIGNvbnN0IHsgc2FuZGJveCB9ID0gb3B0cztcblxuICBjb25zdCBydW4gPSBhc3luYyAocnVuT3B0czogUnVuT3B0aW9ucyk6IFByb21pc2U8QmFzaFJlc3VsdD4gPT4ge1xuICAgIGNvbnN0IHsgY29tbWFuZCwgd2FpdFVudGlsID0gREVGQVVMVF9XQUlUX1VOVElMIH0gPSBydW5PcHRzO1xuXG4gICAgY29uc3QgZXhlY1Jlc3VsdCA9IGF3YWl0IHNhbmRib3guZXhlYyh7XG4gICAgICBjb21tYW5kOiBcInNoXCIsXG4gICAgICBhcmdzOiBbXCItY1wiLCBjb21tYW5kXSxcbiAgICB9KTtcblxuICAgIGlmICh3YWl0VW50aWwgPT09IDApIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIGNvbW1hbmRJZDogZXhlY1Jlc3VsdC5jb21tYW5kSWQsXG4gICAgICAgIHN0ZG91dDogXCJcIixcbiAgICAgICAgc3RkZXJyOiBcIlwiLFxuICAgICAgICBleGl0Q29kZTogLTEsXG4gICAgICAgIHN0YXR1czogXCJydW5uaW5nXCIsXG4gICAgICAgIG91dHB1dERpcjogXCJcIixcbiAgICAgIH07XG4gICAgfVxuXG4gICAgY29uc3QgY29tcGxldGVkID0gYXdhaXQgcmFjZVRpbWVvdXQoZXhlY1Jlc3VsdCwgd2FpdFVudGlsKTtcblxuICAgIGlmICghY29tcGxldGVkKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBjb21tYW5kSWQ6IGV4ZWNSZXN1bHQuY29tbWFuZElkLFxuICAgICAgICBzdGRvdXQ6IFwiXCIsXG4gICAgICAgIHN0ZGVycjogXCJcIixcbiAgICAgICAgZXhpdENvZGU6IC0xLFxuICAgICAgICBzdGF0dXM6IFwicnVubmluZ1wiLFxuICAgICAgICBvdXRwdXREaXI6IFwiXCIsXG4gICAgICB9O1xuICAgIH1cblxuICAgIGNvbnN0IG91dHB1dERpciA9IGAuYWdlbnQvYmFzaC8ke0RhdGUubm93KCl9LSR7ZXhlY1Jlc3VsdC5jb21tYW5kSWR9YDtcbiAgICB3cml0ZU91dHB1dERpcihzYW5kYm94LCBvdXRwdXREaXIsIGNvbXBsZXRlZCwgY29tbWFuZCk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgY29tbWFuZElkOiBleGVjUmVzdWx0LmNvbW1hbmRJZCxcbiAgICAgIHN0ZG91dDogY29tcGxldGVkLnN0ZG91dCxcbiAgICAgIHN0ZGVycjogY29tcGxldGVkLnN0ZGVycixcbiAgICAgIGV4aXRDb2RlOiBjb21wbGV0ZWQuZXhpdENvZGUsXG4gICAgICBzdGF0dXM6IGNvbXBsZXRlZC5leGl0Q29kZSA9PT0gMCA/IFwiY29tcGxldGVkXCIgOiBcImZhaWxlZFwiLFxuICAgICAgb3V0cHV0RGlyLFxuICAgIH07XG4gIH07XG5cbiAgcmV0dXJuIHsgcnVuIH07XG59O1xuXG5hc3luYyBmdW5jdGlvbiByYWNlVGltZW91dChcbiAgZXhlY1Jlc3VsdDogRXhlY1Jlc3VsdCxcbiAgdGltZW91dE1zOiBudW1iZXJcbik6IFByb21pc2U8eyBzdGRvdXQ6IHN0cmluZzsgc3RkZXJyOiBzdHJpbmc7IGV4aXRDb2RlOiBudW1iZXIgfSB8IG51bGw+IHtcbiAgY29uc3QgVElNRURfT1VUID0gU3ltYm9sKFwidGltZW91dFwiKTtcbiAgY29uc3QgdGltZXIgPSBuZXcgUHJvbWlzZTx0eXBlb2YgVElNRURfT1VUPigocmVzb2x2ZSkgPT5cbiAgICBzZXRUaW1lb3V0KCgpID0+IHJlc29sdmUoVElNRURfT1VUKSwgdGltZW91dE1zKVxuICApO1xuICBjb25zdCB3aW5uZXIgPSBhd2FpdCBQcm9taXNlLnJhY2UoW2V4ZWNSZXN1bHQucmVzdWx0LCB0aW1lcl0pO1xuICByZXR1cm4gd2lubmVyID09PSBUSU1FRF9PVVQgPyBudWxsIDogd2lubmVyO1xufVxuXG5mdW5jdGlvbiB3cml0ZU91dHB1dERpcihcbiAgc2FuZGJveDogU2FuZGJveEhhbmRsZSxcbiAgZGlyOiBzdHJpbmcsXG4gIHJlc3VsdDogeyBzdGRvdXQ6IHN0cmluZzsgc3RkZXJyOiBzdHJpbmc7IGV4aXRDb2RlOiBudW1iZXIgfSxcbiAgY29tbWFuZDogc3RyaW5nXG4pOiB2b2lkIHtcbiAgY29uc3QgZmlsZXM6IHsgcGF0aDogc3RyaW5nOyBjb250ZW50OiBzdHJpbmcgfVtdID0gW1xuICAgIHsgcGF0aDogXCJzdGRvdXQudHh0XCIsIGNvbnRlbnQ6IHJlc3VsdC5zdGRvdXQgfSxcbiAgICB7XG4gICAgICBwYXRoOiBcIm1ldGFkYXRhLmpzb25cIixcbiAgICAgIGNvbnRlbnQ6IEpTT04uc3RyaW5naWZ5KHsgZXhpdENvZGU6IHJlc3VsdC5leGl0Q29kZSwgY29tbWFuZCB9LCBudWxsLCAyKSxcbiAgICB9LFxuICBdO1xuICBpZiAocmVzdWx0LnN0ZGVycikge1xuICAgIGZpbGVzLnB1c2goeyBwYXRoOiBcInN0ZGVyci50eHRcIiwgY29udGVudDogcmVzdWx0LnN0ZGVyciB9KTtcbiAgfVxuXG4gIHNhbmRib3gud3JpdGVGaWxlcyh7IGZpbGVzLCBkZXN0UGF0aDogZGlyIH0pLmNhdGNoKCgpID0+IHtcbiAgICAvLyBpbnRlbnRpb25hbCBuby1vcCBcdTIwMTQgb3V0cHV0IGZpbGUgd3JpdGUgaXMgYmVzdC1lZmZvcnRcbiAgfSk7XG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBRU8sSUFBTSxxQkFBcUI7QUF3QjNCLElBQU0sdUJBQXVCLENBQUMsU0FBcUM7QUFDeEUsUUFBTSxFQUFFLFFBQVEsSUFBSTtBQUVwQixRQUFNLE1BQU0sT0FBTyxZQUE2QztBQUM5RCxVQUFNLEVBQUUsU0FBUyxZQUFZLG1CQUFtQixJQUFJO0FBRXBELFVBQU0sYUFBYSxNQUFNLFFBQVEsS0FBSztBQUFBLE1BQ3BDLFNBQVM7QUFBQSxNQUNULE1BQU0sQ0FBQyxNQUFNLE9BQU87QUFBQSxJQUN0QixDQUFDO0FBRUQsUUFBSSxjQUFjLEdBQUc7QUFDbkIsYUFBTztBQUFBLFFBQ0wsV0FBVyxXQUFXO0FBQUEsUUFDdEIsUUFBUTtBQUFBLFFBQ1IsUUFBUTtBQUFBLFFBQ1IsVUFBVTtBQUFBLFFBQ1YsUUFBUTtBQUFBLFFBQ1IsV0FBVztBQUFBLE1BQ2I7QUFBQSxJQUNGO0FBRUEsVUFBTSxZQUFZLE1BQU0sWUFBWSxZQUFZLFNBQVM7QUFFekQsUUFBSSxDQUFDLFdBQVc7QUFDZCxhQUFPO0FBQUEsUUFDTCxXQUFXLFdBQVc7QUFBQSxRQUN0QixRQUFRO0FBQUEsUUFDUixRQUFRO0FBQUEsUUFDUixVQUFVO0FBQUEsUUFDVixRQUFRO0FBQUEsUUFDUixXQUFXO0FBQUEsTUFDYjtBQUFBLElBQ0Y7QUFFQSxVQUFNLFlBQVksZUFBZSxLQUFLLElBQUksQ0FBQyxJQUFJLFdBQVcsU0FBUztBQUNuRSxtQkFBZSxTQUFTLFdBQVcsV0FBVyxPQUFPO0FBRXJELFdBQU87QUFBQSxNQUNMLFdBQVcsV0FBVztBQUFBLE1BQ3RCLFFBQVEsVUFBVTtBQUFBLE1BQ2xCLFFBQVEsVUFBVTtBQUFBLE1BQ2xCLFVBQVUsVUFBVTtBQUFBLE1BQ3BCLFFBQVEsVUFBVSxhQUFhLElBQUksY0FBYztBQUFBLE1BQ2pEO0FBQUEsSUFDRjtBQUFBLEVBQ0Y7QUFFQSxTQUFPLEVBQUUsSUFBSTtBQUNmO0FBRUEsZUFBZSxZQUNiLFlBQ0EsV0FDc0U7QUFDdEUsUUFBTSxZQUFZLHVCQUFPLFNBQVM7QUFDbEMsUUFBTSxRQUFRLElBQUk7QUFBQSxJQUEwQixDQUFDLFlBQzNDLFdBQVcsTUFBTSxRQUFRLFNBQVMsR0FBRyxTQUFTO0FBQUEsRUFDaEQ7QUFDQSxRQUFNLFNBQVMsTUFBTSxRQUFRLEtBQUssQ0FBQyxXQUFXLFFBQVEsS0FBSyxDQUFDO0FBQzVELFNBQU8sV0FBVyxZQUFZLE9BQU87QUFDdkM7QUFFQSxTQUFTLGVBQ1AsU0FDQSxLQUNBLFFBQ0EsU0FDTTtBQUNOLFFBQU0sUUFBNkM7QUFBQSxJQUNqRCxFQUFFLE1BQU0sY0FBYyxTQUFTLE9BQU8sT0FBTztBQUFBLElBQzdDO0FBQUEsTUFDRSxNQUFNO0FBQUEsTUFDTixTQUFTLEtBQUssVUFBVSxFQUFFLFVBQVUsT0FBTyxVQUFVLFFBQVEsR0FBRyxNQUFNLENBQUM7QUFBQSxJQUN6RTtBQUFBLEVBQ0Y7QUFDQSxNQUFJLE9BQU8sUUFBUTtBQUNqQixVQUFNLEtBQUssRUFBRSxNQUFNLGNBQWMsU0FBUyxPQUFPLE9BQU8sQ0FBQztBQUFBLEVBQzNEO0FBRUEsVUFBUSxXQUFXLEVBQUUsT0FBTyxVQUFVLElBQUksQ0FBQyxFQUFFLE1BQU0sTUFBTTtBQUFBLEVBRXpELENBQUM7QUFDSDsiLAogICJuYW1lcyI6IFtdCn0K
|