@pixelbyte-software/pixcode 1.30.0

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 (336) hide show
  1. package/LICENSE +718 -0
  2. package/README.de.md +248 -0
  3. package/README.ja.md +240 -0
  4. package/README.ko.md +240 -0
  5. package/README.md +285 -0
  6. package/README.ru.md +248 -0
  7. package/README.tr.md +250 -0
  8. package/README.zh-CN.md +240 -0
  9. package/dist/api-docs.html +879 -0
  10. package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  11. package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  12. package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  13. package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  14. package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  15. package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  16. package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  17. package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  18. package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  19. package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  20. package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  21. package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  22. package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  23. package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  24. package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  25. package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  26. package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  27. package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  28. package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  29. package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  30. package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  31. package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  32. package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  33. package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  34. package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  35. package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  36. package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  37. package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  38. package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  39. package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  40. package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  41. package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  42. package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  43. package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  44. package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  45. package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  46. package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  47. package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  48. package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  49. package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  50. package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  51. package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  52. package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  53. package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  54. package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  55. package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  56. package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  57. package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  58. package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  59. package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  60. package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  61. package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  62. package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  63. package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  64. package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  65. package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  66. package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  67. package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  68. package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  69. package/dist/assets/index-C2c9QNwK.css +32 -0
  70. package/dist/assets/index-DyXDZED-.js +1277 -0
  71. package/dist/assets/vendor-codemirror-NA4v81it.js +41 -0
  72. package/dist/assets/vendor-react-D7WwDXvu.js +59 -0
  73. package/dist/assets/vendor-xterm-CJZjLICi.js +66 -0
  74. package/dist/clear-cache.html +85 -0
  75. package/dist/convert-icons.md +53 -0
  76. package/dist/favicon.png +0 -0
  77. package/dist/favicon.svg +9 -0
  78. package/dist/generate-icons.js +49 -0
  79. package/dist/icons/claude-ai-icon.svg +1 -0
  80. package/dist/icons/codex-white.svg +3 -0
  81. package/dist/icons/codex.svg +3 -0
  82. package/dist/icons/cursor-white.svg +12 -0
  83. package/dist/icons/cursor.svg +1 -0
  84. package/dist/icons/gemini-ai-icon.svg +1 -0
  85. package/dist/icons/icon-128x128.png +0 -0
  86. package/dist/icons/icon-128x128.svg +12 -0
  87. package/dist/icons/icon-144x144.png +0 -0
  88. package/dist/icons/icon-144x144.svg +12 -0
  89. package/dist/icons/icon-152x152.png +0 -0
  90. package/dist/icons/icon-152x152.svg +12 -0
  91. package/dist/icons/icon-192x192.png +0 -0
  92. package/dist/icons/icon-192x192.svg +12 -0
  93. package/dist/icons/icon-384x384.png +0 -0
  94. package/dist/icons/icon-384x384.svg +12 -0
  95. package/dist/icons/icon-512x512.png +0 -0
  96. package/dist/icons/icon-512x512.svg +12 -0
  97. package/dist/icons/icon-72x72.png +0 -0
  98. package/dist/icons/icon-72x72.svg +12 -0
  99. package/dist/icons/icon-96x96.png +0 -0
  100. package/dist/icons/icon-96x96.svg +12 -0
  101. package/dist/icons/icon-template.svg +12 -0
  102. package/dist/index.html +52 -0
  103. package/dist/logo-128.png +0 -0
  104. package/dist/logo-256.png +0 -0
  105. package/dist/logo-32.png +0 -0
  106. package/dist/logo-512.png +0 -0
  107. package/dist/logo-64.png +0 -0
  108. package/dist/logo.svg +17 -0
  109. package/dist/manifest.json +61 -0
  110. package/dist/screenshots/cli-selection.png +0 -0
  111. package/dist/screenshots/desktop-main.png +0 -0
  112. package/dist/screenshots/mobile-chat.png +0 -0
  113. package/dist/screenshots/tools-modal.png +0 -0
  114. package/dist/sw.js +124 -0
  115. package/dist-server/server/claude-sdk.js +709 -0
  116. package/dist-server/server/claude-sdk.js.map +1 -0
  117. package/dist-server/server/cli.js +854 -0
  118. package/dist-server/server/cli.js.map +1 -0
  119. package/dist-server/server/constants/config.js +6 -0
  120. package/dist-server/server/constants/config.js.map +1 -0
  121. package/dist-server/server/cursor-cli.js +277 -0
  122. package/dist-server/server/cursor-cli.js.map +1 -0
  123. package/dist-server/server/daemon/manager.js +486 -0
  124. package/dist-server/server/daemon/manager.js.map +1 -0
  125. package/dist-server/server/daemon-manager.js +823 -0
  126. package/dist-server/server/daemon-manager.js.map +1 -0
  127. package/dist-server/server/database/db.js +535 -0
  128. package/dist-server/server/database/db.js.map +1 -0
  129. package/dist-server/server/database/schema.js +97 -0
  130. package/dist-server/server/database/schema.js.map +1 -0
  131. package/dist-server/server/gemini-cli.js +408 -0
  132. package/dist-server/server/gemini-cli.js.map +1 -0
  133. package/dist-server/server/gemini-response-handler.js +72 -0
  134. package/dist-server/server/gemini-response-handler.js.map +1 -0
  135. package/dist-server/server/index.js +2325 -0
  136. package/dist-server/server/index.js.map +1 -0
  137. package/dist-server/server/load-env.js +32 -0
  138. package/dist-server/server/load-env.js.map +1 -0
  139. package/dist-server/server/middleware/auth.js +111 -0
  140. package/dist-server/server/middleware/auth.js.map +1 -0
  141. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +103 -0
  142. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -0
  143. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js +103 -0
  144. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js.map +1 -0
  145. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js +266 -0
  146. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js.map +1 -0
  147. package/dist-server/server/modules/providers/list/claude/claude.provider.js +13 -0
  148. package/dist-server/server/modules/providers/list/claude/claude.provider.js.map +1 -0
  149. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +84 -0
  150. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -0
  151. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js +107 -0
  152. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js.map +1 -0
  153. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js +282 -0
  154. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js.map +1 -0
  155. package/dist-server/server/modules/providers/list/codex/codex.provider.js +13 -0
  156. package/dist-server/server/modules/providers/list/codex/codex.provider.js.map +1 -0
  157. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +118 -0
  158. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -0
  159. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js +80 -0
  160. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js.map +1 -0
  161. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js +369 -0
  162. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js.map +1 -0
  163. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js +13 -0
  164. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js.map +1 -0
  165. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +131 -0
  166. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -0
  167. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js +82 -0
  168. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js.map +1 -0
  169. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js +207 -0
  170. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js.map +1 -0
  171. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js +13 -0
  172. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js.map +1 -0
  173. package/dist-server/server/modules/providers/provider.registry.js +31 -0
  174. package/dist-server/server/modules/providers/provider.registry.js.map +1 -0
  175. package/dist-server/server/modules/providers/provider.routes.js +159 -0
  176. package/dist-server/server/modules/providers/provider.routes.js.map +1 -0
  177. package/dist-server/server/modules/providers/services/mcp.service.js +69 -0
  178. package/dist-server/server/modules/providers/services/mcp.service.js.map +1 -0
  179. package/dist-server/server/modules/providers/services/provider-auth.service.js +25 -0
  180. package/dist-server/server/modules/providers/services/provider-auth.service.js.map +1 -0
  181. package/dist-server/server/modules/providers/services/sessions.service.js +29 -0
  182. package/dist-server/server/modules/providers/services/sessions.service.js.map +1 -0
  183. package/dist-server/server/modules/providers/shared/base/abstract.provider.js +14 -0
  184. package/dist-server/server/modules/providers/shared/base/abstract.provider.js.map +1 -0
  185. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js +102 -0
  186. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js.map +1 -0
  187. package/dist-server/server/modules/providers/tests/mcp.test.js +250 -0
  188. package/dist-server/server/modules/providers/tests/mcp.test.js.map +1 -0
  189. package/dist-server/server/openai-codex.js +373 -0
  190. package/dist-server/server/openai-codex.js.map +1 -0
  191. package/dist-server/server/projects.js +2492 -0
  192. package/dist-server/server/projects.js.map +1 -0
  193. package/dist-server/server/routes/agent.js +1147 -0
  194. package/dist-server/server/routes/agent.js.map +1 -0
  195. package/dist-server/server/routes/auth.js +117 -0
  196. package/dist-server/server/routes/auth.js.map +1 -0
  197. package/dist-server/server/routes/cli-auth.js +25 -0
  198. package/dist-server/server/routes/cli-auth.js.map +1 -0
  199. package/dist-server/server/routes/codex.js +18 -0
  200. package/dist-server/server/routes/codex.js.map +1 -0
  201. package/dist-server/server/routes/commands.js +487 -0
  202. package/dist-server/server/routes/commands.js.map +1 -0
  203. package/dist-server/server/routes/cursor.js +49 -0
  204. package/dist-server/server/routes/cursor.js.map +1 -0
  205. package/dist-server/server/routes/gemini.js +21 -0
  206. package/dist-server/server/routes/gemini.js.map +1 -0
  207. package/dist-server/server/routes/git.js +1259 -0
  208. package/dist-server/server/routes/git.js.map +1 -0
  209. package/dist-server/server/routes/mcp-utils.js +29 -0
  210. package/dist-server/server/routes/mcp-utils.js.map +1 -0
  211. package/dist-server/server/routes/messages.js +56 -0
  212. package/dist-server/server/routes/messages.js.map +1 -0
  213. package/dist-server/server/routes/plugins.js +266 -0
  214. package/dist-server/server/routes/plugins.js.map +1 -0
  215. package/dist-server/server/routes/projects.js +566 -0
  216. package/dist-server/server/routes/projects.js.map +1 -0
  217. package/dist-server/server/routes/settings.js +259 -0
  218. package/dist-server/server/routes/settings.js.map +1 -0
  219. package/dist-server/server/routes/taskmaster.js +1373 -0
  220. package/dist-server/server/routes/taskmaster.js.map +1 -0
  221. package/dist-server/server/routes/user.js +115 -0
  222. package/dist-server/server/routes/user.js.map +1 -0
  223. package/dist-server/server/services/notification-orchestrator.js +177 -0
  224. package/dist-server/server/services/notification-orchestrator.js.map +1 -0
  225. package/dist-server/server/services/vapid-keys.js +26 -0
  226. package/dist-server/server/services/vapid-keys.js.map +1 -0
  227. package/dist-server/server/sessionManager.js +194 -0
  228. package/dist-server/server/sessionManager.js.map +1 -0
  229. package/dist-server/server/shared/interfaces.js +2 -0
  230. package/dist-server/server/shared/interfaces.js.map +1 -0
  231. package/dist-server/server/shared/types.js +3 -0
  232. package/dist-server/server/shared/types.js.map +1 -0
  233. package/dist-server/server/shared/utils.js +150 -0
  234. package/dist-server/server/shared/utils.js.map +1 -0
  235. package/dist-server/server/utils/colors.js +20 -0
  236. package/dist-server/server/utils/colors.js.map +1 -0
  237. package/dist-server/server/utils/commandParser.js +255 -0
  238. package/dist-server/server/utils/commandParser.js.map +1 -0
  239. package/dist-server/server/utils/frontmatter.js +16 -0
  240. package/dist-server/server/utils/frontmatter.js.map +1 -0
  241. package/dist-server/server/utils/gitConfig.js +36 -0
  242. package/dist-server/server/utils/gitConfig.js.map +1 -0
  243. package/dist-server/server/utils/mcp-detector.js +134 -0
  244. package/dist-server/server/utils/mcp-detector.js.map +1 -0
  245. package/dist-server/server/utils/plugin-loader.js +413 -0
  246. package/dist-server/server/utils/plugin-loader.js.map +1 -0
  247. package/dist-server/server/utils/plugin-process-manager.js +163 -0
  248. package/dist-server/server/utils/plugin-process-manager.js.map +1 -0
  249. package/dist-server/server/utils/runtime-paths.js +30 -0
  250. package/dist-server/server/utils/runtime-paths.js.map +1 -0
  251. package/dist-server/server/utils/taskmaster-websocket.js +118 -0
  252. package/dist-server/server/utils/taskmaster-websocket.js.map +1 -0
  253. package/dist-server/server/utils/url-detection.js +58 -0
  254. package/dist-server/server/utils/url-detection.js.map +1 -0
  255. package/dist-server/server/vite-daemon.js +75 -0
  256. package/dist-server/server/vite-daemon.js.map +1 -0
  257. package/dist-server/shared/modelConstants.js +90 -0
  258. package/dist-server/shared/modelConstants.js.map +1 -0
  259. package/dist-server/shared/networkHosts.js +20 -0
  260. package/dist-server/shared/networkHosts.js.map +1 -0
  261. package/package.json +168 -0
  262. package/scripts/fix-node-pty.js +67 -0
  263. package/server/claude-sdk.js +834 -0
  264. package/server/cli.js +937 -0
  265. package/server/constants/config.js +5 -0
  266. package/server/cursor-cli.js +342 -0
  267. package/server/daemon/manager.js +564 -0
  268. package/server/daemon-manager.js +920 -0
  269. package/server/database/db.js +593 -0
  270. package/server/database/schema.js +102 -0
  271. package/server/gemini-cli.js +469 -0
  272. package/server/gemini-response-handler.js +79 -0
  273. package/server/index.js +2557 -0
  274. package/server/load-env.js +34 -0
  275. package/server/middleware/auth.js +132 -0
  276. package/server/modules/providers/list/claude/claude-auth.provider.ts +123 -0
  277. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -0
  278. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -0
  279. package/server/modules/providers/list/claude/claude.provider.ts +15 -0
  280. package/server/modules/providers/list/codex/codex-auth.provider.ts +100 -0
  281. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -0
  282. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -0
  283. package/server/modules/providers/list/codex/codex.provider.ts +15 -0
  284. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -0
  285. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -0
  286. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -0
  287. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -0
  288. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +151 -0
  289. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -0
  290. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -0
  291. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -0
  292. package/server/modules/providers/provider.registry.ts +36 -0
  293. package/server/modules/providers/provider.routes.ts +217 -0
  294. package/server/modules/providers/services/mcp.service.ts +94 -0
  295. package/server/modules/providers/services/provider-auth.service.ts +26 -0
  296. package/server/modules/providers/services/sessions.service.ts +45 -0
  297. package/server/modules/providers/shared/base/abstract.provider.ts +20 -0
  298. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -0
  299. package/server/modules/providers/tests/mcp.test.ts +293 -0
  300. package/server/openai-codex.js +426 -0
  301. package/server/projects.js +2792 -0
  302. package/server/routes/agent.js +1245 -0
  303. package/server/routes/auth.js +135 -0
  304. package/server/routes/cli-auth.js +27 -0
  305. package/server/routes/codex.js +19 -0
  306. package/server/routes/commands.js +554 -0
  307. package/server/routes/cursor.js +52 -0
  308. package/server/routes/gemini.js +24 -0
  309. package/server/routes/git.js +1488 -0
  310. package/server/routes/mcp-utils.js +31 -0
  311. package/server/routes/messages.js +61 -0
  312. package/server/routes/plugins.js +307 -0
  313. package/server/routes/projects.js +627 -0
  314. package/server/routes/settings.js +286 -0
  315. package/server/routes/taskmaster.js +1471 -0
  316. package/server/routes/user.js +123 -0
  317. package/server/services/notification-orchestrator.js +227 -0
  318. package/server/services/vapid-keys.js +35 -0
  319. package/server/sessionManager.js +226 -0
  320. package/server/shared/interfaces.ts +54 -0
  321. package/server/shared/types.ts +172 -0
  322. package/server/shared/utils.ts +193 -0
  323. package/server/tsconfig.json +36 -0
  324. package/server/utils/colors.js +21 -0
  325. package/server/utils/commandParser.js +303 -0
  326. package/server/utils/frontmatter.js +18 -0
  327. package/server/utils/gitConfig.js +34 -0
  328. package/server/utils/mcp-detector.js +147 -0
  329. package/server/utils/plugin-loader.js +457 -0
  330. package/server/utils/plugin-process-manager.js +184 -0
  331. package/server/utils/runtime-paths.js +37 -0
  332. package/server/utils/taskmaster-websocket.js +129 -0
  333. package/server/utils/url-detection.js +71 -0
  334. package/server/vite-daemon.js +78 -0
  335. package/shared/modelConstants.js +97 -0
  336. package/shared/networkHosts.js +22 -0
