luca 2.0.0 → 3.0.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 (532) hide show
  1. package/.github/workflows/release.yaml +170 -0
  2. package/AGENTS.md +99 -0
  3. package/CLAUDE.md +123 -0
  4. package/CNAME +1 -0
  5. package/README.md +275 -9
  6. package/RUNME.md +56 -0
  7. package/assistants/codingAssistant/ABOUT.md +5 -0
  8. package/assistants/codingAssistant/CORE.md +33 -0
  9. package/assistants/codingAssistant/hooks.ts +21 -0
  10. package/assistants/codingAssistant/tools.ts +12 -0
  11. package/assistants/inkbot/ABOUT.md +16 -0
  12. package/assistants/inkbot/CORE.md +330 -0
  13. package/assistants/inkbot/hooks.ts +6 -0
  14. package/assistants/inkbot/tools.ts +53 -0
  15. package/assistants/researcher/ABOUT.md +5 -0
  16. package/assistants/researcher/CORE.md +46 -0
  17. package/assistants/researcher/hooks.ts +16 -0
  18. package/assistants/researcher/tools.ts +237 -0
  19. package/bun.lock +2667 -0
  20. package/bunfig.toml +3 -0
  21. package/commands/audit-docs.ts +740 -0
  22. package/commands/build-bootstrap.ts +117 -0
  23. package/commands/build-python-bridge.ts +42 -0
  24. package/commands/build-scaffolds.ts +175 -0
  25. package/commands/bundle-consumer-project.ts +521 -0
  26. package/commands/generate-api-docs.ts +114 -0
  27. package/commands/inkbot.ts +874 -0
  28. package/commands/release.ts +80 -0
  29. package/commands/try-all-challenges.ts +543 -0
  30. package/commands/try-challenge.ts +100 -0
  31. package/dist/agi/container.server.d.ts +63 -0
  32. package/dist/agi/container.server.d.ts.map +1 -0
  33. package/dist/agi/endpoints/ask.d.ts +20 -0
  34. package/dist/agi/endpoints/ask.d.ts.map +1 -0
  35. package/dist/agi/endpoints/conversations/[id].d.ts +27 -0
  36. package/dist/agi/endpoints/conversations/[id].d.ts.map +1 -0
  37. package/dist/agi/endpoints/conversations.d.ts +18 -0
  38. package/dist/agi/endpoints/conversations.d.ts.map +1 -0
  39. package/dist/agi/endpoints/experts.d.ts +8 -0
  40. package/dist/agi/endpoints/experts.d.ts.map +1 -0
  41. package/dist/agi/feature.d.ts +9 -0
  42. package/dist/agi/feature.d.ts.map +1 -0
  43. package/dist/agi/features/assistant.d.ts +509 -0
  44. package/dist/agi/features/assistant.d.ts.map +1 -0
  45. package/dist/agi/features/assistants-manager.d.ts +236 -0
  46. package/dist/agi/features/assistants-manager.d.ts.map +1 -0
  47. package/dist/agi/features/autonomous-assistant.d.ts +281 -0
  48. package/dist/agi/features/autonomous-assistant.d.ts.map +1 -0
  49. package/dist/agi/features/browser-use.d.ts +479 -0
  50. package/dist/agi/features/browser-use.d.ts.map +1 -0
  51. package/dist/agi/features/claude-code.d.ts +824 -0
  52. package/dist/agi/features/claude-code.d.ts.map +1 -0
  53. package/dist/agi/features/conversation-history.d.ts +245 -0
  54. package/dist/agi/features/conversation-history.d.ts.map +1 -0
  55. package/dist/agi/features/conversation.d.ts +464 -0
  56. package/dist/agi/features/conversation.d.ts.map +1 -0
  57. package/dist/agi/features/docs-reader.d.ts +72 -0
  58. package/dist/agi/features/docs-reader.d.ts.map +1 -0
  59. package/dist/agi/features/file-tools.d.ts +110 -0
  60. package/dist/agi/features/file-tools.d.ts.map +1 -0
  61. package/dist/agi/features/luca-coder.d.ts +323 -0
  62. package/dist/agi/features/luca-coder.d.ts.map +1 -0
  63. package/dist/agi/features/openai-codex.d.ts +381 -0
  64. package/dist/agi/features/openai-codex.d.ts.map +1 -0
  65. package/dist/agi/features/openapi.d.ts +200 -0
  66. package/dist/agi/features/openapi.d.ts.map +1 -0
  67. package/dist/agi/features/skills-library.d.ts +167 -0
  68. package/dist/agi/features/skills-library.d.ts.map +1 -0
  69. package/dist/agi/index.d.ts +5 -0
  70. package/dist/agi/index.d.ts.map +1 -0
  71. package/dist/agi/lib/interceptor-chain.d.ts +44 -0
  72. package/dist/agi/lib/interceptor-chain.d.ts.map +1 -0
  73. package/dist/agi/lib/token-counter.d.ts +13 -0
  74. package/dist/agi/lib/token-counter.d.ts.map +1 -0
  75. package/dist/bootstrap/generated.d.ts +5 -0
  76. package/dist/bootstrap/generated.d.ts.map +1 -0
  77. package/dist/browser.d.ts +12 -0
  78. package/dist/browser.d.ts.map +1 -0
  79. package/dist/bus.d.ts +29 -0
  80. package/dist/bus.d.ts.map +1 -0
  81. package/dist/cli/build-info.d.ts +4 -0
  82. package/dist/cli/build-info.d.ts.map +1 -0
  83. package/dist/cli/cli.d.ts +3 -12
  84. package/dist/cli/cli.d.ts.map +1 -0
  85. package/dist/client.d.ts +60 -0
  86. package/dist/client.d.ts.map +1 -0
  87. package/dist/clients/civitai/index.d.ts +472 -0
  88. package/dist/clients/civitai/index.d.ts.map +1 -0
  89. package/dist/clients/client-template.d.ts +30 -0
  90. package/dist/clients/client-template.d.ts.map +1 -0
  91. package/dist/clients/comfyui/index.d.ts +281 -0
  92. package/dist/clients/comfyui/index.d.ts.map +1 -0
  93. package/dist/clients/elevenlabs/index.d.ts +197 -0
  94. package/dist/clients/elevenlabs/index.d.ts.map +1 -0
  95. package/dist/clients/graph.d.ts +64 -0
  96. package/dist/clients/graph.d.ts.map +1 -0
  97. package/dist/clients/openai/index.d.ts +247 -0
  98. package/dist/clients/openai/index.d.ts.map +1 -0
  99. package/dist/clients/rest.d.ts +92 -0
  100. package/dist/clients/rest.d.ts.map +1 -0
  101. package/dist/clients/supabase/index.d.ts +176 -0
  102. package/dist/clients/supabase/index.d.ts.map +1 -0
  103. package/dist/clients/websocket.d.ts +127 -0
  104. package/dist/clients/websocket.d.ts.map +1 -0
  105. package/dist/command.d.ts +163 -0
  106. package/dist/command.d.ts.map +1 -0
  107. package/dist/commands/bootstrap.d.ts +20 -0
  108. package/dist/commands/bootstrap.d.ts.map +1 -0
  109. package/dist/commands/chat.d.ts +37 -0
  110. package/dist/commands/chat.d.ts.map +1 -0
  111. package/dist/commands/code.d.ts +28 -0
  112. package/dist/commands/code.d.ts.map +1 -0
  113. package/dist/commands/console.d.ts +22 -0
  114. package/dist/commands/console.d.ts.map +1 -0
  115. package/dist/commands/describe.d.ts +50 -0
  116. package/dist/commands/describe.d.ts.map +1 -0
  117. package/dist/commands/eval.d.ts +23 -0
  118. package/dist/commands/eval.d.ts.map +1 -0
  119. package/dist/commands/help.d.ts +25 -0
  120. package/dist/commands/help.d.ts.map +1 -0
  121. package/dist/commands/index.d.ts +18 -0
  122. package/dist/commands/index.d.ts.map +1 -0
  123. package/dist/commands/introspect.d.ts +24 -0
  124. package/dist/commands/introspect.d.ts.map +1 -0
  125. package/dist/commands/mcp.d.ts +35 -0
  126. package/dist/commands/mcp.d.ts.map +1 -0
  127. package/dist/commands/prompt.d.ts +38 -0
  128. package/dist/commands/prompt.d.ts.map +1 -0
  129. package/dist/commands/run.d.ts +24 -0
  130. package/dist/commands/run.d.ts.map +1 -0
  131. package/dist/commands/sandbox-mcp.d.ts +34 -0
  132. package/dist/commands/sandbox-mcp.d.ts.map +1 -0
  133. package/dist/commands/save-api-docs.d.ts +21 -0
  134. package/dist/commands/save-api-docs.d.ts.map +1 -0
  135. package/dist/commands/scaffold.d.ts +24 -0
  136. package/dist/commands/scaffold.d.ts.map +1 -0
  137. package/dist/commands/select.d.ts +22 -0
  138. package/dist/commands/select.d.ts.map +1 -0
  139. package/dist/commands/serve.d.ts +29 -0
  140. package/dist/commands/serve.d.ts.map +1 -0
  141. package/dist/container-describer.d.ts +144 -0
  142. package/dist/container-describer.d.ts.map +1 -0
  143. package/dist/container.d.ts +451 -0
  144. package/dist/container.d.ts.map +1 -0
  145. package/dist/endpoint.d.ts +113 -0
  146. package/dist/endpoint.d.ts.map +1 -0
  147. package/dist/feature.d.ts +47 -0
  148. package/dist/feature.d.ts.map +1 -0
  149. package/dist/graft.d.ts +29 -0
  150. package/dist/graft.d.ts.map +1 -0
  151. package/dist/hash-object.d.ts +8 -0
  152. package/dist/hash-object.d.ts.map +1 -0
  153. package/dist/helper.d.ts +209 -0
  154. package/dist/helper.d.ts.map +1 -0
  155. package/dist/introspection/generated.node.d.ts +44623 -0
  156. package/dist/introspection/generated.node.d.ts.map +1 -0
  157. package/dist/introspection/generated.web.d.ts +1412 -0
  158. package/dist/introspection/generated.web.d.ts.map +1 -0
  159. package/dist/introspection/index.d.ts +156 -0
  160. package/dist/introspection/index.d.ts.map +1 -0
  161. package/dist/introspection/scan.d.ts +147 -0
  162. package/dist/introspection/scan.d.ts.map +1 -0
  163. package/dist/node/container.d.ts +256 -0
  164. package/dist/node/container.d.ts.map +1 -0
  165. package/dist/node/feature.d.ts +9 -0
  166. package/dist/node/feature.d.ts.map +1 -0
  167. package/dist/node/features/container-link.d.ts +213 -0
  168. package/dist/node/features/container-link.d.ts.map +1 -0
  169. package/dist/node/features/content-db.d.ts +354 -0
  170. package/dist/node/features/content-db.d.ts.map +1 -0
  171. package/dist/node/features/disk-cache.d.ts +236 -0
  172. package/dist/node/features/disk-cache.d.ts.map +1 -0
  173. package/dist/node/features/dns.d.ts +511 -0
  174. package/dist/node/features/dns.d.ts.map +1 -0
  175. package/dist/node/features/docker.d.ts +485 -0
  176. package/dist/node/features/docker.d.ts.map +1 -0
  177. package/dist/node/features/downloader.d.ts +73 -0
  178. package/dist/node/features/downloader.d.ts.map +1 -0
  179. package/dist/node/features/figlet-fonts.d.ts +4 -0
  180. package/dist/node/features/figlet-fonts.d.ts.map +1 -0
  181. package/dist/node/features/file-manager.d.ts +177 -0
  182. package/dist/node/features/file-manager.d.ts.map +1 -0
  183. package/dist/node/features/fs.d.ts +635 -0
  184. package/dist/node/features/fs.d.ts.map +1 -0
  185. package/dist/node/features/git.d.ts +329 -0
  186. package/dist/node/features/git.d.ts.map +1 -0
  187. package/dist/node/features/google-auth.d.ts +200 -0
  188. package/dist/node/features/google-auth.d.ts.map +1 -0
  189. package/dist/node/features/google-calendar.d.ts +194 -0
  190. package/dist/node/features/google-calendar.d.ts.map +1 -0
  191. package/dist/node/features/google-docs.d.ts +138 -0
  192. package/dist/node/features/google-docs.d.ts.map +1 -0
  193. package/dist/node/features/google-drive.d.ts +202 -0
  194. package/dist/node/features/google-drive.d.ts.map +1 -0
  195. package/dist/node/features/google-mail.d.ts +221 -0
  196. package/dist/node/features/google-mail.d.ts.map +1 -0
  197. package/dist/node/features/google-sheets.d.ts +157 -0
  198. package/dist/node/features/google-sheets.d.ts.map +1 -0
  199. package/dist/node/features/grep.d.ts +207 -0
  200. package/dist/node/features/grep.d.ts.map +1 -0
  201. package/dist/node/features/helpers.d.ts +236 -0
  202. package/dist/node/features/helpers.d.ts.map +1 -0
  203. package/dist/node/features/ink.d.ts +332 -0
  204. package/dist/node/features/ink.d.ts.map +1 -0
  205. package/dist/node/features/ipc-socket.d.ts +298 -0
  206. package/dist/node/features/ipc-socket.d.ts.map +1 -0
  207. package/dist/node/features/json-tree.d.ts +140 -0
  208. package/dist/node/features/json-tree.d.ts.map +1 -0
  209. package/dist/node/features/networking.d.ts +373 -0
  210. package/dist/node/features/networking.d.ts.map +1 -0
  211. package/dist/node/features/nlp.d.ts +125 -0
  212. package/dist/node/features/nlp.d.ts.map +1 -0
  213. package/dist/node/features/opener.d.ts +93 -0
  214. package/dist/node/features/opener.d.ts.map +1 -0
  215. package/dist/node/features/os.d.ts +168 -0
  216. package/dist/node/features/os.d.ts.map +1 -0
  217. package/dist/node/features/package-finder.d.ts +419 -0
  218. package/dist/node/features/package-finder.d.ts.map +1 -0
  219. package/dist/node/features/postgres.d.ts +173 -0
  220. package/dist/node/features/postgres.d.ts.map +1 -0
  221. package/dist/node/features/proc.d.ts +285 -0
  222. package/dist/node/features/proc.d.ts.map +1 -0
  223. package/dist/node/features/process-manager.d.ts +427 -0
  224. package/dist/node/features/process-manager.d.ts.map +1 -0
  225. package/dist/node/features/python.d.ts +477 -0
  226. package/dist/node/features/python.d.ts.map +1 -0
  227. package/dist/node/features/redis.d.ts +247 -0
  228. package/dist/node/features/redis.d.ts.map +1 -0
  229. package/dist/node/features/repl.d.ts +84 -0
  230. package/dist/node/features/repl.d.ts.map +1 -0
  231. package/dist/node/features/runpod.d.ts +527 -0
  232. package/dist/node/features/runpod.d.ts.map +1 -0
  233. package/dist/node/features/secure-shell.d.ts +145 -0
  234. package/dist/node/features/secure-shell.d.ts.map +1 -0
  235. package/dist/node/features/semantic-search.d.ts +207 -0
  236. package/dist/node/features/semantic-search.d.ts.map +1 -0
  237. package/dist/node/features/sqlite.d.ts +180 -0
  238. package/dist/node/features/sqlite.d.ts.map +1 -0
  239. package/dist/node/features/telegram.d.ts +173 -0
  240. package/dist/node/features/telegram.d.ts.map +1 -0
  241. package/dist/node/features/transpiler.d.ts +51 -0
  242. package/dist/node/features/transpiler.d.ts.map +1 -0
  243. package/dist/node/features/tts.d.ts +108 -0
  244. package/dist/node/features/tts.d.ts.map +1 -0
  245. package/dist/node/features/ui.d.ts +562 -0
  246. package/dist/node/features/ui.d.ts.map +1 -0
  247. package/dist/node/features/vault.d.ts +90 -0
  248. package/dist/node/features/vault.d.ts.map +1 -0
  249. package/dist/node/features/vm.d.ts +285 -0
  250. package/dist/node/features/vm.d.ts.map +1 -0
  251. package/dist/node/features/yaml-tree.d.ts +118 -0
  252. package/dist/node/features/yaml-tree.d.ts.map +1 -0
  253. package/dist/node/features/yaml.d.ts +127 -0
  254. package/dist/node/features/yaml.d.ts.map +1 -0
  255. package/dist/node.d.ts +67 -0
  256. package/dist/node.d.ts.map +1 -0
  257. package/dist/python/generated.d.ts +2 -0
  258. package/dist/python/generated.d.ts.map +1 -0
  259. package/dist/react/index.d.ts +36 -0
  260. package/dist/react/index.d.ts.map +1 -0
  261. package/dist/registry.d.ts +97 -0
  262. package/dist/registry.d.ts.map +1 -0
  263. package/dist/scaffolds/generated.d.ts +13 -0
  264. package/dist/scaffolds/generated.d.ts.map +1 -0
  265. package/dist/scaffolds/template.d.ts +11 -0
  266. package/dist/scaffolds/template.d.ts.map +1 -0
  267. package/dist/schemas/base.d.ts +254 -0
  268. package/dist/schemas/base.d.ts.map +1 -0
  269. package/dist/selector.d.ts +130 -0
  270. package/dist/selector.d.ts.map +1 -0
  271. package/dist/server.d.ts +89 -0
  272. package/dist/server.d.ts.map +1 -0
  273. package/dist/servers/express.d.ts +104 -0
  274. package/dist/servers/express.d.ts.map +1 -0
  275. package/dist/servers/mcp.d.ts +201 -0
  276. package/dist/servers/mcp.d.ts.map +1 -0
  277. package/dist/servers/socket.d.ts +121 -0
  278. package/dist/servers/socket.d.ts.map +1 -0
  279. package/dist/state.d.ts +24 -0
  280. package/dist/state.d.ts.map +1 -0
  281. package/dist/web/clients/socket.d.ts +37 -0
  282. package/dist/web/clients/socket.d.ts.map +1 -0
  283. package/dist/web/container.d.ts +55 -0
  284. package/dist/web/container.d.ts.map +1 -0
  285. package/dist/web/extension.d.ts +4 -0
  286. package/dist/web/extension.d.ts.map +1 -0
  287. package/dist/web/feature.d.ts +8 -0
  288. package/dist/web/feature.d.ts.map +1 -0
  289. package/dist/web/features/asset-loader.d.ts +35 -0
  290. package/dist/web/features/asset-loader.d.ts.map +1 -0
  291. package/dist/web/features/container-link.d.ts +167 -0
  292. package/dist/web/features/container-link.d.ts.map +1 -0
  293. package/dist/web/features/esbuild.d.ts +51 -0
  294. package/dist/web/features/esbuild.d.ts.map +1 -0
  295. package/dist/web/features/helpers.d.ts +140 -0
  296. package/dist/web/features/helpers.d.ts.map +1 -0
  297. package/dist/web/features/network.d.ts +69 -0
  298. package/dist/web/features/network.d.ts.map +1 -0
  299. package/dist/web/features/speech.d.ts +71 -0
  300. package/dist/web/features/speech.d.ts.map +1 -0
  301. package/dist/web/features/vault.d.ts +62 -0
  302. package/dist/web/features/vault.d.ts.map +1 -0
  303. package/dist/web/features/vm.d.ts +48 -0
  304. package/dist/web/features/vm.d.ts.map +1 -0
  305. package/dist/web/features/voice-recognition.d.ts +96 -0
  306. package/dist/web/features/voice-recognition.d.ts.map +1 -0
  307. package/dist/web/shims/isomorphic-vm.d.ts +22 -0
  308. package/dist/web/shims/isomorphic-vm.d.ts.map +1 -0
  309. package/index.html +1457 -0
  310. package/index.ts +1 -0
  311. package/install.sh +84 -0
  312. package/luca.cli.ts +16 -0
  313. package/luca.console.ts +9 -0
  314. package/main.py +6 -0
  315. package/package.json +219 -58
  316. package/public/index.html +1457 -0
  317. package/public/slides-ai-native.html +902 -0
  318. package/public/slides-intro.html +974 -0
  319. package/pyproject.toml +7 -0
  320. package/scripts/build-web.ts +28 -0
  321. package/scripts/examples/ask-luca-expert.ts +42 -0
  322. package/scripts/examples/assistant-questions.ts +12 -0
  323. package/scripts/examples/excalidraw-expert.ts +75 -0
  324. package/scripts/examples/expert-chat.ts +0 -0
  325. package/scripts/examples/file-manager.ts +14 -0
  326. package/scripts/examples/ideas.ts +12 -0
  327. package/scripts/examples/interactive-chat.ts +20 -0
  328. package/scripts/examples/openai-tool-calls.ts +113 -0
  329. package/scripts/examples/opening-a-web-browser.ts +5 -0
  330. package/scripts/examples/telegram-bot.ts +79 -0
  331. package/scripts/examples/using-assistant-with-mcp.ts +555 -0
  332. package/scripts/examples/using-claude-code.ts +10 -0
  333. package/scripts/examples/using-contentdb.ts +35 -0
  334. package/scripts/examples/using-conversations.ts +35 -0
  335. package/scripts/examples/using-disk-cache.ts +10 -0
  336. package/scripts/examples/using-docker-shell.ts +75 -0
  337. package/scripts/examples/using-elevenlabs.ts +25 -0
  338. package/scripts/examples/using-google-calendar.ts +57 -0
  339. package/scripts/examples/using-google-docs.ts +74 -0
  340. package/scripts/examples/using-google-drive.ts +74 -0
  341. package/scripts/examples/using-google-sheets.ts +89 -0
  342. package/scripts/examples/using-nlp.ts +55 -0
  343. package/scripts/examples/using-ollama.ts +11 -0
  344. package/scripts/examples/using-postgres.ts +55 -0
  345. package/scripts/examples/using-runpod.ts +32 -0
  346. package/scripts/examples/using-tts.ts +40 -0
  347. package/scripts/scaffold.ts +391 -0
  348. package/scripts/scratch.ts +15 -0
  349. package/scripts/stamp-build.sh +12 -0
  350. package/scripts/test-assistant-hooks.ts +13 -0
  351. package/scripts/test-docs-reader.ts +10 -0
  352. package/scripts/test-linux-binary.sh +80 -0
  353. package/scripts/update-introspection-data.ts +58 -0
  354. package/src/agi/README.md +14 -0
  355. package/src/agi/container.server.ts +156 -0
  356. package/src/agi/feature.ts +13 -0
  357. package/src/agi/features/agent-memory.ts +694 -0
  358. package/src/agi/features/assistant.ts +1653 -0
  359. package/src/agi/features/assistants-manager.ts +534 -0
  360. package/src/agi/features/autonomous-assistant.ts +431 -0
  361. package/src/agi/features/browser-use.ts +672 -0
  362. package/src/agi/features/claude-code.ts +1584 -0
  363. package/src/agi/features/coding-tools.ts +175 -0
  364. package/src/agi/features/conversation-history.ts +672 -0
  365. package/src/agi/features/conversation.ts +1494 -0
  366. package/src/agi/features/docs-reader.ts +167 -0
  367. package/src/agi/features/file-tools.ts +340 -0
  368. package/src/agi/features/luca-coder.ts +641 -0
  369. package/src/agi/features/mcp-bridge.ts +532 -0
  370. package/src/agi/features/openai-codex.ts +651 -0
  371. package/src/agi/features/openapi.ts +445 -0
  372. package/src/agi/features/skills-library.ts +557 -0
  373. package/src/agi/index.ts +6 -0
  374. package/src/agi/lib/interceptor-chain.ts +89 -0
  375. package/src/agi/lib/token-counter.ts +202 -0
  376. package/src/bootstrap/generated.ts +9791 -0
  377. package/src/browser.ts +25 -0
  378. package/src/bus.ts +122 -0
  379. package/src/cli/build-info.ts +4 -0
  380. package/src/cli/cli.ts +355 -0
  381. package/src/client.ts +170 -0
  382. package/src/clients/civitai/index.ts +537 -0
  383. package/src/clients/client-template.ts +41 -0
  384. package/src/clients/comfyui/index.ts +604 -0
  385. package/src/clients/elevenlabs/index.ts +317 -0
  386. package/src/clients/graph.ts +87 -0
  387. package/src/clients/openai/index.ts +456 -0
  388. package/src/clients/rest.ts +207 -0
  389. package/src/clients/supabase/index.ts +357 -0
  390. package/src/clients/voicebox/index.ts +300 -0
  391. package/src/clients/websocket.ts +251 -0
  392. package/src/command.ts +506 -0
  393. package/src/commands/bootstrap.ts +244 -0
  394. package/src/commands/chat.ts +309 -0
  395. package/src/commands/code.ts +371 -0
  396. package/src/commands/console.ts +189 -0
  397. package/src/commands/describe.ts +243 -0
  398. package/src/commands/eval.ts +67 -0
  399. package/src/commands/help.ts +240 -0
  400. package/src/commands/index.ts +19 -0
  401. package/src/commands/introspect.ts +218 -0
  402. package/src/commands/mcp.ts +64 -0
  403. package/src/commands/prompt.ts +1014 -0
  404. package/src/commands/run.ts +278 -0
  405. package/src/commands/sandbox-mcp.ts +343 -0
  406. package/src/commands/save-api-docs.ts +51 -0
  407. package/src/commands/scaffold.ts +225 -0
  408. package/src/commands/select.ts +99 -0
  409. package/src/commands/serve.ts +208 -0
  410. package/src/container-describer.ts +1091 -0
  411. package/src/container.ts +1199 -0
  412. package/src/endpoint.ts +365 -0
  413. package/src/entity.ts +173 -0
  414. package/src/feature.ts +118 -0
  415. package/src/graft.ts +181 -0
  416. package/src/hash-object.ts +97 -0
  417. package/src/helper.ts +849 -0
  418. package/src/introspection/generated.agi.ts +41200 -0
  419. package/src/introspection/generated.node.ts +28773 -0
  420. package/src/introspection/generated.web.ts +2272 -0
  421. package/src/introspection/index.ts +296 -0
  422. package/src/introspection/scan.ts +1136 -0
  423. package/src/node/container.ts +409 -0
  424. package/src/node/feature.ts +13 -0
  425. package/src/node/features/container-link.ts +559 -0
  426. package/src/node/features/content-db.ts +849 -0
  427. package/src/node/features/disk-cache.ts +388 -0
  428. package/src/node/features/display-result.ts +57 -0
  429. package/src/node/features/dns.ts +669 -0
  430. package/src/node/features/docker.ts +921 -0
  431. package/src/node/features/downloader.ts +79 -0
  432. package/src/node/features/figlet-fonts.ts +600 -0
  433. package/src/node/features/file-manager.ts +535 -0
  434. package/src/node/features/fs.ts +1050 -0
  435. package/src/node/features/git.ts +592 -0
  436. package/src/node/features/google-auth.ts +504 -0
  437. package/src/node/features/google-calendar.ts +306 -0
  438. package/src/node/features/google-docs.ts +412 -0
  439. package/src/node/features/google-drive.ts +346 -0
  440. package/src/node/features/google-mail.ts +540 -0
  441. package/src/node/features/google-sheets.ts +286 -0
  442. package/src/node/features/grep.ts +427 -0
  443. package/src/node/features/helpers.ts +762 -0
  444. package/src/node/features/ink.ts +490 -0
  445. package/src/node/features/ipc-socket.ts +649 -0
  446. package/src/node/features/json-tree.ts +170 -0
  447. package/src/node/features/networking.ts +961 -0
  448. package/src/node/features/nlp.ts +212 -0
  449. package/src/node/features/opener.ts +180 -0
  450. package/src/node/features/os.ts +403 -0
  451. package/src/node/features/package-finder.ts +540 -0
  452. package/src/node/features/postgres.ts +289 -0
  453. package/src/node/features/proc.ts +503 -0
  454. package/src/node/features/process-manager.ts +844 -0
  455. package/src/node/features/python.ts +912 -0
  456. package/src/node/features/redis.ts +446 -0
  457. package/src/node/features/repl.ts +212 -0
  458. package/src/node/features/runpod.ts +811 -0
  459. package/src/node/features/secure-shell.ts +261 -0
  460. package/src/node/features/semantic-search.ts +935 -0
  461. package/src/node/features/sqlite.ts +289 -0
  462. package/src/node/features/telegram.ts +343 -0
  463. package/src/node/features/transpiler.ts +160 -0
  464. package/src/node/features/tts.ts +185 -0
  465. package/src/node/features/ui.ts +791 -0
  466. package/src/node/features/vault.ts +153 -0
  467. package/src/node/features/vm.ts +462 -0
  468. package/src/node/features/yaml-tree.ts +148 -0
  469. package/src/node/features/yaml.ts +133 -0
  470. package/src/node.ts +76 -0
  471. package/src/python/bridge.py +220 -0
  472. package/src/python/generated.ts +226 -0
  473. package/src/react/index.ts +175 -0
  474. package/src/registry.ts +210 -0
  475. package/src/scaffolds/generated.ts +1814 -0
  476. package/src/scaffolds/template.ts +46 -0
  477. package/src/schemas/base.ts +296 -0
  478. package/src/selector.ts +352 -0
  479. package/src/server.ts +229 -0
  480. package/src/servers/express.ts +283 -0
  481. package/src/servers/mcp.ts +802 -0
  482. package/src/servers/socket.ts +258 -0
  483. package/src/state.ts +101 -0
  484. package/src/web/clients/socket.ts +99 -0
  485. package/src/web/container.ts +75 -0
  486. package/src/web/extension.ts +30 -0
  487. package/src/web/feature.ts +12 -0
  488. package/src/web/features/asset-loader.ts +72 -0
  489. package/src/web/features/container-link.ts +382 -0
  490. package/src/web/features/esbuild.ts +93 -0
  491. package/src/web/features/helpers.ts +291 -0
  492. package/src/web/features/network.ts +85 -0
  493. package/src/web/features/speech.ts +104 -0
  494. package/src/web/features/vault.ts +207 -0
  495. package/src/web/features/vm.ts +85 -0
  496. package/src/web/features/voice-recognition.ts +161 -0
  497. package/src/web/shims/isomorphic-vm.ts +149 -0
  498. package/tsconfig.build.json +12 -0
  499. package/tsconfig.json +58 -0
  500. package/uv.lock +8 -0
  501. package/LICENSE +0 -21
  502. package/dist/cli/cli.js +0 -48
  503. package/dist/cli/common.d.ts +0 -2
  504. package/dist/cli/common.js +0 -6
  505. package/dist/cli/index.d.ts +0 -2
  506. package/dist/cli/index.js +0 -5
  507. package/dist/cli/run.d.ts +0 -1
  508. package/dist/cli/run.js +0 -38
  509. package/dist/core/index.d.ts +0 -4
  510. package/dist/core/index.js +0 -32
  511. package/dist/core/read.d.ts +0 -2
  512. package/dist/core/read.js +0 -29
  513. package/dist/core/request.d.ts +0 -1
  514. package/dist/core/request.js +0 -2
  515. package/dist/core/write.d.ts +0 -2
  516. package/dist/core/write.js +0 -21
  517. package/dist/index.d.ts +0 -1
  518. package/dist/index.js +0 -5
  519. package/dist/utils/common.d.ts +0 -9
  520. package/dist/utils/common.js +0 -57
  521. package/dist/utils/consts.d.ts +0 -3
  522. package/dist/utils/consts.js +0 -11
  523. package/dist/utils/dict.d.ts +0 -1
  524. package/dist/utils/dict.js +0 -7
  525. package/dist/utils/index.d.ts +0 -5
  526. package/dist/utils/index.js +0 -21
  527. package/dist/utils/log.d.ts +0 -1
  528. package/dist/utils/log.js +0 -5
  529. package/dist/utils/types.d.ts +0 -1
  530. package/dist/utils/types.js +0 -2
  531. package/dist/utils/utils.test.d.ts +0 -1
  532. package/dist/utils/utils.test.js +0 -7
