@cloudflare/sandbox 0.3.1 → 0.3.2

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.
Files changed (59) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/Dockerfile +22 -24
  3. package/README.md +1 -1
  4. package/container_src/bun.lock +31 -77
  5. package/container_src/control-process.ts +1 -1
  6. package/container_src/index.ts +34 -43
  7. package/container_src/interpreter-service.ts +276 -0
  8. package/container_src/isolation.ts +1 -1
  9. package/container_src/mime-processor.ts +1 -1
  10. package/container_src/package.json +4 -4
  11. package/container_src/runtime/executors/javascript/node_executor.ts +123 -0
  12. package/container_src/runtime/executors/python/ipython_executor.py +338 -0
  13. package/container_src/runtime/executors/typescript/ts_executor.ts +138 -0
  14. package/container_src/runtime/process-pool.ts +464 -0
  15. package/container_src/startup.sh +6 -79
  16. package/dist/{chunk-LALY4SFU.js → chunk-FXYPFGOZ.js} +10 -10
  17. package/dist/chunk-FXYPFGOZ.js.map +1 -0
  18. package/dist/{chunk-LFLJGISB.js → chunk-H4PW2LGW.js} +6 -6
  19. package/dist/chunk-H4PW2LGW.js.map +1 -0
  20. package/dist/{chunk-FKBV7CZS.js → chunk-JTKON2SH.js} +9 -9
  21. package/dist/chunk-JTKON2SH.js.map +1 -0
  22. package/dist/{chunk-EGC5IYXA.js → chunk-W7TVRPBG.js} +2 -2
  23. package/dist/chunk-W7TVRPBG.js.map +1 -0
  24. package/dist/{chunk-BEQUGUY4.js → chunk-Z6OZPC6U.js} +9 -6
  25. package/dist/chunk-Z6OZPC6U.js.map +1 -0
  26. package/dist/{client-Dny_ro_v.d.ts → client-COGWU6bz.d.ts} +3 -3
  27. package/dist/client.d.ts +1 -1
  28. package/dist/errors.d.ts +9 -9
  29. package/dist/errors.js +5 -5
  30. package/dist/index.d.ts +2 -2
  31. package/dist/index.js +13 -11
  32. package/dist/interpreter-client.d.ts +4 -0
  33. package/dist/interpreter-client.js +9 -0
  34. package/dist/interpreter-types.d.ts +5 -5
  35. package/dist/interpreter-types.js +1 -1
  36. package/dist/interpreter.d.ts +2 -2
  37. package/dist/interpreter.js +2 -2
  38. package/dist/request-handler.d.ts +1 -1
  39. package/dist/request-handler.js +5 -5
  40. package/dist/sandbox.d.ts +1 -1
  41. package/dist/sandbox.js +5 -5
  42. package/package.json +1 -1
  43. package/src/errors.ts +15 -14
  44. package/src/index.ts +16 -5
  45. package/src/{jupyter-client.ts → interpreter-client.ts} +6 -3
  46. package/src/interpreter-types.ts +102 -95
  47. package/src/interpreter.ts +8 -8
  48. package/src/sandbox.ts +3 -3
  49. package/container_src/jupyter-server.ts +0 -579
  50. package/container_src/jupyter-service.ts +0 -461
  51. package/container_src/jupyter_config.py +0 -48
  52. package/dist/chunk-BEQUGUY4.js.map +0 -1
  53. package/dist/chunk-EGC5IYXA.js.map +0 -1
  54. package/dist/chunk-FKBV7CZS.js.map +0 -1
  55. package/dist/chunk-LALY4SFU.js.map +0 -1
  56. package/dist/chunk-LFLJGISB.js.map +0 -1
  57. package/dist/jupyter-client.d.ts +0 -4
  58. package/dist/jupyter-client.js +0 -9
  59. /package/dist/{jupyter-client.js.map → interpreter-client.js.map} +0 -0
