@computesdk/blaxel 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +666 -0
- package/dist/index.d.mts +60 -0
- package/dist/index.d.ts +60 -0
- package/dist/index.js +403 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +377 -0
- package/dist/index.mjs.map +1 -0
- package/package.json +60 -0
package/dist/index.mjs
ADDED
|
@@ -0,0 +1,377 @@
|
|
|
1
|
+
// src/index.ts
|
|
2
|
+
import { SandboxInstance, settings } from "@blaxel/core";
|
|
3
|
+
import { createProvider, createBackgroundCommand } from "computesdk";
|
|
4
|
+
var blaxel = createProvider({
|
|
5
|
+
name: "blaxel",
|
|
6
|
+
methods: {
|
|
7
|
+
sandbox: {
|
|
8
|
+
// Collection operations (map to compute.sandbox.*)
|
|
9
|
+
create: async (config, options) => {
|
|
10
|
+
await handleBlaxelAuth(config);
|
|
11
|
+
let image = config.image || "blaxel/prod-base:latest";
|
|
12
|
+
if (!config.image && options?.runtime) {
|
|
13
|
+
switch (options.runtime) {
|
|
14
|
+
case "python":
|
|
15
|
+
image = "blaxel/prod-py-app:latest";
|
|
16
|
+
break;
|
|
17
|
+
case "node":
|
|
18
|
+
image = "blaxel/prod-ts-app:latest";
|
|
19
|
+
break;
|
|
20
|
+
default:
|
|
21
|
+
image = "blaxel/prod-base:latest";
|
|
22
|
+
break;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
const memory = config.memory;
|
|
26
|
+
const region = config.region;
|
|
27
|
+
const envs = options?.envs;
|
|
28
|
+
const ttl = options?.timeout ? `${Math.ceil(options.timeout / 1e3)}s` : void 0;
|
|
29
|
+
try {
|
|
30
|
+
let sandbox;
|
|
31
|
+
sandbox = await SandboxInstance.createIfNotExists({
|
|
32
|
+
name: options?.sandboxId || `blaxel-${Date.now()}`,
|
|
33
|
+
image,
|
|
34
|
+
memory,
|
|
35
|
+
envs: Object.entries(envs || {}).map(([name, value]) => ({ name, value })),
|
|
36
|
+
metadata: {
|
|
37
|
+
labels: {
|
|
38
|
+
...options?.metadata?.labels
|
|
39
|
+
}
|
|
40
|
+
},
|
|
41
|
+
ttl,
|
|
42
|
+
ports: config.ports?.map((port) => ({ target: port, protocol: "HTTP" })),
|
|
43
|
+
...region && { region }
|
|
44
|
+
});
|
|
45
|
+
return {
|
|
46
|
+
sandbox,
|
|
47
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown"
|
|
48
|
+
};
|
|
49
|
+
} catch (error) {
|
|
50
|
+
if (error instanceof Error) {
|
|
51
|
+
if (error.message.includes("unauthorized") || error.message.includes("API key")) {
|
|
52
|
+
throw new Error(
|
|
53
|
+
`Blaxel authentication failed. Please check your BLAXEL_API_KEY environment variable.`
|
|
54
|
+
);
|
|
55
|
+
}
|
|
56
|
+
if (error.message.includes("quota") || error.message.includes("limit")) {
|
|
57
|
+
throw new Error(
|
|
58
|
+
`Blaxel quota exceeded. Please check your usage limits.`
|
|
59
|
+
);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
throw new Error(
|
|
63
|
+
`Failed to create Blaxel sandbox: ${error instanceof Error ? error.message : String(error)}`
|
|
64
|
+
);
|
|
65
|
+
}
|
|
66
|
+
},
|
|
67
|
+
getById: async (config, sandboxId) => {
|
|
68
|
+
await handleBlaxelAuth(config);
|
|
69
|
+
try {
|
|
70
|
+
const sandbox = await SandboxInstance.get(sandboxId);
|
|
71
|
+
if (!sandbox) {
|
|
72
|
+
return null;
|
|
73
|
+
}
|
|
74
|
+
return {
|
|
75
|
+
sandbox,
|
|
76
|
+
sandboxId
|
|
77
|
+
};
|
|
78
|
+
} catch (error) {
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
},
|
|
82
|
+
list: async (config) => {
|
|
83
|
+
await handleBlaxelAuth(config);
|
|
84
|
+
const sandboxList = await SandboxInstance.list();
|
|
85
|
+
return sandboxList.map((sandbox) => ({
|
|
86
|
+
sandbox,
|
|
87
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown"
|
|
88
|
+
}));
|
|
89
|
+
},
|
|
90
|
+
destroy: async (config, sandboxId) => {
|
|
91
|
+
await handleBlaxelAuth(config);
|
|
92
|
+
try {
|
|
93
|
+
await SandboxInstance.delete(sandboxId);
|
|
94
|
+
} catch (error) {
|
|
95
|
+
}
|
|
96
|
+
},
|
|
97
|
+
// Instance operations (map to individual Sandbox methods)
|
|
98
|
+
runCode: async (sandbox, code, runtime) => {
|
|
99
|
+
const startTime = Date.now();
|
|
100
|
+
try {
|
|
101
|
+
let effectiveRuntime = runtime;
|
|
102
|
+
if (!effectiveRuntime) {
|
|
103
|
+
const sandboxImage = sandbox.spec?.runtime?.image || "";
|
|
104
|
+
if (sandboxImage.includes("py")) {
|
|
105
|
+
effectiveRuntime = "python";
|
|
106
|
+
} else if (sandboxImage.includes("ts") || sandboxImage.includes("node") || sandboxImage.includes("base")) {
|
|
107
|
+
effectiveRuntime = "node";
|
|
108
|
+
} else {
|
|
109
|
+
effectiveRuntime = // Strong Python indicators
|
|
110
|
+
code.includes("print(") || code.includes("import ") || code.includes("from ") || code.includes("def ") || code.includes("class ") || code.includes("raise ") || code.includes("except ") || code.includes("elif ") || code.includes("lambda ") || code.includes("True") || code.includes("False") || code.includes("None") || code.includes("sys.") || code.includes("json.") || code.includes("__") || code.includes('f"') || code.includes("f'") || code.includes('"""') || code.includes("'''") ? "python" : "node";
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
const escapedCode = code.replace(/\\/g, "\\\\").replace(/"/g, '\\"').replace(/\$/g, "\\$").replace(/`/g, "\\`");
|
|
114
|
+
const command = effectiveRuntime === "python" ? `python3 -c "${escapedCode}"` : `node -e "${escapedCode}"`;
|
|
115
|
+
const { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, false, command);
|
|
116
|
+
if (exitCode !== 0 && stderr) {
|
|
117
|
+
if (stderr.includes("SyntaxError") || stderr.includes("invalid syntax") || stderr.includes("Unexpected token") || stderr.includes("Unexpected identifier")) {
|
|
118
|
+
throw new Error(`Syntax error: ${stderr.trim()}`);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
return {
|
|
122
|
+
stdout,
|
|
123
|
+
stderr,
|
|
124
|
+
exitCode,
|
|
125
|
+
executionTime: Date.now() - startTime,
|
|
126
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown",
|
|
127
|
+
provider: "blaxel"
|
|
128
|
+
};
|
|
129
|
+
} catch (error) {
|
|
130
|
+
if (error instanceof Error && error.message.includes("Syntax error")) {
|
|
131
|
+
throw error;
|
|
132
|
+
}
|
|
133
|
+
return {
|
|
134
|
+
stdout: "",
|
|
135
|
+
stderr: error instanceof Error ? error.message : String(error),
|
|
136
|
+
exitCode: 1,
|
|
137
|
+
executionTime: Date.now() - startTime,
|
|
138
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown",
|
|
139
|
+
provider: "blaxel"
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
},
|
|
143
|
+
runCommand: async (sandbox, command, args = [], options) => {
|
|
144
|
+
const startTime = Date.now();
|
|
145
|
+
try {
|
|
146
|
+
const { command: finalCommand, args: finalArgs, isBackground } = createBackgroundCommand(command, args, options);
|
|
147
|
+
const fullCommand = finalArgs.length > 0 ? `${finalCommand} ${finalArgs.join(" ")}` : finalCommand;
|
|
148
|
+
const { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, isBackground, fullCommand);
|
|
149
|
+
return {
|
|
150
|
+
stdout,
|
|
151
|
+
stderr,
|
|
152
|
+
exitCode,
|
|
153
|
+
executionTime: Date.now() - startTime,
|
|
154
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown",
|
|
155
|
+
provider: "blaxel",
|
|
156
|
+
isBackground,
|
|
157
|
+
...isBackground && { pid: -1 }
|
|
158
|
+
};
|
|
159
|
+
} catch (error) {
|
|
160
|
+
return {
|
|
161
|
+
stdout: "",
|
|
162
|
+
stderr: error instanceof Error ? error.message : String(error),
|
|
163
|
+
exitCode: 127,
|
|
164
|
+
// Command not found exit code
|
|
165
|
+
executionTime: Date.now() - startTime,
|
|
166
|
+
sandboxId: sandbox.metadata?.name || "blaxel-unknown",
|
|
167
|
+
provider: "blaxel"
|
|
168
|
+
};
|
|
169
|
+
}
|
|
170
|
+
},
|
|
171
|
+
getInfo: async (sandbox) => {
|
|
172
|
+
return {
|
|
173
|
+
id: sandbox.metadata?.name || "blaxel-unknown",
|
|
174
|
+
provider: "blaxel",
|
|
175
|
+
runtime: sandbox.spec?.runtime?.image?.includes("py") ? "python" : "node",
|
|
176
|
+
status: convertSandboxStatus(sandbox.status),
|
|
177
|
+
createdAt: sandbox.metadata?.createdAt ? new Date(sandbox.metadata.createdAt) : /* @__PURE__ */ new Date(),
|
|
178
|
+
timeout: parseTTLToMilliseconds(sandbox.spec?.runtime?.ttl),
|
|
179
|
+
metadata: {
|
|
180
|
+
...sandbox.metadata?.labels
|
|
181
|
+
}
|
|
182
|
+
};
|
|
183
|
+
},
|
|
184
|
+
getUrl: async (sandbox, options) => {
|
|
185
|
+
try {
|
|
186
|
+
const isPublic = options.authentication?.public !== void 0 ? options.authentication.public : true;
|
|
187
|
+
const defaultHeaders = {
|
|
188
|
+
"Access-Control-Allow-Origin": "*",
|
|
189
|
+
"Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS, PATCH",
|
|
190
|
+
"Access-Control-Allow-Headers": "Content-Type, Authorization, X-Requested-With, X-Blaxel-Preview-Token, X-Blaxel-Authorization",
|
|
191
|
+
"Access-Control-Allow-Credentials": "true",
|
|
192
|
+
"Access-Control-Expose-Headers": "Content-Length, X-Request-Id",
|
|
193
|
+
"Access-Control-Max-Age": "86400",
|
|
194
|
+
"Vary": "Origin"
|
|
195
|
+
};
|
|
196
|
+
const responseHeaders = options.headers?.response || defaultHeaders;
|
|
197
|
+
const preview = await sandbox.previews.createIfNotExists({
|
|
198
|
+
metadata: {
|
|
199
|
+
name: `preview-port-${options.port}-${isPublic ? "public" : "private"}`
|
|
200
|
+
},
|
|
201
|
+
spec: {
|
|
202
|
+
port: options.port,
|
|
203
|
+
public: isPublic,
|
|
204
|
+
responseHeaders,
|
|
205
|
+
requestHeaders: options.headers?.request || defaultHeaders,
|
|
206
|
+
customDomain: options.customDomain,
|
|
207
|
+
prefixUrl: options.prefixUrl,
|
|
208
|
+
ttl: options.ttl ? `${Math.ceil(options.ttl / 1e3)}s` : void 0
|
|
209
|
+
}
|
|
210
|
+
});
|
|
211
|
+
const url = preview.spec?.url;
|
|
212
|
+
if (!url) {
|
|
213
|
+
throw new Error(`Failed to get preview URL for port ${options.port}`);
|
|
214
|
+
}
|
|
215
|
+
if (!isPublic) {
|
|
216
|
+
const expiryMinutes = options.authentication?.tokenExpiryMinutes || 60;
|
|
217
|
+
const expiresAt = new Date(Date.now() + expiryMinutes * 60 * 1e3);
|
|
218
|
+
const token = await preview.tokens.create(expiresAt);
|
|
219
|
+
const separator = url.includes("?") ? "&" : "?";
|
|
220
|
+
return `${url}${separator}bl_preview_token=${token.value}`;
|
|
221
|
+
}
|
|
222
|
+
return url;
|
|
223
|
+
} catch (error) {
|
|
224
|
+
throw new Error(
|
|
225
|
+
`Failed to get Blaxel preview URL for port ${options.port}: ${error instanceof Error ? error.message : String(error)}`
|
|
226
|
+
);
|
|
227
|
+
}
|
|
228
|
+
},
|
|
229
|
+
// Optional filesystem methods - implement using Blaxel's filesystem API
|
|
230
|
+
filesystem: {
|
|
231
|
+
readFile: async (sandbox, path) => {
|
|
232
|
+
const result = await sandbox.fs.read(path);
|
|
233
|
+
return result || "";
|
|
234
|
+
},
|
|
235
|
+
writeFile: async (sandbox, path, content) => {
|
|
236
|
+
await sandbox.fs.write(path, content);
|
|
237
|
+
},
|
|
238
|
+
mkdir: async (sandbox, path) => {
|
|
239
|
+
await sandbox.fs.mkdir(path);
|
|
240
|
+
},
|
|
241
|
+
readdir: async (sandbox, path) => {
|
|
242
|
+
const result = await sandbox.fs.ls(path);
|
|
243
|
+
const files = result.files || [];
|
|
244
|
+
const directories = result.subdirectories || [];
|
|
245
|
+
let entries = [];
|
|
246
|
+
for (const file of files) {
|
|
247
|
+
entries.push({
|
|
248
|
+
name: file.name,
|
|
249
|
+
path: `${path}/${file.name}`,
|
|
250
|
+
isDirectory: false,
|
|
251
|
+
size: file.size || 0,
|
|
252
|
+
lastModified: new Date(file.lastModified || Date.now())
|
|
253
|
+
});
|
|
254
|
+
}
|
|
255
|
+
for (const directory of directories) {
|
|
256
|
+
entries.push({
|
|
257
|
+
name: directory.name,
|
|
258
|
+
path: `${path}/${directory.name}`,
|
|
259
|
+
isDirectory: true,
|
|
260
|
+
size: 0,
|
|
261
|
+
lastModified: /* @__PURE__ */ new Date()
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
return entries;
|
|
265
|
+
},
|
|
266
|
+
exists: async (sandbox, path) => {
|
|
267
|
+
try {
|
|
268
|
+
await sandbox.fs.read(path);
|
|
269
|
+
return true;
|
|
270
|
+
} catch {
|
|
271
|
+
try {
|
|
272
|
+
await sandbox.fs.ls(path);
|
|
273
|
+
return true;
|
|
274
|
+
} catch {
|
|
275
|
+
return false;
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
},
|
|
279
|
+
remove: async (sandbox, path) => {
|
|
280
|
+
await sandbox.fs.rm(path);
|
|
281
|
+
}
|
|
282
|
+
},
|
|
283
|
+
// Provider-specific typed getInstance method
|
|
284
|
+
getInstance: (sandbox) => {
|
|
285
|
+
return sandbox;
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
});
|
|
290
|
+
function createBlaxelCompute(config) {
|
|
291
|
+
const provider = blaxel(config);
|
|
292
|
+
return {
|
|
293
|
+
sandbox: {
|
|
294
|
+
create: async () => {
|
|
295
|
+
const sandbox = await provider.sandbox.create();
|
|
296
|
+
return {
|
|
297
|
+
...sandbox,
|
|
298
|
+
getInstance: () => {
|
|
299
|
+
return sandbox.getInstance();
|
|
300
|
+
}
|
|
301
|
+
};
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
};
|
|
305
|
+
}
|
|
306
|
+
async function handleBlaxelAuth(config) {
|
|
307
|
+
try {
|
|
308
|
+
await settings.authenticate();
|
|
309
|
+
} catch (error) {
|
|
310
|
+
if (config.workspace || process.env.BLAXEL_WORKSPACE && typeof process !== "undefined") {
|
|
311
|
+
process.env.BL_WORKSPACE = config.workspace || process.env.BLAXEL_WORKSPACE;
|
|
312
|
+
}
|
|
313
|
+
if (config.apiKey || process.env.BLAXEL_API_KEY && typeof process !== "undefined") {
|
|
314
|
+
process.env.BL_API_KEY = config.apiKey || process.env.BLAXEL_API_KEY;
|
|
315
|
+
}
|
|
316
|
+
try {
|
|
317
|
+
await settings.authenticate();
|
|
318
|
+
} catch (error2) {
|
|
319
|
+
throw new Error("Blaxel authentication failed. Please check the following documents for more information: https://docs.blaxel.ai/Security/Access-tokens#using-api-keys");
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
}
|
|
323
|
+
function parseTTLToMilliseconds(ttl) {
|
|
324
|
+
if (!ttl) return 3e5;
|
|
325
|
+
if (typeof ttl === "number") {
|
|
326
|
+
return ttl * 1e3;
|
|
327
|
+
}
|
|
328
|
+
const match = ttl.match(/^(\d+)([smhd])?$/);
|
|
329
|
+
if (!match) return 3e5;
|
|
330
|
+
const value = parseInt(match[1], 10);
|
|
331
|
+
const unit = match[2] || "s";
|
|
332
|
+
switch (unit) {
|
|
333
|
+
case "s":
|
|
334
|
+
return value * 1e3;
|
|
335
|
+
// seconds to ms
|
|
336
|
+
case "m":
|
|
337
|
+
return value * 60 * 1e3;
|
|
338
|
+
// minutes to ms
|
|
339
|
+
case "h":
|
|
340
|
+
return value * 60 * 60 * 1e3;
|
|
341
|
+
// hours to ms
|
|
342
|
+
case "d":
|
|
343
|
+
return value * 24 * 60 * 60 * 1e3;
|
|
344
|
+
// days to ms
|
|
345
|
+
default:
|
|
346
|
+
return 3e5;
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
function convertSandboxStatus(status) {
|
|
350
|
+
switch (status?.toLowerCase()) {
|
|
351
|
+
case "deployed":
|
|
352
|
+
return "running";
|
|
353
|
+
case "deleting":
|
|
354
|
+
return "stopped";
|
|
355
|
+
case "failed":
|
|
356
|
+
return "error";
|
|
357
|
+
default:
|
|
358
|
+
return "running";
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
async function executeWithStreaming(sandbox, isBackground, command) {
|
|
362
|
+
const result = await sandbox.process.exec({ command });
|
|
363
|
+
if (!isBackground) {
|
|
364
|
+
await sandbox.process.wait(result.name);
|
|
365
|
+
}
|
|
366
|
+
const processResult = await sandbox.process.get(result.name);
|
|
367
|
+
return {
|
|
368
|
+
stdout: processResult.logs,
|
|
369
|
+
stderr: processResult.logs,
|
|
370
|
+
exitCode: processResult.exitCode || 0
|
|
371
|
+
};
|
|
372
|
+
}
|
|
373
|
+
export {
|
|
374
|
+
blaxel,
|
|
375
|
+
createBlaxelCompute
|
|
376
|
+
};
|
|
377
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["/**\n * Blaxel Provider - Factory-based Implementation\n * \n * Full-featured provider with filesystem support using the factory pattern.\n */\n\nimport { SandboxInstance, settings } from '@blaxel/core';\nimport { createProvider, createBackgroundCommand } from 'computesdk';\nimport type {\n\tExecutionResult,\n\tSandboxInfo,\n\tRuntime,\n\tCreateSandboxOptions,\n\tFileEntry,\n\tRunCommandOptions,\n\tSandboxStatus\n} from 'computesdk';\n\n/**\n * Blaxel-specific configuration options\n */\nexport interface BlaxelConfig {\n\t/** Blaxel workspace ID - if not provided, will fallback to BL_WORKSPACE environment variable */\n\tworkspace?: string;\n\t/** Blaxel API key - if not provided, will fallback to BL_API_KEY environment variable */\n\tapiKey?: string;\n\t/** Default image for sandboxes */\n\timage?: string;\n\t/** Default region for sandbox deployment */\n\tregion?: string;\n\t/** Default memory allocation in MB */\n\tmemory?: number | 4096;\n\t/** Default ports for sandbox */\n\tports?: number[] | [3000];\n}\n\n/**\n * Create a Blaxel provider instance using the factory pattern\n */\nexport const blaxel = createProvider<SandboxInstance, BlaxelConfig>({\n\tname: 'blaxel',\n\tmethods: {\n\t\tsandbox: {\n\t\t\t// Collection operations (map to compute.sandbox.*)\n\t\t\tcreate: async (config: BlaxelConfig, options?: CreateSandboxOptions) => {\n\t\t\t\tawait handleBlaxelAuth(config);\n\n\t\t\t\t// Determine the image to use\n\t\t\t\tlet image = config.image || 'blaxel/prod-base:latest'; // Default to prod-base\n\n\t\t\t\t// Override with runtime-specific image if runtime is specified and no explicit image\n\t\t\t\tif (!config.image && options?.runtime) {\n\t\t\t\t\tswitch (options.runtime) {\n\t\t\t\t\t\tcase 'python':\n\t\t\t\t\t\t\timage = 'blaxel/prod-py-app:latest';\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\tcase 'node':\n\t\t\t\t\t\t\timage = 'blaxel/prod-ts-app:latest';\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\tdefault:\n\t\t\t\t\t\t\timage = 'blaxel/prod-base:latest';\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tconst memory = config.memory;\n\t\t\t\tconst region = config.region;\n\t\t\t\tconst envs = options?.envs;\n\t\t\t\tconst ttl = options?.timeout ? `${Math.ceil(options.timeout / 1000)}s` : undefined;\n\n\t\t\t\ttry {\n\t\t\t\t\tlet sandbox: SandboxInstance;\n\n\t\t\t\t\t// Create new Blaxel sandbox\n\t\t\t\t\tsandbox = await SandboxInstance.createIfNotExists({\n\t\t\t\t\t\tname: options?.sandboxId || `blaxel-${Date.now()}`,\n\t\t\t\t\t\timage,\n\t\t\t\t\t\tmemory,\n\t\t\t\t\t\tenvs: Object.entries(envs || {}).map(([name, value]) => ({ name, value })),\n\t\t\t\t\t\tmetadata: {\n\t\t\t\t\t\t\tlabels: {\n\t\t\t\t\t\t\t\t...options?.metadata?.labels,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t},\n\t\t\t\t\t\tttl,\n\t\t\t\t\t\tports: config.ports?.map(port => ({ target: port, protocol: 'HTTP' })),\n\t\t\t\t\t\t...(region && { region })\n\t\t\t\t\t});\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\tsandbox,\n\t\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown'\n\t\t\t\t\t};\n\t\t\t\t} catch (error) {\n\t\t\t\t\tif (error instanceof Error) {\n\t\t\t\t\t\tif (error.message.includes('unauthorized') || error.message.includes('API key')) {\n\t\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t\t`Blaxel authentication failed. Please check your BLAXEL_API_KEY environment variable.`\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (error.message.includes('quota') || error.message.includes('limit')) {\n\t\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t\t`Blaxel quota exceeded. Please check your usage limits.`\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t`Failed to create Blaxel sandbox: ${error instanceof Error ? error.message : String(error)}`\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t},\n\n\t\t\tgetById: async (config: BlaxelConfig, sandboxId: string) => {\n\t\t\t\tawait handleBlaxelAuth(config);\n\n\t\t\t\ttry {\n\t\t\t\t\tconst sandbox = await SandboxInstance.get(sandboxId);\n\n\t\t\t\t\tif (!sandbox) {\n\t\t\t\t\t\treturn null;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\tsandbox,\n\t\t\t\t\t\tsandboxId\n\t\t\t\t\t};\n\t\t\t\t} catch (error) {\n\t\t\t\t\t// Sandbox doesn't exist or can't be accessed\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t},\n\n\t\t\tlist: async (config: BlaxelConfig) => {\n\t\t\t\tawait handleBlaxelAuth(config);\n\n\t\t\t\tconst sandboxList = await SandboxInstance.list();\n\t\t\t\treturn sandboxList.map(sandbox => ({\n\t\t\t\t\tsandbox,\n\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown'\n\t\t\t\t}));\n\t\t\t},\n\n\t\t\tdestroy: async (config: BlaxelConfig, sandboxId: string) => {\n\t\t\t\tawait handleBlaxelAuth(config);\n\n\t\t\t\ttry {\n\t\t\t\t\tawait SandboxInstance.delete(sandboxId);\n\t\t\t\t} catch (error) {\n\t\t\t\t\t// Sandbox might already be destroyed or doesn't exist\n\t\t\t\t\t// This is acceptable for destroy operations\n\t\t\t\t}\n\t\t\t},\n\n\t\t\t// Instance operations (map to individual Sandbox methods)\n\t\t\trunCode: async (sandbox: SandboxInstance, code: string, runtime?: Runtime): Promise<ExecutionResult> => {\n\t\t\t\tconst startTime = Date.now();\n\n\t\t\t\ttry {\n\t\t\t\t\t// Determine runtime: \n\t\t\t\t\t// 1. Use explicitly passed runtime if provided\n\t\t\t\t\t// 2. Check sandbox's actual runtime based on its image\n\t\t\t\t\t// 3. Fall back to auto-detection from code content\n\t\t\t\t\tlet effectiveRuntime = runtime;\n\n\t\t\t\t\tif (!effectiveRuntime) {\n\t\t\t\t\t\t// Check sandbox's image to determine its runtime\n\t\t\t\t\t\tconst sandboxImage = sandbox.spec?.runtime?.image || '';\n\t\t\t\t\t\tif (sandboxImage.includes('py')) {\n\t\t\t\t\t\t\teffectiveRuntime = 'python';\n\t\t\t\t\t\t} else if (sandboxImage.includes('ts') || sandboxImage.includes('node') || sandboxImage.includes('base')) {\n\t\t\t\t\t\t\t// prod-base, prod-ts-app are both Node/TypeScript environments\n\t\t\t\t\t\t\teffectiveRuntime = 'node';\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t// Fall back to auto-detection with improved patterns for unknown images\n\t\t\t\t\t\t\teffectiveRuntime = (\n\t\t\t\t\t\t\t\t// Strong Python indicators\n\t\t\t\t\t\t\t\tcode.includes('print(') ||\n\t\t\t\t\t\t\t\t\tcode.includes('import ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('from ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('def ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('class ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('raise ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('except ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('elif ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('lambda ') ||\n\t\t\t\t\t\t\t\t\tcode.includes('True') ||\n\t\t\t\t\t\t\t\t\tcode.includes('False') ||\n\t\t\t\t\t\t\t\t\tcode.includes('None') ||\n\t\t\t\t\t\t\t\t\tcode.includes('sys.') ||\n\t\t\t\t\t\t\t\t\tcode.includes('json.') ||\n\t\t\t\t\t\t\t\t\tcode.includes('__') ||\n\t\t\t\t\t\t\t\t\tcode.includes('f\"') ||\n\t\t\t\t\t\t\t\t\tcode.includes(\"f'\") ||\n\t\t\t\t\t\t\t\t\tcode.includes('\"\"\"') ||\n\t\t\t\t\t\t\t\t\tcode.includes(\"'''\")\n\t\t\t\t\t\t\t\t\t? 'python'\n\t\t\t\t\t\t\t\t\t// Default to Node.js for all other cases (including ambiguous)\n\t\t\t\t\t\t\t\t\t: 'node'\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Execute code using Blaxel's process execution\n\t\t\t\t\t// Escape the code properly for shell execution\n\t\t\t\t\tconst escapedCode = code.replace(/\\\\/g, '\\\\\\\\').replace(/\"/g, '\\\\\"').replace(/\\$/g, '\\\\$').replace(/`/g, '\\\\`');\n\n\t\t\t\t\tconst command = effectiveRuntime === 'python'\n\t\t\t\t\t\t? `python3 -c \"${escapedCode}\"`\n\t\t\t\t\t\t: `node -e \"${escapedCode}\"`;\n\n\t\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, false, command);\n\n\t\t\t\t\t// Check for syntax errors and throw them\n\t\t\t\t\tif (exitCode !== 0 && stderr) {\n\t\t\t\t\t\t// Check for common syntax error patterns\n\t\t\t\t\t\tif (stderr.includes('SyntaxError') ||\n\t\t\t\t\t\t\tstderr.includes('invalid syntax') ||\n\t\t\t\t\t\t\tstderr.includes('Unexpected token') ||\n\t\t\t\t\t\t\tstderr.includes('Unexpected identifier')) {\n\t\t\t\t\t\t\tthrow new Error(`Syntax error: ${stderr.trim()}`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\tstdout,\n\t\t\t\t\t\tstderr,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\texecutionTime: Date.now() - startTime,\n\t\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown',\n\t\t\t\t\t\tprovider: 'blaxel'\n\t\t\t\t\t};\n\t\t\t\t} catch (error) {\n\t\t\t\t\t// Re-throw syntax errors\n\t\t\t\t\tif (error instanceof Error && error.message.includes('Syntax error')) {\n\t\t\t\t\t\tthrow error;\n\t\t\t\t\t}\n\n\t\t\t\t\t// For runtime errors, return a result instead of throwing\n\t\t\t\t\treturn {\n\t\t\t\t\t\tstdout: '',\n\t\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 1,\n\t\t\t\t\t\texecutionTime: Date.now() - startTime,\n\t\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown',\n\t\t\t\t\t\tprovider: 'blaxel'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\t\trunCommand: async (sandbox: SandboxInstance, command: string, args: string[] = [], options?: RunCommandOptions): Promise<ExecutionResult> => {\n\t\t\t\tconst startTime = Date.now();\n\n\t\t\t\ttry {\n\t\t\t\t\t// Handle background command execution\n\t\t\t\t\tconst { command: finalCommand, args: finalArgs, isBackground } = createBackgroundCommand(command, args, options);\n\n\t\t\t\t\t// Construct full command\n\t\t\t\t\tconst fullCommand = finalArgs.length > 0 ? `${finalCommand} ${finalArgs.join(' ')}` : finalCommand;\n\n\t\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, isBackground, fullCommand);\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\tstdout,\n\t\t\t\t\t\tstderr,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\texecutionTime: Date.now() - startTime,\n\t\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown',\n\t\t\t\t\t\tprovider: 'blaxel',\n\t\t\t\t\t\tisBackground,\n\t\t\t\t\t\t...(isBackground && { pid: -1 })\n\t\t\t\t\t};\n\t\t\t\t} catch (error) {\n\t\t\t\t\t// For command failures, return error info instead of throwing\n\t\t\t\t\treturn {\n\t\t\t\t\t\tstdout: '',\n\t\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 127, // Command not found exit code\n\t\t\t\t\t\texecutionTime: Date.now() - startTime,\n\t\t\t\t\t\tsandboxId: sandbox.metadata?.name || 'blaxel-unknown',\n\t\t\t\t\t\tprovider: 'blaxel'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\t\tgetInfo: async (sandbox: SandboxInstance): Promise<SandboxInfo> => {\n\t\t\t\treturn {\n\t\t\t\t\tid: sandbox.metadata?.name || 'blaxel-unknown',\n\t\t\t\t\tprovider: 'blaxel',\n\t\t\t\t\truntime: sandbox.spec?.runtime?.image?.includes('py') ? 'python' : 'node',\n\t\t\t\t\tstatus: convertSandboxStatus(sandbox.status),\n\t\t\t\t\tcreatedAt: sandbox.metadata?.createdAt ? new Date(sandbox.metadata.createdAt) : new Date(),\n\t\t\t\t\ttimeout: parseTTLToMilliseconds(sandbox.spec?.runtime?.ttl),\n\t\t\t\t\tmetadata: {\n\t\t\t\t\t\t...sandbox.metadata?.labels\n\t\t\t\t\t}\n\t\t\t\t};\n\t\t\t},\n\n\t\t\tgetUrl: async (sandbox: SandboxInstance, options: {\n\t\t\t\tport: number;\n\t\t\t\tttl?: number;\n\t\t\t\tprefixUrl?: string;\n\t\t\t\theaders?: {\n\t\t\t\t\tresponse?: Record<string, string>;\n\t\t\t\t\trequest?: Record<string, string>;\n\t\t\t\t};\n\t\t\t\tcustomDomain?: string;\n\t\t\t\tauthentication?: {\n\t\t\t\t\tpublic?: boolean;\n\t\t\t\t\ttokenExpiryMinutes?: number;\n\t\t\t\t};\n\t\t\t}): Promise<string> => {\n\t\t\t\ttry {\n\t\t\t\t\t// If public is not set, default to true\n\t\t\t\t\tconst isPublic = options.authentication?.public !== undefined ? options.authentication.public : true;\n\n\t\t\t\t\t// Default CORS headers for broad compatibility\n\t\t\t\t\tconst defaultHeaders = {\n\t\t\t\t\t\t\"Access-Control-Allow-Origin\": \"*\",\n\t\t\t\t\t\t\"Access-Control-Allow-Methods\": \"GET, POST, PUT, DELETE, OPTIONS, PATCH\",\n\t\t\t\t\t\t\"Access-Control-Allow-Headers\": \"Content-Type, Authorization, X-Requested-With, X-Blaxel-Preview-Token, X-Blaxel-Authorization\",\n\t\t\t\t\t\t\"Access-Control-Allow-Credentials\": \"true\",\n\t\t\t\t\t\t\"Access-Control-Expose-Headers\": \"Content-Length, X-Request-Id\",\n\t\t\t\t\t\t\"Access-Control-Max-Age\": \"86400\",\n\t\t\t\t\t\t\"Vary\": \"Origin\"\n\t\t\t\t\t};\n\n\t\t\t\t\t// Use custom headers if provided, otherwise use defaults\n\t\t\t\t\tconst responseHeaders = options.headers?.response || defaultHeaders;\n\n\t\t\t\t\t// Create or get existing preview URL using Blaxel's preview API\n\t\t\t\t\tconst preview = await sandbox.previews.createIfNotExists({\n\t\t\t\t\t\tmetadata: {\n\t\t\t\t\t\t\tname: `preview-port-${options.port}-${isPublic ? 'public' : 'private'}`\n\t\t\t\t\t\t},\n\t\t\t\t\t\tspec: {\n\t\t\t\t\t\t\tport: options.port,\n\t\t\t\t\t\t\tpublic: isPublic,\n\t\t\t\t\t\t\tresponseHeaders,\n\t\t\t\t\t\t\trequestHeaders: options.headers?.request || defaultHeaders,\n\t\t\t\t\t\t\tcustomDomain: options.customDomain,\n\t\t\t\t\t\t\tprefixUrl: options.prefixUrl,\n\t\t\t\t\t\t\tttl: options.ttl ? `${Math.ceil(options.ttl / 1000)}s` : undefined\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\n\t\t\t\t\t// Get the preview URL\n\t\t\t\t\tconst url = preview.spec?.url;\n\t\t\t\t\tif (!url) {\n\t\t\t\t\t\tthrow new Error(`Failed to get preview URL for port ${options.port}`);\n\t\t\t\t\t}\n\n\t\t\t\t\t// For private previews, create an access token and append it to the URL\n\t\t\t\t\tif (!isPublic) {\n\t\t\t\t\t\t// Create token with specified expiry (default 60 minutes)\n\t\t\t\t\t\tconst expiryMinutes = options.authentication?.tokenExpiryMinutes || 60;\n\t\t\t\t\t\tconst expiresAt = new Date(Date.now() + expiryMinutes * 60 * 1000);\n\t\t\t\t\t\tconst token = await preview.tokens.create(expiresAt);\n\n\t\t\t\t\t\t// Return URL with token as query parameter\n\t\t\t\t\t\tconst separator = url.includes('?') ? '&' : '?';\n\t\t\t\t\t\treturn `${url}${separator}bl_preview_token=${token.value}`;\n\t\t\t\t\t}\n\n\t\t\t\t\t// For public previews, just return the URL\n\t\t\t\t\treturn url;\n\t\t\t\t} catch (error) {\n\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t`Failed to get Blaxel preview URL for port ${options.port}: ${error instanceof Error ? error.message : String(error)}`\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t},\n\n\t\t\t// Optional filesystem methods - implement using Blaxel's filesystem API\n\t\t\tfilesystem: {\n\t\t\t\treadFile: async (sandbox: SandboxInstance, path: string): Promise<string> => {\n\t\t\t\t\tconst result = await sandbox.fs.read(path);\n\t\t\t\t\treturn result || '';\n\t\t\t\t},\n\n\t\t\t\twriteFile: async (sandbox: SandboxInstance, path: string, content: string): Promise<void> => {\n\t\t\t\t\tawait sandbox.fs.write(path, content);\n\t\t\t\t},\n\n\t\t\t\tmkdir: async (sandbox: SandboxInstance, path: string): Promise<void> => {\n\t\t\t\t\tawait sandbox.fs.mkdir(path);\n\t\t\t\t},\n\n\t\t\t\treaddir: async (sandbox: SandboxInstance, path: string): Promise<FileEntry[]> => {\n\t\t\t\t\tconst result = await sandbox.fs.ls(path);\n\t\t\t\t\tconst files = result.files || [];\n\t\t\t\t\tconst directories = result.subdirectories || [];\n\t\t\t\t\tlet entries = [];\n\t\t\t\t\tfor (const file of files) {\n\t\t\t\t\t\tentries.push({\n\t\t\t\t\t\t\tname: file.name,\n\t\t\t\t\t\t\tpath: `${path}/${file.name}`,\n\t\t\t\t\t\t\tisDirectory: false,\n\t\t\t\t\t\t\tsize: file.size || 0,\n\t\t\t\t\t\t\tlastModified: new Date(file.lastModified || Date.now())\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\t\t\t\t\tfor (const directory of directories) {\n\t\t\t\t\t\tentries.push({\n\t\t\t\t\t\t\tname: directory.name,\n\t\t\t\t\t\t\tpath: `${path}/${directory.name}`,\n\t\t\t\t\t\t\tisDirectory: true,\n\t\t\t\t\t\t\tsize: 0,\n\t\t\t\t\t\t\tlastModified: new Date()\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\t\t\t\t\treturn entries;\n\t\t\t\t},\n\n\t\t\t\texists: async (sandbox: SandboxInstance, path: string): Promise<boolean> => {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tawait sandbox.fs.read(path);\n\t\t\t\t\t\treturn true; // It's a file and exists\n\t\t\t\t\t} catch {\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tawait sandbox.fs.ls(path);\n\t\t\t\t\t\t\treturn true; // It's a directory and exists\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\treturn false; // Path doesn't exist\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t},\n\n\t\t\t\tremove: async (sandbox: SandboxInstance, path: string): Promise<void> => {\n\t\t\t\t\tawait sandbox.fs.rm(path);\n\t\t\t\t}\n\t\t\t},\n\n\t\t\t// Provider-specific typed getInstance method\n\t\t\tgetInstance: (sandbox: SandboxInstance): SandboxInstance => {\n\t\t\t\treturn sandbox;\n\t\t\t},\n\t\t}\n\t}\n});\n\n/**\n * Create a properly typed compute instance for Blaxel\n * This version provides full type safety for getInstance() calls\n * \n * @example\n * ```typescript\n * import { createBlaxelCompute } from '@computesdk/blaxel'\n * \n * const compute = createBlaxelCompute({ workspace: 'your-workspace', apiKey: 'your-key' });\n * const sandbox = await compute.sandbox.create();\n * const instance = sandbox.getInstance(); // ✅ Properly typed as SandboxInstance!\n * ```\n */\nexport function createBlaxelCompute(config: BlaxelConfig): {\n\tsandbox: {\n\t\tcreate(): Promise<{\n\t\t\tsandboxId: string;\n\t\t\tprovider: string;\n\t\t\trunCode(code: string, runtime?: import('computesdk').Runtime): Promise<import('computesdk').ExecutionResult>;\n\t\t\trunCommand(command: string, args?: string[]): Promise<import('computesdk').ExecutionResult>;\n\t\t\tgetInfo(): Promise<import('computesdk').SandboxInfo>;\n\t\t\tgetUrl(options: { port: number; protocol?: string }): Promise<string>;\n\t\t\tgetProvider(): ReturnType<typeof blaxel>;\n\t\t\tgetInstance(): SandboxInstance; // ✅ Properly typed!\n\t\t\tkill(): Promise<void>;\n\t\t\tdestroy(): Promise<void>;\n\t\t\tfilesystem: import('computesdk').SandboxFileSystem;\n\t\t}>;\n\t};\n} {\n\tconst provider = blaxel(config);\n\n\treturn {\n\t\tsandbox: {\n\t\t\tcreate: async () => {\n\t\t\t\tconst sandbox = await provider.sandbox.create();\n\t\t\t\treturn {\n\t\t\t\t\t...sandbox,\n\t\t\t\t\tgetInstance: (): SandboxInstance => {\n\t\t\t\t\t\treturn sandbox.getInstance() as SandboxInstance;\n\t\t\t\t\t}\n\t\t\t\t};\n\t\t\t}\n\t\t}\n\t};\n}\n\nasync function handleBlaxelAuth(config: BlaxelConfig) {\n\t// Check if auth is already set in the SDK\n\ttry {\n\t\tawait settings.authenticate();\n\t} catch (error) {\n\t\t// If not, set the auth from the config\n\t\tif (config.workspace || process.env.BLAXEL_WORKSPACE && typeof process !== 'undefined') {\n\t\t\tprocess.env.BL_WORKSPACE = config.workspace || process.env.BLAXEL_WORKSPACE;\n\t\t}\n\t\tif (config.apiKey || process.env.BLAXEL_API_KEY && typeof process !== 'undefined') {\n\t\t\tprocess.env.BL_API_KEY = config.apiKey || process.env.BLAXEL_API_KEY;\n\t\t}\n\t\ttry {\n\t\t\tawait settings.authenticate();\n\t\t} catch (error) {\n\t\t\tthrow new Error('Blaxel authentication failed. Please check the following documents for more information: https://docs.blaxel.ai/Security/Access-tokens#using-api-keys');\n\t\t}\n\t}\n}\n\n/**\n * Parse TTL value from Blaxel's format to milliseconds\n * Supports formats like \"30m\", \"24h\", \"7d\" or plain numbers (seconds)\n */\nfunction parseTTLToMilliseconds(ttl: string | number | undefined): number {\n\tif (!ttl) return 300000; // Default to 5 minutes\n\n\t// If it's already a number, treat it as seconds and convert to milliseconds\n\tif (typeof ttl === 'number') {\n\t\treturn ttl * 1000;\n\t}\n\n\t// Parse string formats like \"30m\", \"24h\", \"7d\"\n\tconst match = ttl.match(/^(\\d+)([smhd])?$/);\n\tif (!match) return 300000; // Default if format is invalid\n\n\tconst value = parseInt(match[1], 10);\n\tconst unit = match[2] || 's'; // Default to seconds if no unit\n\n\tswitch (unit) {\n\t\tcase 's': return value * 1000; // seconds to ms\n\t\tcase 'm': return value * 60 * 1000; // minutes to ms\n\t\tcase 'h': return value * 60 * 60 * 1000; // hours to ms\n\t\tcase 'd': return value * 24 * 60 * 60 * 1000; // days to ms\n\t\tdefault: return 300000; // Default fallback\n\t}\n}\n\nfunction convertSandboxStatus(status: string | undefined): SandboxStatus {\n\tswitch (status?.toLowerCase()) {\n\t\tcase 'deployed': return 'running';\n\t\tcase 'deleting': return 'stopped';\n\t\tcase 'failed': return 'error';\n\t\tdefault: return 'running';\n\t}\n}\n\n/**\n * Execute a command in the sandbox and capture stdout/stderr\n * Handles the common pattern of executing, streaming logs, and waiting for completion\n */\nasync function executeWithStreaming(\n\tsandbox: SandboxInstance,\n\tisBackground: boolean,\n\tcommand: string\n): Promise<{ stdout: string; stderr: string; exitCode: number }> {\n\t// Execute the command\n\tconst result = await sandbox.process.exec({ command });\n\n\tif (!isBackground) {\n\t\t// Wait for process completion\n\t\tawait sandbox.process.wait(result.name);\n\t}\n\n\t// Get final process result for exit code\n\tconst processResult = await sandbox.process.get(result.name);\n\n\t// TODO: Handle proper stdout/stderr streaming\n\treturn {\n\t\tstdout: processResult.logs,\n\t\tstderr: processResult.logs,\n\t\texitCode: processResult.exitCode || 0\n\t};\n}\n\n// Export the Blaxel SandboxInstance type for explicit typing\nexport type { SandboxInstance as BlaxelSandbox } from '@blaxel/core';\n\n"],"mappings":";AAMA,SAAS,iBAAiB,gBAAgB;AAC1C,SAAS,gBAAgB,+BAA+B;AAgCjD,IAAM,SAAS,eAA8C;AAAA,EACnE,MAAM;AAAA,EACN,SAAS;AAAA,IACR,SAAS;AAAA;AAAA,MAER,QAAQ,OAAO,QAAsB,YAAmC;AACvE,cAAM,iBAAiB,MAAM;AAG7B,YAAI,QAAQ,OAAO,SAAS;AAG5B,YAAI,CAAC,OAAO,SAAS,SAAS,SAAS;AACtC,kBAAQ,QAAQ,SAAS;AAAA,YACxB,KAAK;AACJ,sBAAQ;AACR;AAAA,YACD,KAAK;AACJ,sBAAQ;AACR;AAAA,YACD;AACC,sBAAQ;AACR;AAAA,UACF;AAAA,QACD;AACA,cAAM,SAAS,OAAO;AACtB,cAAM,SAAS,OAAO;AACtB,cAAM,OAAO,SAAS;AACtB,cAAM,MAAM,SAAS,UAAU,GAAG,KAAK,KAAK,QAAQ,UAAU,GAAI,CAAC,MAAM;AAEzE,YAAI;AACH,cAAI;AAGJ,oBAAU,MAAM,gBAAgB,kBAAkB;AAAA,YACjD,MAAM,SAAS,aAAa,UAAU,KAAK,IAAI,CAAC;AAAA,YAChD;AAAA,YACA;AAAA,YACA,MAAM,OAAO,QAAQ,QAAQ,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,MAAM,KAAK,OAAO,EAAE,MAAM,MAAM,EAAE;AAAA,YACzE,UAAU;AAAA,cACT,QAAQ;AAAA,gBACP,GAAG,SAAS,UAAU;AAAA,cACvB;AAAA,YACD;AAAA,YACA;AAAA,YACA,OAAO,OAAO,OAAO,IAAI,WAAS,EAAE,QAAQ,MAAM,UAAU,OAAO,EAAE;AAAA,YACrE,GAAI,UAAU,EAAE,OAAO;AAAA,UACxB,CAAC;AAED,iBAAO;AAAA,YACN;AAAA,YACA,WAAW,QAAQ,UAAU,QAAQ;AAAA,UACtC;AAAA,QACD,SAAS,OAAO;AACf,cAAI,iBAAiB,OAAO;AAC3B,gBAAI,MAAM,QAAQ,SAAS,cAAc,KAAK,MAAM,QAAQ,SAAS,SAAS,GAAG;AAChF,oBAAM,IAAI;AAAA,gBACT;AAAA,cACD;AAAA,YACD;AACA,gBAAI,MAAM,QAAQ,SAAS,OAAO,KAAK,MAAM,QAAQ,SAAS,OAAO,GAAG;AACvE,oBAAM,IAAI;AAAA,gBACT;AAAA,cACD;AAAA,YACD;AAAA,UACD;AACA,gBAAM,IAAI;AAAA,YACT,oCAAoC,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,UAC3F;AAAA,QACD;AAAA,MACD;AAAA,MAEA,SAAS,OAAO,QAAsB,cAAsB;AAC3D,cAAM,iBAAiB,MAAM;AAE7B,YAAI;AACH,gBAAM,UAAU,MAAM,gBAAgB,IAAI,SAAS;AAEnD,cAAI,CAAC,SAAS;AACb,mBAAO;AAAA,UACR;AAEA,iBAAO;AAAA,YACN;AAAA,YACA;AAAA,UACD;AAAA,QACD,SAAS,OAAO;AAEf,iBAAO;AAAA,QACR;AAAA,MACD;AAAA,MAEA,MAAM,OAAO,WAAyB;AACrC,cAAM,iBAAiB,MAAM;AAE7B,cAAM,cAAc,MAAM,gBAAgB,KAAK;AAC/C,eAAO,YAAY,IAAI,cAAY;AAAA,UAClC;AAAA,UACA,WAAW,QAAQ,UAAU,QAAQ;AAAA,QACtC,EAAE;AAAA,MACH;AAAA,MAEA,SAAS,OAAO,QAAsB,cAAsB;AAC3D,cAAM,iBAAiB,MAAM;AAE7B,YAAI;AACH,gBAAM,gBAAgB,OAAO,SAAS;AAAA,QACvC,SAAS,OAAO;AAAA,QAGhB;AAAA,MACD;AAAA;AAAA,MAGA,SAAS,OAAO,SAA0B,MAAc,YAAgD;AACvG,cAAM,YAAY,KAAK,IAAI;AAE3B,YAAI;AAKH,cAAI,mBAAmB;AAEvB,cAAI,CAAC,kBAAkB;AAEtB,kBAAM,eAAe,QAAQ,MAAM,SAAS,SAAS;AACrD,gBAAI,aAAa,SAAS,IAAI,GAAG;AAChC,iCAAmB;AAAA,YACpB,WAAW,aAAa,SAAS,IAAI,KAAK,aAAa,SAAS,MAAM,KAAK,aAAa,SAAS,MAAM,GAAG;AAEzG,iCAAmB;AAAA,YACpB,OAAO;AAEN;AAAA,cAEC,KAAK,SAAS,QAAQ,KACrB,KAAK,SAAS,SAAS,KACvB,KAAK,SAAS,OAAO,KACrB,KAAK,SAAS,MAAM,KACpB,KAAK,SAAS,QAAQ,KACtB,KAAK,SAAS,QAAQ,KACtB,KAAK,SAAS,SAAS,KACvB,KAAK,SAAS,OAAO,KACrB,KAAK,SAAS,SAAS,KACvB,KAAK,SAAS,MAAM,KACpB,KAAK,SAAS,OAAO,KACrB,KAAK,SAAS,MAAM,KACpB,KAAK,SAAS,MAAM,KACpB,KAAK,SAAS,OAAO,KACrB,KAAK,SAAS,IAAI,KAClB,KAAK,SAAS,IAAI,KAClB,KAAK,SAAS,IAAI,KAClB,KAAK,SAAS,KAAK,KACnB,KAAK,SAAS,KAAK,IACjB,WAEA;AAAA,YAEL;AAAA,UACD;AAIA,gBAAM,cAAc,KAAK,QAAQ,OAAO,MAAM,EAAE,QAAQ,MAAM,KAAK,EAAE,QAAQ,OAAO,KAAK,EAAE,QAAQ,MAAM,KAAK;AAE9G,gBAAM,UAAU,qBAAqB,WAClC,eAAe,WAAW,MAC1B,YAAY,WAAW;AAE1B,gBAAM,EAAE,QAAQ,QAAQ,SAAS,IAAI,MAAM,qBAAqB,SAAS,OAAO,OAAO;AAGvF,cAAI,aAAa,KAAK,QAAQ;AAE7B,gBAAI,OAAO,SAAS,aAAa,KAChC,OAAO,SAAS,gBAAgB,KAChC,OAAO,SAAS,kBAAkB,KAClC,OAAO,SAAS,uBAAuB,GAAG;AAC1C,oBAAM,IAAI,MAAM,iBAAiB,OAAO,KAAK,CAAC,EAAE;AAAA,YACjD;AAAA,UACD;AAEA,iBAAO;AAAA,YACN;AAAA,YACA;AAAA,YACA;AAAA,YACA,eAAe,KAAK,IAAI,IAAI;AAAA,YAC5B,WAAW,QAAQ,UAAU,QAAQ;AAAA,YACrC,UAAU;AAAA,UACX;AAAA,QACD,SAAS,OAAO;AAEf,cAAI,iBAAiB,SAAS,MAAM,QAAQ,SAAS,cAAc,GAAG;AACrE,kBAAM;AAAA,UACP;AAGA,iBAAO;AAAA,YACN,QAAQ;AAAA,YACR,QAAQ,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,YAC7D,UAAU;AAAA,YACV,eAAe,KAAK,IAAI,IAAI;AAAA,YAC5B,WAAW,QAAQ,UAAU,QAAQ;AAAA,YACrC,UAAU;AAAA,UACX;AAAA,QACD;AAAA,MACD;AAAA,MAEA,YAAY,OAAO,SAA0B,SAAiB,OAAiB,CAAC,GAAG,YAA0D;AAC5I,cAAM,YAAY,KAAK,IAAI;AAE3B,YAAI;AAEH,gBAAM,EAAE,SAAS,cAAc,MAAM,WAAW,aAAa,IAAI,wBAAwB,SAAS,MAAM,OAAO;AAG/G,gBAAM,cAAc,UAAU,SAAS,IAAI,GAAG,YAAY,IAAI,UAAU,KAAK,GAAG,CAAC,KAAK;AAEtF,gBAAM,EAAE,QAAQ,QAAQ,SAAS,IAAI,MAAM,qBAAqB,SAAS,cAAc,WAAW;AAElG,iBAAO;AAAA,YACN;AAAA,YACA;AAAA,YACA;AAAA,YACA,eAAe,KAAK,IAAI,IAAI;AAAA,YAC5B,WAAW,QAAQ,UAAU,QAAQ;AAAA,YACrC,UAAU;AAAA,YACV;AAAA,YACA,GAAI,gBAAgB,EAAE,KAAK,GAAG;AAAA,UAC/B;AAAA,QACD,SAAS,OAAO;AAEf,iBAAO;AAAA,YACN,QAAQ;AAAA,YACR,QAAQ,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,YAC7D,UAAU;AAAA;AAAA,YACV,eAAe,KAAK,IAAI,IAAI;AAAA,YAC5B,WAAW,QAAQ,UAAU,QAAQ;AAAA,YACrC,UAAU;AAAA,UACX;AAAA,QACD;AAAA,MACD;AAAA,MAEA,SAAS,OAAO,YAAmD;AAClE,eAAO;AAAA,UACN,IAAI,QAAQ,UAAU,QAAQ;AAAA,UAC9B,UAAU;AAAA,UACV,SAAS,QAAQ,MAAM,SAAS,OAAO,SAAS,IAAI,IAAI,WAAW;AAAA,UACnE,QAAQ,qBAAqB,QAAQ,MAAM;AAAA,UAC3C,WAAW,QAAQ,UAAU,YAAY,IAAI,KAAK,QAAQ,SAAS,SAAS,IAAI,oBAAI,KAAK;AAAA,UACzF,SAAS,uBAAuB,QAAQ,MAAM,SAAS,GAAG;AAAA,UAC1D,UAAU;AAAA,YACT,GAAG,QAAQ,UAAU;AAAA,UACtB;AAAA,QACD;AAAA,MACD;AAAA,MAEA,QAAQ,OAAO,SAA0B,YAalB;AACtB,YAAI;AAEH,gBAAM,WAAW,QAAQ,gBAAgB,WAAW,SAAY,QAAQ,eAAe,SAAS;AAGhG,gBAAM,iBAAiB;AAAA,YACtB,+BAA+B;AAAA,YAC/B,gCAAgC;AAAA,YAChC,gCAAgC;AAAA,YAChC,oCAAoC;AAAA,YACpC,iCAAiC;AAAA,YACjC,0BAA0B;AAAA,YAC1B,QAAQ;AAAA,UACT;AAGA,gBAAM,kBAAkB,QAAQ,SAAS,YAAY;AAGrD,gBAAM,UAAU,MAAM,QAAQ,SAAS,kBAAkB;AAAA,YACxD,UAAU;AAAA,cACT,MAAM,gBAAgB,QAAQ,IAAI,IAAI,WAAW,WAAW,SAAS;AAAA,YACtE;AAAA,YACA,MAAM;AAAA,cACL,MAAM,QAAQ;AAAA,cACd,QAAQ;AAAA,cACR;AAAA,cACA,gBAAgB,QAAQ,SAAS,WAAW;AAAA,cAC5C,cAAc,QAAQ;AAAA,cACtB,WAAW,QAAQ;AAAA,cACnB,KAAK,QAAQ,MAAM,GAAG,KAAK,KAAK,QAAQ,MAAM,GAAI,CAAC,MAAM;AAAA,YAC1D;AAAA,UACD,CAAC;AAGD,gBAAM,MAAM,QAAQ,MAAM;AAC1B,cAAI,CAAC,KAAK;AACT,kBAAM,IAAI,MAAM,sCAAsC,QAAQ,IAAI,EAAE;AAAA,UACrE;AAGA,cAAI,CAAC,UAAU;AAEd,kBAAM,gBAAgB,QAAQ,gBAAgB,sBAAsB;AACpE,kBAAM,YAAY,IAAI,KAAK,KAAK,IAAI,IAAI,gBAAgB,KAAK,GAAI;AACjE,kBAAM,QAAQ,MAAM,QAAQ,OAAO,OAAO,SAAS;AAGnD,kBAAM,YAAY,IAAI,SAAS,GAAG,IAAI,MAAM;AAC5C,mBAAO,GAAG,GAAG,GAAG,SAAS,oBAAoB,MAAM,KAAK;AAAA,UACzD;AAGA,iBAAO;AAAA,QACR,SAAS,OAAO;AACf,gBAAM,IAAI;AAAA,YACT,6CAA6C,QAAQ,IAAI,KAAK,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,UACrH;AAAA,QACD;AAAA,MACD;AAAA;AAAA,MAGA,YAAY;AAAA,QACX,UAAU,OAAO,SAA0B,SAAkC;AAC5E,gBAAM,SAAS,MAAM,QAAQ,GAAG,KAAK,IAAI;AACzC,iBAAO,UAAU;AAAA,QAClB;AAAA,QAEA,WAAW,OAAO,SAA0B,MAAc,YAAmC;AAC5F,gBAAM,QAAQ,GAAG,MAAM,MAAM,OAAO;AAAA,QACrC;AAAA,QAEA,OAAO,OAAO,SAA0B,SAAgC;AACvE,gBAAM,QAAQ,GAAG,MAAM,IAAI;AAAA,QAC5B;AAAA,QAEA,SAAS,OAAO,SAA0B,SAAuC;AAChF,gBAAM,SAAS,MAAM,QAAQ,GAAG,GAAG,IAAI;AACvC,gBAAM,QAAQ,OAAO,SAAS,CAAC;AAC/B,gBAAM,cAAc,OAAO,kBAAkB,CAAC;AAC9C,cAAI,UAAU,CAAC;AACf,qBAAW,QAAQ,OAAO;AACzB,oBAAQ,KAAK;AAAA,cACZ,MAAM,KAAK;AAAA,cACX,MAAM,GAAG,IAAI,IAAI,KAAK,IAAI;AAAA,cAC1B,aAAa;AAAA,cACb,MAAM,KAAK,QAAQ;AAAA,cACnB,cAAc,IAAI,KAAK,KAAK,gBAAgB,KAAK,IAAI,CAAC;AAAA,YACvD,CAAC;AAAA,UACF;AACA,qBAAW,aAAa,aAAa;AACpC,oBAAQ,KAAK;AAAA,cACZ,MAAM,UAAU;AAAA,cAChB,MAAM,GAAG,IAAI,IAAI,UAAU,IAAI;AAAA,cAC/B,aAAa;AAAA,cACb,MAAM;AAAA,cACN,cAAc,oBAAI,KAAK;AAAA,YACxB,CAAC;AAAA,UACF;AACA,iBAAO;AAAA,QACR;AAAA,QAEA,QAAQ,OAAO,SAA0B,SAAmC;AAC3E,cAAI;AACH,kBAAM,QAAQ,GAAG,KAAK,IAAI;AAC1B,mBAAO;AAAA,UACR,QAAQ;AACP,gBAAI;AACH,oBAAM,QAAQ,GAAG,GAAG,IAAI;AACxB,qBAAO;AAAA,YACR,QAAQ;AACP,qBAAO;AAAA,YACR;AAAA,UACD;AAAA,QACD;AAAA,QAEA,QAAQ,OAAO,SAA0B,SAAgC;AACxE,gBAAM,QAAQ,GAAG,GAAG,IAAI;AAAA,QACzB;AAAA,MACD;AAAA;AAAA,MAGA,aAAa,CAAC,YAA8C;AAC3D,eAAO;AAAA,MACR;AAAA,IACD;AAAA,EACD;AACD,CAAC;AAeM,SAAS,oBAAoB,QAgBlC;AACD,QAAM,WAAW,OAAO,MAAM;AAE9B,SAAO;AAAA,IACN,SAAS;AAAA,MACR,QAAQ,YAAY;AACnB,cAAM,UAAU,MAAM,SAAS,QAAQ,OAAO;AAC9C,eAAO;AAAA,UACN,GAAG;AAAA,UACH,aAAa,MAAuB;AACnC,mBAAO,QAAQ,YAAY;AAAA,UAC5B;AAAA,QACD;AAAA,MACD;AAAA,IACD;AAAA,EACD;AACD;AAEA,eAAe,iBAAiB,QAAsB;AAErD,MAAI;AACH,UAAM,SAAS,aAAa;AAAA,EAC7B,SAAS,OAAO;AAEf,QAAI,OAAO,aAAa,QAAQ,IAAI,oBAAoB,OAAO,YAAY,aAAa;AACvF,cAAQ,IAAI,eAAe,OAAO,aAAa,QAAQ,IAAI;AAAA,IAC5D;AACA,QAAI,OAAO,UAAU,QAAQ,IAAI,kBAAkB,OAAO,YAAY,aAAa;AAClF,cAAQ,IAAI,aAAa,OAAO,UAAU,QAAQ,IAAI;AAAA,IACvD;AACA,QAAI;AACH,YAAM,SAAS,aAAa;AAAA,IAC7B,SAASA,QAAO;AACf,YAAM,IAAI,MAAM,uJAAuJ;AAAA,IACxK;AAAA,EACD;AACD;AAMA,SAAS,uBAAuB,KAA0C;AACzE,MAAI,CAAC,IAAK,QAAO;AAGjB,MAAI,OAAO,QAAQ,UAAU;AAC5B,WAAO,MAAM;AAAA,EACd;AAGA,QAAM,QAAQ,IAAI,MAAM,kBAAkB;AAC1C,MAAI,CAAC,MAAO,QAAO;AAEnB,QAAM,QAAQ,SAAS,MAAM,CAAC,GAAG,EAAE;AACnC,QAAM,OAAO,MAAM,CAAC,KAAK;AAEzB,UAAQ,MAAM;AAAA,IACb,KAAK;AAAK,aAAO,QAAQ;AAAA;AAAA,IACzB,KAAK;AAAK,aAAO,QAAQ,KAAK;AAAA;AAAA,IAC9B,KAAK;AAAK,aAAO,QAAQ,KAAK,KAAK;AAAA;AAAA,IACnC,KAAK;AAAK,aAAO,QAAQ,KAAK,KAAK,KAAK;AAAA;AAAA,IACxC;AAAS,aAAO;AAAA,EACjB;AACD;AAEA,SAAS,qBAAqB,QAA2C;AACxE,UAAQ,QAAQ,YAAY,GAAG;AAAA,IAC9B,KAAK;AAAY,aAAO;AAAA,IACxB,KAAK;AAAY,aAAO;AAAA,IACxB,KAAK;AAAU,aAAO;AAAA,IACtB;AAAS,aAAO;AAAA,EACjB;AACD;AAMA,eAAe,qBACd,SACA,cACA,SACgE;AAEhE,QAAM,SAAS,MAAM,QAAQ,QAAQ,KAAK,EAAE,QAAQ,CAAC;AAErD,MAAI,CAAC,cAAc;AAElB,UAAM,QAAQ,QAAQ,KAAK,OAAO,IAAI;AAAA,EACvC;AAGA,QAAM,gBAAgB,MAAM,QAAQ,QAAQ,IAAI,OAAO,IAAI;AAG3D,SAAO;AAAA,IACN,QAAQ,cAAc;AAAA,IACtB,QAAQ,cAAc;AAAA,IACtB,UAAU,cAAc,YAAY;AAAA,EACrC;AACD;","names":["error"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@computesdk/blaxel",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"description": "Blaxel provider for ComputeSDK",
|
|
5
|
+
"author": "ComputeSDK",
|
|
6
|
+
"license": "MIT",
|
|
7
|
+
"main": "./dist/index.js",
|
|
8
|
+
"module": "./dist/index.mjs",
|
|
9
|
+
"types": "./dist/index.d.ts",
|
|
10
|
+
"exports": {
|
|
11
|
+
".": {
|
|
12
|
+
"types": "./dist/index.d.ts",
|
|
13
|
+
"import": "./dist/index.mjs",
|
|
14
|
+
"require": "./dist/index.js"
|
|
15
|
+
}
|
|
16
|
+
},
|
|
17
|
+
"files": [
|
|
18
|
+
"dist"
|
|
19
|
+
],
|
|
20
|
+
"dependencies": {
|
|
21
|
+
"@blaxel/core": "^0.2.38",
|
|
22
|
+
"computesdk": "1.7.0"
|
|
23
|
+
},
|
|
24
|
+
"keywords": [
|
|
25
|
+
"blaxel",
|
|
26
|
+
"sandbox",
|
|
27
|
+
"code-execution",
|
|
28
|
+
"cloud",
|
|
29
|
+
"compute"
|
|
30
|
+
],
|
|
31
|
+
"repository": {
|
|
32
|
+
"type": "git",
|
|
33
|
+
"url": "https://github.com/computesdk/computesdk.git",
|
|
34
|
+
"directory": "packages/blaxel"
|
|
35
|
+
},
|
|
36
|
+
"homepage": "https://github.com/computesdk/computesdk/tree/main/packages/blaxel",
|
|
37
|
+
"bugs": {
|
|
38
|
+
"url": "https://github.com/computesdk/computesdk/issues"
|
|
39
|
+
},
|
|
40
|
+
"devDependencies": {
|
|
41
|
+
"@types/node": "^20.0.0",
|
|
42
|
+
"@vitest/coverage-v8": "^1.0.0",
|
|
43
|
+
"eslint": "^8.37.0",
|
|
44
|
+
"rimraf": "^5.0.0",
|
|
45
|
+
"tsup": "^8.0.0",
|
|
46
|
+
"typescript": "^5.0.0",
|
|
47
|
+
"vitest": "^1.0.0",
|
|
48
|
+
"@computesdk/test-utils": "1.3.1"
|
|
49
|
+
},
|
|
50
|
+
"scripts": {
|
|
51
|
+
"build": "tsup",
|
|
52
|
+
"clean": "rimraf dist",
|
|
53
|
+
"dev": "tsup --watch",
|
|
54
|
+
"test": "vitest run",
|
|
55
|
+
"test:watch": "vitest watch",
|
|
56
|
+
"test:coverage": "vitest run --coverage",
|
|
57
|
+
"typecheck": "tsc --noEmit",
|
|
58
|
+
"lint": "eslint"
|
|
59
|
+
}
|
|
60
|
+
}
|