@cloudflare/sandbox 0.2.1 → 0.2.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.
- package/CHANGELOG.md +6 -0
- package/README.md +1 -1
- package/container_src/circuit-breaker.ts +121 -0
- package/container_src/index.ts +228 -103
- package/container_src/jupyter-server.ts +289 -46
- package/container_src/jupyter-service.ts +448 -0
- package/container_src/startup.sh +59 -28
- package/dist/chunk-LALY4SFU.js +129 -0
- package/dist/chunk-LALY4SFU.js.map +1 -0
- package/dist/{chunk-SYMWNYWA.js → chunk-VTKZL632.js} +116 -64
- package/dist/chunk-VTKZL632.js.map +1 -0
- package/dist/{chunk-IATLC32Y.js → chunk-ZMPO44U4.js} +8 -8
- package/dist/{client-C7rKCYBD.d.ts → client-bzEV222a.d.ts} +10 -0
- package/dist/client.d.ts +1 -1
- package/dist/errors.d.ts +95 -0
- package/dist/errors.js +27 -0
- package/dist/errors.js.map +1 -0
- package/dist/index.d.ts +2 -1
- package/dist/index.js +27 -3
- package/dist/interpreter.d.ts +1 -1
- package/dist/jupyter-client.d.ts +1 -1
- package/dist/jupyter-client.js +2 -1
- package/dist/request-handler.d.ts +1 -1
- package/dist/request-handler.js +4 -3
- package/dist/sandbox.d.ts +1 -1
- package/dist/sandbox.js +4 -3
- package/package.json +1 -1
- package/src/errors.ts +218 -0
- package/src/index.ts +33 -8
- package/src/jupyter-client.ts +225 -142
- package/dist/chunk-SYMWNYWA.js.map +0 -1
- /package/dist/{chunk-IATLC32Y.js.map → chunk-ZMPO44U4.js.map} +0 -0
|
@@ -1,55 +1,63 @@
|
|
|
1
1
|
import {
|
|
2
2
|
HttpClient
|
|
3
3
|
} from "./chunk-CUHYLCMT.js";
|
|
4
|
+
import {
|
|
5
|
+
isRetryableError,
|
|
6
|
+
parseErrorResponse
|
|
7
|
+
} from "./chunk-LALY4SFU.js";
|
|
4
8
|
|
|
5
9
|
// src/jupyter-client.ts
|
|
6
10
|
var JupyterClient = class extends HttpClient {
|
|
11
|
+
maxRetries = 3;
|
|
12
|
+
retryDelayMs = 1e3;
|
|
7
13
|
async createCodeContext(options = {}) {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
14
|
+
return this.executeWithRetry(async () => {
|
|
15
|
+
const response = await this.doFetch("/api/contexts", {
|
|
16
|
+
method: "POST",
|
|
17
|
+
headers: { "Content-Type": "application/json" },
|
|
18
|
+
body: JSON.stringify({
|
|
19
|
+
language: options.language || "python",
|
|
20
|
+
cwd: options.cwd || "/workspace",
|
|
21
|
+
env_vars: options.envVars
|
|
22
|
+
})
|
|
23
|
+
});
|
|
24
|
+
if (!response.ok) {
|
|
25
|
+
throw await parseErrorResponse(response);
|
|
26
|
+
}
|
|
27
|
+
const data = await response.json();
|
|
28
|
+
return {
|
|
29
|
+
id: data.id,
|
|
30
|
+
language: data.language,
|
|
31
|
+
cwd: data.cwd,
|
|
32
|
+
createdAt: new Date(data.createdAt),
|
|
33
|
+
lastUsed: new Date(data.lastUsed)
|
|
34
|
+
};
|
|
16
35
|
});
|
|
17
|
-
if (!response.ok) {
|
|
18
|
-
const errorData = await response.json().catch(() => ({ error: "Unknown error" }));
|
|
19
|
-
throw new Error(errorData.error || `Failed to create context: ${response.status}`);
|
|
20
|
-
}
|
|
21
|
-
const data = await response.json();
|
|
22
|
-
return {
|
|
23
|
-
id: data.id,
|
|
24
|
-
language: data.language,
|
|
25
|
-
cwd: data.cwd,
|
|
26
|
-
createdAt: new Date(data.createdAt),
|
|
27
|
-
lastUsed: new Date(data.lastUsed)
|
|
28
|
-
};
|
|
29
36
|
}
|
|
30
37
|
async runCodeStream(contextId, code, language, callbacks) {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
38
|
+
return this.executeWithRetry(async () => {
|
|
39
|
+
const response = await this.doFetch("/api/execute/code", {
|
|
40
|
+
method: "POST",
|
|
41
|
+
headers: {
|
|
42
|
+
"Content-Type": "application/json",
|
|
43
|
+
Accept: "text/event-stream"
|
|
44
|
+
},
|
|
45
|
+
body: JSON.stringify({
|
|
46
|
+
context_id: contextId,
|
|
47
|
+
code,
|
|
48
|
+
language
|
|
49
|
+
})
|
|
50
|
+
});
|
|
51
|
+
if (!response.ok) {
|
|
52
|
+
throw await parseErrorResponse(response);
|
|
53
|
+
}
|
|
54
|
+
if (!response.body) {
|
|
55
|
+
throw new Error("No response body for streaming execution");
|
|
56
|
+
}
|
|
57
|
+
for await (const chunk of this.readLines(response.body)) {
|
|
58
|
+
await this.parseExecutionResult(chunk, callbacks);
|
|
59
|
+
}
|
|
42
60
|
});
|
|
43
|
-
if (!response.ok) {
|
|
44
|
-
const errorData = await response.json().catch(() => ({ error: "Unknown error" }));
|
|
45
|
-
throw new Error(errorData.error || `Failed to execute code: ${response.status}`);
|
|
46
|
-
}
|
|
47
|
-
if (!response.body) {
|
|
48
|
-
throw new Error("No response body for streaming execution");
|
|
49
|
-
}
|
|
50
|
-
for await (const chunk of this.readLines(response.body)) {
|
|
51
|
-
await this.parseExecutionResult(chunk, callbacks);
|
|
52
|
-
}
|
|
53
61
|
}
|
|
54
62
|
async *readLines(stream) {
|
|
55
63
|
const reader = stream.getReader();
|
|
@@ -146,40 +154,84 @@ var JupyterClient = class extends HttpClient {
|
|
|
146
154
|
}
|
|
147
155
|
}
|
|
148
156
|
async listCodeContexts() {
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
157
|
+
return this.executeWithRetry(async () => {
|
|
158
|
+
const response = await this.doFetch("/api/contexts", {
|
|
159
|
+
method: "GET",
|
|
160
|
+
headers: { "Content-Type": "application/json" }
|
|
161
|
+
});
|
|
162
|
+
if (!response.ok) {
|
|
163
|
+
throw await parseErrorResponse(response);
|
|
164
|
+
}
|
|
165
|
+
const data = await response.json();
|
|
166
|
+
return data.contexts.map((ctx) => ({
|
|
167
|
+
id: ctx.id,
|
|
168
|
+
language: ctx.language,
|
|
169
|
+
cwd: ctx.cwd,
|
|
170
|
+
createdAt: new Date(ctx.createdAt),
|
|
171
|
+
lastUsed: new Date(ctx.lastUsed)
|
|
172
|
+
}));
|
|
152
173
|
});
|
|
153
|
-
if (!response.ok) {
|
|
154
|
-
const errorData = await response.json().catch(() => ({ error: "Unknown error" }));
|
|
155
|
-
throw new Error(errorData.error || `Failed to list contexts: ${response.status}`);
|
|
156
|
-
}
|
|
157
|
-
const data = await response.json();
|
|
158
|
-
return data.contexts.map((ctx) => ({
|
|
159
|
-
id: ctx.id,
|
|
160
|
-
language: ctx.language,
|
|
161
|
-
cwd: ctx.cwd,
|
|
162
|
-
createdAt: new Date(ctx.createdAt),
|
|
163
|
-
lastUsed: new Date(ctx.lastUsed)
|
|
164
|
-
}));
|
|
165
174
|
}
|
|
166
175
|
async deleteCodeContext(contextId) {
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
176
|
+
return this.executeWithRetry(async () => {
|
|
177
|
+
const response = await this.doFetch(`/api/contexts/${contextId}`, {
|
|
178
|
+
method: "DELETE",
|
|
179
|
+
headers: { "Content-Type": "application/json" }
|
|
180
|
+
});
|
|
181
|
+
if (!response.ok) {
|
|
182
|
+
throw await parseErrorResponse(response);
|
|
183
|
+
}
|
|
170
184
|
});
|
|
171
|
-
if (!response.ok) {
|
|
172
|
-
const errorData = await response.json().catch(() => ({ error: "Unknown error" }));
|
|
173
|
-
throw new Error(errorData.error || `Failed to delete context: ${response.status}`);
|
|
174
|
-
}
|
|
175
185
|
}
|
|
176
186
|
// Override parent doFetch to be public for this class
|
|
177
187
|
async doFetch(path, options) {
|
|
178
188
|
return super.doFetch(path, options);
|
|
179
189
|
}
|
|
190
|
+
/**
|
|
191
|
+
* Execute an operation with automatic retry for transient errors
|
|
192
|
+
*/
|
|
193
|
+
async executeWithRetry(operation) {
|
|
194
|
+
let lastError;
|
|
195
|
+
for (let attempt = 0; attempt < this.maxRetries; attempt++) {
|
|
196
|
+
try {
|
|
197
|
+
return await operation();
|
|
198
|
+
} catch (error) {
|
|
199
|
+
lastError = error;
|
|
200
|
+
if (this.isRetryableError(error)) {
|
|
201
|
+
if (attempt < this.maxRetries - 1) {
|
|
202
|
+
const delay = this.retryDelayMs * 2 ** attempt + Math.random() * 1e3;
|
|
203
|
+
await new Promise((resolve) => setTimeout(resolve, delay));
|
|
204
|
+
continue;
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
throw error;
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
if (lastError?.message.includes("Code execution")) {
|
|
211
|
+
throw lastError;
|
|
212
|
+
}
|
|
213
|
+
throw new Error("Unable to execute code at this time");
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* Check if an error is retryable
|
|
217
|
+
*/
|
|
218
|
+
isRetryableError(error) {
|
|
219
|
+
if (isRetryableError(error)) {
|
|
220
|
+
return true;
|
|
221
|
+
}
|
|
222
|
+
if (error instanceof Error) {
|
|
223
|
+
if (error.message.includes("Circuit breaker is open")) {
|
|
224
|
+
return true;
|
|
225
|
+
}
|
|
226
|
+
if ("status" in error && error.status === "circuit_open") {
|
|
227
|
+
return true;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
return false;
|
|
231
|
+
}
|
|
180
232
|
};
|
|
181
233
|
|
|
182
234
|
export {
|
|
183
235
|
JupyterClient
|
|
184
236
|
};
|
|
185
|
-
//# sourceMappingURL=chunk-
|
|
237
|
+
//# sourceMappingURL=chunk-VTKZL632.js.map
|
|
@@ -0,0 +1 @@
|
|
|
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,3 +1,9 @@
|
|
|
1
|
+
import {
|
|
2
|
+
SecurityError,
|
|
3
|
+
logSecurityEvent,
|
|
4
|
+
sanitizeSandboxId,
|
|
5
|
+
validatePort
|
|
6
|
+
} from "./chunk-6UAWTJ5S.js";
|
|
1
7
|
import {
|
|
2
8
|
parseSSEStream
|
|
3
9
|
} from "./chunk-NNGBXDMY.js";
|
|
@@ -10,13 +16,7 @@ import {
|
|
|
10
16
|
} from "./chunk-FKBV7CZS.js";
|
|
11
17
|
import {
|
|
12
18
|
JupyterClient
|
|
13
|
-
} from "./chunk-
|
|
14
|
-
import {
|
|
15
|
-
SecurityError,
|
|
16
|
-
logSecurityEvent,
|
|
17
|
-
sanitizeSandboxId,
|
|
18
|
-
validatePort
|
|
19
|
-
} from "./chunk-6UAWTJ5S.js";
|
|
19
|
+
} from "./chunk-VTKZL632.js";
|
|
20
20
|
|
|
21
21
|
// src/sandbox.ts
|
|
22
22
|
import { Container, getContainer } from "@cloudflare/containers";
|
|
@@ -687,4 +687,4 @@ export {
|
|
|
687
687
|
proxyToSandbox,
|
|
688
688
|
isLocalhostPattern
|
|
689
689
|
};
|
|
690
|
-
//# sourceMappingURL=chunk-
|
|
690
|
+
//# sourceMappingURL=chunk-ZMPO44U4.js.map
|
|
@@ -9,6 +9,8 @@ interface ExecutionCallbacks {
|
|
|
9
9
|
onError?: (error: ExecutionError) => void | Promise<void>;
|
|
10
10
|
}
|
|
11
11
|
declare class JupyterClient extends HttpClient {
|
|
12
|
+
private readonly maxRetries;
|
|
13
|
+
private readonly retryDelayMs;
|
|
12
14
|
createCodeContext(options?: CreateContextOptions): Promise<CodeContext>;
|
|
13
15
|
runCodeStream(contextId: string | undefined, code: string, language: string | undefined, callbacks: ExecutionCallbacks): Promise<void>;
|
|
14
16
|
private readLines;
|
|
@@ -16,6 +18,14 @@ declare class JupyterClient extends HttpClient {
|
|
|
16
18
|
listCodeContexts(): Promise<CodeContext[]>;
|
|
17
19
|
deleteCodeContext(contextId: string): Promise<void>;
|
|
18
20
|
doFetch(path: string, options?: RequestInit): Promise<Response>;
|
|
21
|
+
/**
|
|
22
|
+
* Execute an operation with automatic retry for transient errors
|
|
23
|
+
*/
|
|
24
|
+
private executeWithRetry;
|
|
25
|
+
/**
|
|
26
|
+
* Check if an error is retryable
|
|
27
|
+
*/
|
|
28
|
+
private isRetryableError;
|
|
19
29
|
}
|
|
20
30
|
|
|
21
31
|
declare function getSandbox(ns: DurableObjectNamespace<Sandbox>, id: string): DurableObjectStub<Sandbox<unknown>>;
|
package/dist/client.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export { D as DeleteFileResponse, E as ExecuteResponse, G as GitCheckoutResponse, H as HttpClient, M as MkdirResponse, a as MoveFileResponse, R as ReadFileResponse, b as RenameFileResponse, W as WriteFileResponse } from './client-
|
|
1
|
+
export { D as DeleteFileResponse, E as ExecuteResponse, G as GitCheckoutResponse, H as HttpClient, M as MkdirResponse, a as MoveFileResponse, R as ReadFileResponse, b as RenameFileResponse, W as WriteFileResponse } from './client-bzEV222a.js';
|
|
2
2
|
import './types.js';
|
|
3
3
|
import '@cloudflare/containers';
|
|
4
4
|
import './interpreter-types.js';
|
package/dist/errors.d.ts
ADDED
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Standard error response from the sandbox API
|
|
3
|
+
*/
|
|
4
|
+
interface SandboxErrorResponse {
|
|
5
|
+
error?: string;
|
|
6
|
+
status?: string;
|
|
7
|
+
progress?: number;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Base error class for all Sandbox-related errors
|
|
11
|
+
*/
|
|
12
|
+
declare class SandboxError extends Error {
|
|
13
|
+
constructor(message: string);
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Error thrown when Jupyter functionality is requested but the service is still initializing.
|
|
17
|
+
*
|
|
18
|
+
* Note: With the current implementation, requests wait for Jupyter to be ready.
|
|
19
|
+
* This error is only thrown when:
|
|
20
|
+
* 1. The request times out waiting for Jupyter (default: 30 seconds)
|
|
21
|
+
* 2. Jupyter initialization actually fails
|
|
22
|
+
*
|
|
23
|
+
* Most requests will succeed after a delay, not throw this error.
|
|
24
|
+
*/
|
|
25
|
+
declare class JupyterNotReadyError extends SandboxError {
|
|
26
|
+
readonly code = "JUPYTER_NOT_READY";
|
|
27
|
+
readonly retryAfter: number;
|
|
28
|
+
readonly progress?: number;
|
|
29
|
+
constructor(message?: string, options?: {
|
|
30
|
+
retryAfter?: number;
|
|
31
|
+
progress?: number;
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Error thrown when a context is not found
|
|
36
|
+
*/
|
|
37
|
+
declare class ContextNotFoundError extends SandboxError {
|
|
38
|
+
readonly code = "CONTEXT_NOT_FOUND";
|
|
39
|
+
readonly contextId: string;
|
|
40
|
+
constructor(contextId: string);
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Error thrown when code execution fails
|
|
44
|
+
*/
|
|
45
|
+
declare class CodeExecutionError extends SandboxError {
|
|
46
|
+
readonly code = "CODE_EXECUTION_ERROR";
|
|
47
|
+
readonly executionError?: {
|
|
48
|
+
ename?: string;
|
|
49
|
+
evalue?: string;
|
|
50
|
+
traceback?: string[];
|
|
51
|
+
};
|
|
52
|
+
constructor(message: string, executionError?: any);
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Error thrown when the sandbox container is not ready
|
|
56
|
+
*/
|
|
57
|
+
declare class ContainerNotReadyError extends SandboxError {
|
|
58
|
+
readonly code = "CONTAINER_NOT_READY";
|
|
59
|
+
constructor(message?: string);
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Error thrown when a network request to the sandbox fails
|
|
63
|
+
*/
|
|
64
|
+
declare class SandboxNetworkError extends SandboxError {
|
|
65
|
+
readonly code = "NETWORK_ERROR";
|
|
66
|
+
readonly statusCode?: number;
|
|
67
|
+
readonly statusText?: string;
|
|
68
|
+
constructor(message: string, statusCode?: number, statusText?: string);
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Error thrown when service is temporarily unavailable (e.g., circuit breaker open)
|
|
72
|
+
*/
|
|
73
|
+
declare class ServiceUnavailableError extends SandboxError {
|
|
74
|
+
readonly code = "SERVICE_UNAVAILABLE";
|
|
75
|
+
readonly retryAfter?: number;
|
|
76
|
+
constructor(message?: string, retryAfter?: number);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Type guard to check if an error is a JupyterNotReadyError
|
|
80
|
+
*/
|
|
81
|
+
declare function isJupyterNotReadyError(error: unknown): error is JupyterNotReadyError;
|
|
82
|
+
/**
|
|
83
|
+
* Type guard to check if an error is any SandboxError
|
|
84
|
+
*/
|
|
85
|
+
declare function isSandboxError(error: unknown): error is SandboxError;
|
|
86
|
+
/**
|
|
87
|
+
* Helper to determine if an error is retryable
|
|
88
|
+
*/
|
|
89
|
+
declare function isRetryableError(error: unknown): boolean;
|
|
90
|
+
/**
|
|
91
|
+
* Parse error response from the sandbox API and return appropriate error instance
|
|
92
|
+
*/
|
|
93
|
+
declare function parseErrorResponse(response: Response): Promise<SandboxError>;
|
|
94
|
+
|
|
95
|
+
export { CodeExecutionError, ContainerNotReadyError, ContextNotFoundError, JupyterNotReadyError, SandboxError, type SandboxErrorResponse, SandboxNetworkError, ServiceUnavailableError, isJupyterNotReadyError, isRetryableError, isSandboxError, parseErrorResponse };
|
package/dist/errors.js
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import {
|
|
2
|
+
CodeExecutionError,
|
|
3
|
+
ContainerNotReadyError,
|
|
4
|
+
ContextNotFoundError,
|
|
5
|
+
JupyterNotReadyError,
|
|
6
|
+
SandboxError,
|
|
7
|
+
SandboxNetworkError,
|
|
8
|
+
ServiceUnavailableError,
|
|
9
|
+
isJupyterNotReadyError,
|
|
10
|
+
isRetryableError,
|
|
11
|
+
isSandboxError,
|
|
12
|
+
parseErrorResponse
|
|
13
|
+
} from "./chunk-LALY4SFU.js";
|
|
14
|
+
export {
|
|
15
|
+
CodeExecutionError,
|
|
16
|
+
ContainerNotReadyError,
|
|
17
|
+
ContextNotFoundError,
|
|
18
|
+
JupyterNotReadyError,
|
|
19
|
+
SandboxError,
|
|
20
|
+
SandboxNetworkError,
|
|
21
|
+
ServiceUnavailableError,
|
|
22
|
+
isJupyterNotReadyError,
|
|
23
|
+
isRetryableError,
|
|
24
|
+
isSandboxError,
|
|
25
|
+
parseErrorResponse
|
|
26
|
+
};
|
|
27
|
+
//# sourceMappingURL=errors.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
export { D as DeleteFileResponse, E as ExecuteResponse, G as GitCheckoutResponse, M as MkdirResponse, a as MoveFileResponse, R as ReadFileResponse, b as RenameFileResponse, S as Sandbox, W as WriteFileResponse, g as getSandbox } from './client-
|
|
1
|
+
export { D as DeleteFileResponse, E as ExecuteResponse, G as GitCheckoutResponse, M as MkdirResponse, a as MoveFileResponse, R as ReadFileResponse, b as RenameFileResponse, S as Sandbox, W as WriteFileResponse, g as getSandbox } from './client-bzEV222a.js';
|
|
2
|
+
export { CodeExecutionError, ContainerNotReadyError, ContextNotFoundError, JupyterNotReadyError, SandboxError, SandboxErrorResponse, SandboxNetworkError, ServiceUnavailableError, isJupyterNotReadyError, isRetryableError, isSandboxError, parseErrorResponse } from './errors.js';
|
|
2
3
|
export { ChartData, CodeContext, CreateContextOptions, Execution, ExecutionError, OutputMessage, Result, ResultImpl, RunCodeOptions } from './interpreter-types.js';
|
|
3
4
|
export { RouteInfo, SandboxEnv, proxyToSandbox } from './request-handler.js';
|
|
4
5
|
export { asyncIterableToSSEStream, parseSSEStream, responseToAsyncIterable } from './sse-parser.js';
|
package/dist/index.js
CHANGED
|
@@ -2,7 +2,8 @@ import {
|
|
|
2
2
|
Sandbox,
|
|
3
3
|
getSandbox,
|
|
4
4
|
proxyToSandbox
|
|
5
|
-
} from "./chunk-
|
|
5
|
+
} from "./chunk-ZMPO44U4.js";
|
|
6
|
+
import "./chunk-6UAWTJ5S.js";
|
|
6
7
|
import {
|
|
7
8
|
asyncIterableToSSEStream,
|
|
8
9
|
parseSSEStream,
|
|
@@ -13,14 +14,37 @@ import "./chunk-FKBV7CZS.js";
|
|
|
13
14
|
import {
|
|
14
15
|
ResultImpl
|
|
15
16
|
} from "./chunk-EGC5IYXA.js";
|
|
16
|
-
import "./chunk-
|
|
17
|
+
import "./chunk-VTKZL632.js";
|
|
17
18
|
import "./chunk-CUHYLCMT.js";
|
|
18
|
-
import
|
|
19
|
+
import {
|
|
20
|
+
CodeExecutionError,
|
|
21
|
+
ContainerNotReadyError,
|
|
22
|
+
ContextNotFoundError,
|
|
23
|
+
JupyterNotReadyError,
|
|
24
|
+
SandboxError,
|
|
25
|
+
SandboxNetworkError,
|
|
26
|
+
ServiceUnavailableError,
|
|
27
|
+
isJupyterNotReadyError,
|
|
28
|
+
isRetryableError,
|
|
29
|
+
isSandboxError,
|
|
30
|
+
parseErrorResponse
|
|
31
|
+
} from "./chunk-LALY4SFU.js";
|
|
19
32
|
export {
|
|
33
|
+
CodeExecutionError,
|
|
34
|
+
ContainerNotReadyError,
|
|
35
|
+
ContextNotFoundError,
|
|
36
|
+
JupyterNotReadyError,
|
|
20
37
|
ResultImpl,
|
|
21
38
|
Sandbox,
|
|
39
|
+
SandboxError,
|
|
40
|
+
SandboxNetworkError,
|
|
41
|
+
ServiceUnavailableError,
|
|
22
42
|
asyncIterableToSSEStream,
|
|
23
43
|
getSandbox,
|
|
44
|
+
isJupyterNotReadyError,
|
|
45
|
+
isRetryableError,
|
|
46
|
+
isSandboxError,
|
|
47
|
+
parseErrorResponse,
|
|
24
48
|
parseSSEStream,
|
|
25
49
|
proxyToSandbox,
|
|
26
50
|
responseToAsyncIterable
|
package/dist/interpreter.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { CreateContextOptions, CodeContext, RunCodeOptions, Execution } from './interpreter-types.js';
|
|
2
|
-
import { S as Sandbox } from './client-
|
|
2
|
+
import { S as Sandbox } from './client-bzEV222a.js';
|
|
3
3
|
import '@cloudflare/containers';
|
|
4
4
|
import './types.js';
|
|
5
5
|
|
package/dist/jupyter-client.d.ts
CHANGED
package/dist/jupyter-client.js
CHANGED
package/dist/request-handler.js
CHANGED
|
@@ -1,14 +1,15 @@
|
|
|
1
1
|
import {
|
|
2
2
|
isLocalhostPattern,
|
|
3
3
|
proxyToSandbox
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-ZMPO44U4.js";
|
|
5
|
+
import "./chunk-6UAWTJ5S.js";
|
|
5
6
|
import "./chunk-NNGBXDMY.js";
|
|
6
7
|
import "./chunk-S5FFBU4Y.js";
|
|
7
8
|
import "./chunk-FKBV7CZS.js";
|
|
8
9
|
import "./chunk-EGC5IYXA.js";
|
|
9
|
-
import "./chunk-
|
|
10
|
+
import "./chunk-VTKZL632.js";
|
|
10
11
|
import "./chunk-CUHYLCMT.js";
|
|
11
|
-
import "./chunk-
|
|
12
|
+
import "./chunk-LALY4SFU.js";
|
|
12
13
|
export {
|
|
13
14
|
isLocalhostPattern,
|
|
14
15
|
proxyToSandbox
|
package/dist/sandbox.d.ts
CHANGED
package/dist/sandbox.js
CHANGED
|
@@ -1,14 +1,15 @@
|
|
|
1
1
|
import {
|
|
2
2
|
Sandbox,
|
|
3
3
|
getSandbox
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-ZMPO44U4.js";
|
|
5
|
+
import "./chunk-6UAWTJ5S.js";
|
|
5
6
|
import "./chunk-NNGBXDMY.js";
|
|
6
7
|
import "./chunk-S5FFBU4Y.js";
|
|
7
8
|
import "./chunk-FKBV7CZS.js";
|
|
8
9
|
import "./chunk-EGC5IYXA.js";
|
|
9
|
-
import "./chunk-
|
|
10
|
+
import "./chunk-VTKZL632.js";
|
|
10
11
|
import "./chunk-CUHYLCMT.js";
|
|
11
|
-
import "./chunk-
|
|
12
|
+
import "./chunk-LALY4SFU.js";
|
|
12
13
|
export {
|
|
13
14
|
Sandbox,
|
|
14
15
|
getSandbox
|