@cloudflare/sandbox 0.3.1 → 0.3.3

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 +12 -0
  2. package/Dockerfile +22 -24
  3. package/README.md +2 -2
  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 +2 -2
  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
package/CHANGELOG.md CHANGED
@@ -1,5 +1,17 @@
1
1
  # @cloudflare/sandbox
2
2
 
3
+ ## 0.3.3
4
+
5
+ ### Patch Changes
6
+
7
+ - [#83](https://github.com/cloudflare/sandbox-sdk/pull/83) [`eec5bb6`](https://github.com/cloudflare/sandbox-sdk/commit/eec5bb6203dd5d775b4b54e91c26de25eeb767ce) Thanks [@mikenomitch](https://github.com/mikenomitch)! - Bump containers package version
8
+
9
+ ## 0.3.2
10
+
11
+ ### Patch Changes
12
+
13
+ - [#76](https://github.com/cloudflare/sandbox-sdk/pull/76) [`ef9e320`](https://github.com/cloudflare/sandbox-sdk/commit/ef9e320dcef30e57797fef6ebd9a9383fa9720d9) Thanks [@ghostwriternr](https://github.com/ghostwriternr)! - Replace Jupyter with lightweight interpreters for >90% faster cold starts for `.runCode` calls, while maintaining full code execution capabilities and rich output support.
14
+
3
15
  ## 0.3.1
4
16
 
5
17
  ### Patch Changes
package/Dockerfile CHANGED
@@ -51,24 +51,12 @@ RUN curl -fsSL https://deb.nodesource.com/setup_20.x | bash - \
51
51
  COPY --from=bun-source /usr/local/bin/bun /usr/local/bin/bun
52
52
  COPY --from=bun-source /usr/local/bin/bunx /usr/local/bin/bunx
53
53
 
54
- # Install minimal Jupyter components
55
- RUN pip3 install --no-cache-dir \
56
- jupyter-server \
57
- jupyter-client \
58
- ipykernel \
59
- orjson \
60
- && python3 -m ipykernel install --user --name python3
61
-
62
- # Install scientific packages
54
+ # Install essential Python packages for code execution
63
55
  RUN pip3 install --no-cache-dir \
64
56
  matplotlib \
65
57
  numpy \
66
58
  pandas \
67
- seaborn
68
-
69
- # Install JavaScript kernel (ijavascript) - using E2B's fork
70
- RUN npm install -g git+https://github.com/e2b-dev/ijavascript.git \
71
- && ijsinstall --install=global
59
+ ipython
72
60
 
73
61
  # Set up container server directory
74
62
  WORKDIR /container-server
@@ -77,9 +65,7 @@ WORKDIR /container-server
77
65
  RUN python3 --version && \
78
66
  node --version && \
79
67
  npm --version && \
80
- bun --version && \
81
- jupyter --version && \
82
- jupyter kernelspec list
68
+ bun --version
83
69
 
84
70
  # Copy container source files to server directory
85
71
  COPY container_src/package.json container_src/bun.lock ./
@@ -87,15 +73,27 @@ RUN bun install --frozen-lockfile
87
73
 
88
74
  COPY container_src/ ./
89
75
 
90
- # Compile TypeScript control process
91
- # Use npx -p typescript to ensure we get the right tsc command
92
- RUN npx -p typescript tsc control-process.ts --outDir . --module commonjs --target es2020 --esModuleInterop --skipLibCheck
93
-
94
- # Create clean workspace directory for users
76
+ # Compile TypeScript files using the locally installed TypeScript
77
+ RUN npx tsc control-process.ts --outDir . --module commonjs --target es2020 --esModuleInterop --skipLibCheck
78
+ RUN cd runtime/executors/javascript && npx tsc node_executor.ts --module commonjs --target es2020 --esModuleInterop --skipLibCheck
79
+ RUN cd runtime/executors/typescript && npx tsc ts_executor.ts --module commonjs --target es2020 --esModuleInterop --skipLibCheck
80
+
81
+ # Configure process pool sizes (can be overridden at runtime)
82
+ ENV PYTHON_POOL_MIN_SIZE=3
83
+ ENV PYTHON_POOL_MAX_SIZE=15
84
+ ENV JAVASCRIPT_POOL_MIN_SIZE=3
85
+ ENV JAVASCRIPT_POOL_MAX_SIZE=10
86
+ ENV TYPESCRIPT_POOL_MIN_SIZE=3
87
+ ENV TYPESCRIPT_POOL_MAX_SIZE=10
88
+
89
+ # Create clean workspace directory for user code
90
+ # Architecture:
91
+ # /container-server/ - SDK infrastructure (server, executors, dependencies)
92
+ # /workspace/ - User's clean workspace for their code
95
93
  RUN mkdir -p /workspace
96
94
 
97
- # Expose the application port (3000 for control, 8888 for Jupyter)
98
- EXPOSE 3000 8888
95
+ # Expose the application port (3000 for control)
96
+ EXPOSE 3000
99
97
 
100
98
  # Make startup script executable
101
99
  RUN chmod +x startup.sh
package/README.md CHANGED
@@ -72,7 +72,7 @@ npm install @cloudflare/sandbox
72
72
  1. **Create a Dockerfile** (temporary requirement, will be removed in future releases):
73
73
 
74
74
  ```dockerfile
75
- FROM docker.io/cloudflare/sandbox:0.3.1
75
+ FROM docker.io/cloudflare/sandbox:0.3.3
76
76
 
77
77
  # Expose the ports you want to expose
78
78
  EXPOSE 3000
@@ -89,7 +89,7 @@ EXPOSE 3000
89
89
  {
90
90
  "class_name": "Sandbox",
91
91
  "image": "./Dockerfile",
92
- "max_instances": 1
92
+ "max_instances": 20
93
93
  }
94
94
  ],
95
95
  "durable_objects": {
@@ -4,119 +4,73 @@
4
4
  "": {
5
5
  "name": "sandbox-server",
6
6
  "dependencies": {
7
- "@jupyterlab/services": "^7.0.0",
7
+ "esbuild": "^0.21.5",
8
8
  "uuid": "^9.0.1",
9
- "ws": "^8.16.0",
10
9
  },
11
10
  "devDependencies": {
11
+ "@types/node": "^20.0.0",
12
12
  "@types/uuid": "^9.0.7",
13
- "@types/ws": "^8.5.10",
13
+ "typescript": "^5.3.0",
14
14
  },
15
15
  },
16
16
  },
17
17
  "packages": {
18
- "@jupyter/ydoc": ["@jupyter/ydoc@3.1.0", "", { "dependencies": { "@jupyterlab/nbformat": "^3.0.0 || ^4.0.0-alpha.21 || ^4.0.0", "@lumino/coreutils": "^1.11.0 || ^2.0.0", "@lumino/disposable": "^1.10.0 || ^2.0.0", "@lumino/signaling": "^1.10.0 || ^2.0.0", "y-protocols": "^1.0.5", "yjs": "^13.5.40" } }, "sha512-+hLNUBZr8Zz8NiuaoYKINDURDJHbjFGxg8EcSU52y+rBe412TsoFxfPSng4eP7w3cZFrVlm7D+8K0nAMHxj0ZQ=="],
18
+ "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.21.5", "", { "os": "aix", "cpu": "ppc64" }, "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ=="],
19
19
 
20
- "@jupyterlab/coreutils": ["@jupyterlab/coreutils@6.4.5", "", { "dependencies": { "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/signaling": "^2.1.4", "minimist": "~1.2.0", "path-browserify": "^1.0.0", "url-parse": "~1.5.4" } }, "sha512-fNEefnqTNP/3alrmGYDb0Mu1heS5Cu39FIDiDDoZuuWv4bsdE2xf7VaStuwjljsQm7VZwi5aeholi1+ciBkZKg=="],
20
+ "@esbuild/android-arm": ["@esbuild/android-arm@0.21.5", "", { "os": "android", "cpu": "arm" }, "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg=="],
21
21
 
22
- "@jupyterlab/nbformat": ["@jupyterlab/nbformat@4.4.5", "", { "dependencies": { "@lumino/coreutils": "^2.2.1" } }, "sha512-VOlm1klsb1LUNp47AL8iDZPP71pXu9a+WqukJFuuWVxg//+iR6wdqTYcunE9Lx/7SZDqMsXGD+doBZC36Lzenw=="],
22
+ "@esbuild/android-arm64": ["@esbuild/android-arm64@0.21.5", "", { "os": "android", "cpu": "arm64" }, "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A=="],
23
23
 
24
- "@jupyterlab/services": ["@jupyterlab/services@7.4.5", "", { "dependencies": { "@jupyter/ydoc": "^3.0.4", "@jupyterlab/coreutils": "^6.4.5", "@jupyterlab/nbformat": "^4.4.5", "@jupyterlab/settingregistry": "^4.4.5", "@jupyterlab/statedb": "^4.4.5", "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/polling": "^2.1.4", "@lumino/properties": "^2.0.3", "@lumino/signaling": "^2.1.4", "ws": "^8.11.0" } }, "sha512-JEr8+4VS2MqFyYHh4AnpK/g5V0qRBpmnfXmU0p9YFjcIgGR365D8JF0880IJ0e8+jaW88gSAvZFB8x3o9xRoAQ=="],
24
+ "@esbuild/android-x64": ["@esbuild/android-x64@0.21.5", "", { "os": "android", "cpu": "x64" }, "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA=="],
25
25
 
26
- "@jupyterlab/settingregistry": ["@jupyterlab/settingregistry@4.4.5", "", { "dependencies": { "@jupyterlab/nbformat": "^4.4.5", "@jupyterlab/statedb": "^4.4.5", "@lumino/commands": "^2.3.2", "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/signaling": "^2.1.4", "@rjsf/utils": "^5.13.4", "ajv": "^8.12.0", "json5": "^2.2.3" }, "peerDependencies": { "react": ">=16" } }, "sha512-6hEBq3qI99VZwO97W0AM0mP1is57bWC8Vk2xudic1a90wA9G6ovUBzayqSMm/6QhTuIbCY+vruznwhhsLrVMeg=="],
26
+ "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.21.5", "", { "os": "darwin", "cpu": "arm64" }, "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ=="],
27
27
 
28
- "@jupyterlab/statedb": ["@jupyterlab/statedb@4.4.5", "", { "dependencies": { "@lumino/commands": "^2.3.2", "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/properties": "^2.0.3", "@lumino/signaling": "^2.1.4" } }, "sha512-UpqhOujKwoWoxtNBL2Qk0nrw+ORLJ3ckwiJg2eA0CI+n5kO3IBYAnPzak0tSS0mQNspayXr3KAm4GQ80op5yuA=="],
28
+ "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.21.5", "", { "os": "darwin", "cpu": "x64" }, "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw=="],
29
29
 
30
- "@lumino/algorithm": ["@lumino/algorithm@2.0.3", "", {}, "sha512-DIcF7cIrGEC1Wh8DNjdwaL7IdcNs4Jj1VjO/90oHefeQPszKgc6DSfCxvbQiRanKR6tl/JL7tq4ZRPZES2oVAA=="],
30
+ "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.21.5", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g=="],
31
31
 
32
- "@lumino/commands": ["@lumino/commands@2.3.2", "", { "dependencies": { "@lumino/algorithm": "^2.0.3", "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/domutils": "^2.0.3", "@lumino/keyboard": "^2.0.3", "@lumino/signaling": "^2.1.4", "@lumino/virtualdom": "^2.0.3" } }, "sha512-aAFEiUpp2hrkQU82Z85w1L80g0iDzsQRncLBa+pqVR/k0k1lz6H9F6xZ1ff+lBumZKKtsxBxNomvd0hfxLLqGw=="],
32
+ "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.21.5", "", { "os": "freebsd", "cpu": "x64" }, "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ=="],
33
33
 
34
- "@lumino/coreutils": ["@lumino/coreutils@2.2.1", "", { "dependencies": { "@lumino/algorithm": "^2.0.3" } }, "sha512-yij4TnxDIum7xfFUsVvZB0oLv4shs2mNbn3juwtEIsruvVBPmurNzKX0Y8z2QetbP2AZ6MSFtBzEKsihf0H0VA=="],
34
+ "@esbuild/linux-arm": ["@esbuild/linux-arm@0.21.5", "", { "os": "linux", "cpu": "arm" }, "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA=="],
35
35
 
36
- "@lumino/disposable": ["@lumino/disposable@2.1.4", "", { "dependencies": { "@lumino/signaling": "^2.1.4" } }, "sha512-qTJiDbglPE2QnG4x4gtBcRbcfKQibxyyinNGKcNDrcK2TGTbbhK5PpMQ8d70l2V2Xw2pb/LfksBAg5pxkJ/G4A=="],
36
+ "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.21.5", "", { "os": "linux", "cpu": "arm64" }, "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q=="],
37
37
 
38
- "@lumino/domutils": ["@lumino/domutils@2.0.3", "", {}, "sha512-bXAbZg3mf2ZDNdBBpCGDike3U+osRGHePTh8H2Ud2KwaN4g/5IryFJm/TiO4K5IYs91bWF2Zqhf3FsdbZKHlGw=="],
38
+ "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.21.5", "", { "os": "linux", "cpu": "ia32" }, "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg=="],
39
39
 
40
- "@lumino/keyboard": ["@lumino/keyboard@2.0.3", "", {}, "sha512-bU2OxAR8a9eNBdV0YFjU6/lVVpbOw1gM7yHOuDGDdNu4J0UpKapFoR9gopNGSaHTmTwDtx9RHdFfIAgHwjZ+VQ=="],
40
+ "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.21.5", "", { "os": "linux", "cpu": "none" }, "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg=="],
41
41
 
42
- "@lumino/polling": ["@lumino/polling@2.1.4", "", { "dependencies": { "@lumino/coreutils": "^2.2.1", "@lumino/disposable": "^2.1.4", "@lumino/signaling": "^2.1.4" } }, "sha512-gSkxlIJ/4/esY2G7bsRrY9A4KimDMHTo0shaD+MCbhd09fZMCWJoDMcA447/dykB1rM5NXgugNLjpdGGL/e8cw=="],
42
+ "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.21.5", "", { "os": "linux", "cpu": "none" }, "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg=="],
43
43
 
44
- "@lumino/properties": ["@lumino/properties@2.0.3", "", {}, "sha512-zkXIU5uYz/ScHCHGl5Bt4gMYsfPxZEduZd80zqDslBWvTIMro3NnzLe66NMnecbdr5N3hDJagYyA8//Qy3XjiA=="],
44
+ "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.21.5", "", { "os": "linux", "cpu": "ppc64" }, "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w=="],
45
45
 
46
- "@lumino/signaling": ["@lumino/signaling@2.1.4", "", { "dependencies": { "@lumino/algorithm": "^2.0.3", "@lumino/coreutils": "^2.2.1" } }, "sha512-nC5Z6d9om369Jkh1Vp3b7C89hV4cjr1fQDVcxhemyKXwc9r6VW7FpKixC+jElcAknP5KLj1FAa8Np+K06mMkEA=="],
46
+ "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.21.5", "", { "os": "linux", "cpu": "none" }, "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA=="],
47
47
 
48
- "@lumino/virtualdom": ["@lumino/virtualdom@2.0.3", "", { "dependencies": { "@lumino/algorithm": "^2.0.3" } }, "sha512-q2C8eBxPvvOOQjN3KuxZ+vJi082JH/GF7KwMdaWsy5g+7wjKdnXPuLQFTBLOrVqIzmbxBDlLeFr93CEhdQXcyQ=="],
48
+ "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.21.5", "", { "os": "linux", "cpu": "s390x" }, "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A=="],
49
49
 
50
- "@rjsf/utils": ["@rjsf/utils@5.24.12", "", { "dependencies": { "json-schema-merge-allof": "^0.8.1", "jsonpointer": "^5.0.1", "lodash": "^4.17.21", "lodash-es": "^4.17.21", "react-is": "^18.2.0" }, "peerDependencies": { "react": "^16.14.0 || >=17" } }, "sha512-fDwQB0XkjZjpdFUz5UAnuZj8nnbxDbX5tp+jTOjjJKw2TMQ9gFFYCQ12lSpdhezA2YgEGZfxyYTGW0DKDL5Drg=="],
50
+ "@esbuild/linux-x64": ["@esbuild/linux-x64@0.21.5", "", { "os": "linux", "cpu": "x64" }, "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ=="],
51
51
 
52
- "@types/node": ["@types/node@24.1.0", "", { "dependencies": { "undici-types": "~7.8.0" } }, "sha512-ut5FthK5moxFKH2T1CUOC6ctR67rQRvvHdFLCD2Ql6KXmMuCrjsSsRI9UsLCm9M18BMwClv4pn327UvB7eeO1w=="],
52
+ "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.21.5", "", { "os": "none", "cpu": "x64" }, "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg=="],
53
53
 
54
- "@types/uuid": ["@types/uuid@9.0.8", "", {}, "sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA=="],
55
-
56
- "@types/ws": ["@types/ws@8.18.1", "", { "dependencies": { "@types/node": "*" } }, "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg=="],
57
-
58
- "ajv": ["ajv@8.17.1", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", "require-from-string": "^2.0.2" } }, "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g=="],
59
-
60
- "compute-gcd": ["compute-gcd@1.2.1", "", { "dependencies": { "validate.io-array": "^1.0.3", "validate.io-function": "^1.0.2", "validate.io-integer-array": "^1.0.0" } }, "sha512-TwMbxBNz0l71+8Sc4czv13h4kEqnchV9igQZBi6QUaz09dnz13juGnnaWWJTRsP3brxOoxeB4SA2WELLw1hCtg=="],
61
-
62
- "compute-lcm": ["compute-lcm@1.1.2", "", { "dependencies": { "compute-gcd": "^1.2.1", "validate.io-array": "^1.0.3", "validate.io-function": "^1.0.2", "validate.io-integer-array": "^1.0.0" } }, "sha512-OFNPdQAXnQhDSKioX8/XYT6sdUlXwpeMjfd6ApxMJfyZ4GxmLR1xvMERctlYhlHwIiz6CSpBc2+qYKjHGZw4TQ=="],
63
-
64
- "fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="],
65
-
66
- "fast-uri": ["fast-uri@3.0.6", "", {}, "sha512-Atfo14OibSv5wAp4VWNsFYE1AchQRTv9cBGWET4pZWHzYshFSS9NQI6I57rdKn9croWVMbYFbLhJ+yJvmZIIHw=="],
67
-
68
- "isomorphic.js": ["isomorphic.js@0.2.5", "", {}, "sha512-PIeMbHqMt4DnUP3MA/Flc0HElYjMXArsw1qwJZcm9sqR8mq3l8NYizFMty0pWwE/tzIGH3EKK5+jes5mAr85yw=="],
69
-
70
- "json-schema-compare": ["json-schema-compare@0.2.2", "", { "dependencies": { "lodash": "^4.17.4" } }, "sha512-c4WYmDKyJXhs7WWvAWm3uIYnfyWFoIp+JEoX34rctVvEkMYCPGhXtvmFFXiffBbxfZsvQ0RNnV5H7GvDF5HCqQ=="],
71
-
72
- "json-schema-merge-allof": ["json-schema-merge-allof@0.8.1", "", { "dependencies": { "compute-lcm": "^1.1.2", "json-schema-compare": "^0.2.2", "lodash": "^4.17.20" } }, "sha512-CTUKmIlPJbsWfzRRnOXz+0MjIqvnleIXwFTzz+t9T86HnYX/Rozria6ZVGLktAU9e+NygNljveP+yxqtQp/Q4w=="],
73
-
74
- "json-schema-traverse": ["json-schema-traverse@1.0.0", "", {}, "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="],
75
-
76
- "json5": ["json5@2.2.3", "", { "bin": { "json5": "lib/cli.js" } }, "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg=="],
77
-
78
- "jsonpointer": ["jsonpointer@5.0.1", "", {}, "sha512-p/nXbhSEcu3pZRdkW1OfJhpsVtW1gd4Wa1fnQc9YLiTfAjn0312eMKimbdIQzuZl9aa9xUGaRlP9T/CJE/ditQ=="],
54
+ "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.21.5", "", { "os": "openbsd", "cpu": "x64" }, "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow=="],
79
55
 
80
- "lib0": ["lib0@0.2.114", "", { "dependencies": { "isomorphic.js": "^0.2.4" }, "bin": { "0gentesthtml": "bin/gentesthtml.js", "0serve": "bin/0serve.js", "0ecdsa-generate-keypair": "bin/0ecdsa-generate-keypair.js" } }, "sha512-gcxmNFzA4hv8UYi8j43uPlQ7CGcyMJ2KQb5kZASw6SnAKAf10hK12i2fjrS3Cl/ugZa5Ui6WwIu1/6MIXiHttQ=="],
56
+ "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.21.5", "", { "os": "sunos", "cpu": "x64" }, "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg=="],
81
57
 
82
- "lodash": ["lodash@4.17.21", "", {}, "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="],
58
+ "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.21.5", "", { "os": "win32", "cpu": "arm64" }, "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A=="],
83
59
 
84
- "lodash-es": ["lodash-es@4.17.21", "", {}, "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw=="],
60
+ "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.21.5", "", { "os": "win32", "cpu": "ia32" }, "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA=="],
85
61
 
86
- "minimist": ["minimist@1.2.8", "", {}, "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA=="],
62
+ "@esbuild/win32-x64": ["@esbuild/win32-x64@0.21.5", "", { "os": "win32", "cpu": "x64" }, "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw=="],
87
63
 
88
- "path-browserify": ["path-browserify@1.0.1", "", {}, "sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g=="],
64
+ "@types/node": ["@types/node@20.19.16", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-VS6TTONVdgwJwtJr7U+ghEjpfmQdqehLLpg/iMYGOd1+ilaFjdBJwFuPggJ4EAYPDCzWfDUHoIxyVnu+tOWVuQ=="],
89
65
 
90
- "querystringify": ["querystringify@2.2.0", "", {}, "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ=="],
91
-
92
- "react": ["react@19.1.1", "", {}, "sha512-w8nqGImo45dmMIfljjMwOGtbmC/mk4CMYhWIicdSflH91J9TyCyczcPFXJzrZ/ZXcgGRFeP6BU0BEJTw6tZdfQ=="],
93
-
94
- "react-is": ["react-is@18.3.1", "", {}, "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg=="],
95
-
96
- "require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="],
66
+ "@types/uuid": ["@types/uuid@9.0.8", "", {}, "sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA=="],
97
67
 
98
- "requires-port": ["requires-port@1.0.0", "", {}, "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ=="],
68
+ "esbuild": ["esbuild@0.21.5", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.21.5", "@esbuild/android-arm": "0.21.5", "@esbuild/android-arm64": "0.21.5", "@esbuild/android-x64": "0.21.5", "@esbuild/darwin-arm64": "0.21.5", "@esbuild/darwin-x64": "0.21.5", "@esbuild/freebsd-arm64": "0.21.5", "@esbuild/freebsd-x64": "0.21.5", "@esbuild/linux-arm": "0.21.5", "@esbuild/linux-arm64": "0.21.5", "@esbuild/linux-ia32": "0.21.5", "@esbuild/linux-loong64": "0.21.5", "@esbuild/linux-mips64el": "0.21.5", "@esbuild/linux-ppc64": "0.21.5", "@esbuild/linux-riscv64": "0.21.5", "@esbuild/linux-s390x": "0.21.5", "@esbuild/linux-x64": "0.21.5", "@esbuild/netbsd-x64": "0.21.5", "@esbuild/openbsd-x64": "0.21.5", "@esbuild/sunos-x64": "0.21.5", "@esbuild/win32-arm64": "0.21.5", "@esbuild/win32-ia32": "0.21.5", "@esbuild/win32-x64": "0.21.5" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw=="],
99
69
 
100
- "undici-types": ["undici-types@7.8.0", "", {}, "sha512-9UJ2xGDvQ43tYyVMpuHlsgApydB8ZKfVYTsLDhXkFL/6gfkp+U8xTGdh8pMJv1SpZna0zxG1DwsKZsreLbXBxw=="],
70
+ "typescript": ["typescript@5.9.2", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-CWBzXQrc/qOkhidw1OzBTQuYRbfyxDXJMVJ1XNwUHGROVmuaeiEm3OslpZ1RV96d7SKKjZKrSJu3+t/xlw3R9A=="],
101
71
 
102
- "url-parse": ["url-parse@1.5.10", "", { "dependencies": { "querystringify": "^2.1.1", "requires-port": "^1.0.0" } }, "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ=="],
72
+ "undici-types": ["undici-types@6.21.0", "", {}, "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ=="],
103
73
 
104
74
  "uuid": ["uuid@9.0.1", "", { "bin": { "uuid": "dist/bin/uuid" } }, "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA=="],
105
-
106
- "validate.io-array": ["validate.io-array@1.0.6", "", {}, "sha512-DeOy7CnPEziggrOO5CZhVKJw6S3Yi7e9e65R1Nl/RTN1vTQKnzjfvks0/8kQ40FP/dsjRAOd4hxmJ7uLa6vxkg=="],
107
-
108
- "validate.io-function": ["validate.io-function@1.0.2", "", {}, "sha512-LlFybRJEriSuBnUhQyG5bwglhh50EpTL2ul23MPIuR1odjO7XaMLFV8vHGwp7AZciFxtYOeiSCT5st+XSPONiQ=="],
109
-
110
- "validate.io-integer": ["validate.io-integer@1.0.5", "", { "dependencies": { "validate.io-number": "^1.0.3" } }, "sha512-22izsYSLojN/P6bppBqhgUDjCkr5RY2jd+N2a3DCAUey8ydvrZ/OkGvFPR7qfOpwR2LC5p4Ngzxz36g5Vgr/hQ=="],
111
-
112
- "validate.io-integer-array": ["validate.io-integer-array@1.0.0", "", { "dependencies": { "validate.io-array": "^1.0.3", "validate.io-integer": "^1.0.4" } }, "sha512-mTrMk/1ytQHtCY0oNO3dztafHYyGU88KL+jRxWuzfOmQb+4qqnWmI+gykvGp8usKZOM0H7keJHEbRaFiYA0VrA=="],
113
-
114
- "validate.io-number": ["validate.io-number@1.0.3", "", {}, "sha512-kRAyotcbNaSYoDnXvb4MHg/0a1egJdLwS6oJ38TJY7aw9n93Fl/3blIXdyYvPOp55CNxywooG/3BcrwNrBpcSg=="],
115
-
116
- "ws": ["ws@8.18.3", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-PEIGCY5tSlUt50cqyMXfCzX+oOPqN0vuGqWzbcJ2xvnkzkq46oOpz7dQaTDBdfICb4N14+GARUDw2XV2N4tvzg=="],
117
-
118
- "y-protocols": ["y-protocols@1.0.6", "", { "dependencies": { "lib0": "^0.2.85" }, "peerDependencies": { "yjs": "^13.0.0" } }, "sha512-vHRF2L6iT3rwj1jub/K5tYcTT/mEYDUppgNPXwp8fmLpui9f7Yeq3OEtTLVF012j39QnV+KEQpNqoN7CWU7Y9Q=="],
119
-
120
- "yjs": ["yjs@13.6.27", "", { "dependencies": { "lib0": "^0.2.99" } }, "sha512-OIDwaflOaq4wC6YlPBy2L6ceKeKuF7DeTxx+jPzv1FHn9tCZ0ZwSRnUBxD05E3yed46fv/FWJbvR+Ud7x0L7zw=="],
121
75
  }
122
76
  }
@@ -654,7 +654,7 @@ async function handleControlMessage(msg: ControlMessage): Promise<void> {
654
654
  *
655
655
  * Creates either an isolated shell using 'unshare' or a regular bash shell.
656
656
  * Isolation uses Linux namespaces (PID) to prevent the shell from:
657
- * - Seeing control plane processes (Jupyter, Bun server)
657
+ * - Seeing control plane processes (Bun server)
658
658
  * - Accessing platform secrets in /proc/1/environ
659
659
  * - Hijacking control plane ports
660
660
  */
@@ -1,5 +1,8 @@
1
1
  import { serve } from "bun";
2
- import { handleExecuteRequest, handleStreamingExecuteRequest } from "./handler/exec";
2
+ import {
3
+ handleExecuteRequest,
4
+ handleStreamingExecuteRequest,
5
+ } from "./handler/exec";
3
6
  import {
4
7
  handleDeleteFileRequest,
5
8
  handleListFilesRequest,
@@ -26,9 +29,12 @@ import {
26
29
  handleStreamProcessLogsRequest,
27
30
  } from "./handler/process";
28
31
  import { handleCreateSession, handleListSessions } from "./handler/session";
32
+ import type { CreateContextRequest } from "./interpreter-service";
33
+ import {
34
+ InterpreterNotReadyError,
35
+ InterpreterService,
36
+ } from "./interpreter-service";
29
37
  import { hasNamespaceSupport, SessionManager } from "./isolation";
30
- import type { CreateContextRequest } from "./jupyter-server";
31
- import { JupyterNotReadyError, JupyterService } from "./jupyter-service";
32
38
 
33
39
  // In-memory storage for exposed ports
34
40
  const exposedPorts = new Map<number, { name?: string; exposedAt: Date }>();
@@ -72,28 +78,12 @@ process.on("uncaughtException", async (error) => {
72
78
  process.exit(1);
73
79
  });
74
80
 
75
- // Initialize Jupyter service with graceful degradation
76
- const jupyterService = new JupyterService();
77
-
78
- // Start Jupyter initialization in background (non-blocking)
79
- console.log("[Container] Starting Jupyter initialization in background...");
80
- console.log(
81
- "[Container] API endpoints are available immediately. Jupyter-dependent features will be available shortly."
82
- );
81
+ // Initialize interpreter service
82
+ const interpreterService = new InterpreterService();
83
83
 
84
- jupyterService
85
- .initialize()
86
- .then(() => {
87
- console.log(
88
- "[Container] Jupyter fully initialized - all features available"
89
- );
90
- })
91
- .catch((error) => {
92
- console.error("[Container] Jupyter initialization failed:", error.message);
93
- console.error(
94
- "[Container] The API will continue in degraded mode without code execution capabilities"
95
- );
96
- });
84
+ // No initialization needed - service is ready immediately!
85
+ console.log("[Container] Interpreter service ready - no cold start!");
86
+ console.log("[Container] All API endpoints available immediately");
97
87
 
98
88
  const server = serve({
99
89
  async fetch(req: Request) {
@@ -144,26 +134,27 @@ const server = serve({
144
134
  return handleExecuteRequest(req, corsHeaders, sessionManager);
145
135
  }
146
136
  break;
147
-
137
+
148
138
  case "/api/execute/stream":
149
139
  if (req.method === "POST") {
150
- return handleStreamingExecuteRequest(req, sessionManager, corsHeaders);
140
+ return handleStreamingExecuteRequest(
141
+ req,
142
+ sessionManager,
143
+ corsHeaders
144
+ );
151
145
  }
152
146
  break;
153
147
 
154
148
  case "/api/ping":
155
149
  if (req.method === "GET") {
156
- const health = await jupyterService.getHealthStatus();
150
+ const health = await interpreterService.getHealthStatus();
157
151
  return new Response(
158
152
  JSON.stringify({
159
153
  message: "pong",
160
154
  timestamp: new Date().toISOString(),
161
- jupyter: health.ready
162
- ? "ready"
163
- : health.initializing
164
- ? "initializing"
165
- : "not ready",
166
- jupyterHealth: health,
155
+ system: "interpreter (70x faster)",
156
+ status: health.ready ? "ready" : "initializing",
157
+ progress: health.progress,
167
158
  }),
168
159
  {
169
160
  headers: {
@@ -267,7 +258,7 @@ const server = serve({
267
258
  if (req.method === "POST") {
268
259
  try {
269
260
  const body = (await req.json()) as CreateContextRequest;
270
- const context = await jupyterService.createContext(body);
261
+ const context = await interpreterService.createContext(body);
271
262
  return new Response(
272
263
  JSON.stringify({
273
264
  id: context.id,
@@ -284,9 +275,9 @@ const server = serve({
284
275
  }
285
276
  );
286
277
  } catch (error) {
287
- if (error instanceof JupyterNotReadyError) {
278
+ if (error instanceof InterpreterNotReadyError) {
288
279
  console.log(
289
- `[Container] Request timed out waiting for Jupyter (${error.progress}% complete)`
280
+ `[Container] Request timed out waiting for interpreter (${error.progress}% complete)`
290
281
  );
291
282
  return new Response(
292
283
  JSON.stringify({
@@ -351,7 +342,7 @@ const server = serve({
351
342
  );
352
343
  }
353
344
  } else if (req.method === "GET") {
354
- const contexts = await jupyterService.listContexts();
345
+ const contexts = await interpreterService.listContexts();
355
346
  return new Response(JSON.stringify({ contexts }), {
356
347
  headers: {
357
348
  "Content-Type": "application/json",
@@ -369,7 +360,7 @@ const server = serve({
369
360
  code: string;
370
361
  language?: string;
371
362
  };
372
- return await jupyterService.executeCode(
363
+ return await interpreterService.executeCode(
373
364
  body.context_id,
374
365
  body.code,
375
366
  body.language
@@ -408,12 +399,12 @@ const server = serve({
408
399
  error.message.includes("initializing")
409
400
  ) {
410
401
  console.log(
411
- "[Container] Code execution deferred - Jupyter still initializing"
402
+ "[Container] Code execution deferred - service still initializing"
412
403
  );
413
404
  } else {
414
405
  console.error("[Container] Error executing code:", error);
415
406
  }
416
- // Error response is already handled by jupyterService.executeCode for not ready state
407
+ // Error response is already handled by service.executeCode for not ready state
417
408
  return new Response(
418
409
  JSON.stringify({
419
410
  error:
@@ -442,7 +433,7 @@ const server = serve({
442
433
  const contextId = pathname.split("/")[3];
443
434
  if (req.method === "DELETE") {
444
435
  try {
445
- await jupyterService.deleteContext(contextId);
436
+ await interpreterService.deleteContext(contextId);
446
437
  return new Response(JSON.stringify({ success: true }), {
447
438
  headers: {
448
439
  "Content-Type": "application/json",
@@ -450,9 +441,9 @@ const server = serve({
450
441
  },
451
442
  });
452
443
  } catch (error) {
453
- if (error instanceof JupyterNotReadyError) {
444
+ if (error instanceof InterpreterNotReadyError) {
454
445
  console.log(
455
- `[Container] Request timed out waiting for Jupyter (${error.progress}% complete)`
446
+ `[Container] Request timed out waiting for interpreter (${error.progress}% complete)`
456
447
  );
457
448
  return new Response(
458
449
  JSON.stringify({