@glwhappen/web-code 1.32.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 (479) hide show
  1. package/LICENSE +718 -0
  2. package/README.de.md +250 -0
  3. package/README.ja.md +242 -0
  4. package/README.ko.md +242 -0
  5. package/README.md +252 -0
  6. package/README.ru.md +250 -0
  7. package/README.tr.md +252 -0
  8. package/README.zh-CN.md +242 -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-Ct6oPUQk.css +32 -0
  70. package/dist/assets/index-u6XmIqLb.js +1346 -0
  71. package/dist/assets/vendor-codemirror-OwyKSvPE.js +41 -0
  72. package/dist/assets/vendor-react-BGZc9oRE.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 +738 -0
  116. package/dist-server/server/claude-sdk.js.map +1 -0
  117. package/dist-server/server/cli.js +641 -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 +271 -0
  122. package/dist-server/server/cursor-cli.js.map +1 -0
  123. package/dist-server/server/gemini-cli.js +539 -0
  124. package/dist-server/server/gemini-cli.js.map +1 -0
  125. package/dist-server/server/gemini-response-handler.js +72 -0
  126. package/dist-server/server/gemini-response-handler.js.map +1 -0
  127. package/dist-server/server/index.js +1340 -0
  128. package/dist-server/server/index.js.map +1 -0
  129. package/dist-server/server/load-env.js +32 -0
  130. package/dist-server/server/load-env.js.map +1 -0
  131. package/dist-server/server/middleware/auth.js +117 -0
  132. package/dist-server/server/middleware/auth.js.map +1 -0
  133. package/dist-server/server/modules/database/connection.js +125 -0
  134. package/dist-server/server/modules/database/connection.js.map +1 -0
  135. package/dist-server/server/modules/database/index.js +13 -0
  136. package/dist-server/server/modules/database/index.js.map +1 -0
  137. package/dist-server/server/modules/database/init-db.js +18 -0
  138. package/dist-server/server/modules/database/init-db.js.map +1 -0
  139. package/dist-server/server/modules/database/migrations.js +419 -0
  140. package/dist-server/server/modules/database/migrations.js.map +1 -0
  141. package/dist-server/server/modules/database/repositories/api-keys.js +72 -0
  142. package/dist-server/server/modules/database/repositories/api-keys.js.map +1 -0
  143. package/dist-server/server/modules/database/repositories/app-config.js +47 -0
  144. package/dist-server/server/modules/database/repositories/app-config.js.map +1 -0
  145. package/dist-server/server/modules/database/repositories/credentials.js +68 -0
  146. package/dist-server/server/modules/database/repositories/credentials.js.map +1 -0
  147. package/dist-server/server/modules/database/repositories/github-tokens.js +54 -0
  148. package/dist-server/server/modules/database/repositories/github-tokens.js.map +1 -0
  149. package/dist-server/server/modules/database/repositories/notification-preferences.js +72 -0
  150. package/dist-server/server/modules/database/repositories/notification-preferences.js.map +1 -0
  151. package/dist-server/server/modules/database/repositories/projects.db.integration.test.js +67 -0
  152. package/dist-server/server/modules/database/repositories/projects.db.integration.test.js.map +1 -0
  153. package/dist-server/server/modules/database/repositories/projects.db.js +185 -0
  154. package/dist-server/server/modules/database/repositories/projects.db.js.map +1 -0
  155. package/dist-server/server/modules/database/repositories/push-subscriptions.js +49 -0
  156. package/dist-server/server/modules/database/repositories/push-subscriptions.js.map +1 -0
  157. package/dist-server/server/modules/database/repositories/scan-state.db.js +31 -0
  158. package/dist-server/server/modules/database/repositories/scan-state.db.js.map +1 -0
  159. package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js +64 -0
  160. package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js.map +1 -0
  161. package/dist-server/server/modules/database/repositories/sessions.db.js +150 -0
  162. package/dist-server/server/modules/database/repositories/sessions.db.js.map +1 -0
  163. package/dist-server/server/modules/database/repositories/users.js +116 -0
  164. package/dist-server/server/modules/database/repositories/users.js.map +1 -0
  165. package/dist-server/server/modules/database/repositories/vapid-keys.js +38 -0
  166. package/dist-server/server/modules/database/repositories/vapid-keys.js.map +1 -0
  167. package/dist-server/server/modules/database/schema.js +150 -0
  168. package/dist-server/server/modules/database/schema.js.map +1 -0
  169. package/dist-server/server/modules/projects/index.js +4 -0
  170. package/dist-server/server/modules/projects/index.js.map +1 -0
  171. package/dist-server/server/modules/projects/projects.routes.js +225 -0
  172. package/dist-server/server/modules/projects/projects.routes.js.map +1 -0
  173. package/dist-server/server/modules/projects/services/project-clone.service.js +220 -0
  174. package/dist-server/server/modules/projects/services/project-clone.service.js.map +1 -0
  175. package/dist-server/server/modules/projects/services/project-delete.service.js +83 -0
  176. package/dist-server/server/modules/projects/services/project-delete.service.js.map +1 -0
  177. package/dist-server/server/modules/projects/services/project-management.service.js +99 -0
  178. package/dist-server/server/modules/projects/services/project-management.service.js.map +1 -0
  179. package/dist-server/server/modules/projects/services/project-star.service.js +60 -0
  180. package/dist-server/server/modules/projects/services/project-star.service.js.map +1 -0
  181. package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js +171 -0
  182. package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js.map +1 -0
  183. package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js +213 -0
  184. package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js.map +1 -0
  185. package/dist-server/server/modules/projects/tests/project-clone.service.test.js +129 -0
  186. package/dist-server/server/modules/projects/tests/project-clone.service.test.js.map +1 -0
  187. package/dist-server/server/modules/projects/tests/project-management.service.test.js +89 -0
  188. package/dist-server/server/modules/projects/tests/project-management.service.test.js.map +1 -0
  189. package/dist-server/server/modules/projects/tests/project-star.service.test.js +99 -0
  190. package/dist-server/server/modules/projects/tests/project-star.service.test.js.map +1 -0
  191. package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js +88 -0
  192. package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js.map +1 -0
  193. package/dist-server/server/modules/providers/index.js +5 -0
  194. package/dist-server/server/modules/providers/index.js.map +1 -0
  195. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +104 -0
  196. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -0
  197. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js +103 -0
  198. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js.map +1 -0
  199. package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js +116 -0
  200. package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js.map +1 -0
  201. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js +546 -0
  202. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js.map +1 -0
  203. package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js +198 -0
  204. package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js.map +1 -0
  205. package/dist-server/server/modules/providers/list/claude/claude.provider.js +17 -0
  206. package/dist-server/server/modules/providers/list/claude/claude.provider.js.map +1 -0
  207. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +84 -0
  208. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -0
  209. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js +107 -0
  210. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js.map +1 -0
  211. package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js +123 -0
  212. package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js.map +1 -0
  213. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js +513 -0
  214. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js.map +1 -0
  215. package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js +82 -0
  216. package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js.map +1 -0
  217. package/dist-server/server/modules/providers/list/codex/codex.provider.js +17 -0
  218. package/dist-server/server/modules/providers/list/codex/codex.provider.js.map +1 -0
  219. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +118 -0
  220. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -0
  221. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js +80 -0
  222. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js.map +1 -0
  223. package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js +105 -0
  224. package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js.map +1 -0
  225. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js +545 -0
  226. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js.map +1 -0
  227. package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js +28 -0
  228. package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js.map +1 -0
  229. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js +17 -0
  230. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js.map +1 -0
  231. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +254 -0
  232. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -0
  233. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js +82 -0
  234. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js.map +1 -0
  235. package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js +312 -0
  236. package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js.map +1 -0
  237. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js +484 -0
  238. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js.map +1 -0
  239. package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js +33 -0
  240. package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js.map +1 -0
  241. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js +17 -0
  242. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js.map +1 -0
  243. package/dist-server/server/modules/providers/provider.registry.js +31 -0
  244. package/dist-server/server/modules/providers/provider.registry.js.map +1 -0
  245. package/dist-server/server/modules/providers/provider.routes.js +377 -0
  246. package/dist-server/server/modules/providers/provider.routes.js.map +1 -0
  247. package/dist-server/server/modules/providers/services/mcp.service.js +69 -0
  248. package/dist-server/server/modules/providers/services/mcp.service.js.map +1 -0
  249. package/dist-server/server/modules/providers/services/provider-auth.service.js +25 -0
  250. package/dist-server/server/modules/providers/services/provider-auth.service.js.map +1 -0
  251. package/dist-server/server/modules/providers/services/session-conversations-search.service.js +984 -0
  252. package/dist-server/server/modules/providers/services/session-conversations-search.service.js.map +1 -0
  253. package/dist-server/server/modules/providers/services/session-synchronizer.service.js +56 -0
  254. package/dist-server/server/modules/providers/services/session-synchronizer.service.js.map +1 -0
  255. package/dist-server/server/modules/providers/services/sessions-watcher.service.js +269 -0
  256. package/dist-server/server/modules/providers/services/sessions-watcher.service.js.map +1 -0
  257. package/dist-server/server/modules/providers/services/sessions.service.js +179 -0
  258. package/dist-server/server/modules/providers/services/sessions.service.js.map +1 -0
  259. package/dist-server/server/modules/providers/services/skills.service.js +11 -0
  260. package/dist-server/server/modules/providers/services/skills.service.js.map +1 -0
  261. package/dist-server/server/modules/providers/shared/base/abstract.provider.js +14 -0
  262. package/dist-server/server/modules/providers/shared/base/abstract.provider.js.map +1 -0
  263. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js +102 -0
  264. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js.map +1 -0
  265. package/dist-server/server/modules/providers/shared/skills/skills.provider.js +45 -0
  266. package/dist-server/server/modules/providers/shared/skills/skills.provider.js.map +1 -0
  267. package/dist-server/server/modules/providers/tests/mcp.test.js +250 -0
  268. package/dist-server/server/modules/providers/tests/mcp.test.js.map +1 -0
  269. package/dist-server/server/modules/providers/tests/skills.test.js +226 -0
  270. package/dist-server/server/modules/providers/tests/skills.test.js.map +1 -0
  271. package/dist-server/server/modules/websocket/index.js +3 -0
  272. package/dist-server/server/modules/websocket/index.js.map +1 -0
  273. package/dist-server/server/modules/websocket/services/chat-websocket.service.js +192 -0
  274. package/dist-server/server/modules/websocket/services/chat-websocket.service.js.map +1 -0
  275. package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js +52 -0
  276. package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js.map +1 -0
  277. package/dist-server/server/modules/websocket/services/shell-websocket.service.js +360 -0
  278. package/dist-server/server/modules/websocket/services/shell-websocket.service.js.map +1 -0
  279. package/dist-server/server/modules/websocket/services/websocket-auth.service.js +32 -0
  280. package/dist-server/server/modules/websocket/services/websocket-auth.service.js.map +1 -0
  281. package/dist-server/server/modules/websocket/services/websocket-server.service.js +36 -0
  282. package/dist-server/server/modules/websocket/services/websocket-server.service.js.map +1 -0
  283. package/dist-server/server/modules/websocket/services/websocket-state.service.js +14 -0
  284. package/dist-server/server/modules/websocket/services/websocket-state.service.js.map +1 -0
  285. package/dist-server/server/modules/websocket/services/websocket-writer.service.js +32 -0
  286. package/dist-server/server/modules/websocket/services/websocket-writer.service.js.map +1 -0
  287. package/dist-server/server/openai-codex.js +418 -0
  288. package/dist-server/server/openai-codex.js.map +1 -0
  289. package/dist-server/server/routes/admin.js +109 -0
  290. package/dist-server/server/routes/admin.js.map +1 -0
  291. package/dist-server/server/routes/agent.js +1145 -0
  292. package/dist-server/server/routes/agent.js.map +1 -0
  293. package/dist-server/server/routes/auth.js +123 -0
  294. package/dist-server/server/routes/auth.js.map +1 -0
  295. package/dist-server/server/routes/commands.js +487 -0
  296. package/dist-server/server/routes/commands.js.map +1 -0
  297. package/dist-server/server/routes/cursor.js +49 -0
  298. package/dist-server/server/routes/cursor.js.map +1 -0
  299. package/dist-server/server/routes/gemini.js +25 -0
  300. package/dist-server/server/routes/gemini.js.map +1 -0
  301. package/dist-server/server/routes/git.js +1263 -0
  302. package/dist-server/server/routes/git.js.map +1 -0
  303. package/dist-server/server/routes/mcp-utils.js +29 -0
  304. package/dist-server/server/routes/mcp-utils.js.map +1 -0
  305. package/dist-server/server/routes/plugins.js +266 -0
  306. package/dist-server/server/routes/plugins.js.map +1 -0
  307. package/dist-server/server/routes/settings.js +259 -0
  308. package/dist-server/server/routes/settings.js.map +1 -0
  309. package/dist-server/server/routes/taskmaster.js +1360 -0
  310. package/dist-server/server/routes/taskmaster.js.map +1 -0
  311. package/dist-server/server/routes/user.js +115 -0
  312. package/dist-server/server/routes/user.js.map +1 -0
  313. package/dist-server/server/services/notification-orchestrator.js +177 -0
  314. package/dist-server/server/services/notification-orchestrator.js.map +1 -0
  315. package/dist-server/server/services/vapid-keys.js +27 -0
  316. package/dist-server/server/services/vapid-keys.js.map +1 -0
  317. package/dist-server/server/sessionManager.js +215 -0
  318. package/dist-server/server/sessionManager.js.map +1 -0
  319. package/dist-server/server/shared/claude-cli-path.js +103 -0
  320. package/dist-server/server/shared/claude-cli-path.js.map +1 -0
  321. package/dist-server/server/shared/claude-cli-path.test.js +45 -0
  322. package/dist-server/server/shared/claude-cli-path.test.js.map +1 -0
  323. package/dist-server/server/shared/default-user.js +29 -0
  324. package/dist-server/server/shared/default-user.js.map +1 -0
  325. package/dist-server/server/shared/frontmatter.js +16 -0
  326. package/dist-server/server/shared/frontmatter.js.map +1 -0
  327. package/dist-server/server/shared/interfaces.js +2 -0
  328. package/dist-server/server/shared/interfaces.js.map +1 -0
  329. package/dist-server/server/shared/types.js +2 -0
  330. package/dist-server/server/shared/types.js.map +1 -0
  331. package/dist-server/server/shared/utils.js +633 -0
  332. package/dist-server/server/shared/utils.js.map +1 -0
  333. package/dist-server/server/utils/colors.js +20 -0
  334. package/dist-server/server/utils/colors.js.map +1 -0
  335. package/dist-server/server/utils/commandParser.js +255 -0
  336. package/dist-server/server/utils/commandParser.js.map +1 -0
  337. package/dist-server/server/utils/gitConfig.js +36 -0
  338. package/dist-server/server/utils/gitConfig.js.map +1 -0
  339. package/dist-server/server/utils/mcp-detector.js +134 -0
  340. package/dist-server/server/utils/mcp-detector.js.map +1 -0
  341. package/dist-server/server/utils/plugin-loader.js +413 -0
  342. package/dist-server/server/utils/plugin-loader.js.map +1 -0
  343. package/dist-server/server/utils/plugin-process-manager.js +163 -0
  344. package/dist-server/server/utils/plugin-process-manager.js.map +1 -0
  345. package/dist-server/server/utils/runtime-paths.js +30 -0
  346. package/dist-server/server/utils/runtime-paths.js.map +1 -0
  347. package/dist-server/server/utils/taskmaster-websocket.js +124 -0
  348. package/dist-server/server/utils/taskmaster-websocket.js.map +1 -0
  349. package/dist-server/server/utils/url-detection.js +58 -0
  350. package/dist-server/server/utils/url-detection.js.map +1 -0
  351. package/dist-server/shared/modelConstants.js +99 -0
  352. package/dist-server/shared/modelConstants.js.map +1 -0
  353. package/dist-server/shared/networkHosts.js +20 -0
  354. package/dist-server/shared/networkHosts.js.map +1 -0
  355. package/package.json +169 -0
  356. package/scripts/fix-node-pty.js +67 -0
  357. package/server/claude-sdk.js +864 -0
  358. package/server/cli.js +688 -0
  359. package/server/constants/config.js +5 -0
  360. package/server/cursor-cli.js +334 -0
  361. package/server/gemini-cli.js +622 -0
  362. package/server/gemini-response-handler.js +79 -0
  363. package/server/index.js +1505 -0
  364. package/server/load-env.js +34 -0
  365. package/server/middleware/auth.js +142 -0
  366. package/server/modules/database/connection.ts +143 -0
  367. package/server/modules/database/index.ts +12 -0
  368. package/server/modules/database/init-db.ts +17 -0
  369. package/server/modules/database/migrations.ts +496 -0
  370. package/server/modules/database/repositories/api-keys.ts +119 -0
  371. package/server/modules/database/repositories/app-config.ts +53 -0
  372. package/server/modules/database/repositories/credentials.ts +106 -0
  373. package/server/modules/database/repositories/github-tokens.ts +100 -0
  374. package/server/modules/database/repositories/notification-preferences.ts +103 -0
  375. package/server/modules/database/repositories/projects.db.integration.test.ts +78 -0
  376. package/server/modules/database/repositories/projects.db.ts +210 -0
  377. package/server/modules/database/repositories/push-subscriptions.ts +80 -0
  378. package/server/modules/database/repositories/scan-state.db.ts +42 -0
  379. package/server/modules/database/repositories/sessions.db.integration.test.ts +78 -0
  380. package/server/modules/database/repositories/sessions.db.ts +230 -0
  381. package/server/modules/database/repositories/users.ts +186 -0
  382. package/server/modules/database/repositories/vapid-keys.ts +57 -0
  383. package/server/modules/database/schema.ts +159 -0
  384. package/server/modules/projects/index.ts +6 -0
  385. package/server/modules/projects/projects.routes.ts +292 -0
  386. package/server/modules/projects/services/project-clone.service.ts +327 -0
  387. package/server/modules/projects/services/project-delete.service.ts +95 -0
  388. package/server/modules/projects/services/project-management.service.ts +158 -0
  389. package/server/modules/projects/services/project-star.service.ts +78 -0
  390. package/server/modules/projects/services/projects-has-taskmaster.service.ts +257 -0
  391. package/server/modules/projects/services/projects-with-sessions-fetch.service.ts +355 -0
  392. package/server/modules/projects/tests/project-clone.service.test.ts +186 -0
  393. package/server/modules/projects/tests/project-management.service.test.ts +122 -0
  394. package/server/modules/projects/tests/project-star.service.test.ts +128 -0
  395. package/server/modules/projects/tests/projects-has-taskmaster.service.test.ts +107 -0
  396. package/server/modules/providers/README.md +346 -0
  397. package/server/modules/providers/index.ts +5 -0
  398. package/server/modules/providers/list/claude/claude-auth.provider.ts +124 -0
  399. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -0
  400. package/server/modules/providers/list/claude/claude-session-synchronizer.provider.ts +179 -0
  401. package/server/modules/providers/list/claude/claude-sessions.provider.ts +642 -0
  402. package/server/modules/providers/list/claude/claude-skills.provider.ts +257 -0
  403. package/server/modules/providers/list/claude/claude.provider.ts +24 -0
  404. package/server/modules/providers/list/codex/codex-auth.provider.ts +100 -0
  405. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -0
  406. package/server/modules/providers/list/codex/codex-session-synchronizer.provider.ts +182 -0
  407. package/server/modules/providers/list/codex/codex-sessions.provider.ts +589 -0
  408. package/server/modules/providers/list/codex/codex-skills.provider.ts +100 -0
  409. package/server/modules/providers/list/codex/codex.provider.ts +24 -0
  410. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -0
  411. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -0
  412. package/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.ts +155 -0
  413. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +624 -0
  414. package/server/modules/providers/list/cursor/cursor-skills.provider.ts +31 -0
  415. package/server/modules/providers/list/cursor/cursor.provider.ts +24 -0
  416. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +307 -0
  417. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -0
  418. package/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.ts +407 -0
  419. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +552 -0
  420. package/server/modules/providers/list/gemini/gemini-skills.provider.ts +36 -0
  421. package/server/modules/providers/list/gemini/gemini.provider.ts +24 -0
  422. package/server/modules/providers/provider.registry.ts +36 -0
  423. package/server/modules/providers/provider.routes.ts +488 -0
  424. package/server/modules/providers/services/mcp.service.ts +94 -0
  425. package/server/modules/providers/services/provider-auth.service.ts +26 -0
  426. package/server/modules/providers/services/session-conversations-search.service.ts +1319 -0
  427. package/server/modules/providers/services/session-synchronizer.service.ts +75 -0
  428. package/server/modules/providers/services/sessions-watcher.service.ts +318 -0
  429. package/server/modules/providers/services/sessions.service.ts +240 -0
  430. package/server/modules/providers/services/skills.service.ts +15 -0
  431. package/server/modules/providers/shared/base/abstract.provider.ts +29 -0
  432. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -0
  433. package/server/modules/providers/shared/skills/skills.provider.ts +64 -0
  434. package/server/modules/providers/tests/mcp.test.ts +293 -0
  435. package/server/modules/providers/tests/skills.test.ts +446 -0
  436. package/server/modules/websocket/README.md +267 -0
  437. package/server/modules/websocket/index.ts +2 -0
  438. package/server/modules/websocket/services/chat-websocket.service.ts +275 -0
  439. package/server/modules/websocket/services/plugin-websocket-proxy.service.ts +65 -0
  440. package/server/modules/websocket/services/shell-websocket.service.ts +489 -0
  441. package/server/modules/websocket/services/websocket-auth.service.ts +54 -0
  442. package/server/modules/websocket/services/websocket-server.service.ts +58 -0
  443. package/server/modules/websocket/services/websocket-state.service.ts +16 -0
  444. package/server/modules/websocket/services/websocket-writer.service.ts +38 -0
  445. package/server/openai-codex.js +474 -0
  446. package/server/routes/admin.js +128 -0
  447. package/server/routes/agent.js +1246 -0
  448. package/server/routes/auth.js +144 -0
  449. package/server/routes/commands.js +556 -0
  450. package/server/routes/cursor.js +52 -0
  451. package/server/routes/gemini.js +30 -0
  452. package/server/routes/git.js +1493 -0
  453. package/server/routes/mcp-utils.js +31 -0
  454. package/server/routes/plugins.js +307 -0
  455. package/server/routes/settings.js +286 -0
  456. package/server/routes/taskmaster.js +1468 -0
  457. package/server/routes/user.js +123 -0
  458. package/server/services/notification-orchestrator.js +228 -0
  459. package/server/services/vapid-keys.js +36 -0
  460. package/server/sessionManager.js +248 -0
  461. package/server/shared/claude-cli-path.test.ts +61 -0
  462. package/server/shared/claude-cli-path.ts +139 -0
  463. package/server/shared/default-user.ts +30 -0
  464. package/server/shared/frontmatter.ts +18 -0
  465. package/server/shared/interfaces.ts +111 -0
  466. package/server/shared/types.ts +406 -0
  467. package/server/shared/utils.ts +763 -0
  468. package/server/tsconfig.json +36 -0
  469. package/server/utils/colors.js +21 -0
  470. package/server/utils/commandParser.js +305 -0
  471. package/server/utils/gitConfig.js +34 -0
  472. package/server/utils/mcp-detector.js +147 -0
  473. package/server/utils/plugin-loader.js +457 -0
  474. package/server/utils/plugin-process-manager.js +184 -0
  475. package/server/utils/runtime-paths.js +37 -0
  476. package/server/utils/taskmaster-websocket.js +135 -0
  477. package/server/utils/url-detection.js +71 -0
  478. package/shared/modelConstants.js +107 -0
  479. package/shared/networkHosts.js +22 -0
