@computesdk/blaxel 1.3.28 → 1.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -31,7 +31,6 @@ var blaxel = (0, import_provider.defineProvider)({
31
31
  sandbox: {
32
32
  // Collection operations (map to compute.sandbox.*)
33
33
  create: async (config, options) => {
34
- await handleBlaxelAuth(config);
35
34
  let image = config.image || "blaxel/prod-base:latest";
36
35
  if (!config.image && options?.runtime) {
37
36
  switch (options.runtime) {
@@ -89,7 +88,6 @@ var blaxel = (0, import_provider.defineProvider)({
89
88
  }
90
89
  },
91
90
  getById: async (config, sandboxId) => {
92
- await handleBlaxelAuth(config);
93
91
  try {
94
92
  const sandbox = await import_core.SandboxInstance.get(sandboxId);
95
93
  if (!sandbox) {
@@ -104,7 +102,6 @@ var blaxel = (0, import_provider.defineProvider)({
104
102
  }
105
103
  },
106
104
  list: async (config) => {
107
- await handleBlaxelAuth(config);
108
105
  const sandboxList = await import_core.SandboxInstance.list();
109
106
  return sandboxList.map((sandbox) => ({
110
107
  sandbox,
@@ -112,7 +109,6 @@ var blaxel = (0, import_provider.defineProvider)({
112
109
  }));
113
110
  },
114
111
  destroy: async (config, sandboxId) => {
115
- await handleBlaxelAuth(config);
116
112
  try {
117
113
  await import_core.SandboxInstance.delete(sandboxId);
118
114
  } catch (error) {
@@ -307,23 +303,6 @@ ${stderr}`.trim() : stdout;
307
303
  }
308
304
  }
309
305
  });
310
- async function handleBlaxelAuth(config) {
311
- if (config.workspace) {
312
- process.env.BL_WORKSPACE = config.workspace;
313
- } else if (!process.env.BL_WORKSPACE && process.env.BLAXEL_WORKSPACE) {
314
- process.env.BL_WORKSPACE = process.env.BLAXEL_WORKSPACE;
315
- }
316
- if (config.apiKey) {
317
- process.env.BL_API_KEY = config.apiKey;
318
- } else if (!process.env.BL_API_KEY && process.env.BLAXEL_API_KEY) {
319
- process.env.BL_API_KEY = process.env.BLAXEL_API_KEY;
320
- }
321
- try {
322
- await import_core.settings.authenticate();
323
- } catch (error) {
324
- throw new Error("Blaxel authentication failed. Please check the following documents for more information: https://docs.blaxel.ai/Security/Access-tokens#using-api-keys");
325
- }
326
- }
327
306
  function parseTTLToMilliseconds(ttl) {
328
307
  if (!ttl) return 3e5;
329
308
  if (typeof ttl === "number") {
package/dist/index.js.map CHANGED
@@ -1 +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 { defineProvider, escapeShellArg } from '@computesdk/provider';\n\nimport type { Runtime, CodeResult, CommandResult, SandboxInfo, CreateSandboxOptions, FileEntry, RunCommandOptions } from '@computesdk/provider';\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 = defineProvider<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\tname: options?.sandboxId || `blaxel-${Date.now()}`,\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\tconst errorDetail = error instanceof Error\n\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t: typeof error === 'object' && error !== null\n\t\t\t\t\t\t\t? JSON.stringify(error)\n\t\t\t\t\t\t\t: String(error);\n\n\t\t\t\t\tif (\n\t\t\t\t\t\terrorDetail.includes('unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Forbidden') ||\n\t\t\t\t\t\terrorDetail.includes('API key')\n\t\t\t\t\t) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel authentication failed: ${errorDetail}`\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t\tif (errorDetail.includes('quota') || errorDetail.includes('limit')) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel quota exceeded: ${errorDetail}`\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: ${errorDetail}`\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<CodeResult> => {\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, 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\t// Combine stdout and stderr into output\n\t\t\t\t\tconst output = stderr ? `${stdout}\\n${stderr}`.trim() : stdout;\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\toutput,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\tlanguage: effectiveRuntime\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\toutput: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 1,\n\t\t\t\t\t\tlanguage: runtime || 'node'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\trunCommand: async (sandbox: SandboxInstance, command: string, options?: RunCommandOptions): Promise<CommandResult> => {\n\t\t\tconst startTime = Date.now();\n\n\t\t\ttry {\n\t\t\t\t// Build command with options\n\t\t\t\tlet fullCommand = command;\n\t\t\t\t\n\t\t\t\t// Handle environment variables\n\t\t\t\tif (options?.env && Object.keys(options.env).length > 0) {\n\t\t\t\t\tconst envPrefix = Object.entries(options.env)\n\t\t\t\t\t\t.map(([k, v]) => `${k}=\"${escapeShellArg(v)}\"`)\n\t\t\t\t\t\t.join(' ');\n\t\t\t\t\tfullCommand = `${envPrefix} ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle working directory\n\t\t\t\tif (options?.cwd) {\n\t\t\t\t\tfullCommand = `cd \"${escapeShellArg(options.cwd)}\" && ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle background execution\n\t\t\t\tif (options?.background) {\n\t\t\t\t\tfullCommand = `nohup ${fullCommand} > /dev/null 2>&1 &`;\n\t\t\t\t}\n\n\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, fullCommand);\n\n\t\t\t\treturn {\n\t\t\t\t\tstdout,\n\t\t\t\t\tstderr,\n\t\t\t\t\texitCode,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t} catch (error) {\n\t\t\t\treturn {\n\t\t\t\t\tstdout: '',\n\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\texitCode: 127,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t}\n\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\ttype: 'file' as const,\n\t\t\t\t\t\t\tsize: file.size || 0,\n\t\t\t\t\t\t\tmodified: 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\ttype: 'directory' as const,\n\t\t\t\t\t\t\tsize: 0,\n\t\t\t\t\t\t\tmodified: 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\nasync function handleBlaxelAuth(config: BlaxelConfig) {\n\t// Always apply config values to environment before authenticating.\n\t// In the gateway path, credentials come from HTTP headers via config\n\t// and must be set as env vars before @blaxel/core's settings.authenticate() is called.\n\tif (config.workspace) {\n\t\tprocess.env.BL_WORKSPACE = config.workspace;\n\t} else if (!process.env.BL_WORKSPACE && process.env.BLAXEL_WORKSPACE) {\n\t\tprocess.env.BL_WORKSPACE = process.env.BLAXEL_WORKSPACE;\n\t}\n\n\tif (config.apiKey) {\n\t\tprocess.env.BL_API_KEY = config.apiKey;\n\t} else if (!process.env.BL_API_KEY && process.env.BLAXEL_API_KEY) {\n\t\tprocess.env.BL_API_KEY = process.env.BLAXEL_API_KEY;\n\t}\n\n\ttry {\n\t\tawait settings.authenticate();\n\t} catch (error) {\n\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}\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): 'running' | 'stopped' | 'error' {\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\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\t// Wait for process completion\n\tawait sandbox.process.wait(result.name);\n\n\t// Get final process result for exit code\n\tconst processResult = await sandbox.process.get(result.name);\n\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":"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","names":[]}
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 } from '@blaxel/core';\nimport { defineProvider, escapeShellArg } from '@computesdk/provider';\n\nimport type { Runtime, CodeResult, CommandResult, SandboxInfo, CreateSandboxOptions, FileEntry, RunCommandOptions } from '@computesdk/provider';\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 = defineProvider<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\t\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\tname: options?.sandboxId || `blaxel-${Date.now()}`,\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\tconst errorDetail = error instanceof Error\n\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t: typeof error === 'object' && error !== null\n\t\t\t\t\t\t\t? JSON.stringify(error)\n\t\t\t\t\t\t\t: String(error);\n\n\t\t\t\t\tif (\n\t\t\t\t\t\terrorDetail.includes('unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Forbidden') ||\n\t\t\t\t\t\terrorDetail.includes('API key')\n\t\t\t\t\t) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel authentication failed: ${errorDetail}`\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t\tif (errorDetail.includes('quota') || errorDetail.includes('limit')) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel quota exceeded: ${errorDetail}`\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: ${errorDetail}`\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\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\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\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<CodeResult> => {\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, 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\t// Combine stdout and stderr into output\n\t\t\t\t\tconst output = stderr ? `${stdout}\\n${stderr}`.trim() : stdout;\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\toutput,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\tlanguage: effectiveRuntime\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\toutput: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 1,\n\t\t\t\t\t\tlanguage: runtime || 'node'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\trunCommand: async (sandbox: SandboxInstance, command: string, options?: RunCommandOptions): Promise<CommandResult> => {\n\t\t\tconst startTime = Date.now();\n\n\t\t\ttry {\n\t\t\t\t// Build command with options\n\t\t\t\tlet fullCommand = command;\n\t\t\t\t\n\t\t\t\t// Handle environment variables\n\t\t\t\tif (options?.env && Object.keys(options.env).length > 0) {\n\t\t\t\t\tconst envPrefix = Object.entries(options.env)\n\t\t\t\t\t\t.map(([k, v]) => `${k}=\"${escapeShellArg(v)}\"`)\n\t\t\t\t\t\t.join(' ');\n\t\t\t\t\tfullCommand = `${envPrefix} ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle working directory\n\t\t\t\tif (options?.cwd) {\n\t\t\t\t\tfullCommand = `cd \"${escapeShellArg(options.cwd)}\" && ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle background execution\n\t\t\t\tif (options?.background) {\n\t\t\t\t\tfullCommand = `nohup ${fullCommand} > /dev/null 2>&1 &`;\n\t\t\t\t}\n\n\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, fullCommand);\n\n\t\t\t\treturn {\n\t\t\t\t\tstdout,\n\t\t\t\t\tstderr,\n\t\t\t\t\texitCode,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t} catch (error) {\n\t\t\t\treturn {\n\t\t\t\t\tstdout: '',\n\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\texitCode: 127,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t}\n\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\ttype: 'file' as const,\n\t\t\t\t\t\t\tsize: file.size || 0,\n\t\t\t\t\t\t\tmodified: 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\ttype: 'directory' as const,\n\t\t\t\t\t\t\tsize: 0,\n\t\t\t\t\t\t\tmodified: 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 * 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): 'running' | 'stopped' | 'error' {\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\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\t// Wait for process completion\n\tawait sandbox.process.wait(result.name);\n\n\t// Get final process result for exit code\n\tconst processResult = await sandbox.process.get(result.name);\n\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":"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","names":[]}
package/dist/index.mjs CHANGED
@@ -1,5 +1,5 @@
1
1
  // src/index.ts
2
- import { SandboxInstance, settings } from "@blaxel/core";
2
+ import { SandboxInstance } from "@blaxel/core";
3
3
  import { defineProvider, escapeShellArg } from "@computesdk/provider";
4
4
  var blaxel = defineProvider({
5
5
  name: "blaxel",
@@ -7,7 +7,6 @@ var blaxel = defineProvider({
7
7
  sandbox: {
8
8
  // Collection operations (map to compute.sandbox.*)
9
9
  create: async (config, options) => {
10
- await handleBlaxelAuth(config);
11
10
  let image = config.image || "blaxel/prod-base:latest";
12
11
  if (!config.image && options?.runtime) {
13
12
  switch (options.runtime) {
@@ -65,7 +64,6 @@ var blaxel = defineProvider({
65
64
  }
66
65
  },
67
66
  getById: async (config, sandboxId) => {
68
- await handleBlaxelAuth(config);
69
67
  try {
70
68
  const sandbox = await SandboxInstance.get(sandboxId);
71
69
  if (!sandbox) {
@@ -80,7 +78,6 @@ var blaxel = defineProvider({
80
78
  }
81
79
  },
82
80
  list: async (config) => {
83
- await handleBlaxelAuth(config);
84
81
  const sandboxList = await SandboxInstance.list();
85
82
  return sandboxList.map((sandbox) => ({
86
83
  sandbox,
@@ -88,7 +85,6 @@ var blaxel = defineProvider({
88
85
  }));
89
86
  },
90
87
  destroy: async (config, sandboxId) => {
91
- await handleBlaxelAuth(config);
92
88
  try {
93
89
  await SandboxInstance.delete(sandboxId);
94
90
  } catch (error) {
@@ -283,23 +279,6 @@ ${stderr}`.trim() : stdout;
283
279
  }
284
280
  }
285
281
  });
286
- async function handleBlaxelAuth(config) {
287
- if (config.workspace) {
288
- process.env.BL_WORKSPACE = config.workspace;
289
- } else if (!process.env.BL_WORKSPACE && process.env.BLAXEL_WORKSPACE) {
290
- process.env.BL_WORKSPACE = process.env.BLAXEL_WORKSPACE;
291
- }
292
- if (config.apiKey) {
293
- process.env.BL_API_KEY = config.apiKey;
294
- } else if (!process.env.BL_API_KEY && process.env.BLAXEL_API_KEY) {
295
- process.env.BL_API_KEY = process.env.BLAXEL_API_KEY;
296
- }
297
- try {
298
- await settings.authenticate();
299
- } catch (error) {
300
- throw new Error("Blaxel authentication failed. Please check the following documents for more information: https://docs.blaxel.ai/Security/Access-tokens#using-api-keys");
301
- }
302
- }
303
282
  function parseTTLToMilliseconds(ttl) {
304
283
  if (!ttl) return 3e5;
305
284
  if (typeof ttl === "number") {
@@ -1 +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 { defineProvider, escapeShellArg } from '@computesdk/provider';\n\nimport type { Runtime, CodeResult, CommandResult, SandboxInfo, CreateSandboxOptions, FileEntry, RunCommandOptions } from '@computesdk/provider';\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 = defineProvider<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\tname: options?.sandboxId || `blaxel-${Date.now()}`,\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\tconst errorDetail = error instanceof Error\n\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t: typeof error === 'object' && error !== null\n\t\t\t\t\t\t\t? JSON.stringify(error)\n\t\t\t\t\t\t\t: String(error);\n\n\t\t\t\t\tif (\n\t\t\t\t\t\terrorDetail.includes('unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Forbidden') ||\n\t\t\t\t\t\terrorDetail.includes('API key')\n\t\t\t\t\t) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel authentication failed: ${errorDetail}`\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t\tif (errorDetail.includes('quota') || errorDetail.includes('limit')) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel quota exceeded: ${errorDetail}`\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: ${errorDetail}`\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<CodeResult> => {\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, 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\t// Combine stdout and stderr into output\n\t\t\t\t\tconst output = stderr ? `${stdout}\\n${stderr}`.trim() : stdout;\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\toutput,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\tlanguage: effectiveRuntime\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\toutput: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 1,\n\t\t\t\t\t\tlanguage: runtime || 'node'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\trunCommand: async (sandbox: SandboxInstance, command: string, options?: RunCommandOptions): Promise<CommandResult> => {\n\t\t\tconst startTime = Date.now();\n\n\t\t\ttry {\n\t\t\t\t// Build command with options\n\t\t\t\tlet fullCommand = command;\n\t\t\t\t\n\t\t\t\t// Handle environment variables\n\t\t\t\tif (options?.env && Object.keys(options.env).length > 0) {\n\t\t\t\t\tconst envPrefix = Object.entries(options.env)\n\t\t\t\t\t\t.map(([k, v]) => `${k}=\"${escapeShellArg(v)}\"`)\n\t\t\t\t\t\t.join(' ');\n\t\t\t\t\tfullCommand = `${envPrefix} ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle working directory\n\t\t\t\tif (options?.cwd) {\n\t\t\t\t\tfullCommand = `cd \"${escapeShellArg(options.cwd)}\" && ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle background execution\n\t\t\t\tif (options?.background) {\n\t\t\t\t\tfullCommand = `nohup ${fullCommand} > /dev/null 2>&1 &`;\n\t\t\t\t}\n\n\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, fullCommand);\n\n\t\t\t\treturn {\n\t\t\t\t\tstdout,\n\t\t\t\t\tstderr,\n\t\t\t\t\texitCode,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t} catch (error) {\n\t\t\t\treturn {\n\t\t\t\t\tstdout: '',\n\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\texitCode: 127,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t}\n\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\ttype: 'file' as const,\n\t\t\t\t\t\t\tsize: file.size || 0,\n\t\t\t\t\t\t\tmodified: 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\ttype: 'directory' as const,\n\t\t\t\t\t\t\tsize: 0,\n\t\t\t\t\t\t\tmodified: 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\nasync function handleBlaxelAuth(config: BlaxelConfig) {\n\t// Always apply config values to environment before authenticating.\n\t// In the gateway path, credentials come from HTTP headers via config\n\t// and must be set as env vars before @blaxel/core's settings.authenticate() is called.\n\tif (config.workspace) {\n\t\tprocess.env.BL_WORKSPACE = config.workspace;\n\t} else if (!process.env.BL_WORKSPACE && process.env.BLAXEL_WORKSPACE) {\n\t\tprocess.env.BL_WORKSPACE = process.env.BLAXEL_WORKSPACE;\n\t}\n\n\tif (config.apiKey) {\n\t\tprocess.env.BL_API_KEY = config.apiKey;\n\t} else if (!process.env.BL_API_KEY && process.env.BLAXEL_API_KEY) {\n\t\tprocess.env.BL_API_KEY = process.env.BLAXEL_API_KEY;\n\t}\n\n\ttry {\n\t\tawait settings.authenticate();\n\t} catch (error) {\n\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}\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): 'running' | 'stopped' | 'error' {\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\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\t// Wait for process completion\n\tawait sandbox.process.wait(result.name);\n\n\t// Get final process result for exit code\n\tconst processResult = await sandbox.process.get(result.name);\n\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":"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","names":[]}
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 } from '@blaxel/core';\nimport { defineProvider, escapeShellArg } from '@computesdk/provider';\n\nimport type { Runtime, CodeResult, CommandResult, SandboxInfo, CreateSandboxOptions, FileEntry, RunCommandOptions } from '@computesdk/provider';\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 = defineProvider<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\t\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\tname: options?.sandboxId || `blaxel-${Date.now()}`,\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\tconst errorDetail = error instanceof Error\n\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t: typeof error === 'object' && error !== null\n\t\t\t\t\t\t\t? JSON.stringify(error)\n\t\t\t\t\t\t\t: String(error);\n\n\t\t\t\t\tif (\n\t\t\t\t\t\terrorDetail.includes('unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Unauthorized') ||\n\t\t\t\t\t\terrorDetail.includes('Forbidden') ||\n\t\t\t\t\t\terrorDetail.includes('API key')\n\t\t\t\t\t) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel authentication failed: ${errorDetail}`\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t\tif (errorDetail.includes('quota') || errorDetail.includes('limit')) {\n\t\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t\t`Blaxel quota exceeded: ${errorDetail}`\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: ${errorDetail}`\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\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\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\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<CodeResult> => {\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, 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\t// Combine stdout and stderr into output\n\t\t\t\t\tconst output = stderr ? `${stdout}\\n${stderr}`.trim() : stdout;\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\toutput,\n\t\t\t\t\t\texitCode,\n\t\t\t\t\t\tlanguage: effectiveRuntime\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\toutput: error instanceof Error ? error.message : String(error),\n\t\t\t\t\t\texitCode: 1,\n\t\t\t\t\t\tlanguage: runtime || 'node'\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t},\n\n\t\trunCommand: async (sandbox: SandboxInstance, command: string, options?: RunCommandOptions): Promise<CommandResult> => {\n\t\t\tconst startTime = Date.now();\n\n\t\t\ttry {\n\t\t\t\t// Build command with options\n\t\t\t\tlet fullCommand = command;\n\t\t\t\t\n\t\t\t\t// Handle environment variables\n\t\t\t\tif (options?.env && Object.keys(options.env).length > 0) {\n\t\t\t\t\tconst envPrefix = Object.entries(options.env)\n\t\t\t\t\t\t.map(([k, v]) => `${k}=\"${escapeShellArg(v)}\"`)\n\t\t\t\t\t\t.join(' ');\n\t\t\t\t\tfullCommand = `${envPrefix} ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle working directory\n\t\t\t\tif (options?.cwd) {\n\t\t\t\t\tfullCommand = `cd \"${escapeShellArg(options.cwd)}\" && ${fullCommand}`;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// Handle background execution\n\t\t\t\tif (options?.background) {\n\t\t\t\t\tfullCommand = `nohup ${fullCommand} > /dev/null 2>&1 &`;\n\t\t\t\t}\n\n\t\t\t\tconst { stdout, stderr, exitCode } = await executeWithStreaming(sandbox, fullCommand);\n\n\t\t\t\treturn {\n\t\t\t\t\tstdout,\n\t\t\t\t\tstderr,\n\t\t\t\t\texitCode,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t} catch (error) {\n\t\t\t\treturn {\n\t\t\t\t\tstdout: '',\n\t\t\t\t\tstderr: error instanceof Error ? error.message : String(error),\n\t\t\t\t\texitCode: 127,\n\t\t\t\t\tdurationMs: Date.now() - startTime\n\t\t\t\t};\n\t\t\t}\n\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\ttype: 'file' as const,\n\t\t\t\t\t\t\tsize: file.size || 0,\n\t\t\t\t\t\t\tmodified: 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\ttype: 'directory' as const,\n\t\t\t\t\t\t\tsize: 0,\n\t\t\t\t\t\t\tmodified: 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 * 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): 'running' | 'stopped' | 'error' {\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\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\t// Wait for process completion\n\tawait sandbox.process.wait(result.name);\n\n\t// Get final process result for exit code\n\tconst processResult = await sandbox.process.get(result.name);\n\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":"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","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@computesdk/blaxel",
3
- "version": "1.3.28",
3
+ "version": "1.4.1",
4
4
  "description": "Blaxel provider for ComputeSDK - lightweight cloud sandboxes for code execution",
5
5
  "author": "ComputeSDK",
6
6
  "license": "MIT",
@@ -19,8 +19,8 @@
19
19
  ],
20
20
  "dependencies": {
21
21
  "@blaxel/core": "^0.2.42",
22
- "computesdk": "1.21.1",
23
- "@computesdk/provider": "1.0.20"
22
+ "@computesdk/provider": "1.0.21",
23
+ "computesdk": "2.0.1"
24
24
  },
25
25
  "keywords": [
26
26
  "computesdk",