zeitlich 0.2.13 → 0.2.14
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 +49 -38
- package/dist/adapters/sandbox/daytona/index.cjs +205 -0
- package/dist/adapters/sandbox/daytona/index.cjs.map +1 -0
- package/dist/adapters/sandbox/daytona/index.d.cts +86 -0
- package/dist/adapters/sandbox/daytona/index.d.ts +86 -0
- package/dist/adapters/sandbox/daytona/index.js +202 -0
- package/dist/adapters/sandbox/daytona/index.js.map +1 -0
- package/dist/adapters/sandbox/inmemory/index.cjs +174 -0
- package/dist/adapters/sandbox/inmemory/index.cjs.map +1 -0
- package/dist/adapters/sandbox/inmemory/index.d.cts +28 -0
- package/dist/adapters/sandbox/inmemory/index.d.ts +28 -0
- package/dist/adapters/sandbox/inmemory/index.js +172 -0
- package/dist/adapters/sandbox/inmemory/index.js.map +1 -0
- package/dist/adapters/sandbox/virtual/index.cjs +405 -0
- package/dist/adapters/sandbox/virtual/index.cjs.map +1 -0
- package/dist/adapters/sandbox/virtual/index.d.cts +85 -0
- package/dist/adapters/sandbox/virtual/index.d.ts +85 -0
- package/dist/adapters/sandbox/virtual/index.js +400 -0
- package/dist/adapters/sandbox/virtual/index.js.map +1 -0
- package/dist/adapters/thread/google-genai/index.cjs +284 -0
- package/dist/adapters/thread/google-genai/index.cjs.map +1 -0
- package/dist/adapters/thread/google-genai/index.d.cts +145 -0
- package/dist/adapters/thread/google-genai/index.d.ts +145 -0
- package/dist/adapters/thread/google-genai/index.js +278 -0
- package/dist/adapters/thread/google-genai/index.js.map +1 -0
- package/dist/adapters/{langchain → thread/langchain}/index.cjs +7 -9
- package/dist/adapters/thread/langchain/index.cjs.map +1 -0
- package/dist/adapters/{langchain → thread/langchain}/index.d.cts +17 -21
- package/dist/adapters/{langchain → thread/langchain}/index.d.ts +17 -21
- package/dist/adapters/{langchain → thread/langchain}/index.js +7 -9
- package/dist/adapters/thread/langchain/index.js.map +1 -0
- package/dist/index.cjs +816 -545
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +235 -74
- package/dist/index.d.ts +235 -74
- package/dist/index.js +804 -540
- package/dist/index.js.map +1 -1
- package/dist/types-B4C9txdq.d.ts +389 -0
- package/dist/{thread-manager-qc0g5Rvd.d.cts → types-B9ljZewB.d.cts} +1 -6
- package/dist/{thread-manager-qc0g5Rvd.d.ts → types-B9ljZewB.d.ts} +1 -6
- package/dist/types-BMXzv7TN.d.cts +476 -0
- package/dist/types-BMXzv7TN.d.ts +476 -0
- package/dist/types-BVP87m_W.d.cts +121 -0
- package/dist/types-CDubRtad.d.cts +115 -0
- package/dist/types-CDubRtad.d.ts +115 -0
- package/dist/types-CwwgQ_9H.d.ts +121 -0
- package/dist/types-GpMU4b0w.d.cts +389 -0
- package/dist/workflow.cjs +444 -318
- package/dist/workflow.cjs.map +1 -1
- package/dist/workflow.d.cts +271 -222
- package/dist/workflow.d.ts +271 -222
- package/dist/workflow.js +440 -316
- package/dist/workflow.js.map +1 -1
- package/package.json +59 -6
- package/src/adapters/sandbox/daytona/filesystem.ts +136 -0
- package/src/adapters/sandbox/daytona/index.ts +149 -0
- package/src/adapters/sandbox/daytona/types.ts +34 -0
- package/src/adapters/sandbox/inmemory/index.ts +213 -0
- package/src/adapters/sandbox/virtual/filesystem.ts +345 -0
- package/src/adapters/sandbox/virtual/index.ts +88 -0
- package/src/adapters/sandbox/virtual/mutations.ts +38 -0
- package/src/adapters/sandbox/virtual/provider.ts +101 -0
- package/src/adapters/sandbox/virtual/tree.ts +82 -0
- package/src/adapters/sandbox/virtual/types.ts +127 -0
- package/src/adapters/sandbox/virtual/virtual-sandbox.test.ts +523 -0
- package/src/adapters/sandbox/virtual/with-virtual-sandbox.ts +91 -0
- package/src/adapters/thread/google-genai/activities.ts +121 -0
- package/src/adapters/thread/google-genai/index.ts +41 -0
- package/src/adapters/thread/google-genai/model-invoker.ts +154 -0
- package/src/adapters/thread/google-genai/thread-manager.ts +169 -0
- package/src/adapters/{langchain → thread/langchain}/activities.ts +11 -15
- package/src/adapters/{langchain → thread/langchain}/index.ts +1 -1
- package/src/adapters/{langchain → thread/langchain}/model-invoker.ts +15 -18
- package/src/adapters/{langchain → thread/langchain}/thread-manager.ts +1 -1
- package/src/index.ts +32 -24
- package/src/lib/activity.ts +87 -0
- package/src/lib/hooks/index.ts +11 -0
- package/src/lib/hooks/types.ts +98 -0
- package/src/lib/model/helpers.ts +6 -0
- package/src/lib/model/index.ts +13 -0
- package/src/lib/{model-invoker.ts → model/types.ts} +18 -1
- package/src/lib/sandbox/index.ts +19 -0
- package/src/lib/sandbox/manager.ts +76 -0
- package/src/lib/sandbox/sandbox.test.ts +158 -0
- package/src/lib/{fs.ts → sandbox/tree.ts} +6 -6
- package/src/lib/sandbox/types.ts +164 -0
- package/src/lib/session/index.ts +11 -0
- package/src/lib/{session.ts → session/session.ts} +76 -48
- package/src/lib/session/types.ts +93 -0
- package/src/lib/skills/fs-provider.ts +16 -15
- package/src/lib/skills/handler.ts +31 -0
- package/src/lib/skills/index.ts +5 -1
- package/src/lib/skills/register.ts +20 -0
- package/src/lib/skills/tool.ts +47 -0
- package/src/lib/state/index.ts +9 -0
- package/src/lib/{state-manager.ts → state/manager.ts} +10 -147
- package/src/lib/state/types.ts +134 -0
- package/src/lib/subagent/define.ts +71 -0
- package/src/lib/subagent/handler.ts +99 -0
- package/src/lib/subagent/index.ts +13 -0
- package/src/lib/subagent/register.ts +53 -0
- package/src/lib/subagent/tool.ts +80 -0
- package/src/lib/subagent/types.ts +92 -0
- package/src/lib/thread/index.ts +7 -0
- package/src/lib/{thread-manager.ts → thread/manager.ts} +1 -33
- package/src/lib/thread/types.ts +33 -0
- package/src/lib/tool-router/auto-append.ts +55 -0
- package/src/lib/tool-router/index.ts +41 -0
- package/src/lib/tool-router/router.ts +462 -0
- package/src/lib/tool-router/types.ts +478 -0
- package/src/lib/tool-router/with-sandbox.ts +70 -0
- package/src/lib/types.ts +5 -382
- package/src/tools/bash/bash.test.ts +53 -55
- package/src/tools/bash/handler.ts +23 -51
- package/src/tools/edit/handler.ts +67 -81
- package/src/tools/glob/handler.ts +60 -17
- package/src/tools/read-file/handler.ts +67 -0
- package/src/tools/read-skill/handler.ts +1 -31
- package/src/tools/read-skill/tool.ts +5 -47
- package/src/tools/subagent/handler.ts +1 -100
- package/src/tools/subagent/tool.ts +5 -93
- package/src/tools/task-create/handler.ts +1 -1
- package/src/tools/task-get/handler.ts +1 -1
- package/src/tools/task-list/handler.ts +1 -1
- package/src/tools/task-update/handler.ts +1 -1
- package/src/tools/write-file/handler.ts +47 -0
- package/src/workflow.ts +88 -47
- package/tsup.config.ts +8 -1
- package/dist/adapters/langchain/index.cjs.map +0 -1
- package/dist/adapters/langchain/index.js.map +0 -1
- package/dist/model-invoker-y_zlyMqu.d.cts +0 -892
- package/dist/model-invoker-y_zlyMqu.d.ts +0 -892
- package/src/lib/tool-router.ts +0 -977
- package/src/lib/workflow-helpers.ts +0 -50
- /package/src/lib/{thread-id.ts → thread/id.ts} +0 -0
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import { Daytona } from '@daytonaio/sdk';
|
|
2
|
+
import { ApplicationFailure } from '@temporalio/common';
|
|
3
|
+
import { posix } from 'path';
|
|
4
|
+
|
|
5
|
+
// src/adapters/sandbox/daytona/index.ts
|
|
6
|
+
var SandboxNotSupportedError = class extends ApplicationFailure {
|
|
7
|
+
constructor(operation) {
|
|
8
|
+
super(
|
|
9
|
+
`Sandbox does not support: ${operation}`,
|
|
10
|
+
"SandboxNotSupportedError",
|
|
11
|
+
true
|
|
12
|
+
);
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
var SandboxNotFoundError = class extends ApplicationFailure {
|
|
16
|
+
constructor(sandboxId) {
|
|
17
|
+
super(`Sandbox not found: ${sandboxId}`, "SandboxNotFoundError", true);
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
var DaytonaSandboxFileSystem = class {
|
|
21
|
+
constructor(sandbox) {
|
|
22
|
+
this.sandbox = sandbox;
|
|
23
|
+
}
|
|
24
|
+
async readFile(path) {
|
|
25
|
+
const buf = await this.sandbox.fs.downloadFile(path);
|
|
26
|
+
return buf.toString("utf-8");
|
|
27
|
+
}
|
|
28
|
+
async readFileBuffer(path) {
|
|
29
|
+
const buf = await this.sandbox.fs.downloadFile(path);
|
|
30
|
+
return new Uint8Array(buf.buffer, buf.byteOffset, buf.byteLength);
|
|
31
|
+
}
|
|
32
|
+
async writeFile(path, content) {
|
|
33
|
+
const buf = typeof content === "string" ? Buffer.from(content, "utf-8") : Buffer.from(content);
|
|
34
|
+
await this.sandbox.fs.uploadFile(buf, path);
|
|
35
|
+
}
|
|
36
|
+
async appendFile(path, content) {
|
|
37
|
+
let existing;
|
|
38
|
+
try {
|
|
39
|
+
existing = await this.sandbox.fs.downloadFile(path);
|
|
40
|
+
} catch {
|
|
41
|
+
return this.writeFile(path, content);
|
|
42
|
+
}
|
|
43
|
+
const addition = typeof content === "string" ? Buffer.from(content, "utf-8") : content;
|
|
44
|
+
const merged = Buffer.concat([existing, Buffer.from(addition)]);
|
|
45
|
+
await this.sandbox.fs.uploadFile(merged, path);
|
|
46
|
+
}
|
|
47
|
+
async exists(path) {
|
|
48
|
+
try {
|
|
49
|
+
await this.sandbox.fs.getFileDetails(path);
|
|
50
|
+
return true;
|
|
51
|
+
} catch {
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
async stat(path) {
|
|
56
|
+
const info = await this.sandbox.fs.getFileDetails(path);
|
|
57
|
+
return {
|
|
58
|
+
isFile: !info.isDir,
|
|
59
|
+
isDirectory: info.isDir,
|
|
60
|
+
isSymbolicLink: false,
|
|
61
|
+
size: info.size,
|
|
62
|
+
mtime: new Date(info.modTime)
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
async mkdir(path, _options) {
|
|
66
|
+
await this.sandbox.fs.createFolder(path, "755");
|
|
67
|
+
}
|
|
68
|
+
async readdir(path) {
|
|
69
|
+
const entries = await this.sandbox.fs.listFiles(path);
|
|
70
|
+
return entries.map((e) => e.name);
|
|
71
|
+
}
|
|
72
|
+
async readdirWithFileTypes(path) {
|
|
73
|
+
const entries = await this.sandbox.fs.listFiles(path);
|
|
74
|
+
return entries.map((e) => ({
|
|
75
|
+
name: e.name,
|
|
76
|
+
isFile: !e.isDir,
|
|
77
|
+
isDirectory: e.isDir,
|
|
78
|
+
isSymbolicLink: false
|
|
79
|
+
}));
|
|
80
|
+
}
|
|
81
|
+
async rm(path, options) {
|
|
82
|
+
try {
|
|
83
|
+
await this.sandbox.fs.deleteFile(path, options?.recursive);
|
|
84
|
+
} catch (err) {
|
|
85
|
+
if (!options?.force) throw err;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
async cp(src, dest, options) {
|
|
89
|
+
const info = await this.sandbox.fs.getFileDetails(src);
|
|
90
|
+
if (info.isDir) {
|
|
91
|
+
if (!options?.recursive) {
|
|
92
|
+
throw new Error(`EISDIR: is a directory (use recursive): ${src}`);
|
|
93
|
+
}
|
|
94
|
+
await this.sandbox.process.executeCommand(`cp -r "${src}" "${dest}"`);
|
|
95
|
+
} else {
|
|
96
|
+
await this.sandbox.process.executeCommand(`cp "${src}" "${dest}"`);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
async mv(src, dest) {
|
|
100
|
+
await this.sandbox.fs.moveFiles(src, dest);
|
|
101
|
+
}
|
|
102
|
+
async readlink(_path) {
|
|
103
|
+
throw new SandboxNotSupportedError("readlink");
|
|
104
|
+
}
|
|
105
|
+
resolvePath(base, path) {
|
|
106
|
+
if (posix.isAbsolute(path)) return posix.normalize(path);
|
|
107
|
+
return posix.resolve(base, path);
|
|
108
|
+
}
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
// src/adapters/sandbox/daytona/index.ts
|
|
112
|
+
var DaytonaSandboxImpl = class {
|
|
113
|
+
constructor(id, sdkSandbox) {
|
|
114
|
+
this.id = id;
|
|
115
|
+
this.sdkSandbox = sdkSandbox;
|
|
116
|
+
this.fs = new DaytonaSandboxFileSystem(sdkSandbox);
|
|
117
|
+
}
|
|
118
|
+
capabilities = {
|
|
119
|
+
filesystem: true,
|
|
120
|
+
execution: true,
|
|
121
|
+
persistence: false
|
|
122
|
+
};
|
|
123
|
+
fs;
|
|
124
|
+
async exec(command, options) {
|
|
125
|
+
const response = await this.sdkSandbox.process.executeCommand(
|
|
126
|
+
command,
|
|
127
|
+
options?.cwd,
|
|
128
|
+
options?.env,
|
|
129
|
+
options?.timeout
|
|
130
|
+
);
|
|
131
|
+
return {
|
|
132
|
+
exitCode: response.exitCode ?? 0,
|
|
133
|
+
stdout: response.result ?? "",
|
|
134
|
+
stderr: ""
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
async destroy() {
|
|
138
|
+
await this.sdkSandbox.delete(60);
|
|
139
|
+
}
|
|
140
|
+
};
|
|
141
|
+
var DaytonaSandboxProvider = class {
|
|
142
|
+
id = "daytona";
|
|
143
|
+
capabilities = {
|
|
144
|
+
filesystem: true,
|
|
145
|
+
execution: true,
|
|
146
|
+
persistence: false
|
|
147
|
+
};
|
|
148
|
+
client;
|
|
149
|
+
constructor(config) {
|
|
150
|
+
this.client = new Daytona(config);
|
|
151
|
+
}
|
|
152
|
+
async create(options) {
|
|
153
|
+
const sdkSandbox = await this.client.create(
|
|
154
|
+
{
|
|
155
|
+
language: options?.language,
|
|
156
|
+
snapshot: options?.snapshot,
|
|
157
|
+
envVars: options?.env,
|
|
158
|
+
labels: options?.labels,
|
|
159
|
+
autoStopInterval: options?.autoStopInterval,
|
|
160
|
+
autoArchiveInterval: options?.autoArchiveInterval,
|
|
161
|
+
autoDeleteInterval: options?.autoDeleteInterval
|
|
162
|
+
},
|
|
163
|
+
{ timeout: options?.timeout ?? 60 }
|
|
164
|
+
);
|
|
165
|
+
const sandbox = new DaytonaSandboxImpl(sdkSandbox.id, sdkSandbox);
|
|
166
|
+
if (options?.initialFiles) {
|
|
167
|
+
for (const [path, content] of Object.entries(options.initialFiles)) {
|
|
168
|
+
await sandbox.fs.writeFile(path, content);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
return { sandbox };
|
|
172
|
+
}
|
|
173
|
+
async get(sandboxId) {
|
|
174
|
+
try {
|
|
175
|
+
const sdkSandbox = await this.client.get(sandboxId);
|
|
176
|
+
return new DaytonaSandboxImpl(sdkSandbox.id, sdkSandbox);
|
|
177
|
+
} catch {
|
|
178
|
+
throw new SandboxNotFoundError(sandboxId);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
async destroy(sandboxId) {
|
|
182
|
+
try {
|
|
183
|
+
const sdkSandbox = await this.client.get(sandboxId);
|
|
184
|
+
await this.client.delete(sdkSandbox);
|
|
185
|
+
} catch {
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
async snapshot(_sandboxId) {
|
|
189
|
+
throw new SandboxNotSupportedError(
|
|
190
|
+
"snapshot (use Daytona's native snapshot API directly)"
|
|
191
|
+
);
|
|
192
|
+
}
|
|
193
|
+
async restore(_snapshot) {
|
|
194
|
+
throw new SandboxNotSupportedError(
|
|
195
|
+
"restore (use Daytona's native snapshot API directly)"
|
|
196
|
+
);
|
|
197
|
+
}
|
|
198
|
+
};
|
|
199
|
+
|
|
200
|
+
export { DaytonaSandboxFileSystem, DaytonaSandboxProvider };
|
|
201
|
+
//# sourceMappingURL=index.js.map
|
|
202
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/lib/sandbox/types.ts","../../../../src/adapters/sandbox/daytona/filesystem.ts","../../../../src/adapters/sandbox/daytona/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["// ============================================================================\n// Sandbox Filesystem\n// ============================================================================\n\nexport interface DirentEntry {\n name: string;\n isFile: boolean;\n isDirectory: boolean;\n isSymbolicLink: boolean;\n}\n\nexport interface FileStat {\n isFile: boolean;\n isDirectory: boolean;\n isSymbolicLink: boolean;\n size: number;\n mtime: Date;\n}\n\n/**\n * Provider-agnostic filesystem interface.\n *\n * Implementations that don't support a method should throw\n * {@link SandboxNotSupportedError}.\n */\nexport interface SandboxFileSystem {\n readFile(path: string): Promise<string>;\n readFileBuffer(path: string): Promise<Uint8Array>;\n writeFile(path: string, content: string | Uint8Array): Promise<void>;\n appendFile(path: string, content: string | Uint8Array): Promise<void>;\n exists(path: string): Promise<boolean>;\n stat(path: string): Promise<FileStat>;\n mkdir(path: string, options?: { recursive?: boolean }): Promise<void>;\n readdir(path: string): Promise<string[]>;\n readdirWithFileTypes(path: string): Promise<DirentEntry[]>;\n rm(path: string, options?: { recursive?: boolean; force?: boolean }): Promise<void>;\n cp(src: string, dest: string, options?: { recursive?: boolean }): Promise<void>;\n mv(src: string, dest: string): Promise<void>;\n readlink(path: string): Promise<string>;\n resolvePath(base: string, path: string): string;\n}\n\n// ============================================================================\n// Execution\n// ============================================================================\n\nexport interface ExecOptions {\n timeout?: number;\n cwd?: string;\n env?: Record<string, string>;\n}\n\nexport interface ExecResult {\n exitCode: number;\n stdout: string;\n stderr: string;\n}\n\n// ============================================================================\n// Capabilities\n// ============================================================================\n\nexport interface SandboxCapabilities {\n /** Sandbox supports filesystem operations */\n filesystem: boolean;\n /** Sandbox supports shell/command execution */\n execution: boolean;\n /** Sandbox state can be persisted and restored */\n persistence: boolean;\n}\n\n// ============================================================================\n// Sandbox\n// ============================================================================\n\nexport interface Sandbox {\n readonly id: string;\n readonly capabilities: SandboxCapabilities;\n readonly fs: SandboxFileSystem;\n\n exec(command: string, options?: ExecOptions): Promise<ExecResult>;\n destroy(): Promise<void>;\n}\n\n// ============================================================================\n// Snapshots\n// ============================================================================\n\nexport interface SandboxSnapshot {\n sandboxId: string;\n providerId: string;\n /** Provider-specific serialised state */\n data: unknown;\n createdAt: string;\n}\n\n// ============================================================================\n// Provider\n// ============================================================================\n\nexport interface SandboxCreateOptions {\n /** Preferred sandbox ID (provider may ignore) */\n id?: string;\n /** Seed the filesystem with these files */\n initialFiles?: Record<string, string | Uint8Array>;\n /** Environment variables available inside the sandbox */\n env?: Record<string, string>;\n}\n\nexport interface SandboxCreateResult {\n sandbox: Sandbox;\n /** Optional state to merge into the workflow's `AgentState` via the session. */\n stateUpdate?: Record<string, unknown>;\n}\n\nexport interface SandboxProvider<\n TOptions extends SandboxCreateOptions = SandboxCreateOptions,\n TSandbox extends Sandbox = Sandbox,\n> {\n readonly id: string;\n readonly capabilities: SandboxCapabilities;\n\n create(options?: TOptions): Promise<SandboxCreateResult>;\n get(sandboxId: string): Promise<TSandbox>;\n destroy(sandboxId: string): Promise<void>;\n snapshot(sandboxId: string): Promise<SandboxSnapshot>;\n restore(snapshot: SandboxSnapshot): Promise<Sandbox>;\n}\n\n// ============================================================================\n// SandboxOps — workflow-side activity interface (like ThreadOps)\n// ============================================================================\n\nexport interface SandboxOps<\n TOptions extends SandboxCreateOptions = SandboxCreateOptions,\n> {\n createSandbox(\n options?: TOptions,\n ): Promise<{ sandboxId: string; stateUpdate?: Record<string, unknown> }>;\n destroySandbox(sandboxId: string): Promise<void>;\n snapshotSandbox(sandboxId: string): Promise<SandboxSnapshot>;\n}\n\n// ============================================================================\n// Errors\n// ============================================================================\n\nimport { ApplicationFailure } from \"@temporalio/common\";\n\nexport class SandboxNotSupportedError extends ApplicationFailure {\n constructor(operation: string) {\n super(\n `Sandbox does not support: ${operation}`,\n \"SandboxNotSupportedError\",\n true,\n );\n }\n}\n\nexport class SandboxNotFoundError extends ApplicationFailure {\n constructor(sandboxId: string) {\n super(`Sandbox not found: ${sandboxId}`, \"SandboxNotFoundError\", true);\n }\n}\n","import type { Sandbox as DaytonaSdkSandbox } from \"@daytonaio/sdk\";\nimport type {\n SandboxFileSystem,\n DirentEntry,\n FileStat,\n} from \"../../../lib/sandbox/types\";\nimport { SandboxNotSupportedError } from \"../../../lib/sandbox/types\";\nimport { posix } from \"node:path\";\n\n/**\n * {@link SandboxFileSystem} backed by a Daytona SDK sandbox.\n *\n * Maps zeitlich's filesystem interface to Daytona's `sandbox.fs` and\n * `sandbox.process` APIs. Operations that have no direct Daytona equivalent\n * (e.g. `appendFile`, `cp`) are composed from primitives.\n */\nexport class DaytonaSandboxFileSystem implements SandboxFileSystem {\n constructor(private sandbox: DaytonaSdkSandbox) {}\n\n async readFile(path: string): Promise<string> {\n const buf = await this.sandbox.fs.downloadFile(path);\n return buf.toString(\"utf-8\");\n }\n\n async readFileBuffer(path: string): Promise<Uint8Array> {\n const buf = await this.sandbox.fs.downloadFile(path);\n return new Uint8Array(buf.buffer, buf.byteOffset, buf.byteLength);\n }\n\n async writeFile(path: string, content: string | Uint8Array): Promise<void> {\n const buf =\n typeof content === \"string\"\n ? Buffer.from(content, \"utf-8\")\n : Buffer.from(content);\n await this.sandbox.fs.uploadFile(buf, path);\n }\n\n async appendFile(\n path: string,\n content: string | Uint8Array,\n ): Promise<void> {\n let existing: Buffer;\n try {\n existing = await this.sandbox.fs.downloadFile(path);\n } catch {\n return this.writeFile(path, content);\n }\n\n const addition =\n typeof content === \"string\" ? Buffer.from(content, \"utf-8\") : content;\n const merged = Buffer.concat([existing, Buffer.from(addition)]);\n await this.sandbox.fs.uploadFile(merged, path);\n }\n\n async exists(path: string): Promise<boolean> {\n try {\n await this.sandbox.fs.getFileDetails(path);\n return true;\n } catch {\n return false;\n }\n }\n\n async stat(path: string): Promise<FileStat> {\n const info = await this.sandbox.fs.getFileDetails(path);\n return {\n isFile: !info.isDir,\n isDirectory: info.isDir,\n isSymbolicLink: false,\n size: info.size,\n mtime: new Date(info.modTime),\n };\n }\n\n async mkdir(\n path: string,\n _options?: { recursive?: boolean },\n ): Promise<void> {\n await this.sandbox.fs.createFolder(path, \"755\");\n }\n\n async readdir(path: string): Promise<string[]> {\n const entries = await this.sandbox.fs.listFiles(path);\n return entries.map((e) => e.name);\n }\n\n async readdirWithFileTypes(path: string): Promise<DirentEntry[]> {\n const entries = await this.sandbox.fs.listFiles(path);\n return entries.map((e) => ({\n name: e.name,\n isFile: !e.isDir,\n isDirectory: e.isDir,\n isSymbolicLink: false,\n }));\n }\n\n async rm(\n path: string,\n options?: { recursive?: boolean; force?: boolean },\n ): Promise<void> {\n try {\n await this.sandbox.fs.deleteFile(path, options?.recursive);\n } catch (err) {\n if (!options?.force) throw err;\n }\n }\n\n async cp(\n src: string,\n dest: string,\n options?: { recursive?: boolean },\n ): Promise<void> {\n const info = await this.sandbox.fs.getFileDetails(src);\n if (info.isDir) {\n if (!options?.recursive) {\n throw new Error(`EISDIR: is a directory (use recursive): ${src}`);\n }\n await this.sandbox.process.executeCommand(`cp -r \"${src}\" \"${dest}\"`);\n } else {\n await this.sandbox.process.executeCommand(`cp \"${src}\" \"${dest}\"`);\n }\n }\n\n async mv(src: string, dest: string): Promise<void> {\n await this.sandbox.fs.moveFiles(src, dest);\n }\n\n async readlink(_path: string): Promise<string> {\n throw new SandboxNotSupportedError(\"readlink\");\n }\n\n resolvePath(base: string, path: string): string {\n if (posix.isAbsolute(path)) return posix.normalize(path);\n return posix.resolve(base, path);\n }\n}\n","import {\n Daytona,\n type Sandbox as DaytonaSdkSandbox,\n} from \"@daytonaio/sdk\";\nimport type {\n Sandbox,\n SandboxCapabilities,\n SandboxCreateResult,\n SandboxProvider,\n SandboxSnapshot,\n ExecOptions,\n ExecResult,\n} from \"../../../lib/sandbox/types\";\nimport {\n SandboxNotFoundError,\n SandboxNotSupportedError,\n} from \"../../../lib/sandbox/types\";\nimport { DaytonaSandboxFileSystem } from \"./filesystem\";\nimport type {\n DaytonaSandbox,\n DaytonaSandboxConfig,\n DaytonaSandboxCreateOptions,\n} from \"./types\";\n\n// ============================================================================\n// DaytonaSandbox\n// ============================================================================\n\nclass DaytonaSandboxImpl implements Sandbox {\n readonly capabilities: SandboxCapabilities = {\n filesystem: true,\n execution: true,\n persistence: false,\n };\n\n readonly fs: DaytonaSandboxFileSystem;\n\n constructor(\n readonly id: string,\n private sdkSandbox: DaytonaSdkSandbox,\n ) {\n this.fs = new DaytonaSandboxFileSystem(sdkSandbox);\n }\n\n async exec(command: string, options?: ExecOptions): Promise<ExecResult> {\n const response = await this.sdkSandbox.process.executeCommand(\n command,\n options?.cwd,\n options?.env,\n options?.timeout,\n );\n\n return {\n exitCode: response.exitCode ?? 0,\n stdout: response.result ?? \"\",\n stderr: \"\",\n };\n }\n\n async destroy(): Promise<void> {\n await this.sdkSandbox.delete(60);\n }\n}\n\n// ============================================================================\n// DaytonaSandboxProvider\n// ============================================================================\n\nexport class DaytonaSandboxProvider\n implements SandboxProvider<DaytonaSandboxCreateOptions, DaytonaSandbox>\n{\n readonly id = \"daytona\";\n readonly capabilities: SandboxCapabilities = {\n filesystem: true,\n execution: true,\n persistence: false,\n };\n\n private client: Daytona;\n\n constructor(config?: DaytonaSandboxConfig) {\n this.client = new Daytona(config);\n }\n\n async create(\n options?: DaytonaSandboxCreateOptions,\n ): Promise<SandboxCreateResult> {\n const sdkSandbox = await this.client.create(\n {\n language: options?.language,\n snapshot: options?.snapshot,\n envVars: options?.env,\n labels: options?.labels,\n autoStopInterval: options?.autoStopInterval,\n autoArchiveInterval: options?.autoArchiveInterval,\n autoDeleteInterval: options?.autoDeleteInterval,\n },\n { timeout: options?.timeout ?? 60 },\n );\n\n const sandbox = new DaytonaSandboxImpl(sdkSandbox.id, sdkSandbox);\n\n if (options?.initialFiles) {\n for (const [path, content] of Object.entries(options.initialFiles)) {\n await sandbox.fs.writeFile(path, content);\n }\n }\n\n return { sandbox };\n }\n\n async get(sandboxId: string): Promise<DaytonaSandbox> {\n try {\n const sdkSandbox = await this.client.get(sandboxId);\n return new DaytonaSandboxImpl(sdkSandbox.id, sdkSandbox);\n } catch {\n throw new SandboxNotFoundError(sandboxId);\n }\n }\n\n async destroy(sandboxId: string): Promise<void> {\n try {\n const sdkSandbox = await this.client.get(sandboxId);\n await this.client.delete(sdkSandbox);\n } catch {\n // Already gone\n }\n }\n\n async snapshot(_sandboxId: string): Promise<SandboxSnapshot> {\n throw new SandboxNotSupportedError(\n \"snapshot (use Daytona's native snapshot API directly)\",\n );\n }\n\n async restore(_snapshot: SandboxSnapshot): Promise<never> {\n throw new SandboxNotSupportedError(\n \"restore (use Daytona's native snapshot API directly)\",\n );\n }\n}\n\n// Re-exports\nexport { DaytonaSandboxFileSystem } from \"./filesystem\";\nexport type {\n DaytonaSandbox,\n DaytonaSandboxConfig,\n DaytonaSandboxCreateOptions,\n} from \"./types\";\n"]}
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var justBash = require('just-bash');
|
|
4
|
+
var common = require('@temporalio/common');
|
|
5
|
+
var workflow = require('@temporalio/workflow');
|
|
6
|
+
|
|
7
|
+
// src/adapters/sandbox/inmemory/index.ts
|
|
8
|
+
var SandboxNotFoundError = class extends common.ApplicationFailure {
|
|
9
|
+
constructor(sandboxId) {
|
|
10
|
+
super(`Sandbox not found: ${sandboxId}`, "SandboxNotFoundError", true);
|
|
11
|
+
}
|
|
12
|
+
};
|
|
13
|
+
var BASE62 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
14
|
+
function getShortId(length = 12) {
|
|
15
|
+
const hex = workflow.uuid4().replace(/-/g, "");
|
|
16
|
+
let result = "";
|
|
17
|
+
for (let i = 0; i < length; i++) {
|
|
18
|
+
const byte = parseInt(hex.slice(i * 2, i * 2 + 2), 16);
|
|
19
|
+
result += BASE62[byte % BASE62.length];
|
|
20
|
+
}
|
|
21
|
+
return result;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// src/adapters/sandbox/inmemory/index.ts
|
|
25
|
+
function toSandboxFs(fs) {
|
|
26
|
+
return {
|
|
27
|
+
readFile: (path) => fs.readFile(path),
|
|
28
|
+
readFileBuffer: (path) => fs.readFileBuffer(path),
|
|
29
|
+
writeFile: (path, content) => fs.writeFile(path, content),
|
|
30
|
+
appendFile: (path, content) => fs.appendFile(path, content),
|
|
31
|
+
exists: (path) => fs.exists(path),
|
|
32
|
+
stat: async (path) => {
|
|
33
|
+
const s = await fs.stat(path);
|
|
34
|
+
return {
|
|
35
|
+
isFile: s.isFile,
|
|
36
|
+
isDirectory: s.isDirectory,
|
|
37
|
+
isSymbolicLink: s.isSymbolicLink,
|
|
38
|
+
size: s.size,
|
|
39
|
+
mtime: s.mtime
|
|
40
|
+
};
|
|
41
|
+
},
|
|
42
|
+
mkdir: (path, opts) => fs.mkdir(path, opts),
|
|
43
|
+
readdir: (path) => fs.readdir(path),
|
|
44
|
+
readdirWithFileTypes: async (path) => {
|
|
45
|
+
if (!fs.readdirWithFileTypes) {
|
|
46
|
+
const names = await fs.readdir(path);
|
|
47
|
+
return Promise.all(
|
|
48
|
+
names.map(async (name) => {
|
|
49
|
+
const s = await fs.stat(`${path}/${name}`);
|
|
50
|
+
return {
|
|
51
|
+
name,
|
|
52
|
+
isFile: s.isFile,
|
|
53
|
+
isDirectory: s.isDirectory,
|
|
54
|
+
isSymbolicLink: s.isSymbolicLink
|
|
55
|
+
};
|
|
56
|
+
})
|
|
57
|
+
);
|
|
58
|
+
}
|
|
59
|
+
return fs.readdirWithFileTypes(path);
|
|
60
|
+
},
|
|
61
|
+
rm: (path, opts) => fs.rm(path, opts),
|
|
62
|
+
cp: (src, dest, opts) => fs.cp(src, dest, opts),
|
|
63
|
+
mv: (src, dest) => fs.mv(src, dest),
|
|
64
|
+
readlink: (path) => fs.readlink(path),
|
|
65
|
+
resolvePath: (base, p) => fs.resolvePath(base, p)
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
var InMemorySandboxImpl = class {
|
|
69
|
+
constructor(id, justBashFs, options) {
|
|
70
|
+
this.id = id;
|
|
71
|
+
this.justBashFs = justBashFs;
|
|
72
|
+
this.fs = toSandboxFs(justBashFs);
|
|
73
|
+
this.bashOptions = {
|
|
74
|
+
executionLimits: { maxStringLength: 52428800 },
|
|
75
|
+
...options?.bashOptions
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
capabilities = {
|
|
79
|
+
filesystem: true,
|
|
80
|
+
execution: true,
|
|
81
|
+
persistence: true
|
|
82
|
+
};
|
|
83
|
+
fs;
|
|
84
|
+
bashOptions;
|
|
85
|
+
async exec(command, _options) {
|
|
86
|
+
const bash = new justBash.Bash({ ...this.bashOptions, fs: this.justBashFs });
|
|
87
|
+
const { exitCode, stderr, stdout } = await bash.exec(command);
|
|
88
|
+
return { exitCode, stdout, stderr };
|
|
89
|
+
}
|
|
90
|
+
async destroy() {
|
|
91
|
+
}
|
|
92
|
+
/** Expose the underlying IFileSystem for snapshot serialisation */
|
|
93
|
+
_getJustBashFs() {
|
|
94
|
+
return this.justBashFs;
|
|
95
|
+
}
|
|
96
|
+
};
|
|
97
|
+
var InMemorySandboxProvider = class {
|
|
98
|
+
constructor(defaultOptions) {
|
|
99
|
+
this.defaultOptions = defaultOptions;
|
|
100
|
+
}
|
|
101
|
+
id = "inmemory";
|
|
102
|
+
capabilities = {
|
|
103
|
+
filesystem: true,
|
|
104
|
+
execution: true,
|
|
105
|
+
persistence: true
|
|
106
|
+
};
|
|
107
|
+
sandboxes = /* @__PURE__ */ new Map();
|
|
108
|
+
async get(id) {
|
|
109
|
+
const sandbox = this.sandboxes.get(id);
|
|
110
|
+
if (!sandbox) throw new SandboxNotFoundError(id);
|
|
111
|
+
return sandbox;
|
|
112
|
+
}
|
|
113
|
+
async destroy(id) {
|
|
114
|
+
const sandbox = this.sandboxes.get(id);
|
|
115
|
+
if (sandbox) {
|
|
116
|
+
await sandbox.destroy();
|
|
117
|
+
this.sandboxes.delete(id);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
async create(options) {
|
|
121
|
+
const id = options?.id ?? getShortId();
|
|
122
|
+
const initialFiles = {};
|
|
123
|
+
if (options?.initialFiles) {
|
|
124
|
+
for (const [path, content] of Object.entries(options.initialFiles)) {
|
|
125
|
+
initialFiles[path] = content;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
const fs = new justBash.InMemoryFs(initialFiles);
|
|
129
|
+
const sandbox = new InMemorySandboxImpl(id, fs, this.defaultOptions);
|
|
130
|
+
this.sandboxes.set(id, sandbox);
|
|
131
|
+
return { sandbox };
|
|
132
|
+
}
|
|
133
|
+
async snapshot(sandboxId) {
|
|
134
|
+
const sandbox = this.sandboxes.get(sandboxId);
|
|
135
|
+
if (!sandbox) throw new SandboxNotFoundError(sandboxId);
|
|
136
|
+
const fs = sandbox._getJustBashFs();
|
|
137
|
+
const paths = fs.getAllPaths();
|
|
138
|
+
const files = {};
|
|
139
|
+
for (const p of paths) {
|
|
140
|
+
try {
|
|
141
|
+
const stat = await fs.stat(p);
|
|
142
|
+
if (stat.isFile) {
|
|
143
|
+
files[p] = await fs.readFile(p);
|
|
144
|
+
}
|
|
145
|
+
} catch {
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
return {
|
|
149
|
+
sandboxId,
|
|
150
|
+
providerId: this.id,
|
|
151
|
+
data: { files },
|
|
152
|
+
createdAt: (/* @__PURE__ */ new Date()).toISOString()
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
async restore(snapshot) {
|
|
156
|
+
const { files } = snapshot.data;
|
|
157
|
+
const initialFiles = {};
|
|
158
|
+
for (const [path, content] of Object.entries(files)) {
|
|
159
|
+
initialFiles[path] = content;
|
|
160
|
+
}
|
|
161
|
+
const fs = new justBash.InMemoryFs(initialFiles);
|
|
162
|
+
const sandbox = new InMemorySandboxImpl(
|
|
163
|
+
snapshot.sandboxId,
|
|
164
|
+
fs,
|
|
165
|
+
this.defaultOptions
|
|
166
|
+
);
|
|
167
|
+
this.sandboxes.set(sandbox.id, sandbox);
|
|
168
|
+
return sandbox;
|
|
169
|
+
}
|
|
170
|
+
};
|
|
171
|
+
|
|
172
|
+
exports.InMemorySandboxProvider = InMemorySandboxProvider;
|
|
173
|
+
//# sourceMappingURL=index.cjs.map
|
|
174
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/lib/sandbox/types.ts","../../../../src/lib/thread/id.ts","../../../../src/adapters/sandbox/inmemory/index.ts"],"names":["ApplicationFailure","uuid4","Bash","InMemoryFs"],"mappings":"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","file":"index.cjs","sourcesContent":["// ============================================================================\n// Sandbox Filesystem\n// ============================================================================\n\nexport interface DirentEntry {\n name: string;\n isFile: boolean;\n isDirectory: boolean;\n isSymbolicLink: boolean;\n}\n\nexport interface FileStat {\n isFile: boolean;\n isDirectory: boolean;\n isSymbolicLink: boolean;\n size: number;\n mtime: Date;\n}\n\n/**\n * Provider-agnostic filesystem interface.\n *\n * Implementations that don't support a method should throw\n * {@link SandboxNotSupportedError}.\n */\nexport interface SandboxFileSystem {\n readFile(path: string): Promise<string>;\n readFileBuffer(path: string): Promise<Uint8Array>;\n writeFile(path: string, content: string | Uint8Array): Promise<void>;\n appendFile(path: string, content: string | Uint8Array): Promise<void>;\n exists(path: string): Promise<boolean>;\n stat(path: string): Promise<FileStat>;\n mkdir(path: string, options?: { recursive?: boolean }): Promise<void>;\n readdir(path: string): Promise<string[]>;\n readdirWithFileTypes(path: string): Promise<DirentEntry[]>;\n rm(path: string, options?: { recursive?: boolean; force?: boolean }): Promise<void>;\n cp(src: string, dest: string, options?: { recursive?: boolean }): Promise<void>;\n mv(src: string, dest: string): Promise<void>;\n readlink(path: string): Promise<string>;\n resolvePath(base: string, path: string): string;\n}\n\n// ============================================================================\n// Execution\n// ============================================================================\n\nexport interface ExecOptions {\n timeout?: number;\n cwd?: string;\n env?: Record<string, string>;\n}\n\nexport interface ExecResult {\n exitCode: number;\n stdout: string;\n stderr: string;\n}\n\n// ============================================================================\n// Capabilities\n// ============================================================================\n\nexport interface SandboxCapabilities {\n /** Sandbox supports filesystem operations */\n filesystem: boolean;\n /** Sandbox supports shell/command execution */\n execution: boolean;\n /** Sandbox state can be persisted and restored */\n persistence: boolean;\n}\n\n// ============================================================================\n// Sandbox\n// ============================================================================\n\nexport interface Sandbox {\n readonly id: string;\n readonly capabilities: SandboxCapabilities;\n readonly fs: SandboxFileSystem;\n\n exec(command: string, options?: ExecOptions): Promise<ExecResult>;\n destroy(): Promise<void>;\n}\n\n// ============================================================================\n// Snapshots\n// ============================================================================\n\nexport interface SandboxSnapshot {\n sandboxId: string;\n providerId: string;\n /** Provider-specific serialised state */\n data: unknown;\n createdAt: string;\n}\n\n// ============================================================================\n// Provider\n// ============================================================================\n\nexport interface SandboxCreateOptions {\n /** Preferred sandbox ID (provider may ignore) */\n id?: string;\n /** Seed the filesystem with these files */\n initialFiles?: Record<string, string | Uint8Array>;\n /** Environment variables available inside the sandbox */\n env?: Record<string, string>;\n}\n\nexport interface SandboxCreateResult {\n sandbox: Sandbox;\n /** Optional state to merge into the workflow's `AgentState` via the session. */\n stateUpdate?: Record<string, unknown>;\n}\n\nexport interface SandboxProvider<\n TOptions extends SandboxCreateOptions = SandboxCreateOptions,\n TSandbox extends Sandbox = Sandbox,\n> {\n readonly id: string;\n readonly capabilities: SandboxCapabilities;\n\n create(options?: TOptions): Promise<SandboxCreateResult>;\n get(sandboxId: string): Promise<TSandbox>;\n destroy(sandboxId: string): Promise<void>;\n snapshot(sandboxId: string): Promise<SandboxSnapshot>;\n restore(snapshot: SandboxSnapshot): Promise<Sandbox>;\n}\n\n// ============================================================================\n// SandboxOps — workflow-side activity interface (like ThreadOps)\n// ============================================================================\n\nexport interface SandboxOps<\n TOptions extends SandboxCreateOptions = SandboxCreateOptions,\n> {\n createSandbox(\n options?: TOptions,\n ): Promise<{ sandboxId: string; stateUpdate?: Record<string, unknown> }>;\n destroySandbox(sandboxId: string): Promise<void>;\n snapshotSandbox(sandboxId: string): Promise<SandboxSnapshot>;\n}\n\n// ============================================================================\n// Errors\n// ============================================================================\n\nimport { ApplicationFailure } from \"@temporalio/common\";\n\nexport class SandboxNotSupportedError extends ApplicationFailure {\n constructor(operation: string) {\n super(\n `Sandbox does not support: ${operation}`,\n \"SandboxNotSupportedError\",\n true,\n );\n }\n}\n\nexport class SandboxNotFoundError extends ApplicationFailure {\n constructor(sandboxId: string) {\n super(`Sandbox not found: ${sandboxId}`, \"SandboxNotFoundError\", true);\n }\n}\n","import { uuid4 } from \"@temporalio/workflow\";\n\nconst BASE62 =\n \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\";\n\n/**\n * Generate a compact, workflow-deterministic identifier.\n *\n * Uses Temporal's `uuid4()` internally (seeded by the workflow's RNG),\n * then re-encodes the hex bytes into a base-62 alphabet for a shorter,\n * more token-efficient identifier (~3 tokens vs ~10 for a full UUID).\n *\n * Suitable for thread IDs, child workflow IDs, or any workflow-scoped identifier.\n *\n * @param length - Number of base-62 characters (default 12, ~71 bits of entropy)\n */\nexport function getShortId(length = 12): string {\n const hex = uuid4().replace(/-/g, \"\");\n let result = \"\";\n for (let i = 0; i < length; i++) {\n const byte = parseInt(hex.slice(i * 2, i * 2 + 2), 16);\n result += BASE62[byte % BASE62.length];\n }\n return result;\n}\n","import {\n Bash,\n InMemoryFs,\n type BashOptions,\n type IFileSystem,\n type InitialFiles,\n} from \"just-bash\";\nimport type {\n Sandbox,\n SandboxCapabilities,\n SandboxCreateOptions,\n SandboxCreateResult,\n SandboxFileSystem,\n SandboxProvider,\n SandboxSnapshot,\n ExecOptions,\n ExecResult,\n DirentEntry,\n FileStat,\n} from \"../../../lib/sandbox/types\";\nimport { SandboxNotFoundError } from \"../../../lib/sandbox/types\";\nimport { getShortId } from \"../../../lib/thread/id\";\n\n// ============================================================================\n// Adapter: IFileSystem → SandboxFileSystem\n// ============================================================================\n\nfunction toSandboxFs(fs: IFileSystem): SandboxFileSystem {\n return {\n readFile: (path) => fs.readFile(path),\n readFileBuffer: (path) => fs.readFileBuffer(path),\n writeFile: (path, content) => fs.writeFile(path, content),\n appendFile: (path, content) => fs.appendFile(path, content),\n exists: (path) => fs.exists(path),\n stat: async (path): Promise<FileStat> => {\n const s = await fs.stat(path);\n return {\n isFile: s.isFile,\n isDirectory: s.isDirectory,\n isSymbolicLink: s.isSymbolicLink,\n size: s.size,\n mtime: s.mtime,\n };\n },\n mkdir: (path, opts) => fs.mkdir(path, opts),\n readdir: (path) => fs.readdir(path),\n readdirWithFileTypes: async (path): Promise<DirentEntry[]> => {\n if (!fs.readdirWithFileTypes) {\n const names = await fs.readdir(path);\n return Promise.all(\n names.map(async (name) => {\n const s = await fs.stat(`${path}/${name}`);\n return {\n name,\n isFile: s.isFile,\n isDirectory: s.isDirectory,\n isSymbolicLink: s.isSymbolicLink,\n };\n })\n );\n }\n return fs.readdirWithFileTypes(path);\n },\n rm: (path, opts) => fs.rm(path, opts),\n cp: (src, dest, opts) => fs.cp(src, dest, opts),\n mv: (src, dest) => fs.mv(src, dest),\n readlink: (path) => fs.readlink(path),\n resolvePath: (base, p) => fs.resolvePath(base, p),\n };\n}\n\n// ============================================================================\n// InMemorySandbox\n// ============================================================================\n\nexport interface InMemorySandboxOptions {\n /** Options forwarded to `just-bash` `Bash` (minus `fs` which is managed) */\n bashOptions?: Omit<BashOptions, \"fs\">;\n}\n\n/**\n * An in-memory {@link Sandbox} backed by `just-bash`.\n */\nexport type InMemorySandbox = Sandbox & { fs: SandboxFileSystem };\n\nclass InMemorySandboxImpl implements Sandbox {\n readonly capabilities: SandboxCapabilities = {\n filesystem: true,\n execution: true,\n persistence: true,\n };\n\n readonly fs: SandboxFileSystem;\n private bashOptions: Omit<BashOptions, \"fs\">;\n\n constructor(\n readonly id: string,\n private justBashFs: IFileSystem,\n options?: InMemorySandboxOptions\n ) {\n this.fs = toSandboxFs(justBashFs);\n this.bashOptions = {\n executionLimits: { maxStringLength: 52_428_800 },\n ...options?.bashOptions,\n };\n }\n\n async exec(command: string, _options?: ExecOptions): Promise<ExecResult> {\n const bash = new Bash({ ...this.bashOptions, fs: this.justBashFs });\n const { exitCode, stderr, stdout } = await bash.exec(command);\n return { exitCode, stdout, stderr };\n }\n\n async destroy(): Promise<void> {\n // In-memory: nothing to clean up\n }\n\n /** Expose the underlying IFileSystem for snapshot serialisation */\n _getJustBashFs(): IFileSystem {\n return this.justBashFs;\n }\n}\n\n// ============================================================================\n// InMemorySandboxProvider\n// ============================================================================\n\nexport class InMemorySandboxProvider implements SandboxProvider {\n readonly id = \"inmemory\";\n readonly capabilities: SandboxCapabilities = {\n filesystem: true,\n execution: true,\n persistence: true,\n };\n\n private sandboxes = new Map<string, InMemorySandboxImpl>();\n\n constructor(private defaultOptions?: InMemorySandboxOptions) {}\n\n async get(id: string): Promise<Sandbox> {\n const sandbox = this.sandboxes.get(id);\n if (!sandbox) throw new SandboxNotFoundError(id);\n return sandbox;\n }\n\n async destroy(id: string): Promise<void> {\n const sandbox = this.sandboxes.get(id);\n if (sandbox) {\n await sandbox.destroy();\n this.sandboxes.delete(id);\n }\n }\n\n async create(options?: SandboxCreateOptions): Promise<SandboxCreateResult> {\n const id = options?.id ?? getShortId();\n const initialFiles: InitialFiles = {};\n\n if (options?.initialFiles) {\n for (const [path, content] of Object.entries(options.initialFiles)) {\n initialFiles[path] = content;\n }\n }\n\n const fs = new InMemoryFs(initialFiles);\n const sandbox = new InMemorySandboxImpl(id, fs, this.defaultOptions);\n this.sandboxes.set(id, sandbox);\n return { sandbox };\n }\n\n async snapshot(sandboxId: string): Promise<SandboxSnapshot> {\n const sandbox = this.sandboxes.get(sandboxId);\n if (!sandbox) throw new SandboxNotFoundError(sandboxId);\n\n const fs = sandbox._getJustBashFs();\n const paths = fs.getAllPaths();\n const files: Record<string, string> = {};\n\n for (const p of paths) {\n try {\n const stat = await fs.stat(p);\n if (stat.isFile) {\n files[p] = await fs.readFile(p);\n }\n } catch {\n // skip entries that can't be read (e.g. broken symlinks)\n }\n }\n\n return {\n sandboxId,\n providerId: this.id,\n data: { files },\n createdAt: new Date().toISOString(),\n };\n }\n\n async restore(snapshot: SandboxSnapshot): Promise<Sandbox> {\n const { files } = snapshot.data as { files: Record<string, string> };\n const initialFiles: InitialFiles = {};\n for (const [path, content] of Object.entries(files)) {\n initialFiles[path] = content;\n }\n\n const fs = new InMemoryFs(initialFiles);\n const sandbox = new InMemorySandboxImpl(\n snapshot.sandboxId,\n fs,\n this.defaultOptions\n );\n this.sandboxes.set(sandbox.id, sandbox);\n return sandbox;\n }\n}\n"]}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { BashOptions } from 'just-bash';
|
|
2
|
+
import { S as Sandbox, a as SandboxFileSystem, b as SandboxProvider, c as SandboxCapabilities, d as SandboxCreateOptions, e as SandboxCreateResult, f as SandboxSnapshot } from '../../../types-CDubRtad.cjs';
|
|
3
|
+
import '@temporalio/common';
|
|
4
|
+
|
|
5
|
+
interface InMemorySandboxOptions {
|
|
6
|
+
/** Options forwarded to `just-bash` `Bash` (minus `fs` which is managed) */
|
|
7
|
+
bashOptions?: Omit<BashOptions, "fs">;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* An in-memory {@link Sandbox} backed by `just-bash`.
|
|
11
|
+
*/
|
|
12
|
+
type InMemorySandbox = Sandbox & {
|
|
13
|
+
fs: SandboxFileSystem;
|
|
14
|
+
};
|
|
15
|
+
declare class InMemorySandboxProvider implements SandboxProvider {
|
|
16
|
+
private defaultOptions?;
|
|
17
|
+
readonly id = "inmemory";
|
|
18
|
+
readonly capabilities: SandboxCapabilities;
|
|
19
|
+
private sandboxes;
|
|
20
|
+
constructor(defaultOptions?: InMemorySandboxOptions | undefined);
|
|
21
|
+
get(id: string): Promise<Sandbox>;
|
|
22
|
+
destroy(id: string): Promise<void>;
|
|
23
|
+
create(options?: SandboxCreateOptions): Promise<SandboxCreateResult>;
|
|
24
|
+
snapshot(sandboxId: string): Promise<SandboxSnapshot>;
|
|
25
|
+
restore(snapshot: SandboxSnapshot): Promise<Sandbox>;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export { type InMemorySandbox, type InMemorySandboxOptions, InMemorySandboxProvider };
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { BashOptions } from 'just-bash';
|
|
2
|
+
import { S as Sandbox, a as SandboxFileSystem, b as SandboxProvider, c as SandboxCapabilities, d as SandboxCreateOptions, e as SandboxCreateResult, f as SandboxSnapshot } from '../../../types-CDubRtad.js';
|
|
3
|
+
import '@temporalio/common';
|
|
4
|
+
|
|
5
|
+
interface InMemorySandboxOptions {
|
|
6
|
+
/** Options forwarded to `just-bash` `Bash` (minus `fs` which is managed) */
|
|
7
|
+
bashOptions?: Omit<BashOptions, "fs">;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* An in-memory {@link Sandbox} backed by `just-bash`.
|
|
11
|
+
*/
|
|
12
|
+
type InMemorySandbox = Sandbox & {
|
|
13
|
+
fs: SandboxFileSystem;
|
|
14
|
+
};
|
|
15
|
+
declare class InMemorySandboxProvider implements SandboxProvider {
|
|
16
|
+
private defaultOptions?;
|
|
17
|
+
readonly id = "inmemory";
|
|
18
|
+
readonly capabilities: SandboxCapabilities;
|
|
19
|
+
private sandboxes;
|
|
20
|
+
constructor(defaultOptions?: InMemorySandboxOptions | undefined);
|
|
21
|
+
get(id: string): Promise<Sandbox>;
|
|
22
|
+
destroy(id: string): Promise<void>;
|
|
23
|
+
create(options?: SandboxCreateOptions): Promise<SandboxCreateResult>;
|
|
24
|
+
snapshot(sandboxId: string): Promise<SandboxSnapshot>;
|
|
25
|
+
restore(snapshot: SandboxSnapshot): Promise<Sandbox>;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export { type InMemorySandbox, type InMemorySandboxOptions, InMemorySandboxProvider };
|