@@ -1,461 +0,0 @@
1
- import { existsSync } from "node:fs";
2
- import { CircuitBreaker } from "./circuit-breaker";
3
- import { type CreateContextRequest, JupyterServer } from "./jupyter-server";
4
-
5
- interface PoolStats {
6
- available: number;
7
- inUse: number;
8
- total: number;
9
- minSize: number;
10
- maxSize: number;
11
- warming: boolean;
12
- }
13
-
14
- interface CircuitBreakerState {
15
- state: string;
16
- failures: number;
17
- lastFailure: number;
18
- isOpen: boolean;
19
- }
20
-
21
- export interface JupyterHealthStatus {
22
- ready: boolean;
23
- initializing: boolean;
24
- error?: string;
25
- checks?: {
26
- httpApi: boolean;
27
- kernelManager: boolean;
28
- markerFile: boolean;
29
- kernelSpawn?: boolean;
30
- websocket?: boolean;
31
- };
32
- timestamp: number;
33
- performance?: {
34
- poolStats?: Record<string, PoolStats>;
35
- activeContexts?: number;
36
- uptime?: number;
37
- circuitBreakers?: {
38
- contextCreation: CircuitBreakerState;
39
- codeExecution: CircuitBreakerState;
40
- kernelCommunication: CircuitBreakerState;
41
- };
42
- };
43
- }
44
-
45
- /**
46
- * Wrapper service that provides graceful degradation for Jupyter functionality
47
- */
48
- export class JupyterService {
49
- private jupyterServer: JupyterServer;
50
- private initPromise: Promise<void> | null = null;
51
- private initialized = false;
52
- private initError: Error | null = null;
53
- private startTime = Date.now();
54
- private circuitBreakers: {
55
- contextCreation: CircuitBreaker;
56
- codeExecution: CircuitBreaker;
57
- kernelCommunication: CircuitBreaker;
58
- };
59
-
60
- constructor() {
61
- this.jupyterServer = new JupyterServer();
62
-
63
- // Initialize circuit breakers for different operations
64
- this.circuitBreakers = {
65
- contextCreation: new CircuitBreaker({
66
- name: "context-creation",
67
- threshold: 3,
68
- timeout: 60000, // 1 minute
69
- }),
70
- codeExecution: new CircuitBreaker({
71
- name: "code-execution",
72
- threshold: 5,
73
- timeout: 30000, // 30 seconds
74
- }),
75
- kernelCommunication: new CircuitBreaker({
76
- name: "kernel-communication",
77
- threshold: 10,
78
- timeout: 20000, // 20 seconds
79
- }),
80
- };
81
- }
82
-
83
- /**
84
- * Initialize Jupyter server with retry logic
85
- */
86
- async initialize(): Promise<void> {
87
- if (this.initialized) return;
88
-
89
- if (!this.initPromise) {
90
- this.initPromise = this.doInitialize()
91
- .then(() => {
92
- this.initialized = true;
93
- console.log("[JupyterService] Initialization complete");
94
- })
95
- .catch((err) => {
96
- this.initError = err;
97
- // Don't null out initPromise on error - keep it so we can return the same error
98
- console.error("[JupyterService] Initialization failed:", err);
99
- throw err;
100
- });
101
- }
102
-
103
- return this.initPromise;
104
- }
105
-
106
- private async doInitialize(): Promise<void> {
107
- // Wait for Jupyter marker file or timeout
108
- const markerCheckPromise = this.waitForMarkerFile();
109
- const timeoutPromise = new Promise<void>((_, reject) => {
110
- setTimeout(
111
- () => reject(new Error("Jupyter initialization timeout")),
112
- 60000
113
- );
114
- });
115
-
116
- try {
117
- await Promise.race([markerCheckPromise, timeoutPromise]);
118
- console.log("[JupyterService] Jupyter process detected via marker file");
119
- } catch (error) {
120
- console.log(
121
- "[JupyterService] Marker file not found yet - proceeding with initialization"
122
- );
123
- }
124
-
125
- // Initialize Jupyter server
126
- await this.jupyterServer.initialize();
127
-
128
- // Pre-warm context pools in background after Jupyter is ready
129
- this.warmContextPools();
130
- }
131
-
132
- /**
133
- * Pre-warm context pools for better performance
134
- */
135
- private async warmContextPools() {
136
- // Delay pre-warming to avoid startup rush that triggers Bun WebSocket bug
137
- await new Promise((resolve) => setTimeout(resolve, 2000));
138
-
139
- try {
140
- console.log(
141
- "[JupyterService] Pre-warming context pools for better performance"
142
- );
143
-
144
- // Only pre-warm Python contexts to avoid Bun WebSocket validation errors
145
- // JavaScript contexts will be created on-demand
146
- await this.jupyterServer.enablePoolWarming("python", 1);
147
-
148
- // Commenting out JavaScript pre-warming due to kernel message validation errors
149
- // The errors appear to be related to Bun's handling of WebSocket messages
150
- // JavaScript contexts still work fine when created on-demand
151
- //
152
- // await new Promise((resolve) => setTimeout(resolve, 1000));
153
- // await this.jupyterServer.enablePoolWarming("javascript", 1);
154
- } catch (error) {
155
- console.error("[JupyterService] Error pre-warming context pools:", error);
156
- console.error(
157
- "[JupyterService] Pre-warming failed but service continues"
158
- );
159
- }
160
- }
161
-
162
- private async waitForMarkerFile(): Promise<void> {
163
- const markerPath = "/tmp/jupyter-ready";
164
- let attempts = 0;
165
- const maxAttempts = 120; // 2 minutes with 1s intervals
166
-
167
- while (attempts < maxAttempts) {
168
- if (existsSync(markerPath)) {
169
- return;
170
- }
171
- attempts++;
172
- await new Promise((resolve) => setTimeout(resolve, 1000));
173
- }
174
-
175
- throw new Error("Marker file not found within timeout");
176
- }
177
-
178
- /**
179
- * Get current health status
180
- */
181
- async getHealthStatus(): Promise<JupyterHealthStatus> {
182
- const status: JupyterHealthStatus = {
183
- ready: this.initialized,
184
- initializing: this.initPromise !== null && !this.initialized,
185
- timestamp: Date.now(),
186
- };
187
-
188
- if (this.initError) {
189
- status.error = this.initError.message;
190
- }
191
-
192
- // Detailed health checks
193
- if (this.initialized || this.initPromise) {
194
- status.checks = {
195
- httpApi: await this.checkHttpApi(),
196
- kernelManager: this.initialized,
197
- markerFile: existsSync("/tmp/jupyter-ready"),
198
- };
199
-
200
- // Advanced health checks only if fully initialized
201
- if (this.initialized) {
202
- const advancedChecks = await this.performAdvancedHealthChecks();
203
- status.checks.kernelSpawn = advancedChecks.kernelSpawn;
204
- status.checks.websocket = advancedChecks.websocket;
205
-
206
- // Performance metrics
207
- status.performance = {
208
- poolStats: await this.jupyterServer.getPoolStats(),
209
- activeContexts: (await this.jupyterServer.listContexts()).length,
210
- uptime: Date.now() - this.startTime,
211
- };
212
- }
213
- }
214
-
215
- // Add circuit breaker status
216
- if (status.performance) {
217
- status.performance.circuitBreakers = {
218
- contextCreation: this.circuitBreakers.contextCreation.getState(),
219
- codeExecution: this.circuitBreakers.codeExecution.getState(),
220
- kernelCommunication:
221
- this.circuitBreakers.kernelCommunication.getState(),
222
- };
223
- }
224
-
225
- return status;
226
- }
227
-
228
- private async checkHttpApi(): Promise<boolean> {
229
- try {
230
- const response = await fetch("http://localhost:8888/api", {
231
- signal: AbortSignal.timeout(2000),
232
- });
233
- return response.ok;
234
- } catch {
235
- return false;
236
- }
237
- }
238
-
239
- /**
240
- * Perform advanced health checks including kernel spawn and WebSocket
241
- */
242
- private async performAdvancedHealthChecks(): Promise<{
243
- kernelSpawn: boolean;
244
- websocket: boolean;
245
- }> {
246
- const checks = {
247
- kernelSpawn: false,
248
- websocket: false,
249
- };
250
-
251
- try {
252
- // Test kernel spawn with timeout
253
- const testContext = await Promise.race([
254
- this.jupyterServer.createContext({ language: "python" }),
255
- new Promise<null>((_, reject) =>
256
- setTimeout(() => reject(new Error("Kernel spawn timeout")), 5000)
257
- ),
258
- ]);
259
-
260
- if (testContext) {
261
- checks.kernelSpawn = true;
262
-
263
- // Test WebSocket by executing simple code
264
- try {
265
- const result = await Promise.race([
266
- this.jupyterServer.executeCode(
267
- testContext.id,
268
- "print('health_check')"
269
- ),
270
- new Promise<null>((_, reject) =>
271
- setTimeout(() => reject(new Error("WebSocket timeout")), 3000)
272
- ),
273
- ]);
274
-
275
- checks.websocket = result !== null;
276
- } catch {
277
- // WebSocket test failed
278
- }
279
-
280
- // Clean up test context
281
- try {
282
- await this.jupyterServer.deleteContext(testContext.id);
283
- } catch {
284
- // Ignore cleanup errors
285
- }
286
- }
287
- } catch (error) {
288
- console.log("[JupyterService] Advanced health check failed:", error);
289
- }
290
-
291
- return checks;
292
- }
293
-
294
- /**
295
- * Ensure Jupyter is initialized before proceeding
296
- * This will wait for initialization to complete or fail
297
- */
298
- private async ensureInitialized(timeoutMs: number = 30000): Promise<void> {
299
- if (this.initialized) return;
300
-
301
- // Start initialization if not already started
302
- if (!this.initPromise) {
303
- this.initPromise = this.initialize();
304
- }
305
-
306
- // Wait for initialization with timeout
307
- try {
308
- await Promise.race([
309
- this.initPromise,
310
- new Promise<never>((_, reject) =>
311
- setTimeout(
312
- () =>
313
- reject(new Error("Timeout waiting for Jupyter initialization")),
314
- timeoutMs
315
- )
316
- ),
317
- ]);
318
- } catch (error) {
319
- // If it's a timeout and Jupyter is still initializing, throw a retryable error
320
- if (
321
- error instanceof Error &&
322
- error.message.includes("Timeout") &&
323
- !this.initError
324
- ) {
325
- throw new JupyterNotReadyError(
326
- "Jupyter is taking longer than expected to initialize. Please try again.",
327
- {
328
- retryAfter: 10,
329
- progress: this.getInitializationProgress(),
330
- }
331
- );
332
- }
333
- // If initialization actually failed, throw the real error
334
- throw new Error(
335
- `Jupyter initialization failed: ${
336
- error instanceof Error ? error.message : "Unknown error"
337
- }`
338
- );
339
- }
340
- }
341
-
342
- /**
343
- * Create context - will wait for Jupyter if still initializing
344
- */
345
- async createContext(req: CreateContextRequest): Promise<any> {
346
- if (!this.initialized) {
347
- console.log(
348
- "[JupyterService] Context creation requested while Jupyter is initializing - waiting..."
349
- );
350
- const startWait = Date.now();
351
- await this.ensureInitialized();
352
- const waitTime = Date.now() - startWait;
353
- console.log(
354
- `[JupyterService] Jupyter ready after ${waitTime}ms wait - proceeding with context creation`
355
- );
356
- }
357
-
358
- // Use circuit breaker for context creation
359
- return await this.circuitBreakers.contextCreation.execute(async () => {
360
- return await this.jupyterServer.createContext(req);
361
- });
362
- }
363
-
364
- /**
365
- * Execute code - will wait for Jupyter if still initializing
366
- */
367
- async executeCode(
368
- contextId: string | undefined,
369
- code: string,
370
- language?: string
371
- ): Promise<Response> {
372
- if (!this.initialized) {
373
- console.log(
374
- "[JupyterService] Code execution requested while Jupyter is initializing - waiting..."
375
- );
376
- const startWait = Date.now();
377
- await this.ensureInitialized();
378
- const waitTime = Date.now() - startWait;
379
- console.log(
380
- `[JupyterService] Jupyter ready after ${waitTime}ms wait - proceeding with code execution`
381
- );
382
- }
383
-
384
- // Use circuit breaker for code execution
385
- return await this.circuitBreakers.codeExecution.execute(async () => {
386
- return await this.jupyterServer.executeCode(contextId, code, language);
387
- });
388
- }
389
-
390
- /**
391
- * List contexts with graceful degradation
392
- */
393
- async listContexts(): Promise<any[]> {
394
- if (!this.initialized) {
395
- return [];
396
- }
397
- return await this.jupyterServer.listContexts();
398
- }
399
-
400
- /**
401
- * Delete context - will wait for Jupyter if still initializing
402
- */
403
- async deleteContext(contextId: string): Promise<void> {
404
- if (!this.initialized) {
405
- console.log(
406
- "[JupyterService] Context deletion requested while Jupyter is initializing - waiting..."
407
- );
408
- const startWait = Date.now();
409
- await this.ensureInitialized();
410
- const waitTime = Date.now() - startWait;
411
- console.log(
412
- `[JupyterService] Jupyter ready after ${waitTime}ms wait - proceeding with context deletion`
413
- );
414
- }
415
-
416
- // Use circuit breaker for kernel communication
417
- return await this.circuitBreakers.kernelCommunication.execute(async () => {
418
- return await this.jupyterServer.deleteContext(contextId);
419
- });
420
- }
421
-
422
- /**
423
- * Shutdown the service
424
- */
425
- async shutdown(): Promise<void> {
426
- if (this.initialized) {
427
- await this.jupyterServer.shutdown();
428
- }
429
- }
430
-
431
- /**
432
- * Get initialization progress
433
- */
434
- private getInitializationProgress(): number {
435
- if (this.initialized) return 100;
436
- if (!this.initPromise) return 0;
437
-
438
- const elapsed = Date.now() - this.startTime;
439
- const estimatedTotal = 20000; // 20 seconds estimated
440
- return Math.min(95, Math.round((elapsed / estimatedTotal) * 100));
441
- }
442
- }
443
-
444
- /**
445
- * Error thrown when Jupyter is not ready yet
446
- * This matches the interface of the SDK's JupyterNotReadyError
447
- */
448
- export class JupyterNotReadyError extends Error {
449
- public readonly retryAfter: number;
450
- public readonly progress?: number;
451
-
452
- constructor(
453
- message: string,
454
- options?: { retryAfter?: number; progress?: number }
455
- ) {
456
- super(message);
457
- this.name = "JupyterNotReadyError";
458
- this.retryAfter = options?.retryAfter || 5;
459
- this.progress = options?.progress;
460
- }
461
- }
@@ -1,48 +0,0 @@
1
- """
2
- Minimal Jupyter configuration focused on kernel-only usage
3
- """
4
-
5
- c = get_config() # noqa
6
-
7
- # Disable all authentication - we handle security at container level
8
- c.ServerApp.token = ''
9
- c.ServerApp.password = ''
10
- c.IdentityProvider.token = ''
11
- c.ServerApp.allow_origin = '*'
12
- c.ServerApp.allow_remote_access = True
13
- c.ServerApp.disable_check_xsrf = True
14
- c.ServerApp.allow_root = True
15
- c.ServerApp.allow_credentials = True
16
-
17
- # Also set NotebookApp settings for compatibility
18
- c.NotebookApp.token = ''
19
- c.NotebookApp.password = ''
20
- c.NotebookApp.allow_origin = '*'
21
- c.NotebookApp.allow_remote_access = True
22
- c.NotebookApp.disable_check_xsrf = True
23
- c.NotebookApp.allow_credentials = True
24
-
25
- # Performance settings
26
- c.ServerApp.iopub_data_rate_limit = 1000000000 # E2B uses 1GB/s
27
-
28
- # Minimal logging
29
- c.Application.log_level = 'ERROR'
30
-
31
- # Disable browser
32
- c.ServerApp.open_browser = False
33
-
34
- # Optimize for container environment
35
- c.ServerApp.ip = '0.0.0.0'
36
- c.ServerApp.port = 8888
37
-
38
- # Kernel optimizations
39
- c.KernelManager.shutdown_wait_time = 0.0
40
- c.MappingKernelManager.cull_idle_timeout = 0
41
- c.MappingKernelManager.cull_interval = 0
42
-
43
- # Disable terminals
44
- c.ServerApp.terminals_enabled = False
45
-
46
- # Disable all extensions to speed up startup
47
- c.ServerApp.jpserver_extensions = {}
48
- c.ServerApp.nbserver_extensions = {}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/jupyter-client.ts"],"sourcesContent":["import { HttpClient } from \"./client.js\";\nimport { isRetryableError, parseErrorResponse } from \"./errors.js\";\nimport type {\n CodeContext,\n CreateContextOptions,\n ExecutionError,\n OutputMessage,\n Result,\n} from \"./interpreter-types.js\";\n\n// API Response types\ninterface ContextResponse {\n id: string;\n language: string;\n cwd: string;\n createdAt: string; // ISO date string from JSON\n lastUsed: string; // ISO date string from JSON\n}\n\ninterface ContextListResponse {\n contexts: ContextResponse[];\n}\n\n// Streaming execution data from the server\ninterface StreamingExecutionData {\n type: \"result\" | \"stdout\" | \"stderr\" | \"error\" | \"execution_complete\";\n text?: string;\n html?: string;\n png?: string; // base64\n jpeg?: string; // base64\n svg?: string;\n latex?: string;\n markdown?: string;\n javascript?: string;\n json?: unknown;\n chart?: {\n type:\n | \"line\"\n | \"bar\"\n | \"scatter\"\n | \"pie\"\n | \"histogram\"\n | \"heatmap\"\n | \"unknown\";\n data: unknown;\n options?: unknown;\n };\n data?: unknown;\n metadata?: Record<string, unknown>;\n execution_count?: number;\n ename?: string;\n evalue?: string;\n traceback?: string[];\n lineNumber?: number;\n timestamp?: number;\n}\n\nexport interface ExecutionCallbacks {\n onStdout?: (output: OutputMessage) => void | Promise<void>;\n onStderr?: (output: OutputMessage) => void | Promise<void>;\n onResult?: (result: Result) => void | Promise<void>;\n onError?: (error: ExecutionError) => void | Promise<void>;\n}\n\nexport class JupyterClient extends HttpClient {\n private readonly maxRetries = 3;\n private readonly retryDelayMs = 1000;\n\n async createCodeContext(\n options: CreateContextOptions = {}\n ): Promise<CodeContext> {\n return this.executeWithRetry(async () => {\n const response = await this.doFetch(\"/api/contexts\", {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n language: options.language || \"python\",\n cwd: options.cwd || \"/workspace\",\n env_vars: options.envVars,\n }),\n });\n\n if (!response.ok) {\n throw await parseErrorResponse(response);\n }\n\n const data = (await response.json()) as ContextResponse;\n return {\n id: data.id,\n language: data.language,\n cwd: data.cwd,\n createdAt: new Date(data.createdAt),\n lastUsed: new Date(data.lastUsed),\n };\n });\n }\n\n async runCodeStream(\n contextId: string | undefined,\n code: string,\n language: string | undefined,\n callbacks: ExecutionCallbacks\n ): Promise<void> {\n return this.executeWithRetry(async () => {\n const response = await this.doFetch(\"/api/execute/code\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"text/event-stream\",\n },\n body: JSON.stringify({\n context_id: contextId,\n code,\n language,\n }),\n });\n\n if (!response.ok) {\n throw await parseErrorResponse(response);\n }\n\n if (!response.body) {\n throw new Error(\"No response body for streaming execution\");\n }\n\n // Process streaming response\n for await (const chunk of this.readLines(response.body)) {\n await this.parseExecutionResult(chunk, callbacks);\n }\n });\n }\n\n private async *readLines(\n stream: ReadableStream<Uint8Array>\n ): AsyncGenerator<string> {\n const reader = stream.getReader();\n let buffer = \"\";\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (value) {\n buffer += new TextDecoder().decode(value);\n }\n if (done) break;\n\n let newlineIdx = buffer.indexOf(\"\\n\");\n while (newlineIdx !== -1) {\n yield buffer.slice(0, newlineIdx);\n buffer = buffer.slice(newlineIdx + 1);\n newlineIdx = buffer.indexOf(\"\\n\");\n }\n }\n\n // Yield any remaining data\n if (buffer.length > 0) {\n yield buffer;\n }\n } finally {\n reader.releaseLock();\n }\n }\n\n private async parseExecutionResult(\n line: string,\n callbacks: ExecutionCallbacks\n ) {\n if (!line.trim()) return;\n\n try {\n const data = JSON.parse(line) as StreamingExecutionData;\n\n switch (data.type) {\n case \"stdout\":\n if (callbacks.onStdout && data.text) {\n await callbacks.onStdout({\n text: data.text,\n timestamp: data.timestamp || Date.now(),\n });\n }\n break;\n\n case \"stderr\":\n if (callbacks.onStderr && data.text) {\n await callbacks.onStderr({\n text: data.text,\n timestamp: data.timestamp || Date.now(),\n });\n }\n break;\n\n case \"result\":\n if (callbacks.onResult) {\n // Convert raw result to Result interface\n const result: Result = {\n text: data.text,\n html: data.html,\n png: data.png,\n jpeg: data.jpeg,\n svg: data.svg,\n latex: data.latex,\n markdown: data.markdown,\n javascript: data.javascript,\n json: data.json,\n chart: data.chart,\n data: data.data,\n formats: () => {\n const formats: string[] = [];\n if (data.text) formats.push(\"text\");\n if (data.html) formats.push(\"html\");\n if (data.png) formats.push(\"png\");\n if (data.jpeg) formats.push(\"jpeg\");\n if (data.svg) formats.push(\"svg\");\n if (data.latex) formats.push(\"latex\");\n if (data.markdown) formats.push(\"markdown\");\n if (data.javascript) formats.push(\"javascript\");\n if (data.json) formats.push(\"json\");\n if (data.chart) formats.push(\"chart\");\n return formats;\n },\n };\n await callbacks.onResult(result);\n }\n break;\n\n case \"error\":\n if (callbacks.onError) {\n await callbacks.onError({\n name: data.ename || \"Error\",\n value: data.evalue || data.text || \"Unknown error\",\n traceback: data.traceback || [],\n lineNumber: data.lineNumber,\n });\n }\n break;\n\n case \"execution_complete\":\n // Execution completed successfully\n break;\n }\n } catch (error) {\n console.error(\"[JupyterClient] Error parsing execution result:\", error);\n }\n }\n\n async listCodeContexts(): Promise<CodeContext[]> {\n return this.executeWithRetry(async () => {\n const response = await this.doFetch(\"/api/contexts\", {\n method: \"GET\",\n headers: { \"Content-Type\": \"application/json\" },\n });\n\n if (!response.ok) {\n throw await parseErrorResponse(response);\n }\n\n const data = (await response.json()) as ContextListResponse;\n return data.contexts.map((ctx) => ({\n id: ctx.id,\n language: ctx.language,\n cwd: ctx.cwd,\n createdAt: new Date(ctx.createdAt),\n lastUsed: new Date(ctx.lastUsed),\n }));\n });\n }\n\n async deleteCodeContext(contextId: string): Promise<void> {\n return this.executeWithRetry(async () => {\n const response = await this.doFetch(`/api/contexts/${contextId}`, {\n method: \"DELETE\",\n headers: { \"Content-Type\": \"application/json\" },\n });\n\n if (!response.ok) {\n throw await parseErrorResponse(response);\n }\n });\n }\n\n // Override parent doFetch to be public for this class\n public async doFetch(path: string, options?: RequestInit): Promise<Response> {\n return super.doFetch(path, options);\n }\n\n /**\n * Execute an operation with automatic retry for transient errors\n */\n private async executeWithRetry<T>(operation: () => Promise<T>): Promise<T> {\n let lastError: Error | undefined;\n\n for (let attempt = 0; attempt < this.maxRetries; attempt++) {\n try {\n return await operation();\n } catch (error) {\n lastError = error as Error;\n\n // Check if it's a retryable error (circuit breaker or Jupyter not ready)\n if (this.isRetryableError(error)) {\n // Don't retry on the last attempt\n if (attempt < this.maxRetries - 1) {\n // Exponential backoff with jitter\n const delay =\n this.retryDelayMs * 2 ** attempt + Math.random() * 1000;\n await new Promise((resolve) => setTimeout(resolve, delay));\n continue;\n }\n }\n\n // Non-retryable error or last attempt - throw immediately\n throw error;\n }\n }\n\n // All retries exhausted - throw a clean error without implementation details\n if (lastError?.message.includes(\"Code execution\")) {\n // If the error already has a clean message about code execution, use it\n throw lastError;\n }\n\n // Otherwise, throw a generic but user-friendly error\n throw new Error(\"Unable to execute code at this time\");\n }\n\n /**\n * Check if an error is retryable\n */\n private isRetryableError(error: unknown): boolean {\n // Use the SDK's built-in retryable check\n if (isRetryableError(error)) {\n return true;\n }\n\n // Also check for circuit breaker specific errors\n if (error instanceof Error) {\n // Circuit breaker errors (from the container's response)\n if (error.message.includes(\"Circuit breaker is open\")) {\n return true;\n }\n\n // Check if error has a status property\n if (\"status\" in error && error.status === \"circuit_open\") {\n return true;\n }\n }\n\n return false;\n }\n}\n"],"mappings":"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","names":[]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/interpreter-types.ts"],"sourcesContent":["// Context Management\nexport interface CreateContextOptions {\n /**\n * Programming language for the context\n * @default 'python'\n */\n language?: 'python' | 'javascript' | 'typescript';\n \n /**\n * Working directory for the context\n * @default '/workspace'\n */\n cwd?: string;\n \n /**\n * Environment variables for the context\n */\n envVars?: Record<string, string>;\n \n /**\n * Request timeout in milliseconds\n * @default 30000\n */\n timeout?: number;\n}\n\nexport interface CodeContext {\n /**\n * Unique identifier for the context\n */\n readonly id: string;\n \n /**\n * Programming language of the context\n */\n readonly language: string;\n \n /**\n * Current working directory\n */\n readonly cwd: string;\n \n /**\n * When the context was created\n */\n readonly createdAt: Date;\n \n /**\n * When the context was last used\n */\n readonly lastUsed: Date;\n}\n\n// Execution Options\nexport interface RunCodeOptions {\n /**\n * Context to run the code in. If not provided, uses default context for the language\n */\n context?: CodeContext;\n \n /**\n * Language to use if context is not provided\n * @default 'python'\n */\n language?: 'python' | 'javascript' | 'typescript';\n \n /**\n * Environment variables for this execution\n */\n envVars?: Record<string, string>;\n \n /**\n * Execution timeout in milliseconds\n * @default 60000\n */\n timeout?: number;\n \n /**\n * AbortSignal for cancelling execution\n */\n signal?: AbortSignal;\n \n /**\n * Callback for stdout output\n */\n onStdout?: (output: OutputMessage) => void | Promise<void>;\n \n /**\n * Callback for stderr output\n */\n onStderr?: (output: OutputMessage) => void | Promise<void>;\n \n /**\n * Callback for execution results (charts, tables, etc)\n */\n onResult?: (result: Result) => void | Promise<void>;\n \n /**\n * Callback for execution errors\n */\n onError?: (error: ExecutionError) => void | Promise<void>;\n}\n\n// Output Messages\nexport interface OutputMessage {\n /**\n * The output text\n */\n text: string;\n \n /**\n * Timestamp of the output\n */\n timestamp: number;\n}\n\n// Execution Results\nexport interface Result {\n /**\n * Plain text representation\n */\n text?: string;\n \n /**\n * HTML representation (tables, formatted output)\n */\n html?: string;\n \n /**\n * PNG image data (base64 encoded)\n */\n png?: string;\n \n /**\n * JPEG image data (base64 encoded)\n */\n jpeg?: string;\n \n /**\n * SVG image data\n */\n svg?: string;\n \n /**\n * LaTeX representation\n */\n latex?: string;\n \n /**\n * Markdown representation\n */\n markdown?: string;\n \n /**\n * JavaScript code to execute\n */\n javascript?: string;\n \n /**\n * JSON data\n */\n json?: any;\n \n /**\n * Chart data if the result is a visualization\n */\n chart?: ChartData;\n \n /**\n * Raw data object\n */\n data?: any;\n \n /**\n * Available output formats\n */\n formats(): string[];\n}\n\n// Chart Data\nexport interface ChartData {\n /**\n * Type of chart\n */\n type: 'line' | 'bar' | 'scatter' | 'pie' | 'histogram' | 'heatmap' | 'unknown';\n \n /**\n * Chart title\n */\n title?: string;\n \n /**\n * Chart data (format depends on library)\n */\n data: any;\n \n /**\n * Chart layout/configuration\n */\n layout?: any;\n \n /**\n * Additional configuration\n */\n config?: any;\n \n /**\n * Library that generated the chart\n */\n library?: 'matplotlib' | 'plotly' | 'altair' | 'seaborn' | 'unknown';\n \n /**\n * Base64 encoded image if available\n */\n image?: string;\n}\n\n// Execution Error\nexport interface ExecutionError {\n /**\n * Error name/type (e.g., 'NameError', 'SyntaxError')\n */\n name: string;\n \n /**\n * Error message\n */\n value: string;\n \n /**\n * Stack trace\n */\n traceback: string[];\n \n /**\n * Line number where error occurred\n */\n lineNumber?: number;\n}\n\n// Serializable execution result\nexport interface ExecutionResult {\n code: string;\n logs: {\n stdout: string[];\n stderr: string[];\n };\n error?: ExecutionError;\n executionCount?: number;\n results: Array<{\n text?: string;\n html?: string;\n png?: string;\n jpeg?: string;\n svg?: string;\n latex?: string;\n markdown?: string;\n javascript?: string;\n json?: any;\n chart?: ChartData;\n data?: any;\n }>;\n}\n\n// Execution Result Container\nexport class Execution {\n /**\n * All results from the execution\n */\n public results: Result[] = [];\n \n /**\n * Accumulated stdout and stderr\n */\n public logs = {\n stdout: [] as string[],\n stderr: [] as string[]\n };\n \n /**\n * Execution error if any\n */\n public error?: ExecutionError;\n \n /**\n * Execution count (for Jupyter)\n */\n public executionCount?: number;\n \n constructor(\n public readonly code: string,\n public readonly context: CodeContext\n ) {}\n \n /**\n * Convert to a plain object for serialization\n */\n toJSON(): ExecutionResult {\n return {\n code: this.code,\n logs: this.logs,\n error: this.error,\n executionCount: this.executionCount,\n results: this.results.map(result => ({\n text: result.text,\n html: result.html,\n png: result.png,\n jpeg: result.jpeg,\n svg: result.svg,\n latex: result.latex,\n markdown: result.markdown,\n javascript: result.javascript,\n json: result.json,\n chart: result.chart,\n data: result.data\n }))\n };\n }\n}\n\n// Implementation of Result\nexport class ResultImpl implements Result {\n constructor(private raw: any) {}\n \n get text(): string | undefined { \n return this.raw.text || this.raw.data?.['text/plain']; \n }\n \n get html(): string | undefined { \n return this.raw.html || this.raw.data?.['text/html']; \n }\n \n get png(): string | undefined { \n return this.raw.png || this.raw.data?.['image/png']; \n }\n \n get jpeg(): string | undefined { \n return this.raw.jpeg || this.raw.data?.['image/jpeg']; \n }\n \n get svg(): string | undefined { \n return this.raw.svg || this.raw.data?.['image/svg+xml']; \n }\n \n get latex(): string | undefined { \n return this.raw.latex || this.raw.data?.['text/latex']; \n }\n \n get markdown(): string | undefined { \n return this.raw.markdown || this.raw.data?.['text/markdown']; \n }\n \n get javascript(): string | undefined { \n return this.raw.javascript || this.raw.data?.['application/javascript']; \n }\n \n get json(): any { \n return this.raw.json || this.raw.data?.['application/json']; \n }\n \n get chart(): ChartData | undefined { \n return this.raw.chart; \n }\n \n get data(): any { \n return this.raw.data; \n }\n \n formats(): string[] {\n const formats: string[] = [];\n if (this.text) formats.push('text');\n if (this.html) formats.push('html');\n if (this.png) formats.push('png');\n if (this.jpeg) formats.push('jpeg');\n if (this.svg) formats.push('svg');\n if (this.latex) formats.push('latex');\n if (this.markdown) formats.push('markdown');\n if (this.javascript) formats.push('javascript');\n if (this.json) formats.push('json');\n if (this.chart) formats.push('chart');\n return formats;\n }\n}"],"mappings":";AAyQO,IAAM,YAAN,MAAgB;AAAA,EAwBrB,YACkB,MACA,SAChB;AAFgB;AACA;AAAA,EACf;AAAA;AAAA;AAAA;AAAA,EAvBI,UAAoB,CAAC;AAAA;AAAA;AAAA;AAAA,EAKrB,OAAO;AAAA,IACZ,QAAQ,CAAC;AAAA,IACT,QAAQ,CAAC;AAAA,EACX;AAAA;AAAA;AAAA;AAAA,EAKO;AAAA;AAAA;AAAA;AAAA,EAKA;AAAA;AAAA;AAAA;AAAA,EAUP,SAA0B;AACxB,WAAO;AAAA,MACL,MAAM,KAAK;AAAA,MACX,MAAM,KAAK;AAAA,MACX,OAAO,KAAK;AAAA,MACZ,gBAAgB,KAAK;AAAA,MACrB,SAAS,KAAK,QAAQ,IAAI,aAAW;AAAA,QACnC,MAAM,OAAO;AAAA,QACb,MAAM,OAAO;AAAA,QACb,KAAK,OAAO;AAAA,QACZ,MAAM,OAAO;AAAA,QACb,KAAK,OAAO;AAAA,QACZ,OAAO,OAAO;AAAA,QACd,UAAU,OAAO;AAAA,QACjB,YAAY,OAAO;AAAA,QACnB,MAAM,OAAO;AAAA,QACb,OAAO,OAAO;AAAA,QACd,MAAM,OAAO;AAAA,MACf,EAAE;AAAA,IACJ;AAAA,EACF;AACF;AAGO,IAAM,aAAN,MAAmC;AAAA,EACxC,YAAoB,KAAU;AAAV;AAAA,EAAW;AAAA,EAE/B,IAAI,OAA2B;AAC7B,WAAO,KAAK,IAAI,QAAQ,KAAK,IAAI,OAAO,YAAY;AAAA,EACtD;AAAA,EAEA,IAAI,OAA2B;AAC7B,WAAO,KAAK,IAAI,QAAQ,KAAK,IAAI,OAAO,WAAW;AAAA,EACrD;AAAA,EAEA,IAAI,MAA0B;AAC5B,WAAO,KAAK,IAAI,OAAO,KAAK,IAAI,OAAO,WAAW;AAAA,EACpD;AAAA,EAEA,IAAI,OAA2B;AAC7B,WAAO,KAAK,IAAI,QAAQ,KAAK,IAAI,OAAO,YAAY;AAAA,EACtD;AAAA,EAEA,IAAI,MAA0B;AAC5B,WAAO,KAAK,IAAI,OAAO,KAAK,IAAI,OAAO,eAAe;AAAA,EACxD;AAAA,EAEA,IAAI,QAA4B;AAC9B,WAAO,KAAK,IAAI,SAAS,KAAK,IAAI,OAAO,YAAY;AAAA,EACvD;AAAA,EAEA,IAAI,WAA+B;AACjC,WAAO,KAAK,IAAI,YAAY,KAAK,IAAI,OAAO,eAAe;AAAA,EAC7D;AAAA,EAEA,IAAI,aAAiC;AACnC,WAAO,KAAK,IAAI,cAAc,KAAK,IAAI,OAAO,wBAAwB;AAAA,EACxE;AAAA,EAEA,IAAI,OAAY;AACd,WAAO,KAAK,IAAI,QAAQ,KAAK,IAAI,OAAO,kBAAkB;AAAA,EAC5D;AAAA,EAEA,IAAI,QAA+B;AACjC,WAAO,KAAK,IAAI;AAAA,EAClB;AAAA,EAEA,IAAI,OAAY;AACd,WAAO,KAAK,IAAI;AAAA,EAClB;AAAA,EAEA,UAAoB;AAClB,UAAM,UAAoB,CAAC;AAC3B,QAAI,KAAK,KAAM,SAAQ,KAAK,MAAM;AAClC,QAAI,KAAK,KAAM,SAAQ,KAAK,MAAM;AAClC,QAAI,KAAK,IAAK,SAAQ,KAAK,KAAK;AAChC,QAAI,KAAK,KAAM,SAAQ,KAAK,MAAM;AAClC,QAAI,KAAK,IAAK,SAAQ,KAAK,KAAK;AAChC,QAAI,KAAK,MAAO,SAAQ,KAAK,OAAO;AACpC,QAAI,KAAK,SAAU,SAAQ,KAAK,UAAU;AAC1C,QAAI,KAAK,WAAY,SAAQ,KAAK,YAAY;AAC9C,QAAI,KAAK,KAAM,SAAQ,KAAK,MAAM;AAClC,QAAI,KAAK,MAAO,SAAQ,KAAK,OAAO;AACpC,WAAO;AAAA,EACT;AACF;","names":[]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/interpreter.ts"],"sourcesContent":["import {\n type CodeContext,\n type CreateContextOptions,\n Execution,\n ResultImpl,\n type RunCodeOptions,\n} from \"./interpreter-types.js\";\nimport type { JupyterClient } from \"./jupyter-client.js\";\nimport type { Sandbox } from \"./sandbox.js\";\n\nexport class CodeInterpreter {\n private jupyterClient: JupyterClient;\n private contexts = new Map<string, CodeContext>();\n\n constructor(sandbox: Sandbox) {\n this.jupyterClient = sandbox.client as JupyterClient;\n }\n\n /**\n * Create a new code execution context\n */\n async createCodeContext(\n options: CreateContextOptions = {}\n ): Promise<CodeContext> {\n const context = await this.jupyterClient.createCodeContext(options);\n this.contexts.set(context.id, context);\n return context;\n }\n\n /**\n * Run code with optional context\n */\n async runCode(\n code: string,\n options: RunCodeOptions = {}\n ): Promise<Execution> {\n // Get or create context\n let context = options.context;\n if (!context) {\n // Try to find or create a default context for the language\n const language = options.language || \"python\";\n context = await this.getOrCreateDefaultContext(language);\n }\n\n // Create execution object to collect results\n const execution = new Execution(code, context);\n\n // Stream execution\n await this.jupyterClient.runCodeStream(context.id, code, options.language, {\n onStdout: (output) => {\n execution.logs.stdout.push(output.text);\n if (options.onStdout) return options.onStdout(output);\n },\n onStderr: (output) => {\n execution.logs.stderr.push(output.text);\n if (options.onStderr) return options.onStderr(output);\n },\n onResult: async (result) => {\n execution.results.push(new ResultImpl(result) as any);\n if (options.onResult) return options.onResult(result);\n },\n onError: (error) => {\n execution.error = error;\n if (options.onError) return options.onError(error);\n },\n });\n\n return execution;\n }\n\n /**\n * Run code and return a streaming response\n */\n async runCodeStream(\n code: string,\n options: RunCodeOptions = {}\n ): Promise<ReadableStream> {\n // Get or create context\n let context = options.context;\n if (!context) {\n const language = options.language || \"python\";\n context = await this.getOrCreateDefaultContext(language);\n }\n\n // Create streaming response\n const response = await this.jupyterClient.doFetch(\"/api/execute/code\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"text/event-stream\",\n },\n body: JSON.stringify({\n context_id: context.id,\n code,\n language: options.language,\n }),\n });\n\n if (!response.ok) {\n const errorData = (await response\n .json()\n .catch(() => ({ error: \"Unknown error\" }))) as { error?: string };\n throw new Error(\n errorData.error || `Failed to execute code: ${response.status}`\n );\n }\n\n if (!response.body) {\n throw new Error(\"No response body for streaming execution\");\n }\n\n return response.body;\n }\n\n /**\n * List all code contexts\n */\n async listCodeContexts(): Promise<CodeContext[]> {\n const contexts = await this.jupyterClient.listCodeContexts();\n\n // Update local cache\n for (const context of contexts) {\n this.contexts.set(context.id, context);\n }\n\n return contexts;\n }\n\n /**\n * Delete a code context\n */\n async deleteCodeContext(contextId: string): Promise<void> {\n await this.jupyterClient.deleteCodeContext(contextId);\n this.contexts.delete(contextId);\n }\n\n private async getOrCreateDefaultContext(\n language: \"python\" | \"javascript\" | \"typescript\"\n ): Promise<CodeContext> {\n // Check if we have a cached context for this language\n for (const context of this.contexts.values()) {\n if (context.language === language) {\n return context;\n }\n }\n\n // Create new default context\n return this.createCodeContext({ language });\n }\n}\n"],"mappings":";;;;;;AAUO,IAAM,kBAAN,MAAsB;AAAA,EACnB;AAAA,EACA,WAAW,oBAAI,IAAyB;AAAA,EAEhD,YAAY,SAAkB;AAC5B,SAAK,gBAAgB,QAAQ;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,kBACJ,UAAgC,CAAC,GACX;AACtB,UAAM,UAAU,MAAM,KAAK,cAAc,kBAAkB,OAAO;AAClE,SAAK,SAAS,IAAI,QAAQ,IAAI,OAAO;AACrC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,QACJ,MACA,UAA0B,CAAC,GACP;AAEpB,QAAI,UAAU,QAAQ;AACtB,QAAI,CAAC,SAAS;AAEZ,YAAM,WAAW,QAAQ,YAAY;AACrC,gBAAU,MAAM,KAAK,0BAA0B,QAAQ;AAAA,IACzD;AAGA,UAAM,YAAY,IAAI,UAAU,MAAM,OAAO;AAG7C,UAAM,KAAK,cAAc,cAAc,QAAQ,IAAI,MAAM,QAAQ,UAAU;AAAA,MACzE,UAAU,CAAC,WAAW;AACpB,kBAAU,KAAK,OAAO,KAAK,OAAO,IAAI;AACtC,YAAI,QAAQ,SAAU,QAAO,QAAQ,SAAS,MAAM;AAAA,MACtD;AAAA,MACA,UAAU,CAAC,WAAW;AACpB,kBAAU,KAAK,OAAO,KAAK,OAAO,IAAI;AACtC,YAAI,QAAQ,SAAU,QAAO,QAAQ,SAAS,MAAM;AAAA,MACtD;AAAA,MACA,UAAU,OAAO,WAAW;AAC1B,kBAAU,QAAQ,KAAK,IAAI,WAAW,MAAM,CAAQ;AACpD,YAAI,QAAQ,SAAU,QAAO,QAAQ,SAAS,MAAM;AAAA,MACtD;AAAA,MACA,SAAS,CAAC,UAAU;AAClB,kBAAU,QAAQ;AAClB,YAAI,QAAQ,QAAS,QAAO,QAAQ,QAAQ,KAAK;AAAA,MACnD;AAAA,IACF,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,cACJ,MACA,UAA0B,CAAC,GACF;AAEzB,QAAI,UAAU,QAAQ;AACtB,QAAI,CAAC,SAAS;AACZ,YAAM,WAAW,QAAQ,YAAY;AACrC,gBAAU,MAAM,KAAK,0BAA0B,QAAQ;AAAA,IACzD;AAGA,UAAM,WAAW,MAAM,KAAK,cAAc,QAAQ,qBAAqB;AAAA,MACrE,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,QAAQ;AAAA,MACV;AAAA,MACA,MAAM,KAAK,UAAU;AAAA,QACnB,YAAY,QAAQ;AAAA,QACpB;AAAA,QACA,UAAU,QAAQ;AAAA,MACpB,CAAC;AAAA,IACH,CAAC;AAED,QAAI,CAAC,SAAS,IAAI;AAChB,YAAM,YAAa,MAAM,SACtB,KAAK,EACL,MAAM,OAAO,EAAE,OAAO,gBAAgB,EAAE;AAC3C,YAAM,IAAI;AAAA,QACR,UAAU,SAAS,2BAA2B,SAAS,MAAM;AAAA,MAC/D;AAAA,IACF;AAEA,QAAI,CAAC,SAAS,MAAM;AAClB,YAAM,IAAI,MAAM,0CAA0C;AAAA,IAC5D;AAEA,WAAO,SAAS;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,mBAA2C;AAC/C,UAAM,WAAW,MAAM,KAAK,cAAc,iBAAiB;AAG3D,eAAW,WAAW,UAAU;AAC9B,WAAK,SAAS,IAAI,QAAQ,IAAI,OAAO;AAAA,IACvC;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,kBAAkB,WAAkC;AACxD,UAAM,KAAK,cAAc,kBAAkB,SAAS;AACpD,SAAK,SAAS,OAAO,SAAS;AAAA,EAChC;AAAA,EAEA,MAAc,0BACZ,UACsB;AAEtB,eAAW,WAAW,KAAK,SAAS,OAAO,GAAG;AAC5C,UAAI,QAAQ,aAAa,UAAU;AACjC,eAAO;AAAA,MACT;AAAA,IACF;AAGA,WAAO,KAAK,kBAAkB,EAAE,SAAS,CAAC;AAAA,EAC5C;AACF;","names":[]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/errors.ts"],"sourcesContent":["/**\n * Standard error response from the sandbox API\n */\nexport interface SandboxErrorResponse {\n error?: string;\n status?: string;\n progress?: number;\n}\n\n/**\n * Base error class for all Sandbox-related errors\n */\nexport class SandboxError extends Error {\n constructor(message: string) {\n super(message);\n this.name = this.constructor.name;\n\n // Maintains proper stack trace for where our error was thrown (only available on V8)\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n }\n }\n}\n\n/**\n * Error thrown when Jupyter functionality is requested but the service is still initializing.\n *\n * Note: With the current implementation, requests wait for Jupyter to be ready.\n * This error is only thrown when:\n * 1. The request times out waiting for Jupyter (default: 30 seconds)\n * 2. Jupyter initialization actually fails\n *\n * Most requests will succeed after a delay, not throw this error.\n */\nexport class JupyterNotReadyError extends SandboxError {\n public readonly code = \"JUPYTER_NOT_READY\";\n public readonly retryAfter: number;\n public readonly progress?: number;\n\n constructor(\n message?: string,\n options?: { retryAfter?: number; progress?: number }\n ) {\n super(\n message || \"Jupyter is still initializing. Please retry in a few seconds.\"\n );\n this.retryAfter = options?.retryAfter || 5;\n this.progress = options?.progress;\n }\n}\n\n/**\n * Error thrown when a context is not found\n */\nexport class ContextNotFoundError extends SandboxError {\n public readonly code = \"CONTEXT_NOT_FOUND\";\n public readonly contextId: string;\n\n constructor(contextId: string) {\n super(`Context ${contextId} not found`);\n this.contextId = contextId;\n }\n}\n\n/**\n * Error thrown when code execution fails\n */\nexport class CodeExecutionError extends SandboxError {\n public readonly code = \"CODE_EXECUTION_ERROR\";\n public readonly executionError?: {\n ename?: string;\n evalue?: string;\n traceback?: string[];\n };\n\n constructor(message: string, executionError?: any) {\n super(message);\n this.executionError = executionError;\n }\n}\n\n/**\n * Error thrown when the sandbox container is not ready\n */\nexport class ContainerNotReadyError extends SandboxError {\n public readonly code = \"CONTAINER_NOT_READY\";\n\n constructor(message?: string) {\n super(\n message ||\n \"Container is not ready. Please wait for initialization to complete.\"\n );\n }\n}\n\n/**\n * Error thrown when a network request to the sandbox fails\n */\nexport class SandboxNetworkError extends SandboxError {\n public readonly code = \"NETWORK_ERROR\";\n public readonly statusCode?: number;\n public readonly statusText?: string;\n\n constructor(message: string, statusCode?: number, statusText?: string) {\n super(message);\n this.statusCode = statusCode;\n this.statusText = statusText;\n }\n}\n\n/**\n * Error thrown when service is temporarily unavailable (e.g., circuit breaker open)\n */\nexport class ServiceUnavailableError extends SandboxError {\n public readonly code = \"SERVICE_UNAVAILABLE\";\n public readonly retryAfter?: number;\n\n constructor(message?: string, retryAfter?: number) {\n // Simple, user-friendly message without implementation details\n super(message || \"Service temporarily unavailable\");\n this.retryAfter = retryAfter;\n }\n}\n\n/**\n * Type guard to check if an error is a JupyterNotReadyError\n */\nexport function isJupyterNotReadyError(\n error: unknown\n): error is JupyterNotReadyError {\n return error instanceof JupyterNotReadyError;\n}\n\n/**\n * Type guard to check if an error is any SandboxError\n */\nexport function isSandboxError(error: unknown): error is SandboxError {\n return error instanceof SandboxError;\n}\n\n/**\n * Helper to determine if an error is retryable\n */\nexport function isRetryableError(error: unknown): boolean {\n if (\n error instanceof JupyterNotReadyError ||\n error instanceof ContainerNotReadyError ||\n error instanceof ServiceUnavailableError\n ) {\n return true;\n }\n\n if (error instanceof SandboxNetworkError) {\n // Retry on 502, 503, 504 (gateway/service unavailable errors)\n return error.statusCode\n ? [502, 503, 504].includes(error.statusCode)\n : false;\n }\n\n return false;\n}\n\n/**\n * Parse error response from the sandbox API and return appropriate error instance\n */\nexport async function parseErrorResponse(\n response: Response\n): Promise<SandboxError> {\n let data: SandboxErrorResponse;\n\n try {\n data = (await response.json()) as SandboxErrorResponse;\n } catch {\n // If JSON parsing fails, return a generic network error\n return new SandboxNetworkError(\n `Request failed with status ${response.status}`,\n response.status,\n response.statusText\n );\n }\n\n // Check for specific error types based on response\n if (response.status === 503) {\n // Circuit breaker error\n if (data.status === \"circuit_open\") {\n return new ServiceUnavailableError(\n \"Service temporarily unavailable\",\n parseInt(response.headers.get(\"Retry-After\") || \"30\")\n );\n }\n\n // Jupyter initialization error\n if (data.status === \"initializing\") {\n return new JupyterNotReadyError(data.error, {\n retryAfter: parseInt(response.headers.get(\"Retry-After\") || \"5\"),\n progress: data.progress,\n });\n }\n }\n\n // Check for context not found\n if (\n response.status === 404 &&\n data.error?.includes(\"Context\") &&\n data.error?.includes(\"not found\")\n ) {\n const contextId =\n data.error.match(/Context (\\S+) not found/)?.[1] || \"unknown\";\n return new ContextNotFoundError(contextId);\n }\n\n // Default network error\n return new SandboxNetworkError(\n data.error || `Request failed with status ${response.status}`,\n response.status,\n response.statusText\n );\n}\n"],"mappings":";AAYO,IAAM,eAAN,cAA2B,MAAM;AAAA,EACtC,YAAY,SAAiB;AAC3B,UAAM,OAAO;AACb,SAAK,OAAO,KAAK,YAAY;AAG7B,QAAI,MAAM,mBAAmB;AAC3B,YAAM,kBAAkB,MAAM,KAAK,WAAW;AAAA,IAChD;AAAA,EACF;AACF;AAYO,IAAM,uBAAN,cAAmC,aAAa;AAAA,EACrC,OAAO;AAAA,EACP;AAAA,EACA;AAAA,EAEhB,YACE,SACA,SACA;AACA;AAAA,MACE,WAAW;AAAA,IACb;AACA,SAAK,aAAa,SAAS,cAAc;AACzC,SAAK,WAAW,SAAS;AAAA,EAC3B;AACF;AAKO,IAAM,uBAAN,cAAmC,aAAa;AAAA,EACrC,OAAO;AAAA,EACP;AAAA,EAEhB,YAAY,WAAmB;AAC7B,UAAM,WAAW,SAAS,YAAY;AACtC,SAAK,YAAY;AAAA,EACnB;AACF;AAKO,IAAM,qBAAN,cAAiC,aAAa;AAAA,EACnC,OAAO;AAAA,EACP;AAAA,EAMhB,YAAY,SAAiB,gBAAsB;AACjD,UAAM,OAAO;AACb,SAAK,iBAAiB;AAAA,EACxB;AACF;AAKO,IAAM,yBAAN,cAAqC,aAAa;AAAA,EACvC,OAAO;AAAA,EAEvB,YAAY,SAAkB;AAC5B;AAAA,MACE,WACE;AAAA,IACJ;AAAA,EACF;AACF;AAKO,IAAM,sBAAN,cAAkC,aAAa;AAAA,EACpC,OAAO;AAAA,EACP;AAAA,EACA;AAAA,EAEhB,YAAY,SAAiB,YAAqB,YAAqB;AACrE,UAAM,OAAO;AACb,SAAK,aAAa;AAClB,SAAK,aAAa;AAAA,EACpB;AACF;AAKO,IAAM,0BAAN,cAAsC,aAAa;AAAA,EACxC,OAAO;AAAA,EACP;AAAA,EAEhB,YAAY,SAAkB,YAAqB;AAEjD,UAAM,WAAW,iCAAiC;AAClD,SAAK,aAAa;AAAA,EACpB;AACF;AAKO,SAAS,uBACd,OAC+B;AAC/B,SAAO,iBAAiB;AAC1B;AAKO,SAAS,eAAe,OAAuC;AACpE,SAAO,iBAAiB;AAC1B;AAKO,SAAS,iBAAiB,OAAyB;AACxD,MACE,iBAAiB,wBACjB,iBAAiB,0BACjB,iBAAiB,yBACjB;AACA,WAAO;AAAA,EACT;AAEA,MAAI,iBAAiB,qBAAqB;AAExC,WAAO,MAAM,aACT,CAAC,KAAK,KAAK,GAAG,EAAE,SAAS,MAAM,UAAU,IACzC;AAAA,EACN;AAEA,SAAO;AACT;AAKA,eAAsB,mBACpB,UACuB;AACvB,MAAI;AAEJ,MAAI;AACF,WAAQ,MAAM,SAAS,KAAK;AAAA,EAC9B,QAAQ;AAEN,WAAO,IAAI;AAAA,MACT,8BAA8B,SAAS,MAAM;AAAA,MAC7C,SAAS;AAAA,MACT,SAAS;AAAA,IACX;AAAA,EACF;AAGA,MAAI,SAAS,WAAW,KAAK;AAE3B,QAAI,KAAK,WAAW,gBAAgB;AAClC,aAAO,IAAI;AAAA,QACT;AAAA,QACA,SAAS,SAAS,QAAQ,IAAI,aAAa,KAAK,IAAI;AAAA,MACtD;AAAA,IACF;AAGA,QAAI,KAAK,WAAW,gBAAgB;AAClC,aAAO,IAAI,qBAAqB,KAAK,OAAO;AAAA,QAC1C,YAAY,SAAS,SAAS,QAAQ,IAAI,aAAa,KAAK,GAAG;AAAA,QAC/D,UAAU,KAAK;AAAA,MACjB,CAAC;AAAA,IACH;AAAA,EACF;AAGA,MACE,SAAS,WAAW,OACpB,KAAK,OAAO,SAAS,SAAS,KAC9B,KAAK,OAAO,SAAS,WAAW,GAChC;AACA,UAAM,YACJ,KAAK,MAAM,MAAM,yBAAyB,IAAI,CAAC,KAAK;AACtD,WAAO,IAAI,qBAAqB,SAAS;AAAA,EAC3C;AAGA,SAAO,IAAI;AAAA,IACT,KAAK,SAAS,8BAA8B,SAAS,MAAM;AAAA,IAC3D,SAAS;AAAA,IACT,SAAS;AAAA,EACX;AACF;","names":[]}