@@ -0,0 +1,912 @@
1
+ import { z } from 'zod'
2
+ import { FeatureStateSchema, FeatureOptionsSchema, FeatureEventsSchema } from '../../schemas/base.js'
3
+ import { Feature } from "../feature.js";
4
+ import { tmpdir } from 'os';
5
+ import { bridgeScript } from '../../python/generated.js';
6
+ import type { ChildProcess } from 'child_process';
7
+
8
+ export const PythonStateSchema = FeatureStateSchema.extend({
9
+ /** Path to the detected Python executable */
10
+ pythonPath: z.string().nullable().default(null).describe('Path to the detected Python executable'),
11
+ /** Root directory of the Python project */
12
+ projectDir: z.string().nullable().default(null).describe('Root directory of the Python project'),
13
+ /** Detected Python environment type */
14
+ environmentType: z.enum(['uv', 'conda', 'venv', 'system']).nullable().default(null).describe('Detected Python environment type (uv, conda, venv, or system)'),
15
+ /** Whether the Python environment is ready for execution */
16
+ isReady: z.boolean().default(false).describe('Whether the Python environment is ready for execution'),
17
+ /** Path to the last executed Python script */
18
+ lastExecutedScript: z.string().nullable().default(null).describe('Path to the last executed Python script'),
19
+ /** Whether a persistent Python session is currently active */
20
+ sessionActive: z.boolean().default(false).describe('Whether a persistent Python session is currently active'),
21
+ /** Unique ID of the current persistent session */
22
+ sessionId: z.string().nullable().default(null).describe('Unique ID of the current persistent session'),
23
+ })
24
+
25
+ export const PythonOptionsSchema = FeatureOptionsSchema.extend({
26
+ /** Directory containing the Python project */
27
+ dir: z.string().optional().describe('Directory containing the Python project'),
28
+ /** Custom install command to override auto-detection */
29
+ installCommand: z.string().optional().describe('Custom install command to override auto-detection'),
30
+ /** Path to Python script that will populate locals/context */
31
+ contextScript: z.string().optional().describe('Path to Python script that will populate locals/context'),
32
+ /** Specific Python executable to use */
33
+ pythonPath: z.string().optional().describe('Specific Python executable path to use'),
34
+ })
35
+
36
+ export type PythonState = z.infer<typeof PythonStateSchema>
37
+ export type PythonOptions = z.infer<typeof PythonOptionsSchema>
38
+
39
+ export const PythonEventsSchema = FeatureEventsSchema.extend({
40
+ ready: z.tuple([]).describe('When the Python environment is ready for execution'),
41
+ environmentDetected: z.tuple([z.object({
42
+ pythonPath: z.string().nullable().describe('Path to the detected Python executable'),
43
+ environmentType: z.enum(['uv', 'conda', 'venv', 'system']).nullable().describe('Detected environment type'),
44
+ }).describe('Environment detection result')]).describe('When the Python environment type is detected'),
45
+ installingDependencies: z.tuple([z.object({
46
+ command: z.string().describe('The install command being run'),
47
+ }).describe('Install details')]).describe('When dependency installation begins'),
48
+ dependenciesInstalled: z.tuple([z.object({
49
+ stdout: z.string().describe('Standard output from install'),
50
+ stderr: z.string().describe('Standard error from install'),
51
+ exitCode: z.number().describe('Process exit code'),
52
+ }).describe('Install result')]).describe('When dependencies are successfully installed'),
53
+ dependencyInstallFailed: z.tuple([z.object({
54
+ stdout: z.string().describe('Standard output from install'),
55
+ stderr: z.string().describe('Standard error from install'),
56
+ exitCode: z.number().describe('Process exit code'),
57
+ }).describe('Install result')]).describe('When dependency installation fails'),
58
+ codeExecuted: z.tuple([z.object({
59
+ code: z.string().describe('The Python code that was executed'),
60
+ variables: z.record(z.string(), z.any()).describe('Variables passed to the execution'),
61
+ result: z.object({
62
+ stdout: z.string().describe('Standard output'),
63
+ stderr: z.string().describe('Standard error'),
64
+ exitCode: z.number().describe('Process exit code'),
65
+ }).describe('Execution result'),
66
+ }).describe('Code execution details')]).describe('When Python code finishes executing'),
67
+ fileExecuted: z.tuple([z.object({
68
+ filePath: z.string().describe('Path to the executed Python file'),
69
+ variables: z.record(z.string(), z.any()).describe('Variables passed as arguments'),
70
+ result: z.object({
71
+ stdout: z.string().describe('Standard output'),
72
+ stderr: z.string().describe('Standard error'),
73
+ exitCode: z.number().describe('Process exit code'),
74
+ }).describe('Execution result'),
75
+ }).describe('File execution details')]).describe('When a Python file finishes executing'),
76
+ localsParseError: z.tuple([z.any().describe('The parse error')]).describe('When captured locals fail to parse as JSON'),
77
+ sessionStarted: z.tuple([z.object({
78
+ sessionId: z.string().describe('Unique session identifier'),
79
+ }).describe('Session start details')]).describe('When a persistent Python session starts'),
80
+ sessionStopped: z.tuple([z.object({
81
+ sessionId: z.string().describe('Session identifier that stopped'),
82
+ }).describe('Session stop details')]).describe('When a persistent Python session stops'),
83
+ sessionError: z.tuple([z.object({
84
+ error: z.string().describe('Error message'),
85
+ sessionId: z.string().nullable().describe('Session identifier, if available'),
86
+ }).describe('Session error details')]).describe('When a session-level error occurs'),
87
+ }).describe('Python events')
88
+
89
+ /** Result from a persistent session run() call. */
90
+ export interface RunResult {
91
+ ok: boolean
92
+ result: any
93
+ stdout: string
94
+ error?: string
95
+ traceback?: string
96
+ }
97
+
98
+ /**
99
+ * The Python VM feature provides Python virtual machine capabilities for executing Python code.
100
+ *
101
+ * This feature automatically detects Python environments (uv, conda, venv, system) and provides
102
+ * methods to install dependencies and execute Python scripts. It can manage project-specific
103
+ * Python environments and maintain context between executions.
104
+ *
105
+ * Supports two modes:
106
+ * - **Stateless** (default): `execute()` and `executeFile()` spawn a fresh process per call
107
+ * - **Persistent session**: `startSession()` spawns a long-lived bridge process that maintains
108
+ * state across `run()` calls, enabling real codebase interaction with imports and session variables
109
+ *
110
+ * @example
111
+ * ```typescript
112
+ * const python = container.feature('python', {
113
+ * dir: "/path/to/python/project",
114
+ * })
115
+ *
116
+ * // Stateless execution
117
+ * const result = await python.execute('print("Hello from Python!")')
118
+ *
119
+ * // Persistent session
120
+ * await python.startSession()
121
+ * await python.run('import myapp.models')
122
+ * await python.run('users = myapp.models.User.objects.all()')
123
+ * const result = await python.run('print(len(users))')
124
+ * await python.stopSession()
125
+ * ```
126
+ *
127
+ * @extends Feature
128
+ */
129
+ export class Python<
130
+ T extends PythonState = PythonState,
131
+ K extends PythonOptions = PythonOptions
132
+ > extends Feature<T, K> {
133
+ static override shortcut = "features.python" as const
134
+ static override stateSchema = PythonStateSchema
135
+ static override optionsSchema = PythonOptionsSchema
136
+ static override eventsSchema = PythonEventsSchema
137
+ static { Feature.register(this, 'python') }
138
+
139
+ private _bridgeProcess: ChildProcess | null = null
140
+ private _bridgeScriptPath: string | null = null
141
+ private _pendingRequests = new Map<string, { resolve: (v: any) => void, reject: (e: any) => void }>()
142
+ private _stdoutBuffer = ''
143
+
144
+ override get initialState(): T {
145
+ return {
146
+ ...super.initialState,
147
+ pythonPath: null,
148
+ projectDir: null,
149
+ environmentType: null,
150
+ isReady: false,
151
+ lastExecutedScript: null,
152
+ sessionActive: false,
153
+ sessionId: null,
154
+ } as T
155
+ }
156
+
157
+ override async enable(options: any = {}): Promise<this> {
158
+ await super.enable(options)
159
+
160
+ // Setup project directory
161
+ if (this.options.dir) {
162
+ this.state.set('projectDir', this.container.paths.resolve(this.options.dir))
163
+ } else {
164
+ this.state.set('projectDir', this.container.cwd)
165
+ }
166
+
167
+ // Detect Python environment
168
+ await this.detectEnvironment()
169
+
170
+ // Execute context script if provided
171
+ if (this.options.contextScript && this.container.feature('fs').exists(this.options.contextScript)) {
172
+ await this.execute(`exec(open('${this.options.contextScript}').read())`)
173
+ }
174
+
175
+ this.state.set('isReady', true)
176
+ this.emit('ready')
177
+
178
+ return this
179
+ }
180
+
181
+ /** Returns the root directory of the Python project. */
182
+ get projectDir() {
183
+ return this.state.get('projectDir') || this.container.cwd
184
+ }
185
+
186
+ /** Returns the path to the Python executable for this environment. */
187
+ get pythonPath() {
188
+ return this.state.get('pythonPath') || 'python'
189
+ }
190
+
191
+ /** Returns the detected environment type: 'uv', 'conda', 'venv', or 'system'. */
192
+ get environmentType() {
193
+ return this.state.get('environmentType') || 'system'
194
+ }
195
+
196
+ /**
197
+ * Detects the Python environment type and sets the appropriate Python path.
198
+ *
199
+ * This method checks for various Python environment managers in order of preference:
200
+ * uv, conda, venv, then falls back to system Python. It sets the pythonPath and
201
+ * environmentType in the state.
202
+ *
203
+ * @returns {Promise<void>}
204
+ *
205
+ * @example
206
+ * ```typescript
207
+ * await python.detectEnvironment()
208
+ * console.log(python.state.get('environmentType')) // 'uv' | 'conda' | 'venv' | 'system'
209
+ * console.log(python.state.get('pythonPath')) // '/path/to/python/executable'
210
+ * ```
211
+ */
212
+ async detectEnvironment(): Promise<void> {
213
+ const projectDir = this.state.get('projectDir')!
214
+ let pythonPath: string | null = null
215
+ let environmentType: PythonState['environmentType'] = null
216
+
217
+ const proc = this.container.feature('proc')
218
+ const fs = this.container.feature('fs')
219
+ const paths = this.container.paths
220
+
221
+ /** Resolve a binary to its full path via `which`, falling back to the bare name. */
222
+ const resolveBin = (name: string): string => {
223
+ try { return proc.exec(`which ${name}`).trim() } catch { return name }
224
+ }
225
+
226
+ // Use explicitly provided Python path
227
+ if (this.options.pythonPath) {
228
+ pythonPath = this.options.pythonPath
229
+ environmentType = 'system'
230
+ }
231
+
232
+ // Check for uv — independent so a missing uv binary falls through to conda/venv
233
+ if (!pythonPath && (fs.exists(paths.resolve(projectDir, 'uv.lock')) || fs.exists(paths.resolve(projectDir, 'pyproject.toml')))) {
234
+ try {
235
+ const uvBin = resolveBin('uv')
236
+ const result = await proc.execAndCapture(`${uvBin} run python --version`)
237
+ // execAndCapture returns exitCode 0 on ENOENT — check result.error to confirm the binary actually ran
238
+ if (result.exitCode === 0 && !result.error) {
239
+ pythonPath = `${uvBin} run python`
240
+ environmentType = 'uv'
241
+ }
242
+ } catch (error) {
243
+ // Fall through to next detection method
244
+ }
245
+ }
246
+
247
+ // Check for conda
248
+ if (!pythonPath && (fs.exists(paths.resolve(projectDir, 'environment.yml')) || fs.exists(paths.resolve(projectDir, 'conda.yml')))) {
249
+ try {
250
+ const condaBin = resolveBin('conda')
251
+ const result = await proc.execAndCapture(`${condaBin} run python --version`)
252
+ if (result.exitCode === 0 && !result.error) {
253
+ pythonPath = `${condaBin} run python`
254
+ environmentType = 'conda'
255
+ }
256
+ } catch (error) {
257
+ // Fall through to next detection method
258
+ }
259
+ }
260
+
261
+ // Check for venv
262
+ if (!pythonPath && (fs.exists(paths.resolve(projectDir, 'venv')) || fs.exists(paths.resolve(projectDir, '.venv')))) {
263
+ const venvPath = fs.exists(paths.resolve(projectDir, 'venv')) ? 'venv' : '.venv'
264
+ const venvPython = process.platform === 'win32'
265
+ ? paths.resolve(projectDir, venvPath, 'Scripts', 'python.exe')
266
+ : paths.resolve(projectDir, venvPath, 'bin', 'python')
267
+
268
+ if (fs.exists(venvPython)) {
269
+ pythonPath = venvPython
270
+ environmentType = 'venv'
271
+ }
272
+ }
273
+
274
+ // Fall back to system Python
275
+ if (!pythonPath) {
276
+ try {
277
+ const python3Bin = resolveBin('python3')
278
+ const result = await proc.execAndCapture(`${python3Bin} --version`)
279
+ if (result.exitCode === 0) {
280
+ pythonPath = python3Bin
281
+ environmentType = 'system'
282
+ } else {
283
+ const pythonBin = resolveBin('python')
284
+ const result2 = await proc.execAndCapture(`${pythonBin} --version`)
285
+ if (result2.exitCode === 0) {
286
+ pythonPath = pythonBin
287
+ environmentType = 'system'
288
+ }
289
+ }
290
+ } catch (error) {
291
+ throw new Error('Could not find Python installation')
292
+ }
293
+ }
294
+
295
+ this.state.set('pythonPath', pythonPath)
296
+ this.state.set('environmentType', environmentType)
297
+
298
+ this.emit('environmentDetected', { pythonPath, environmentType })
299
+ }
300
+
301
+ /**
302
+ * Installs dependencies for the Python project.
303
+ *
304
+ * This method automatically detects the appropriate package manager and install command
305
+ * based on the environment type. If a custom installCommand is provided in options,
306
+ * it will use that instead.
307
+ *
308
+ * @returns {Promise<{ stdout: string; stderr: string; exitCode: number }>}
309
+ *
310
+ * @example
311
+ * ```typescript
312
+ * // Auto-detect and install
313
+ * const result = await python.installDependencies()
314
+ *
315
+ * // With custom install command
316
+ * const python = container.feature('python', {
317
+ * installCommand: 'pip install -r requirements.txt'
318
+ * })
319
+ * const result = await python.installDependencies()
320
+ * ```
321
+ */
322
+ async installDependencies(): Promise<{ stdout: string; stderr: string; exitCode: number }> {
323
+ const proc = this.container.feature('proc')
324
+ const fs = this.container.feature('fs')
325
+ const paths = this.container.paths
326
+ const projectDir = this.state.get('projectDir')!
327
+ const environmentType = this.state.get('environmentType')
328
+
329
+ let installCommand: string
330
+
331
+ if (this.options.installCommand) {
332
+ installCommand = this.options.installCommand
333
+ } else {
334
+ switch (environmentType) {
335
+ case 'uv':
336
+ installCommand = 'uv sync'
337
+ break
338
+ case 'conda':
339
+ if (fs.exists(paths.resolve(projectDir, 'environment.yml'))) {
340
+ installCommand = 'conda env update -f environment.yml'
341
+ } else if (fs.exists(paths.resolve(projectDir, 'conda.yml'))) {
342
+ installCommand = 'conda env update -f conda.yml'
343
+ } else {
344
+ installCommand = 'conda install --file requirements.txt'
345
+ }
346
+ break
347
+ case 'venv':
348
+ case 'system':
349
+ default:
350
+ if (fs.exists(paths.resolve(projectDir, 'requirements.txt'))) {
351
+ const pythonPath = this.state.get('pythonPath')!
352
+ installCommand = `${pythonPath} -m pip install -r requirements.txt`
353
+ } else if (fs.exists(paths.resolve(projectDir, 'pyproject.toml'))) {
354
+ const pythonPath = this.state.get('pythonPath')!
355
+ installCommand = `${pythonPath} -m pip install -e .`
356
+ } else {
357
+ throw new Error('No requirements.txt or pyproject.toml found for dependency installation')
358
+ }
359
+ break
360
+ }
361
+ }
362
+
363
+ this.emit('installingDependencies', { command: installCommand })
364
+
365
+ const result = await proc.execAndCapture(installCommand, { cwd: projectDir })
366
+
367
+ if (result.exitCode === 0) {
368
+ this.emit('dependenciesInstalled', result)
369
+ } else {
370
+ this.emit('dependencyInstallFailed', result)
371
+ }
372
+
373
+ return result
374
+ }
375
+
376
+ /**
377
+ * Executes Python code and returns the result.
378
+ *
379
+ * This method creates a temporary Python script with the provided code and variables,
380
+ * executes it using the detected Python environment, and captures the output.
381
+ *
382
+ * @param {string} code - The Python code to execute
383
+ * @param {Record<string, any>} [variables={}] - Variables to make available to the Python code
384
+ * @param {object} [options] - Execution options
385
+ * @param {boolean} [options.captureLocals=false] - Whether to capture and return local variables after execution
386
+ * @returns {Promise<{ stdout: string; stderr: string; exitCode: number; locals?: any }>}
387
+ *
388
+ * @example
389
+ * ```typescript
390
+ * // Simple execution
391
+ * const result = await python.execute('print("Hello World")')
392
+ * console.log(result.stdout) // 'Hello World'
393
+ *
394
+ * // With variables
395
+ * const result = await python.execute('print(f"Hello {name}!")', { name: 'Alice' })
396
+ *
397
+ * // Capture locals
398
+ * const result = await python.execute('x = 42\ny = x * 2', {}, { captureLocals: true })
399
+ * console.log(result.locals) // { x: 42, y: 84 }
400
+ * ```
401
+ */
402
+ async execute(
403
+ code: string,
404
+ variables: Record<string, any> = {},
405
+ options: { captureLocals?: boolean } = {}
406
+ ): Promise<{ stdout: string; stderr: string; exitCode: number; locals?: any }> {
407
+ const proc = this.container.feature('proc')
408
+ const fs = this.container.feature('fs')
409
+
410
+ const { projectDir, pythonPath } = this
411
+
412
+ // Create temporary script in system temp dir (not inside the project)
413
+ const tempDir = `${tmpdir()}/luca-python-temp`
414
+ await fs.ensureFolder(tempDir)
415
+ const scriptPath = this.container.paths.resolve(tempDir, `script-${Date.now()}.py`)
416
+
417
+ // Build the Python script
418
+ let script = ''
419
+
420
+ // Import json for serialization
421
+ script += 'import json\nimport sys\n\n'
422
+
423
+ // Set up variables
424
+ if (Object.keys(variables).length > 0) {
425
+ script += '# Variables\n'
426
+ for (const [key, value] of Object.entries(variables)) {
427
+ script += `${key} = ${JSON.stringify(value)}\n`
428
+ }
429
+ script += '\n'
430
+ }
431
+
432
+ // Add the user code
433
+ script += '# User code\n'
434
+ script += code
435
+
436
+ // Capture locals if requested
437
+ if (options.captureLocals) {
438
+ script += '\n\n# Capture locals\n'
439
+ script += 'locals_dict = {k: v for k, v in locals().items() if not k.startswith("__")}\n'
440
+ script += 'print("__LOCALS__:" + json.dumps(locals_dict, default=str))\n'
441
+ }
442
+
443
+ await fs.writeFileAsync(scriptPath, script)
444
+
445
+ // Execute the script
446
+ const command = pythonPath.includes(' ') ? pythonPath : `${pythonPath}`
447
+ const result = await proc.execAndCapture(`${command} ${scriptPath}`, { cwd: projectDir })
448
+
449
+ // Parse locals if captured
450
+ let locals: any = undefined
451
+ if (options.captureLocals && result.stdout.includes('__LOCALS__:')) {
452
+ try {
453
+ const localsMatch = result.stdout.match(/__LOCALS__:(.+)$/m)
454
+ if (localsMatch && localsMatch[1]) {
455
+ locals = JSON.parse(localsMatch[1])
456
+ // Remove the locals output from stdout
457
+ result.stdout = result.stdout.replace(/__LOCALS__:.+$/m, '').trim()
458
+ }
459
+ } catch (error) {
460
+ this.emit('localsParseError', error)
461
+ }
462
+ }
463
+
464
+ // Clean up temporary file
465
+ await fs.rm(scriptPath)
466
+
467
+ this.state.set('lastExecutedScript', scriptPath)
468
+ this.emit('codeExecuted', { code, variables, result })
469
+
470
+ return { ...result, locals }
471
+ }
472
+
473
+ /**
474
+ * Executes a Python file and returns the result.
475
+ *
476
+ * @param {string} filePath - Path to the Python file to execute
477
+ * @param {Record<string, any>} [variables={}] - Variables to make available via command line arguments
478
+ * @returns {Promise<{ stdout: string; stderr: string; exitCode: number }>}
479
+ *
480
+ * @example
481
+ * ```typescript
482
+ * const result = await python.executeFile('/path/to/script.py')
483
+ * console.log(result.stdout)
484
+ * ```
485
+ */
486
+ async executeFile(
487
+ filePath: string,
488
+ variables: Record<string, any> = {}
489
+ ): Promise<{ stdout: string; stderr: string; exitCode: number }> {
490
+ const proc = this.container.feature('proc')
491
+ const projectDir = this.state.get('projectDir')!
492
+ const pythonPath = this.state.get('pythonPath')!
493
+
494
+ // Convert variables to command line arguments
495
+ const args = Object.entries(variables).map(([key, value]) => `--${key}=${value}`).join(' ')
496
+ const command = pythonPath.includes(' ') ? pythonPath : `${pythonPath}`
497
+
498
+ const result = await proc.execAndCapture(`${command} ${filePath} ${args}`, { cwd: projectDir })
499
+
500
+ this.emit('fileExecuted', { filePath, variables, result })
501
+
502
+ return result
503
+ }
504
+
505
+ /**
506
+ * Gets information about the current Python environment.
507
+ *
508
+ * @returns {Promise<{ version: string; path: string; packages: string[] }>}
509
+ */
510
+ async getEnvironmentInfo(): Promise<{ version: string; path: string; packages: string[] }> {
511
+ const proc = this.container.feature('proc')
512
+ const pythonPath = this.state.get('pythonPath')!
513
+ const projectDir = this.state.get('projectDir')!
514
+
515
+ // Get Python version
516
+ const versionResult = await proc.execAndCapture(`${pythonPath} --version`, { cwd: projectDir })
517
+ const version = versionResult.stdout.trim()
518
+
519
+ // Get Python path
520
+ const pathResult = await proc.execAndCapture(`${pythonPath} -c "import sys; print(sys.executable)"`, { cwd: projectDir })
521
+ const path = pathResult.stdout.trim()
522
+
523
+ // Get installed packages
524
+ const packagesResult = await proc.execAndCapture(`${pythonPath} -m pip list --format=freeze`, { cwd: projectDir })
525
+ const packages = packagesResult.stdout.trim().split('\n').filter(line => line.length > 0)
526
+
527
+ return { version, path, packages }
528
+ }
529
+
530
+ // ---------------------------------------------------------------------------
531
+ // Persistent session methods
532
+ // ---------------------------------------------------------------------------
533
+
534
+ /**
535
+ * Splits the (possibly multi-word) pythonPath into a command and args array
536
+ * suitable for proc.spawn(). For example, `uv run python` becomes
537
+ * `{ command: 'uv', args: ['run', 'python', ...extraArgs] }`.
538
+ */
539
+ private _parsePythonCommand(extraArgs: string[]): { command: string, args: string[] } {
540
+ const parts = this.pythonPath.split(/\s+/)
541
+ return { command: parts[0] ?? 'python', args: [...parts.slice(1), ...extraArgs] }
542
+ }
543
+
544
+ /**
545
+ * Writes the bundled bridge.py to a temp directory and returns its path.
546
+ * Reuses the same path across calls within a process.
547
+ */
548
+ private async _ensureBridgeScript(): Promise<string> {
549
+ if (this._bridgeScriptPath) return this._bridgeScriptPath
550
+
551
+ const fs = this.container.feature('fs')
552
+ const bridgeDir = `${tmpdir()}/luca-python-bridge`
553
+ await fs.ensureFolder(bridgeDir)
554
+ const scriptPath = `${bridgeDir}/bridge.py`
555
+ await fs.writeFileAsync(scriptPath, bridgeScript)
556
+ this._bridgeScriptPath = scriptPath
557
+ return scriptPath
558
+ }
559
+
560
+ /**
561
+ * Sends a JSON-line request to the bridge process and returns a promise
562
+ * that resolves when the matching response (by id) arrives.
563
+ *
564
+ * @param type - The request type (exec, eval, import, call, get_locals, reset)
565
+ * @param payload - Additional fields to include in the request
566
+ * @param timeout - Timeout in ms (default 30000)
567
+ */
568
+ private _sendRequest(type: string, payload: Record<string, any> = {}, timeout = 30000): Promise<any> {
569
+ if (!this._bridgeProcess || !this._bridgeProcess.stdin) {
570
+ return Promise.reject(new Error('No active Python session. Call startSession() first.'))
571
+ }
572
+
573
+ const id = this.container.utils.uuid()
574
+ const request = JSON.stringify({ id, type, ...payload }) + '\n'
575
+
576
+ return new Promise((resolve, reject) => {
577
+ const timer = setTimeout(() => {
578
+ this._pendingRequests.delete(id)
579
+ reject(new Error(`Python bridge request timed out after ${timeout}ms (type: ${type})`))
580
+ }, timeout)
581
+
582
+ this._pendingRequests.set(id, {
583
+ resolve: (value: any) => {
584
+ clearTimeout(timer)
585
+ resolve(value)
586
+ },
587
+ reject: (err: any) => {
588
+ clearTimeout(timer)
589
+ reject(err)
590
+ },
591
+ })
592
+
593
+ this._bridgeProcess!.stdin!.write(request)
594
+ })
595
+ }
596
+
597
+ /**
598
+ * Handles incoming stdout data from the bridge process. Buffers partial
599
+ * lines and parses complete JSON-line responses, resolving their matching
600
+ * pending requests.
601
+ */
602
+ private _onBridgeData(chunk: Buffer | string): void {
603
+ this._stdoutBuffer += chunk.toString()
604
+
605
+ const lines = this._stdoutBuffer.split('\n')
606
+ // Keep the last (possibly incomplete) segment in the buffer
607
+ this._stdoutBuffer = lines.pop() || ''
608
+
609
+ for (const line of lines) {
610
+ if (!line.trim()) continue
611
+ try {
612
+ const response = JSON.parse(line)
613
+ const id = response.id
614
+ if (id && this._pendingRequests.has(id)) {
615
+ const pending = this._pendingRequests.get(id)!
616
+ this._pendingRequests.delete(id)
617
+ pending.resolve(response)
618
+ }
619
+ // Non-id responses (like the initial "ready") are handled by startSession
620
+ } catch {
621
+ // Not JSON — could be stray output, ignore
622
+ }
623
+ }
624
+ }
625
+
626
+ /**
627
+ * Starts a persistent Python session by spawning the bridge process.
628
+ *
629
+ * The bridge sets up sys.path for the project directory, then enters a
630
+ * JSON-line REPL loop. State (variables, imports) persists across run() calls
631
+ * until stopSession() or resetSession() is called.
632
+ *
633
+ * @example
634
+ * ```typescript
635
+ * const python = container.feature('python', { dir: '/path/to/project' })
636
+ * await python.enable()
637
+ * await python.startSession()
638
+ * await python.run('x = 42')
639
+ * const result = await python.run('print(x)')
640
+ * console.log(result.stdout) // '42\n'
641
+ * await python.stopSession()
642
+ * ```
643
+ */
644
+ async startSession(): Promise<void> {
645
+ if (this.state.get('sessionActive')) {
646
+ throw new Error('A Python session is already active. Call stopSession() first.')
647
+ }
648
+
649
+ const proc = this.container.feature('proc')
650
+ const bridgePath = await this._ensureBridgeScript()
651
+ const { command, args } = this._parsePythonCommand(['-u', bridgePath])
652
+
653
+ const child = proc.spawn(command, args, {
654
+ cwd: this.projectDir,
655
+ stdout: 'pipe',
656
+ stderr: 'pipe',
657
+ })
658
+
659
+ this._bridgeProcess = child
660
+ this._stdoutBuffer = ''
661
+
662
+ // Wait for the ready signal from the bridge
663
+ const readyPromise = new Promise<void>((resolve, reject) => {
664
+ const timer = setTimeout(() => {
665
+ reject(new Error('Python bridge failed to start within 15 seconds'))
666
+ }, 15000)
667
+
668
+ const onData = (chunk: Buffer | string) => {
669
+ this._stdoutBuffer += chunk.toString()
670
+ const lines = this._stdoutBuffer.split('\n')
671
+ this._stdoutBuffer = lines.pop() || ''
672
+
673
+ for (const line of lines) {
674
+ if (!line.trim()) continue
675
+ try {
676
+ const msg = JSON.parse(line)
677
+ if (msg.type === 'ready' && msg.ok) {
678
+ clearTimeout(timer)
679
+ // Switch to the normal data handler
680
+ child.stdout!.removeListener('data', onData)
681
+ child.stdout!.on('data', this._onBridgeData.bind(this))
682
+ resolve()
683
+ return
684
+ }
685
+ } catch {
686
+ // ignore non-JSON during init
687
+ }
688
+ }
689
+ }
690
+
691
+ child.stdout!.on('data', onData)
692
+
693
+ child.on('error', (err: Error) => {
694
+ clearTimeout(timer)
695
+ reject(new Error(`Python bridge process error: ${err.message}`))
696
+ })
697
+
698
+ child.on('exit', (code: number | null) => {
699
+ clearTimeout(timer)
700
+ reject(new Error(`Python bridge exited during startup with code ${code}`))
701
+ })
702
+ })
703
+
704
+ // Send init handshake with project directory
705
+ child.stdin!.write(JSON.stringify({ project_dir: this.projectDir }) + '\n')
706
+
707
+ await readyPromise
708
+
709
+ // Register crash handler (after successful startup)
710
+ child.removeAllListeners('exit')
711
+ child.on('exit', (code: number | null) => {
712
+ const sessionId = this.state.get('sessionId')
713
+ this.state.set('sessionActive', false)
714
+ this._bridgeProcess = null
715
+
716
+ // Reject all pending requests
717
+ for (const [id, pending] of this._pendingRequests) {
718
+ pending.reject(new Error(`Python bridge exited unexpectedly with code ${code}`))
719
+ }
720
+ this._pendingRequests.clear()
721
+
722
+ this.emit('sessionError', { error: `Bridge exited with code ${code}`, sessionId })
723
+ })
724
+
725
+ // Capture stderr for diagnostics (don't interfere with protocol)
726
+ child.stderr!.on('data', (chunk: Buffer | string) => {
727
+ const text = chunk.toString().trim()
728
+ if (text) {
729
+ this.emit('sessionError', { error: text, sessionId: this.state.get('sessionId') })
730
+ }
731
+ })
732
+
733
+ const sessionId = this.container.utils.uuid()
734
+ this.state.set('sessionActive', true)
735
+ this.state.set('sessionId', sessionId)
736
+ this.emit('sessionStarted', { sessionId })
737
+ }
738
+
739
+ /**
740
+ * Stops the persistent Python session and cleans up the bridge process.
741
+ *
742
+ * @example
743
+ * ```typescript
744
+ * await python.stopSession()
745
+ * ```
746
+ */
747
+ async stopSession(): Promise<void> {
748
+ const sessionId = this.state.get('sessionId')
749
+
750
+ if (this._bridgeProcess) {
751
+ this._bridgeProcess.removeAllListeners('exit')
752
+ this._bridgeProcess.kill('SIGTERM')
753
+ this._bridgeProcess = null
754
+ }
755
+
756
+ // Reject any pending requests
757
+ for (const [id, pending] of this._pendingRequests) {
758
+ pending.reject(new Error('Python session stopped'))
759
+ }
760
+ this._pendingRequests.clear()
761
+ this._stdoutBuffer = ''
762
+
763
+ this.state.set('sessionActive', false)
764
+ this.state.set('sessionId', null)
765
+
766
+ if (sessionId) {
767
+ this.emit('sessionStopped', { sessionId })
768
+ }
769
+ }
770
+
771
+ /**
772
+ * Executes Python code in the persistent session. Variables and imports
773
+ * survive across calls. This is the session equivalent of execute().
774
+ *
775
+ * @param code - Python code to execute
776
+ * @param variables - Variables to inject into the namespace before execution
777
+ * @returns The execution result including captured stdout and any error info
778
+ *
779
+ * @example
780
+ * ```typescript
781
+ * await python.startSession()
782
+ *
783
+ * // State persists across calls
784
+ * await python.run('x = 42')
785
+ * const result = await python.run('print(x * 2)')
786
+ * console.log(result.stdout) // '84\n'
787
+ *
788
+ * // Inject variables from JS
789
+ * const result2 = await python.run('print(f"Hello {name}!")', { name: 'World' })
790
+ * console.log(result2.stdout) // 'Hello World!\n'
791
+ * ```
792
+ */
793
+ async run(code: string, variables: Record<string, any> = {}): Promise<RunResult> {
794
+ const response = await this._sendRequest('exec', { code, variables })
795
+ return {
796
+ ok: response.ok,
797
+ result: response.result ?? null,
798
+ stdout: response.stdout ?? '',
799
+ error: response.error,
800
+ traceback: response.traceback,
801
+ }
802
+ }
803
+
804
+ /**
805
+ * Evaluates a Python expression in the persistent session and returns its value.
806
+ *
807
+ * @param expression - Python expression to evaluate
808
+ * @returns The evaluated result (JSON-serializable, or repr() string for complex types)
809
+ *
810
+ * @example
811
+ * ```typescript
812
+ * await python.run('x = 42')
813
+ * const result = await python.eval('x * 2')
814
+ * console.log(result) // 84
815
+ * ```
816
+ */
817
+ async eval(expression: string): Promise<any> {
818
+ const response = await this._sendRequest('eval', { expression })
819
+ if (!response.ok) {
820
+ throw new Error(response.error || 'eval failed')
821
+ }
822
+ return response.result
823
+ }
824
+
825
+ /**
826
+ * Imports a Python module into the persistent session namespace.
827
+ *
828
+ * @param moduleName - Dotted module path (e.g. 'myapp.models')
829
+ * @param alias - Optional alias for the import (defaults to the last segment)
830
+ *
831
+ * @example
832
+ * ```typescript
833
+ * await python.importModule('json')
834
+ * await python.importModule('myapp.models', 'models')
835
+ * const result = await python.eval('models.User')
836
+ * ```
837
+ */
838
+ async importModule(moduleName: string, alias?: string): Promise<void> {
839
+ const response = await this._sendRequest('import', { module: moduleName, alias })
840
+ if (!response.ok) {
841
+ throw new Error(response.error || `Failed to import ${moduleName}`)
842
+ }
843
+ }
844
+
845
+ /**
846
+ * Calls a function by dotted path in the persistent session namespace.
847
+ *
848
+ * @param funcPath - Dotted path to the function (e.g. 'json.dumps' or 'my_func')
849
+ * @param args - Positional arguments
850
+ * @param kwargs - Keyword arguments
851
+ * @returns The function's return value
852
+ *
853
+ * @example
854
+ * ```typescript
855
+ * await python.importModule('json')
856
+ * const result = await python.call('json.dumps', [{ a: 1 }], { indent: 2 })
857
+ * ```
858
+ */
859
+ async call(funcPath: string, args: any[] = [], kwargs: Record<string, any> = {}): Promise<any> {
860
+ const response = await this._sendRequest('call', { function: funcPath, args, kwargs })
861
+ if (!response.ok) {
862
+ throw new Error(response.error || `Failed to call ${funcPath}`)
863
+ }
864
+ return response.result
865
+ }
866
+
867
+ /**
868
+ * Returns all non-dunder variables from the persistent session namespace.
869
+ *
870
+ * @returns A record of variable names to their JSON-serializable values
871
+ *
872
+ * @example
873
+ * ```typescript
874
+ * await python.run('x = 42\ny = "hello"')
875
+ * const locals = await python.getLocals()
876
+ * console.log(locals) // { x: 42, y: 'hello' }
877
+ * ```
878
+ */
879
+ async getLocals(): Promise<Record<string, any>> {
880
+ const response = await this._sendRequest('get_locals')
881
+ if (!response.ok) {
882
+ throw new Error(response.error || 'Failed to get locals')
883
+ }
884
+ return response.result
885
+ }
886
+
887
+ /**
888
+ * Clears all variables and imports from the persistent session namespace.
889
+ * The session remains active — you can continue calling run() after reset.
890
+ *
891
+ * @example
892
+ * ```typescript
893
+ * await python.run('x = 42')
894
+ * await python.resetSession()
895
+ * // x is now undefined
896
+ * ```
897
+ */
898
+ async resetSession(): Promise<void> {
899
+ const response = await this._sendRequest('reset')
900
+ if (!response.ok) {
901
+ throw new Error(response.error || 'Failed to reset session')
902
+ }
903
+ }
904
+ }
905
+
906
+ export default Python
907
+ // Module augmentation for type safety
908
+ declare module '../feature.js' {
909
+ interface AvailableFeatures {
910
+ python: typeof Python;
911
+ }
912
+ }