@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,469 @@
1
+ import { spawn } from 'child_process';
2
+ import crossSpawn from 'cross-spawn';
3
+
4
+ // Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
5
+ const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
6
+ import { promises as fs } from 'fs';
7
+ import path from 'path';
8
+ import os from 'os';
9
+ import sessionManager from './sessionManager.js';
10
+ import GeminiResponseHandler from './gemini-response-handler.js';
11
+ import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
12
+ import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
13
+ import { createNormalizedMessage } from './shared/utils.js';
14
+
15
+ let activeGeminiProcesses = new Map(); // Track active processes by session ID
16
+
17
+ async function spawnGemini(command, options = {}, ws) {
18
+ const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
19
+ let capturedSessionId = sessionId; // Track session ID throughout the process
20
+ let sessionCreatedSent = false; // Track if we've already sent session-created event
21
+ let assistantBlocks = []; // Accumulate the full response blocks including tools
22
+
23
+ // Use tools settings passed from frontend, or defaults
24
+ const settings = toolsSettings || {
25
+ allowedTools: [],
26
+ disallowedTools: [],
27
+ skipPermissions: false
28
+ };
29
+
30
+ // Build Gemini CLI command - start with print/resume flags first
31
+ const args = [];
32
+
33
+ // Add prompt flag with command if we have a command
34
+ if (command && command.trim()) {
35
+ args.push('--prompt', command);
36
+ }
37
+
38
+ // If we have a sessionId, we want to resume
39
+ if (sessionId) {
40
+ const session = sessionManager.getSession(sessionId);
41
+ if (session && session.cliSessionId) {
42
+ args.push('--resume', session.cliSessionId);
43
+ }
44
+ }
45
+
46
+ // Use cwd (actual project directory) instead of projectPath (Gemini's metadata directory)
47
+ // Clean the path by removing any non-printable characters
48
+ const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
49
+ const workingDir = cleanPath;
50
+
51
+ // Handle images by saving them to temporary files and passing paths to Gemini
52
+ const tempImagePaths = [];
53
+ let tempDir = null;
54
+ if (images && images.length > 0) {
55
+ try {
56
+ // Create temp directory in the project directory so Gemini can access it
57
+ tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
58
+ await fs.mkdir(tempDir, { recursive: true });
59
+
60
+ // Save each image to a temp file
61
+ for (const [index, image] of images.entries()) {
62
+ // Extract base64 data and mime type
63
+ const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
64
+ if (!matches) {
65
+ continue;
66
+ }
67
+
68
+ const [, mimeType, base64Data] = matches;
69
+ const extension = mimeType.split('/')[1] || 'png';
70
+ const filename = `image_${index}.${extension}`;
71
+ const filepath = path.join(tempDir, filename);
72
+
73
+ // Write base64 data to file
74
+ await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
75
+ tempImagePaths.push(filepath);
76
+ }
77
+
78
+ // Include the full image paths in the prompt for Gemini to reference
79
+ // Gemini CLI can read images from file paths in the prompt
80
+ if (tempImagePaths.length > 0 && command && command.trim()) {
81
+ const imageNote = `\n\n[Images given: ${tempImagePaths.length} images are located at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
82
+ const modifiedCommand = command + imageNote;
83
+
84
+ // Update the command in args
85
+ const promptIndex = args.indexOf('--prompt');
86
+ if (promptIndex !== -1 && args[promptIndex + 1] === command) {
87
+ args[promptIndex + 1] = modifiedCommand;
88
+ } else if (promptIndex !== -1) {
89
+ // If we're using context, update the full prompt
90
+ args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
91
+ }
92
+ }
93
+ } catch (error) {
94
+ console.error('Error processing images for Gemini:', error);
95
+ }
96
+ }
97
+
98
+ // Add basic flags for Gemini
99
+ if (options.debug) {
100
+ args.push('--debug');
101
+ }
102
+
103
+ // Add MCP config flag only if MCP servers are configured
104
+ try {
105
+ const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
106
+ let hasMcpServers = false;
107
+
108
+ try {
109
+ await fs.access(geminiConfigPath);
110
+ const geminiConfigRaw = await fs.readFile(geminiConfigPath, 'utf8');
111
+ const geminiConfig = JSON.parse(geminiConfigRaw);
112
+
113
+ // Check global MCP servers
114
+ if (geminiConfig.mcpServers && Object.keys(geminiConfig.mcpServers).length > 0) {
115
+ hasMcpServers = true;
116
+ }
117
+
118
+ // Check project-specific MCP servers
119
+ if (!hasMcpServers && geminiConfig.geminiProjects) {
120
+ const currentProjectPath = process.cwd();
121
+ const projectConfig = geminiConfig.geminiProjects[currentProjectPath];
122
+ if (projectConfig && projectConfig.mcpServers && Object.keys(projectConfig.mcpServers).length > 0) {
123
+ hasMcpServers = true;
124
+ }
125
+ }
126
+ } catch (e) {
127
+ // Ignore if file doesn't exist or isn't parsable
128
+ }
129
+
130
+ if (hasMcpServers) {
131
+ args.push('--mcp-config', geminiConfigPath);
132
+ }
133
+ } catch (error) {
134
+ // Ignore outer errors
135
+ }
136
+
137
+ // Add model for all sessions (both new and resumed)
138
+ let modelToUse = options.model || 'gemini-2.5-flash';
139
+ args.push('--model', modelToUse);
140
+ args.push('--output-format', 'stream-json');
141
+
142
+ // Handle approval modes and allowed tools
143
+ if (settings.skipPermissions || options.skipPermissions || permissionMode === 'yolo') {
144
+ args.push('--yolo');
145
+ } else if (permissionMode === 'auto_edit') {
146
+ args.push('--approval-mode', 'auto_edit');
147
+ } else if (permissionMode === 'plan') {
148
+ args.push('--approval-mode', 'plan');
149
+ }
150
+
151
+ if (settings.allowedTools && settings.allowedTools.length > 0) {
152
+ args.push('--allowed-tools', settings.allowedTools.join(','));
153
+ }
154
+
155
+ // Try to find gemini in PATH first, then fall back to environment variable
156
+ const geminiPath = process.env.GEMINI_PATH || 'gemini';
157
+ console.log('Spawning Gemini CLI:', geminiPath, args.join(' '));
158
+ console.log('Working directory:', workingDir);
159
+
160
+ let spawnCmd = geminiPath;
161
+ let spawnArgs = args;
162
+
163
+ // On non-Windows platforms, wrap the execution in a shell to avoid ENOEXEC
164
+ // which happens when the target is a script lacking a shebang.
165
+ if (os.platform() !== 'win32') {
166
+ spawnCmd = 'sh';
167
+ // Use exec to replace the shell process, ensuring signals hit gemini directly
168
+ spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
169
+ }
170
+
171
+ return new Promise((resolve, reject) => {
172
+ const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
173
+ cwd: workingDir,
174
+ stdio: ['pipe', 'pipe', 'pipe'],
175
+ env: { ...process.env } // Inherit all environment variables
176
+ });
177
+ let terminalNotificationSent = false;
178
+ let terminalFailureReason = null;
179
+
180
+ const notifyTerminalState = ({ code = null, error = null } = {}) => {
181
+ if (terminalNotificationSent) {
182
+ return;
183
+ }
184
+
185
+ terminalNotificationSent = true;
186
+
187
+ const finalSessionId = capturedSessionId || sessionId || processKey;
188
+ if (code === 0 && !error) {
189
+ notifyRunStopped({
190
+ userId: ws?.userId || null,
191
+ provider: 'gemini',
192
+ sessionId: finalSessionId,
193
+ sessionName: sessionSummary,
194
+ stopReason: 'completed'
195
+ });
196
+ return;
197
+ }
198
+
199
+ notifyRunFailed({
200
+ userId: ws?.userId || null,
201
+ provider: 'gemini',
202
+ sessionId: finalSessionId,
203
+ sessionName: sessionSummary,
204
+ error: error || terminalFailureReason || `Gemini CLI exited with code ${code}`
205
+ });
206
+ };
207
+
208
+ // Attach temp file info to process for cleanup later
209
+ geminiProcess.tempImagePaths = tempImagePaths;
210
+ geminiProcess.tempDir = tempDir;
211
+
212
+ // Store process reference for potential abort
213
+ const processKey = capturedSessionId || sessionId || Date.now().toString();
214
+ activeGeminiProcesses.set(processKey, geminiProcess);
215
+
216
+ // Store sessionId on the process object for debugging
217
+ geminiProcess.sessionId = processKey;
218
+
219
+ // Close stdin to signal we're done sending input
220
+ geminiProcess.stdin.end();
221
+
222
+ // Add timeout handler
223
+ const timeoutMs = 120000; // 120 seconds for slower models
224
+ let timeout;
225
+
226
+ const startTimeout = () => {
227
+ if (timeout) clearTimeout(timeout);
228
+ timeout = setTimeout(() => {
229
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
230
+ terminalFailureReason = `Gemini CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
231
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
232
+ try {
233
+ geminiProcess.kill('SIGTERM');
234
+ } catch (e) { }
235
+ }, timeoutMs);
236
+ };
237
+
238
+ startTimeout();
239
+
240
+ // Save user message to session when starting
241
+ if (command && capturedSessionId) {
242
+ sessionManager.addMessage(capturedSessionId, 'user', command);
243
+ }
244
+
245
+ // Create response handler for NDJSON buffering
246
+ let responseHandler;
247
+ if (ws) {
248
+ responseHandler = new GeminiResponseHandler(ws, {
249
+ onContentFragment: (content) => {
250
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
251
+ assistantBlocks[assistantBlocks.length - 1].text += content;
252
+ } else {
253
+ assistantBlocks.push({ type: 'text', text: content });
254
+ }
255
+ },
256
+ onToolUse: (event) => {
257
+ assistantBlocks.push({
258
+ type: 'tool_use',
259
+ id: event.tool_id,
260
+ name: event.tool_name,
261
+ input: event.parameters
262
+ });
263
+ },
264
+ onToolResult: (event) => {
265
+ if (capturedSessionId) {
266
+ if (assistantBlocks.length > 0) {
267
+ sessionManager.addMessage(capturedSessionId, 'assistant', [...assistantBlocks]);
268
+ assistantBlocks = [];
269
+ }
270
+ sessionManager.addMessage(capturedSessionId, 'user', [{
271
+ type: 'tool_result',
272
+ tool_use_id: event.tool_id,
273
+ content: event.output === undefined ? null : event.output,
274
+ is_error: event.status === 'error'
275
+ }]);
276
+ }
277
+ },
278
+ onInit: (event) => {
279
+ if (capturedSessionId) {
280
+ const sess = sessionManager.getSession(capturedSessionId);
281
+ if (sess && !sess.cliSessionId) {
282
+ sess.cliSessionId = event.session_id;
283
+ sessionManager.saveSession(capturedSessionId);
284
+ }
285
+ }
286
+ }
287
+ });
288
+ }
289
+
290
+ // Handle stdout
291
+ geminiProcess.stdout.on('data', (data) => {
292
+ const rawOutput = data.toString();
293
+ startTimeout(); // Re-arm the timeout
294
+
295
+ // For new sessions, create a session ID FIRST
296
+ if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
297
+ capturedSessionId = `gemini_${Date.now()}`;
298
+ sessionCreatedSent = true;
299
+
300
+ // Create session in session manager
301
+ sessionManager.createSession(capturedSessionId, cwd || process.cwd());
302
+
303
+ // Save the user message now that we have a session ID
304
+ if (command) {
305
+ sessionManager.addMessage(capturedSessionId, 'user', command);
306
+ }
307
+
308
+ // Update process key with captured session ID
309
+ if (processKey !== capturedSessionId) {
310
+ activeGeminiProcesses.delete(processKey);
311
+ activeGeminiProcesses.set(capturedSessionId, geminiProcess);
312
+ }
313
+
314
+ ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
315
+
316
+ ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
317
+ }
318
+
319
+ if (responseHandler) {
320
+ responseHandler.processData(rawOutput);
321
+ } else if (rawOutput) {
322
+ // Fallback to direct sending for raw CLI mode without WS
323
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
324
+ assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
325
+ } else {
326
+ assistantBlocks.push({ type: 'text', text: rawOutput });
327
+ }
328
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
329
+ ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'gemini' }));
330
+ }
331
+ });
332
+
333
+ // Handle stderr
334
+ geminiProcess.stderr.on('data', (data) => {
335
+ const errorMsg = data.toString();
336
+
337
+ // Filter out deprecation warnings and "Loaded cached credentials" message
338
+ if (errorMsg.includes('[DEP0040]') ||
339
+ errorMsg.includes('DeprecationWarning') ||
340
+ errorMsg.includes('--trace-deprecation') ||
341
+ errorMsg.includes('Loaded cached credentials')) {
342
+ return;
343
+ }
344
+
345
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
346
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'gemini' }));
347
+ });
348
+
349
+ // Handle process completion
350
+ geminiProcess.on('close', async (code) => {
351
+ clearTimeout(timeout);
352
+
353
+ // Flush any remaining buffered content
354
+ if (responseHandler) {
355
+ responseHandler.forceFlush();
356
+ responseHandler.destroy();
357
+ }
358
+
359
+ // Clean up process reference
360
+ const finalSessionId = capturedSessionId || sessionId || processKey;
361
+ activeGeminiProcesses.delete(finalSessionId);
362
+
363
+ // Save assistant response to session if we have one
364
+ if (finalSessionId && assistantBlocks.length > 0) {
365
+ sessionManager.addMessage(finalSessionId, 'assistant', assistantBlocks);
366
+ }
367
+
368
+ ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'gemini' }));
369
+
370
+ // Clean up temporary image files if any
371
+ if (geminiProcess.tempImagePaths && geminiProcess.tempImagePaths.length > 0) {
372
+ for (const imagePath of geminiProcess.tempImagePaths) {
373
+ await fs.unlink(imagePath).catch(err => { });
374
+ }
375
+ if (geminiProcess.tempDir) {
376
+ await fs.rm(geminiProcess.tempDir, { recursive: true, force: true }).catch(err => { });
377
+ }
378
+ }
379
+
380
+ if (code === 0) {
381
+ notifyTerminalState({ code });
382
+ resolve();
383
+ } else {
384
+ // code 127 = shell "command not found" — check installation
385
+ if (code === 127) {
386
+ const installed = await providerAuthService.isProviderInstalled('gemini');
387
+ if (!installed) {
388
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
389
+ ws.send(createNormalizedMessage({ kind: 'error', content: 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli', sessionId: socketSessionId, provider: 'gemini' }));
390
+ }
391
+ }
392
+
393
+ notifyTerminalState({
394
+ code,
395
+ error: code === null ? 'Gemini CLI process was terminated or timed out' : null
396
+ });
397
+ reject(new Error(code === null ? 'Gemini CLI process was terminated or timed out' : `Gemini CLI exited with code ${code}`));
398
+ }
399
+ });
400
+
401
+ // Handle process errors
402
+ geminiProcess.on('error', async (error) => {
403
+ // Clean up process reference on error
404
+ const finalSessionId = capturedSessionId || sessionId || processKey;
405
+ activeGeminiProcesses.delete(finalSessionId);
406
+
407
+ // Check if Gemini CLI is installed for a clearer error message
408
+ const installed = await providerAuthService.isProviderInstalled('gemini');
409
+ const errorContent = !installed
410
+ ? 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli'
411
+ : error.message;
412
+
413
+ const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
414
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'gemini' }));
415
+ notifyTerminalState({ error });
416
+
417
+ reject(error);
418
+ });
419
+
420
+ });
421
+ }
422
+
423
+ function abortGeminiSession(sessionId) {
424
+ let geminiProc = activeGeminiProcesses.get(sessionId);
425
+ let processKey = sessionId;
426
+
427
+ if (!geminiProc) {
428
+ for (const [key, proc] of activeGeminiProcesses.entries()) {
429
+ if (proc.sessionId === sessionId) {
430
+ geminiProc = proc;
431
+ processKey = key;
432
+ break;
433
+ }
434
+ }
435
+ }
436
+
437
+ if (geminiProc) {
438
+ try {
439
+ geminiProc.kill('SIGTERM');
440
+ setTimeout(() => {
441
+ if (activeGeminiProcesses.has(processKey)) {
442
+ try {
443
+ geminiProc.kill('SIGKILL');
444
+ } catch (e) { }
445
+ }
446
+ }, 2000); // Wait 2 seconds before force kill
447
+
448
+ return true;
449
+ } catch (error) {
450
+ return false;
451
+ }
452
+ }
453
+ return false;
454
+ }
455
+
456
+ function isGeminiSessionActive(sessionId) {
457
+ return activeGeminiProcesses.has(sessionId);
458
+ }
459
+
460
+ function getActiveGeminiSessions() {
461
+ return Array.from(activeGeminiProcesses.keys());
462
+ }
463
+
464
+ export {
465
+ spawnGemini,
466
+ abortGeminiSession,
467
+ isGeminiSessionActive,
468
+ getActiveGeminiSessions
469
+ };
@@ -0,0 +1,79 @@
1
+ // Gemini Response Handler - JSON Stream processing
2
+ import { sessionsService } from './modules/providers/services/sessions.service.js';
3
+
4
+ class GeminiResponseHandler {
5
+ constructor(ws, options = {}) {
6
+ this.ws = ws;
7
+ this.buffer = '';
8
+ this.onContentFragment = options.onContentFragment || null;
9
+ this.onInit = options.onInit || null;
10
+ this.onToolUse = options.onToolUse || null;
11
+ this.onToolResult = options.onToolResult || null;
12
+ }
13
+
14
+ // Process incoming raw data from Gemini stream-json
15
+ processData(data) {
16
+ this.buffer += data;
17
+
18
+ // Split by newline
19
+ const lines = this.buffer.split('\n');
20
+
21
+ // Keep the last incomplete line in the buffer
22
+ this.buffer = lines.pop() || '';
23
+
24
+ for (const line of lines) {
25
+ if (!line.trim()) continue;
26
+
27
+ try {
28
+ const event = JSON.parse(line);
29
+ this.handleEvent(event);
30
+ } catch (err) {
31
+ // Not a JSON line, probably debug output or CLI warnings
32
+ }
33
+ }
34
+ }
35
+
36
+ handleEvent(event) {
37
+ const sid = typeof this.ws.getSessionId === 'function' ? this.ws.getSessionId() : null;
38
+
39
+ if (event.type === 'init') {
40
+ if (this.onInit) {
41
+ this.onInit(event);
42
+ }
43
+ return;
44
+ }
45
+
46
+ // Invoke per-type callbacks for session tracking
47
+ if (event.type === 'message' && event.role === 'assistant') {
48
+ const content = event.content || '';
49
+ if (this.onContentFragment && content) {
50
+ this.onContentFragment(content);
51
+ }
52
+ } else if (event.type === 'tool_use' && this.onToolUse) {
53
+ this.onToolUse(event);
54
+ } else if (event.type === 'tool_result' && this.onToolResult) {
55
+ this.onToolResult(event);
56
+ }
57
+
58
+ // Normalize via adapter and send all resulting messages
59
+ const normalized = sessionsService.normalizeMessage('gemini', event, sid);
60
+ for (const msg of normalized) {
61
+ this.ws.send(msg);
62
+ }
63
+ }
64
+
65
+ forceFlush() {
66
+ if (this.buffer.trim()) {
67
+ try {
68
+ const event = JSON.parse(this.buffer);
69
+ this.handleEvent(event);
70
+ } catch (err) { }
71
+ }
72
+ }
73
+
74
+ destroy() {
75
+ this.buffer = '';
76
+ }
77
+ }
78
+
79
+ export default GeminiResponseHandler;