@@ -0,0 +1,709 @@
1
+ /**
2
+ * Claude SDK Integration
3
+ *
4
+ * This module provides SDK-based integration with Claude using the @anthropic-ai/claude-agent-sdk.
5
+ * It mirrors the interface of claude-cli.js but uses the SDK internally for better performance
6
+ * and maintainability.
7
+ *
8
+ * Key features:
9
+ * - Direct SDK integration without child processes
10
+ * - Session management with abort capability
11
+ * - Options mapping between CLI and SDK formats
12
+ * - WebSocket message streaming
13
+ */
14
+ import { query } from '@anthropic-ai/claude-agent-sdk';
15
+ import crypto from 'crypto';
16
+ import { promises as fs } from 'fs';
17
+ import path from 'path';
18
+ import os from 'os';
19
+ import { CLAUDE_MODELS } from '../shared/modelConstants.js';
20
+ import { createNotificationEvent, notifyRunFailed, notifyRunStopped, notifyUserIfEnabled } from './services/notification-orchestrator.js';
21
+ import { sessionsService } from './modules/providers/services/sessions.service.js';
22
+ import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
23
+ import { createNormalizedMessage } from './shared/utils.js';
24
+ const activeSessions = new Map();
25
+ const pendingToolApprovals = new Map();
26
+ const TOOL_APPROVAL_TIMEOUT_MS = parseInt(process.env.CLAUDE_TOOL_APPROVAL_TIMEOUT_MS, 10) || 55000;
27
+ const TOOLS_REQUIRING_INTERACTION = new Set(['AskUserQuestion', 'ExitPlanMode']);
28
+ function createRequestId() {
29
+ if (typeof crypto.randomUUID === 'function') {
30
+ return crypto.randomUUID();
31
+ }
32
+ return crypto.randomBytes(16).toString('hex');
33
+ }
34
+ function waitForToolApproval(requestId, options = {}) {
35
+ const { timeoutMs = TOOL_APPROVAL_TIMEOUT_MS, signal, onCancel, metadata } = options;
36
+ return new Promise(resolve => {
37
+ let settled = false;
38
+ const finalize = (decision) => {
39
+ if (settled)
40
+ return;
41
+ settled = true;
42
+ cleanup();
43
+ resolve(decision);
44
+ };
45
+ let timeout;
46
+ const cleanup = () => {
47
+ pendingToolApprovals.delete(requestId);
48
+ if (timeout)
49
+ clearTimeout(timeout);
50
+ if (signal && abortHandler) {
51
+ signal.removeEventListener('abort', abortHandler);
52
+ }
53
+ };
54
+ // timeoutMs 0 = wait indefinitely (interactive tools)
55
+ if (timeoutMs > 0) {
56
+ timeout = setTimeout(() => {
57
+ onCancel?.('timeout');
58
+ finalize(null);
59
+ }, timeoutMs);
60
+ }
61
+ const abortHandler = () => {
62
+ onCancel?.('cancelled');
63
+ finalize({ cancelled: true });
64
+ };
65
+ if (signal) {
66
+ if (signal.aborted) {
67
+ onCancel?.('cancelled');
68
+ finalize({ cancelled: true });
69
+ return;
70
+ }
71
+ signal.addEventListener('abort', abortHandler, { once: true });
72
+ }
73
+ const resolver = (decision) => {
74
+ finalize(decision);
75
+ };
76
+ // Attach metadata for getPendingApprovalsForSession lookup
77
+ if (metadata) {
78
+ Object.assign(resolver, metadata);
79
+ }
80
+ pendingToolApprovals.set(requestId, resolver);
81
+ });
82
+ }
83
+ function resolveToolApproval(requestId, decision) {
84
+ const resolver = pendingToolApprovals.get(requestId);
85
+ if (resolver) {
86
+ resolver(decision);
87
+ }
88
+ }
89
+ // Match stored permission entries against a tool + input combo.
90
+ // This only supports exact tool names and the Bash(command:*) shorthand
91
+ // used by the UI; it intentionally does not implement full glob semantics,
92
+ // introduced to stay consistent with the UI's "Allow rule" format.
93
+ function matchesToolPermission(entry, toolName, input) {
94
+ if (!entry || !toolName) {
95
+ return false;
96
+ }
97
+ if (entry === toolName) {
98
+ return true;
99
+ }
100
+ const bashMatch = entry.match(/^Bash\((.+):\*\)$/);
101
+ if (toolName === 'Bash' && bashMatch) {
102
+ const allowedPrefix = bashMatch[1];
103
+ let command = '';
104
+ if (typeof input === 'string') {
105
+ command = input.trim();
106
+ }
107
+ else if (input && typeof input === 'object' && typeof input.command === 'string') {
108
+ command = input.command.trim();
109
+ }
110
+ if (!command) {
111
+ return false;
112
+ }
113
+ return command.startsWith(allowedPrefix);
114
+ }
115
+ return false;
116
+ }
117
+ /**
118
+ * Maps CLI options to SDK-compatible options format
119
+ * @param {Object} options - CLI options
120
+ * @returns {Object} SDK-compatible options
121
+ */
122
+ function mapCliOptionsToSDK(options = {}) {
123
+ const { sessionId, cwd, toolsSettings, permissionMode } = options;
124
+ const sdkOptions = {};
125
+ // Forward all host env vars (e.g. ANTHROPIC_BASE_URL, ANTHROPIC_AUTH_TOKEN) to the subprocess.
126
+ // Since claude-agent-sdk 0.2.113+, options.env REPLACES process.env in the subprocess
127
+ // instead of overlaying it. Without spreading process.env here, users who rely on
128
+ // ANTHROPIC_BASE_URL, HTTP(S)_PROXY, etc. would silently lose those settings.
129
+ sdkOptions.env = { ...process.env };
130
+ // Use CLAUDE_CLI_PATH if explicitly set, otherwise fall back to the "claude" binary on PATH.
131
+ // The SDK (>=0.2.113) looks for a bundled native binary as an optional dependency by default;
132
+ // this fallback keeps users who installed via the official installer working even when
133
+ // `npm prune --production` removed those optional deps.
134
+ sdkOptions.pathToClaudeCodeExecutable = process.env.CLAUDE_CLI_PATH || 'claude';
135
+ // Map working directory
136
+ if (cwd) {
137
+ sdkOptions.cwd = cwd;
138
+ }
139
+ // Map permission mode
140
+ if (permissionMode && permissionMode !== 'default') {
141
+ sdkOptions.permissionMode = permissionMode;
142
+ }
143
+ // Map tool settings
144
+ const settings = toolsSettings || {
145
+ allowedTools: [],
146
+ disallowedTools: [],
147
+ skipPermissions: false
148
+ };
149
+ // Handle tool permissions
150
+ if (settings.skipPermissions && permissionMode !== 'plan') {
151
+ // When skipping permissions, use bypassPermissions mode
152
+ sdkOptions.permissionMode = 'bypassPermissions';
153
+ }
154
+ let allowedTools = [...(settings.allowedTools || [])];
155
+ // Add plan mode default tools
156
+ if (permissionMode === 'plan') {
157
+ const planModeTools = ['Read', 'Task', 'exit_plan_mode', 'TodoRead', 'TodoWrite', 'WebFetch', 'WebSearch'];
158
+ for (const tool of planModeTools) {
159
+ if (!allowedTools.includes(tool)) {
160
+ allowedTools.push(tool);
161
+ }
162
+ }
163
+ }
164
+ sdkOptions.allowedTools = allowedTools;
165
+ // Use the tools preset to make all default built-in tools available (including AskUserQuestion).
166
+ // This was introduced in SDK 0.1.57. Omitting this preserves existing behavior (all tools available),
167
+ // but being explicit ensures forward compatibility and clarity.
168
+ sdkOptions.tools = { type: 'preset', preset: 'claude_code' };
169
+ sdkOptions.disallowedTools = settings.disallowedTools || [];
170
+ // Map model (default to sonnet)
171
+ // Valid models: sonnet, opus, haiku, opusplan, sonnet[1m]
172
+ sdkOptions.model = options.model || CLAUDE_MODELS.DEFAULT;
173
+ // Model logged at query start below
174
+ // Map system prompt configuration
175
+ sdkOptions.systemPrompt = {
176
+ type: 'preset',
177
+ preset: 'claude_code' // Required to use CLAUDE.md
178
+ };
179
+ // Map setting sources for CLAUDE.md loading
180
+ // This loads CLAUDE.md from project, user (~/.config/claude/CLAUDE.md), and local directories
181
+ sdkOptions.settingSources = ['project', 'user', 'local'];
182
+ // Map resume session
183
+ if (sessionId) {
184
+ sdkOptions.resume = sessionId;
185
+ }
186
+ return sdkOptions;
187
+ }
188
+ /**
189
+ * Adds a session to the active sessions map
190
+ * @param {string} sessionId - Session identifier
191
+ * @param {Object} queryInstance - SDK query instance
192
+ * @param {Array<string>} tempImagePaths - Temp image file paths for cleanup
193
+ * @param {string} tempDir - Temp directory for cleanup
194
+ */
195
+ function addSession(sessionId, queryInstance, tempImagePaths = [], tempDir = null, writer = null) {
196
+ activeSessions.set(sessionId, {
197
+ instance: queryInstance,
198
+ startTime: Date.now(),
199
+ status: 'active',
200
+ tempImagePaths,
201
+ tempDir,
202
+ writer
203
+ });
204
+ }
205
+ /**
206
+ * Removes a session from the active sessions map
207
+ * @param {string} sessionId - Session identifier
208
+ */
209
+ function removeSession(sessionId) {
210
+ activeSessions.delete(sessionId);
211
+ }
212
+ /**
213
+ * Gets a session from the active sessions map
214
+ * @param {string} sessionId - Session identifier
215
+ * @returns {Object|undefined} Session data or undefined
216
+ */
217
+ function getSession(sessionId) {
218
+ return activeSessions.get(sessionId);
219
+ }
220
+ /**
221
+ * Gets all active session IDs
222
+ * @returns {Array<string>} Array of active session IDs
223
+ */
224
+ function getAllSessions() {
225
+ return Array.from(activeSessions.keys());
226
+ }
227
+ /**
228
+ * Transforms SDK messages to WebSocket format expected by frontend
229
+ * @param {Object} sdkMessage - SDK message object
230
+ * @returns {Object} Transformed message ready for WebSocket
231
+ */
232
+ function transformMessage(sdkMessage) {
233
+ // Extract parent_tool_use_id for subagent tool grouping
234
+ if (sdkMessage.parent_tool_use_id) {
235
+ return {
236
+ ...sdkMessage,
237
+ parentToolUseId: sdkMessage.parent_tool_use_id
238
+ };
239
+ }
240
+ return sdkMessage;
241
+ }
242
+ /**
243
+ * Extracts token usage from SDK result messages
244
+ * @param {Object} resultMessage - SDK result message
245
+ * @returns {Object|null} Token budget object or null
246
+ */
247
+ function extractTokenBudget(resultMessage) {
248
+ if (resultMessage.type !== 'result' || !resultMessage.modelUsage) {
249
+ return null;
250
+ }
251
+ // Get the first model's usage data
252
+ const modelKey = Object.keys(resultMessage.modelUsage)[0];
253
+ const modelData = resultMessage.modelUsage[modelKey];
254
+ if (!modelData) {
255
+ return null;
256
+ }
257
+ // Use cumulative tokens if available (tracks total for the session)
258
+ // Otherwise fall back to per-request tokens
259
+ const inputTokens = modelData.cumulativeInputTokens || modelData.inputTokens || 0;
260
+ const outputTokens = modelData.cumulativeOutputTokens || modelData.outputTokens || 0;
261
+ const cacheReadTokens = modelData.cumulativeCacheReadInputTokens || modelData.cacheReadInputTokens || 0;
262
+ const cacheCreationTokens = modelData.cumulativeCacheCreationInputTokens || modelData.cacheCreationInputTokens || 0;
263
+ // Total used = input + output + cache tokens
264
+ const totalUsed = inputTokens + outputTokens + cacheReadTokens + cacheCreationTokens;
265
+ // Use configured context window budget from environment (default 160000)
266
+ // This is the user's budget limit, not the model's context window
267
+ const contextWindow = parseInt(process.env.CONTEXT_WINDOW) || 160000;
268
+ // Token calc logged via token-budget WS event
269
+ return {
270
+ used: totalUsed,
271
+ total: contextWindow
272
+ };
273
+ }
274
+ /**
275
+ * Handles image processing for SDK queries
276
+ * Saves base64 images to temporary files and returns modified prompt with file paths
277
+ * @param {string} command - Original user prompt
278
+ * @param {Array} images - Array of image objects with base64 data
279
+ * @param {string} cwd - Working directory for temp file creation
280
+ * @returns {Promise<Object>} {modifiedCommand, tempImagePaths, tempDir}
281
+ */
282
+ async function handleImages(command, images, cwd) {
283
+ const tempImagePaths = [];
284
+ let tempDir = null;
285
+ if (!images || images.length === 0) {
286
+ return { modifiedCommand: command, tempImagePaths, tempDir };
287
+ }
288
+ try {
289
+ // Create temp directory in the project directory
290
+ const workingDir = cwd || process.cwd();
291
+ tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
292
+ await fs.mkdir(tempDir, { recursive: true });
293
+ // Save each image to a temp file
294
+ for (const [index, image] of images.entries()) {
295
+ // Extract base64 data and mime type
296
+ const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
297
+ if (!matches) {
298
+ console.error('Invalid image data format');
299
+ continue;
300
+ }
301
+ const [, mimeType, base64Data] = matches;
302
+ const extension = mimeType.split('/')[1] || 'png';
303
+ const filename = `image_${index}.${extension}`;
304
+ const filepath = path.join(tempDir, filename);
305
+ // Write base64 data to file
306
+ await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
307
+ tempImagePaths.push(filepath);
308
+ }
309
+ // Include the full image paths in the prompt
310
+ let modifiedCommand = command;
311
+ if (tempImagePaths.length > 0 && command && command.trim()) {
312
+ const imageNote = `\n\n[Images provided at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
313
+ modifiedCommand = command + imageNote;
314
+ }
315
+ // Images processed
316
+ return { modifiedCommand, tempImagePaths, tempDir };
317
+ }
318
+ catch (error) {
319
+ console.error('Error processing images for SDK:', error);
320
+ return { modifiedCommand: command, tempImagePaths, tempDir };
321
+ }
322
+ }
323
+ /**
324
+ * Cleans up temporary image files
325
+ * @param {Array<string>} tempImagePaths - Array of temp file paths to delete
326
+ * @param {string} tempDir - Temp directory to remove
327
+ */
328
+ async function cleanupTempFiles(tempImagePaths, tempDir) {
329
+ if (!tempImagePaths || tempImagePaths.length === 0) {
330
+ return;
331
+ }
332
+ try {
333
+ // Delete individual temp files
334
+ for (const imagePath of tempImagePaths) {
335
+ await fs.unlink(imagePath).catch(err => console.error(`Failed to delete temp image ${imagePath}:`, err));
336
+ }
337
+ // Delete temp directory
338
+ if (tempDir) {
339
+ await fs.rm(tempDir, { recursive: true, force: true }).catch(err => console.error(`Failed to delete temp directory ${tempDir}:`, err));
340
+ }
341
+ // Temp files cleaned
342
+ }
343
+ catch (error) {
344
+ console.error('Error during temp file cleanup:', error);
345
+ }
346
+ }
347
+ /**
348
+ * Loads MCP server configurations from ~/.claude.json
349
+ * @param {string} cwd - Current working directory for project-specific configs
350
+ * @returns {Object|null} MCP servers object or null if none found
351
+ */
352
+ async function loadMcpConfig(cwd) {
353
+ try {
354
+ const claudeConfigPath = path.join(os.homedir(), '.claude.json');
355
+ // Check if config file exists
356
+ try {
357
+ await fs.access(claudeConfigPath);
358
+ }
359
+ catch (error) {
360
+ // File doesn't exist, return null
361
+ // No config file
362
+ return null;
363
+ }
364
+ // Read and parse config file
365
+ let claudeConfig;
366
+ try {
367
+ const configContent = await fs.readFile(claudeConfigPath, 'utf8');
368
+ claudeConfig = JSON.parse(configContent);
369
+ }
370
+ catch (error) {
371
+ console.error('Failed to parse ~/.claude.json:', error.message);
372
+ return null;
373
+ }
374
+ // Extract MCP servers (merge global and project-specific)
375
+ let mcpServers = {};
376
+ // Add global MCP servers
377
+ if (claudeConfig.mcpServers && typeof claudeConfig.mcpServers === 'object') {
378
+ mcpServers = { ...claudeConfig.mcpServers };
379
+ // Global MCP servers loaded
380
+ }
381
+ // Add/override with project-specific MCP servers
382
+ if (claudeConfig.claudeProjects && cwd) {
383
+ const projectConfig = claudeConfig.claudeProjects[cwd];
384
+ if (projectConfig && projectConfig.mcpServers && typeof projectConfig.mcpServers === 'object') {
385
+ mcpServers = { ...mcpServers, ...projectConfig.mcpServers };
386
+ // Project MCP servers merged
387
+ }
388
+ }
389
+ // Return null if no servers found
390
+ if (Object.keys(mcpServers).length === 0) {
391
+ return null;
392
+ }
393
+ return mcpServers;
394
+ }
395
+ catch (error) {
396
+ console.error('Error loading MCP config:', error.message);
397
+ return null;
398
+ }
399
+ }
400
+ /**
401
+ * Executes a Claude query using the SDK
402
+ * @param {string} command - User prompt/command
403
+ * @param {Object} options - Query options
404
+ * @param {Object} ws - WebSocket connection
405
+ * @returns {Promise<void>}
406
+ */
407
+ async function queryClaudeSDK(command, options = {}, ws) {
408
+ const { sessionId, sessionSummary } = options;
409
+ let capturedSessionId = sessionId;
410
+ let sessionCreatedSent = false;
411
+ let tempImagePaths = [];
412
+ let tempDir = null;
413
+ const emitNotification = (event) => {
414
+ notifyUserIfEnabled({
415
+ userId: ws?.userId || null,
416
+ writer: ws,
417
+ event
418
+ });
419
+ };
420
+ try {
421
+ // Map CLI options to SDK format
422
+ const sdkOptions = mapCliOptionsToSDK(options);
423
+ // Load MCP configuration
424
+ const mcpServers = await loadMcpConfig(options.cwd);
425
+ if (mcpServers) {
426
+ sdkOptions.mcpServers = mcpServers;
427
+ }
428
+ // Handle images - save to temp files and modify prompt
429
+ const imageResult = await handleImages(command, options.images, options.cwd);
430
+ const finalCommand = imageResult.modifiedCommand;
431
+ tempImagePaths = imageResult.tempImagePaths;
432
+ tempDir = imageResult.tempDir;
433
+ sdkOptions.hooks = {
434
+ Notification: [{
435
+ matcher: '',
436
+ hooks: [async (input) => {
437
+ const message = typeof input?.message === 'string' ? input.message : 'Claude requires your attention.';
438
+ emitNotification(createNotificationEvent({
439
+ provider: 'claude',
440
+ sessionId: capturedSessionId || sessionId || null,
441
+ kind: 'action_required',
442
+ code: 'agent.notification',
443
+ meta: { message, sessionName: sessionSummary },
444
+ severity: 'warning',
445
+ requiresUserAction: true,
446
+ dedupeKey: `claude:hook:notification:${capturedSessionId || sessionId || 'none'}:${message}`
447
+ }));
448
+ return {};
449
+ }]
450
+ }]
451
+ };
452
+ sdkOptions.canUseTool = async (toolName, input, context) => {
453
+ const requiresInteraction = TOOLS_REQUIRING_INTERACTION.has(toolName);
454
+ if (!requiresInteraction) {
455
+ if (sdkOptions.permissionMode === 'bypassPermissions') {
456
+ return { behavior: 'allow', updatedInput: input };
457
+ }
458
+ const isDisallowed = (sdkOptions.disallowedTools || []).some(entry => matchesToolPermission(entry, toolName, input));
459
+ if (isDisallowed) {
460
+ return { behavior: 'deny', message: 'Tool disallowed by settings' };
461
+ }
462
+ const isAllowed = (sdkOptions.allowedTools || []).some(entry => matchesToolPermission(entry, toolName, input));
463
+ if (isAllowed) {
464
+ return { behavior: 'allow', updatedInput: input };
465
+ }
466
+ }
467
+ const requestId = createRequestId();
468
+ ws.send(createNormalizedMessage({ kind: 'permission_request', requestId, toolName, input, sessionId: capturedSessionId || sessionId || null, provider: 'claude' }));
469
+ emitNotification(createNotificationEvent({
470
+ provider: 'claude',
471
+ sessionId: capturedSessionId || sessionId || null,
472
+ kind: 'action_required',
473
+ code: 'permission.required',
474
+ meta: { toolName, sessionName: sessionSummary },
475
+ severity: 'warning',
476
+ requiresUserAction: true,
477
+ dedupeKey: `claude:permission:${capturedSessionId || sessionId || 'none'}:${requestId}`
478
+ }));
479
+ const decision = await waitForToolApproval(requestId, {
480
+ timeoutMs: requiresInteraction ? 0 : undefined,
481
+ signal: context?.signal,
482
+ metadata: {
483
+ _sessionId: capturedSessionId || sessionId || null,
484
+ _toolName: toolName,
485
+ _input: input,
486
+ _receivedAt: new Date(),
487
+ },
488
+ onCancel: (reason) => {
489
+ ws.send(createNormalizedMessage({ kind: 'permission_cancelled', requestId, reason, sessionId: capturedSessionId || sessionId || null, provider: 'claude' }));
490
+ }
491
+ });
492
+ if (!decision) {
493
+ return { behavior: 'deny', message: 'Permission request timed out' };
494
+ }
495
+ if (decision.cancelled) {
496
+ return { behavior: 'deny', message: 'Permission request cancelled' };
497
+ }
498
+ if (decision.allow) {
499
+ if (decision.rememberEntry && typeof decision.rememberEntry === 'string') {
500
+ if (!sdkOptions.allowedTools.includes(decision.rememberEntry)) {
501
+ sdkOptions.allowedTools.push(decision.rememberEntry);
502
+ }
503
+ if (Array.isArray(sdkOptions.disallowedTools)) {
504
+ sdkOptions.disallowedTools = sdkOptions.disallowedTools.filter(entry => entry !== decision.rememberEntry);
505
+ }
506
+ }
507
+ return { behavior: 'allow', updatedInput: decision.updatedInput ?? input };
508
+ }
509
+ return { behavior: 'deny', message: decision.message ?? 'User denied tool use' };
510
+ };
511
+ // Set stream-close timeout for interactive tools (Query constructor reads it synchronously). Claude Agent SDK has a default of 5s and this overrides it
512
+ const prevStreamTimeout = process.env.CLAUDE_CODE_STREAM_CLOSE_TIMEOUT;
513
+ process.env.CLAUDE_CODE_STREAM_CLOSE_TIMEOUT = '300000';
514
+ let queryInstance;
515
+ try {
516
+ queryInstance = query({
517
+ prompt: finalCommand,
518
+ options: sdkOptions
519
+ });
520
+ }
521
+ catch (hookError) {
522
+ // Older/newer SDK versions may not accept hook shapes yet.
523
+ // Keep notification behavior operational via runtime events even if hook registration fails.
524
+ console.warn('Failed to initialize Claude query with hooks, retrying without hooks:', hookError?.message || hookError);
525
+ delete sdkOptions.hooks;
526
+ queryInstance = query({
527
+ prompt: finalCommand,
528
+ options: sdkOptions
529
+ });
530
+ }
531
+ // Restore immediately — Query constructor already captured the value
532
+ if (prevStreamTimeout !== undefined) {
533
+ process.env.CLAUDE_CODE_STREAM_CLOSE_TIMEOUT = prevStreamTimeout;
534
+ }
535
+ else {
536
+ delete process.env.CLAUDE_CODE_STREAM_CLOSE_TIMEOUT;
537
+ }
538
+ // Track the query instance for abort capability
539
+ if (capturedSessionId) {
540
+ addSession(capturedSessionId, queryInstance, tempImagePaths, tempDir, ws);
541
+ }
542
+ // Process streaming messages
543
+ console.log('Starting async generator loop for session:', capturedSessionId || 'NEW');
544
+ for await (const message of queryInstance) {
545
+ // Capture session ID from first message
546
+ if (message.session_id && !capturedSessionId) {
547
+ capturedSessionId = message.session_id;
548
+ addSession(capturedSessionId, queryInstance, tempImagePaths, tempDir, ws);
549
+ // Set session ID on writer
550
+ if (ws.setSessionId && typeof ws.setSessionId === 'function') {
551
+ ws.setSessionId(capturedSessionId);
552
+ }
553
+ // Send session-created event only once for new sessions
554
+ if (!sessionId && !sessionCreatedSent) {
555
+ sessionCreatedSent = true;
556
+ ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'claude' }));
557
+ }
558
+ }
559
+ else {
560
+ // session_id already captured
561
+ }
562
+ // Transform and normalize message via adapter
563
+ const transformedMessage = transformMessage(message);
564
+ const sid = capturedSessionId || sessionId || null;
565
+ // Use adapter to normalize SDK events into NormalizedMessage[]
566
+ const normalized = sessionsService.normalizeMessage('claude', transformedMessage, sid);
567
+ for (const msg of normalized) {
568
+ // Preserve parentToolUseId from SDK wrapper for subagent tool grouping
569
+ if (transformedMessage.parentToolUseId && !msg.parentToolUseId) {
570
+ msg.parentToolUseId = transformedMessage.parentToolUseId;
571
+ }
572
+ ws.send(msg);
573
+ }
574
+ // Extract and send token budget updates from result messages
575
+ if (message.type === 'result') {
576
+ const models = Object.keys(message.modelUsage || {});
577
+ if (models.length > 0) {
578
+ // Model info available in result message
579
+ }
580
+ const tokenBudgetData = extractTokenBudget(message);
581
+ if (tokenBudgetData) {
582
+ ws.send(createNormalizedMessage({ kind: 'status', text: 'token_budget', tokenBudget: tokenBudgetData, sessionId: capturedSessionId || sessionId || null, provider: 'claude' }));
583
+ }
584
+ }
585
+ }
586
+ // Clean up session on completion
587
+ if (capturedSessionId) {
588
+ removeSession(capturedSessionId);
589
+ }
590
+ // Clean up temporary image files
591
+ await cleanupTempFiles(tempImagePaths, tempDir);
592
+ // Send completion event
593
+ ws.send(createNormalizedMessage({ kind: 'complete', exitCode: 0, isNewSession: !sessionId && !!command, sessionId: capturedSessionId, provider: 'claude' }));
594
+ notifyRunStopped({
595
+ userId: ws?.userId || null,
596
+ provider: 'claude',
597
+ sessionId: capturedSessionId || sessionId || null,
598
+ sessionName: sessionSummary,
599
+ stopReason: 'completed'
600
+ });
601
+ // Complete
602
+ }
603
+ catch (error) {
604
+ console.error('SDK query error:', error);
605
+ // Clean up session on error
606
+ if (capturedSessionId) {
607
+ removeSession(capturedSessionId);
608
+ }
609
+ // Clean up temporary image files on error
610
+ await cleanupTempFiles(tempImagePaths, tempDir);
611
+ // Check if Claude CLI is installed for a clearer error message
612
+ const installed = await providerAuthService.isProviderInstalled('claude');
613
+ const errorContent = !installed
614
+ ? 'Claude Code is not installed. Please install it first: https://docs.anthropic.com/en/docs/claude-code'
615
+ : error.message;
616
+ // Send error to WebSocket
617
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: capturedSessionId || sessionId || null, provider: 'claude' }));
618
+ notifyRunFailed({
619
+ userId: ws?.userId || null,
620
+ provider: 'claude',
621
+ sessionId: capturedSessionId || sessionId || null,
622
+ sessionName: sessionSummary,
623
+ error
624
+ });
625
+ }
626
+ }
627
+ /**
628
+ * Aborts an active SDK session
629
+ * @param {string} sessionId - Session identifier
630
+ * @returns {boolean} True if session was aborted, false if not found
631
+ */
632
+ async function abortClaudeSDKSession(sessionId) {
633
+ const session = getSession(sessionId);
634
+ if (!session) {
635
+ console.log(`Session ${sessionId} not found`);
636
+ return false;
637
+ }
638
+ try {
639
+ console.log(`Aborting SDK session: ${sessionId}`);
640
+ // Call interrupt() on the query instance
641
+ await session.instance.interrupt();
642
+ // Update session status
643
+ session.status = 'aborted';
644
+ // Clean up temporary image files
645
+ await cleanupTempFiles(session.tempImagePaths, session.tempDir);
646
+ // Clean up session
647
+ removeSession(sessionId);
648
+ return true;
649
+ }
650
+ catch (error) {
651
+ console.error(`Error aborting session ${sessionId}:`, error);
652
+ return false;
653
+ }
654
+ }
655
+ /**
656
+ * Checks if an SDK session is currently active
657
+ * @param {string} sessionId - Session identifier
658
+ * @returns {boolean} True if session is active
659
+ */
660
+ function isClaudeSDKSessionActive(sessionId) {
661
+ const session = getSession(sessionId);
662
+ return session && session.status === 'active';
663
+ }
664
+ /**
665
+ * Gets all active SDK session IDs
666
+ * @returns {Array<string>} Array of active session IDs
667
+ */
668
+ function getActiveClaudeSDKSessions() {
669
+ return getAllSessions();
670
+ }
671
+ /**
672
+ * Get pending tool approvals for a specific session.
673
+ * @param {string} sessionId - The session ID
674
+ * @returns {Array} Array of pending permission request objects
675
+ */
676
+ function getPendingApprovalsForSession(sessionId) {
677
+ const pending = [];
678
+ for (const [requestId, resolver] of pendingToolApprovals.entries()) {
679
+ if (resolver._sessionId === sessionId) {
680
+ pending.push({
681
+ requestId,
682
+ toolName: resolver._toolName || 'UnknownTool',
683
+ input: resolver._input,
684
+ context: resolver._context,
685
+ sessionId,
686
+ receivedAt: resolver._receivedAt || new Date(),
687
+ });
688
+ }
689
+ }
690
+ return pending;
691
+ }
692
+ /**
693
+ * Reconnect a session's WebSocketWriter to a new raw WebSocket.
694
+ * Called when client reconnects (e.g. page refresh) while SDK is still running.
695
+ * @param {string} sessionId - The session ID
696
+ * @param {Object} newRawWs - The new raw WebSocket connection
697
+ * @returns {boolean} True if writer was successfully reconnected
698
+ */
699
+ function reconnectSessionWriter(sessionId, newRawWs) {
700
+ const session = getSession(sessionId);
701
+ if (!session?.writer?.updateWebSocket)
702
+ return false;
703
+ session.writer.updateWebSocket(newRawWs);
704
+ console.log(`[RECONNECT] Writer swapped for session ${sessionId}`);
705
+ return true;
706
+ }
707
+ // Export public API
708
+ export { queryClaudeSDK, abortClaudeSDKSession, isClaudeSDKSessionActive, getActiveClaudeSDKSessions, resolveToolApproval, getPendingApprovalsForSession, reconnectSessionWriter };
709
+ //# sourceMappingURL=claude-sdk.js.map