@@ -0,0 +1,622 @@
1
+ import { spawn } from 'child_process';
2
+ import { promises as fs } from 'fs';
3
+ import os from 'os';
4
+ import path from 'path';
5
+
6
+ import crossSpawn from 'cross-spawn';
7
+
8
+ import sessionManager from './sessionManager.js';
9
+ import GeminiResponseHandler from './gemini-response-handler.js';
10
+ import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
11
+ import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
12
+ import { createNormalizedMessage } from './shared/utils.js';
13
+
14
+ // Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
15
+ const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
16
+
17
+ let activeGeminiProcesses = new Map(); // Track active processes by session ID
18
+
19
+ function mapGeminiExitCodeToMessage(exitCode) {
20
+ switch (exitCode) {
21
+ case 42:
22
+ return 'Gemini rejected the request input (exit code 42).';
23
+ case 44:
24
+ return 'Gemini sandbox error (exit code 44). Check local sandbox/container settings.';
25
+ case 52:
26
+ return 'Gemini configuration error (exit code 52). Check your Gemini settings files for invalid JSON/config.';
27
+ case 53:
28
+ return 'Gemini conversation turn limit reached (exit code 53). Start a new Gemini session.';
29
+ default:
30
+ return null;
31
+ }
32
+ }
33
+
34
+ const GEMINI_AUTH_ENV_KEYS = [
35
+ 'GEMINI_API_KEY',
36
+ 'GOOGLE_API_KEY',
37
+ 'GOOGLE_CLOUD_PROJECT',
38
+ 'GOOGLE_CLOUD_PROJECT_ID',
39
+ 'GOOGLE_CLOUD_LOCATION',
40
+ 'GOOGLE_APPLICATION_CREDENTIALS'
41
+ ];
42
+
43
+ function parseEnvFileContent(content) {
44
+ const parsed = {};
45
+
46
+ for (const rawLine of content.split(/\r?\n/)) {
47
+ const line = rawLine.trim();
48
+ if (!line || line.startsWith('#')) {
49
+ continue;
50
+ }
51
+
52
+ const exportPrefix = 'export ';
53
+ const normalizedLine = line.startsWith(exportPrefix) ? line.slice(exportPrefix.length).trim() : line;
54
+ const separatorIndex = normalizedLine.indexOf('=');
55
+
56
+ if (separatorIndex <= 0) {
57
+ continue;
58
+ }
59
+
60
+ const key = normalizedLine.slice(0, separatorIndex).trim();
61
+ if (!key) {
62
+ continue;
63
+ }
64
+
65
+ let value = normalizedLine.slice(separatorIndex + 1).trim();
66
+ const hasDoubleQuotes = value.startsWith('"') && value.endsWith('"');
67
+ const hasSingleQuotes = value.startsWith('\'') && value.endsWith('\'');
68
+
69
+ if (hasDoubleQuotes || hasSingleQuotes) {
70
+ value = value.slice(1, -1);
71
+ } else {
72
+ // Support inline comments in unquoted values: KEY=value # comment
73
+ value = value.replace(/\s+#.*$/, '').trim();
74
+ }
75
+
76
+ parsed[key] = value;
77
+ }
78
+
79
+ return parsed;
80
+ }
81
+
82
+ async function loadGeminiUserLevelEnv() {
83
+ const geminiCliHome = (process.env.GEMINI_CLI_HOME || '').trim() || os.homedir();
84
+ const envCandidates = [
85
+ path.join(geminiCliHome, '.gemini', '.env'),
86
+ path.join(geminiCliHome, '.env')
87
+ ];
88
+
89
+ for (const envPath of envCandidates) {
90
+ try {
91
+ await fs.access(envPath);
92
+ const content = await fs.readFile(envPath, 'utf8');
93
+ return parseEnvFileContent(content);
94
+ } catch {
95
+ // Keep scanning for the next candidate.
96
+ }
97
+ }
98
+
99
+ return {};
100
+ }
101
+
102
+ async function buildGeminiProcessEnv() {
103
+ const processEnv = { ...process.env };
104
+ if (processEnv.GEMINI_API_KEY || processEnv.GOOGLE_API_KEY || processEnv.GOOGLE_APPLICATION_CREDENTIALS) {
105
+ return processEnv;
106
+ }
107
+
108
+ // Gemini CLI docs recommend ~/.gemini/.env for persistent headless auth settings.
109
+ // When the server process was launched without shell profile variables, we still
110
+ // want the spawned CLI process to inherit those user-level credentials.
111
+ const userEnv = await loadGeminiUserLevelEnv();
112
+ for (const key of GEMINI_AUTH_ENV_KEYS) {
113
+ if (!processEnv[key] && userEnv[key]) {
114
+ processEnv[key] = userEnv[key];
115
+ }
116
+ }
117
+
118
+ return processEnv;
119
+ }
120
+
121
+ async function spawnGemini(command, options = {}, ws) {
122
+ const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
123
+ const userId = ws?.userId ?? null;
124
+ let capturedSessionId = sessionId; // Track session ID throughout the process
125
+ let sessionCreatedSent = false; // Track if we've already sent session-created event
126
+ let assistantBlocks = []; // Accumulate the full response blocks including tools
127
+
128
+ // Use tools settings passed from frontend, or defaults
129
+ const settings = toolsSettings || {
130
+ allowedTools: [],
131
+ disallowedTools: [],
132
+ skipPermissions: false
133
+ };
134
+
135
+ // Build Gemini CLI command - start with print/resume flags first
136
+ const args = [];
137
+
138
+ // Add prompt flag with command if we have a command
139
+ if (command && command.trim()) {
140
+ args.push('--prompt', command);
141
+ }
142
+
143
+ // If we have a sessionId, we want to resume
144
+ if (sessionId) {
145
+ const session = userId !== null ? sessionManager.getSession(userId, sessionId) : null;
146
+ if (session && session.cliSessionId) {
147
+ args.push('--resume', session.cliSessionId);
148
+ }
149
+ }
150
+
151
+ // Use cwd (actual project directory) instead of projectPath (Gemini's metadata directory)
152
+ // Clean the path by removing any non-printable characters
153
+ const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
154
+ const workingDir = cleanPath;
155
+
156
+ // Handle images by saving them to temporary files and passing paths to Gemini
157
+ const tempImagePaths = [];
158
+ let tempDir = null;
159
+ if (images && images.length > 0) {
160
+ try {
161
+ // Create temp directory in the project directory so Gemini can access it
162
+ tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
163
+ await fs.mkdir(tempDir, { recursive: true });
164
+
165
+ // Save each image to a temp file
166
+ for (const [index, image] of images.entries()) {
167
+ // Extract base64 data and mime type
168
+ const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
169
+ if (!matches) {
170
+ continue;
171
+ }
172
+
173
+ const [, mimeType, base64Data] = matches;
174
+ const extension = mimeType.split('/')[1] || 'png';
175
+ const filename = `image_${index}.${extension}`;
176
+ const filepath = path.join(tempDir, filename);
177
+
178
+ // Write base64 data to file
179
+ await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
180
+ tempImagePaths.push(filepath);
181
+ }
182
+
183
+ // Include the full image paths in the prompt for Gemini to reference
184
+ // Gemini CLI can read images from file paths in the prompt
185
+ if (tempImagePaths.length > 0 && command && command.trim()) {
186
+ 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')}`;
187
+ const modifiedCommand = command + imageNote;
188
+
189
+ // Update the command in args
190
+ const promptIndex = args.indexOf('--prompt');
191
+ if (promptIndex !== -1 && args[promptIndex + 1] === command) {
192
+ args[promptIndex + 1] = modifiedCommand;
193
+ } else if (promptIndex !== -1) {
194
+ // If we're using context, update the full prompt
195
+ args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
196
+ }
197
+ }
198
+ } catch (error) {
199
+ console.error('Error processing images for Gemini:', error);
200
+ }
201
+ }
202
+
203
+ // Add basic flags for Gemini
204
+ if (options.debug) {
205
+ args.push('--debug');
206
+ }
207
+
208
+ // This integration runs Gemini in headless mode and cannot answer trust prompts.
209
+ // Skip folder-trust interactivity so authenticated runs don't fail with
210
+ // FatalUntrustedWorkspaceError in previously unseen directories.
211
+ args.push('--skip-trust');
212
+
213
+ // Add MCP config flag only if MCP servers are configured
214
+ try {
215
+ const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
216
+ let hasMcpServers = false;
217
+
218
+ try {
219
+ await fs.access(geminiConfigPath);
220
+ const geminiConfigRaw = await fs.readFile(geminiConfigPath, 'utf8');
221
+ const geminiConfig = JSON.parse(geminiConfigRaw);
222
+
223
+ // Check global MCP servers
224
+ if (geminiConfig.mcpServers && Object.keys(geminiConfig.mcpServers).length > 0) {
225
+ hasMcpServers = true;
226
+ }
227
+
228
+ // Check project-specific MCP servers
229
+ if (!hasMcpServers && geminiConfig.geminiProjects) {
230
+ const currentProjectPath = process.cwd();
231
+ const projectConfig = geminiConfig.geminiProjects[currentProjectPath];
232
+ if (projectConfig && projectConfig.mcpServers && Object.keys(projectConfig.mcpServers).length > 0) {
233
+ hasMcpServers = true;
234
+ }
235
+ }
236
+ } catch (e) {
237
+ // Ignore if file doesn't exist or isn't parsable
238
+ }
239
+
240
+ if (hasMcpServers) {
241
+ args.push('--mcp-config', geminiConfigPath);
242
+ }
243
+ } catch (error) {
244
+ // Ignore outer errors
245
+ }
246
+
247
+ // Add model for all sessions (both new and resumed)
248
+ let modelToUse = options.model || 'gemini-2.5-flash';
249
+ args.push('--model', modelToUse);
250
+ args.push('--output-format', 'stream-json');
251
+
252
+ // Handle approval modes and allowed tools
253
+ if (settings.skipPermissions || options.skipPermissions || permissionMode === 'yolo') {
254
+ args.push('--yolo');
255
+ } else if (permissionMode === 'auto_edit') {
256
+ args.push('--approval-mode', 'auto_edit');
257
+ } else if (permissionMode === 'plan') {
258
+ args.push('--approval-mode', 'plan');
259
+ }
260
+
261
+ if (settings.allowedTools && settings.allowedTools.length > 0) {
262
+ args.push('--allowed-tools', settings.allowedTools.join(','));
263
+ }
264
+
265
+ // Try to find gemini in PATH first, then fall back to environment variable
266
+ const geminiPath = process.env.GEMINI_PATH || 'gemini';
267
+ let spawnCmd = geminiPath;
268
+ let spawnArgs = args;
269
+
270
+ // On non-Windows platforms, wrap the execution in a shell to avoid ENOEXEC
271
+ // which happens when the target is a script lacking a shebang.
272
+ if (os.platform() !== 'win32') {
273
+ spawnCmd = 'sh';
274
+ // Use exec to replace the shell process, ensuring signals hit gemini directly
275
+ spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
276
+ }
277
+
278
+ const spawnEnv = await buildGeminiProcessEnv();
279
+
280
+ return new Promise((resolve, reject) => {
281
+ const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
282
+ cwd: workingDir,
283
+ stdio: ['pipe', 'pipe', 'pipe'],
284
+ env: spawnEnv
285
+ });
286
+ let terminalNotificationSent = false;
287
+ let terminalFailureReason = null;
288
+
289
+ const notifyTerminalState = ({ code = null, error = null } = {}) => {
290
+ if (terminalNotificationSent) {
291
+ return;
292
+ }
293
+
294
+ terminalNotificationSent = true;
295
+
296
+ const finalSessionId = capturedSessionId || sessionId || processKey;
297
+ if (code === 0 && !error) {
298
+ notifyRunStopped({
299
+ userId: ws?.userId || null,
300
+ provider: 'gemini',
301
+ sessionId: finalSessionId,
302
+ sessionName: sessionSummary,
303
+ stopReason: 'completed'
304
+ });
305
+ return;
306
+ }
307
+
308
+ notifyRunFailed({
309
+ userId: ws?.userId || null,
310
+ provider: 'gemini',
311
+ sessionId: finalSessionId,
312
+ sessionName: sessionSummary,
313
+ error: error || terminalFailureReason || `Gemini CLI exited with code ${code}`
314
+ });
315
+ };
316
+
317
+ // Attach temp file info to process for cleanup later
318
+ geminiProcess.tempImagePaths = tempImagePaths;
319
+ geminiProcess.tempDir = tempDir;
320
+
321
+ // Store process reference for potential abort
322
+ const processKey = capturedSessionId || sessionId || Date.now().toString();
323
+ activeGeminiProcesses.set(processKey, geminiProcess);
324
+
325
+ // Store sessionId on the process object for debugging
326
+ geminiProcess.sessionId = processKey;
327
+
328
+ // Close stdin to signal we're done sending input
329
+ geminiProcess.stdin.end();
330
+
331
+ // Add timeout handler
332
+ const timeoutMs = 120000; // 120 seconds for slower models
333
+ let timeout;
334
+
335
+ const startTimeout = () => {
336
+ if (timeout) clearTimeout(timeout);
337
+ timeout = setTimeout(() => {
338
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
339
+ terminalFailureReason = `Gemini CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
340
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
341
+ try {
342
+ geminiProcess.kill('SIGTERM');
343
+ } catch (e) { }
344
+ }, timeoutMs);
345
+ };
346
+
347
+ startTimeout();
348
+
349
+ // Save user message to session when starting
350
+ if (command && capturedSessionId && userId !== null) {
351
+ sessionManager.addMessage(userId, capturedSessionId, 'user', command);
352
+ }
353
+
354
+ // Create response handler for NDJSON buffering
355
+ let responseHandler;
356
+ if (ws) {
357
+ responseHandler = new GeminiResponseHandler(ws, {
358
+ onContentFragment: (content) => {
359
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
360
+ assistantBlocks[assistantBlocks.length - 1].text += content;
361
+ } else {
362
+ assistantBlocks.push({ type: 'text', text: content });
363
+ }
364
+ },
365
+ onToolUse: (event) => {
366
+ assistantBlocks.push({
367
+ type: 'tool_use',
368
+ id: event.tool_id,
369
+ name: event.tool_name,
370
+ input: event.parameters
371
+ });
372
+ },
373
+ onToolResult: (event) => {
374
+ if (capturedSessionId && userId !== null) {
375
+ if (assistantBlocks.length > 0) {
376
+ sessionManager.addMessage(userId, capturedSessionId, 'assistant', [...assistantBlocks]);
377
+ assistantBlocks = [];
378
+ }
379
+ sessionManager.addMessage(userId, capturedSessionId, 'user', [{
380
+ type: 'tool_result',
381
+ tool_use_id: event.tool_id,
382
+ content: event.output === undefined ? null : event.output,
383
+ is_error: event.status === 'error'
384
+ }]);
385
+ }
386
+ },
387
+ onInit: (event) => {
388
+ const discoveredSessionId = event?.session_id;
389
+ if (!discoveredSessionId) {
390
+ return;
391
+ }
392
+
393
+ // New Gemini sessions announce their canonical ID asynchronously via the
394
+ // initial `init` stream event. Avoid synthetic IDs and only register
395
+ // the session once that real ID is known (same model used by Claude/Codex).
396
+ if (!capturedSessionId) {
397
+ capturedSessionId = discoveredSessionId;
398
+
399
+ if (userId !== null) {
400
+ sessionManager.createSession(userId, capturedSessionId, cwd || process.cwd());
401
+ if (command) {
402
+ sessionManager.addMessage(userId, capturedSessionId, 'user', command);
403
+ }
404
+ }
405
+
406
+ if (processKey !== capturedSessionId) {
407
+ activeGeminiProcesses.delete(processKey);
408
+ activeGeminiProcesses.set(capturedSessionId, geminiProcess);
409
+ }
410
+
411
+ geminiProcess.sessionId = capturedSessionId;
412
+
413
+ if (ws.setSessionId && typeof ws.setSessionId === 'function') {
414
+ ws.setSessionId(capturedSessionId);
415
+ }
416
+
417
+ if (!sessionId && !sessionCreatedSent) {
418
+ sessionCreatedSent = true;
419
+ ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
420
+ }
421
+ }
422
+
423
+ if (userId !== null) {
424
+ const sess = sessionManager.getSession(userId, capturedSessionId);
425
+ if (sess && !sess.cliSessionId) {
426
+ sess.cliSessionId = discoveredSessionId;
427
+ sessionManager.saveSession(userId, capturedSessionId);
428
+ }
429
+ }
430
+ }
431
+ });
432
+ }
433
+
434
+ // Handle stdout
435
+ geminiProcess.stdout.on('data', (data) => {
436
+ const rawOutput = data.toString();
437
+ startTimeout(); // Re-arm the timeout
438
+
439
+ if (responseHandler) {
440
+ responseHandler.processData(rawOutput);
441
+ } else if (rawOutput) {
442
+ // Fallback to direct sending for raw CLI mode without WS
443
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
444
+ assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
445
+ } else {
446
+ assistantBlocks.push({ type: 'text', text: rawOutput });
447
+ }
448
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
449
+ ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'gemini' }));
450
+ }
451
+ });
452
+
453
+ // Handle stderr
454
+ geminiProcess.stderr.on('data', (data) => {
455
+ const errorMsg = data.toString();
456
+
457
+ // Filter out deprecation warnings and "Loaded cached credentials" message
458
+ if (errorMsg.includes('[DEP0040]') ||
459
+ errorMsg.includes('DeprecationWarning') ||
460
+ errorMsg.includes('--trace-deprecation') ||
461
+ errorMsg.includes('Loaded cached credentials')) {
462
+ return;
463
+ }
464
+
465
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
466
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'gemini' }));
467
+ });
468
+
469
+ // Handle process completion
470
+ geminiProcess.on('close', async (code) => {
471
+ clearTimeout(timeout);
472
+
473
+ // Flush any remaining buffered content
474
+ if (responseHandler) {
475
+ responseHandler.forceFlush();
476
+ responseHandler.destroy();
477
+ }
478
+
479
+ // Clean up process reference
480
+ const finalSessionId = capturedSessionId || sessionId || processKey;
481
+ activeGeminiProcesses.delete(finalSessionId);
482
+
483
+ // Save assistant response to session if we have one
484
+ if (finalSessionId && assistantBlocks.length > 0 && userId !== null) {
485
+ sessionManager.addMessage(userId, finalSessionId, 'assistant', assistantBlocks);
486
+ }
487
+
488
+ ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'gemini' }));
489
+
490
+ // Clean up temporary image files if any
491
+ if (geminiProcess.tempImagePaths && geminiProcess.tempImagePaths.length > 0) {
492
+ for (const imagePath of geminiProcess.tempImagePaths) {
493
+ await fs.unlink(imagePath).catch(err => { });
494
+ }
495
+ if (geminiProcess.tempDir) {
496
+ await fs.rm(geminiProcess.tempDir, { recursive: true, force: true }).catch(err => { });
497
+ }
498
+ }
499
+
500
+ if (code === 0) {
501
+ notifyTerminalState({ code });
502
+ resolve();
503
+ } else {
504
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
505
+
506
+ // code 127 = shell "command not found" - check installation
507
+ if (code === 127) {
508
+ const installed = await providerAuthService.isProviderInstalled('gemini');
509
+ if (!installed) {
510
+ terminalFailureReason = 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli';
511
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
512
+ }
513
+ } else if (code === 41) {
514
+ // Gemini CLI documents exit code 41 as FatalAuthenticationError.
515
+ // Surface an actionable auth error instead of a generic exit-code message.
516
+ let authErrorSuffix = '';
517
+ try {
518
+ const authStatus = await providerAuthService.getProviderAuthStatus('gemini');
519
+ if (!authStatus?.authenticated && authStatus?.error) {
520
+ authErrorSuffix = ` Details: ${authStatus.error}`;
521
+ }
522
+ } catch {
523
+ // Keep base remediation text when auth status lookup fails.
524
+ }
525
+
526
+ terminalFailureReason =
527
+ 'Gemini authentication failed (exit code 41). '
528
+ + 'Run `gemini` in a terminal to choose an auth method, or configure a valid `GEMINI_API_KEY`.'
529
+ + authErrorSuffix;
530
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
531
+ } else {
532
+ const mappedError = mapGeminiExitCodeToMessage(code);
533
+ if (mappedError) {
534
+ terminalFailureReason = mappedError;
535
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
536
+ }
537
+ }
538
+
539
+ notifyTerminalState({
540
+ code,
541
+ error: code === null ? 'Gemini CLI process was terminated or timed out' : null
542
+ });
543
+ reject(
544
+ new Error(
545
+ terminalFailureReason
546
+ || (code === null
547
+ ? 'Gemini CLI process was terminated or timed out'
548
+ : `Gemini CLI exited with code ${code}`)
549
+ )
550
+ );
551
+ }
552
+ });
553
+
554
+ // Handle process errors
555
+ geminiProcess.on('error', async (error) => {
556
+ // Clean up process reference on error
557
+ const finalSessionId = capturedSessionId || sessionId || processKey;
558
+ activeGeminiProcesses.delete(finalSessionId);
559
+
560
+ // Check if Gemini CLI is installed for a clearer error message
561
+ const installed = await providerAuthService.isProviderInstalled('gemini');
562
+ const errorContent = !installed
563
+ ? 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli'
564
+ : error.message;
565
+
566
+ const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
567
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'gemini' }));
568
+ notifyTerminalState({ error });
569
+
570
+ reject(error);
571
+ });
572
+
573
+ });
574
+ }
575
+
576
+ function abortGeminiSession(sessionId) {
577
+ let geminiProc = activeGeminiProcesses.get(sessionId);
578
+ let processKey = sessionId;
579
+
580
+ if (!geminiProc) {
581
+ for (const [key, proc] of activeGeminiProcesses.entries()) {
582
+ if (proc.sessionId === sessionId) {
583
+ geminiProc = proc;
584
+ processKey = key;
585
+ break;
586
+ }
587
+ }
588
+ }
589
+
590
+ if (geminiProc) {
591
+ try {
592
+ geminiProc.kill('SIGTERM');
593
+ setTimeout(() => {
594
+ if (activeGeminiProcesses.has(processKey)) {
595
+ try {
596
+ geminiProc.kill('SIGKILL');
597
+ } catch (e) { }
598
+ }
599
+ }, 2000); // Wait 2 seconds before force kill
600
+
601
+ return true;
602
+ } catch (error) {
603
+ return false;
604
+ }
605
+ }
606
+ return false;
607
+ }
608
+
609
+ function isGeminiSessionActive(sessionId) {
610
+ return activeGeminiProcesses.has(sessionId);
611
+ }
612
+
613
+ function getActiveGeminiSessions() {
614
+ return Array.from(activeGeminiProcesses.keys());
615
+ }
616
+
617
+ export {
618
+ spawnGemini,
619
+ abortGeminiSession,
620
+ isGeminiSessionActive,
621
+ getActiveGeminiSessions
622
+ };