@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,1246 @@
1
+ import express from 'express';
2
+ import { spawn } from 'child_process';
3
+ import path from 'path';
4
+ import os from 'os';
5
+ import { promises as fs } from 'fs';
6
+ import crypto from 'crypto';
7
+ import { userDb, apiKeysDb, githubTokensDb, projectsDb } from '../modules/database/index.js';
8
+ import { queryClaudeSDK } from '../claude-sdk.js';
9
+ import { spawnCursor } from '../cursor-cli.js';
10
+ import { queryCodex } from '../openai-codex.js';
11
+ import { spawnGemini } from '../gemini-cli.js';
12
+ import { Octokit } from '@octokit/rest';
13
+ import { CLAUDE_MODELS, CURSOR_MODELS, CODEX_MODELS } from '../../shared/modelConstants.js';
14
+ import { IS_PLATFORM } from '../constants/config.js';
15
+ import { normalizeProjectPath } from '../shared/utils.js';
16
+
17
+ const router = express.Router();
18
+
19
+ /**
20
+ * Middleware to authenticate agent API requests.
21
+ *
22
+ * Supports two authentication modes:
23
+ * 1. Platform mode (IS_PLATFORM=true): For managed/hosted deployments where
24
+ * authentication is handled by an external proxy. Requests are trusted and
25
+ * the default user context is used.
26
+ *
27
+ * 2. API key mode (default): For self-hosted deployments where users authenticate
28
+ * via API keys created in the UI. Keys are validated against the local database.
29
+ */
30
+ const validateExternalApiKey = (req, res, next) => {
31
+ // Platform mode: Authentication is handled externally (e.g., by a proxy layer).
32
+ // Trust the request and use the default user context.
33
+ if (IS_PLATFORM) {
34
+ try {
35
+ const user = userDb.getFirstUser();
36
+ if (!user) {
37
+ return res.status(500).json({ error: 'Platform mode: No user found in database' });
38
+ }
39
+ req.user = user;
40
+ return next();
41
+ } catch (error) {
42
+ console.error('Platform mode error:', error);
43
+ return res.status(500).json({ error: 'Platform mode: Failed to fetch user' });
44
+ }
45
+ }
46
+
47
+ // Self-hosted mode: Validate API key from header or query parameter
48
+ const apiKey = req.headers['x-api-key'] || req.query.apiKey;
49
+
50
+ if (!apiKey) {
51
+ return res.status(401).json({ error: 'API key required' });
52
+ }
53
+
54
+ const user = apiKeysDb.validateApiKey(apiKey);
55
+
56
+ if (!user) {
57
+ return res.status(401).json({ error: 'Invalid or inactive API key' });
58
+ }
59
+
60
+ req.user = user;
61
+ next();
62
+ };
63
+
64
+ /**
65
+ * Get the remote URL of a git repository
66
+ * @param {string} repoPath - Path to the git repository
67
+ * @returns {Promise<string>} - Remote URL of the repository
68
+ */
69
+ async function getGitRemoteUrl(repoPath) {
70
+ return new Promise((resolve, reject) => {
71
+ const gitProcess = spawn('git', ['config', '--get', 'remote.origin.url'], {
72
+ cwd: repoPath,
73
+ stdio: ['pipe', 'pipe', 'pipe']
74
+ });
75
+
76
+ let stdout = '';
77
+ let stderr = '';
78
+
79
+ gitProcess.stdout.on('data', (data) => {
80
+ stdout += data.toString();
81
+ });
82
+
83
+ gitProcess.stderr.on('data', (data) => {
84
+ stderr += data.toString();
85
+ });
86
+
87
+ gitProcess.on('close', (code) => {
88
+ if (code === 0) {
89
+ resolve(stdout.trim());
90
+ } else {
91
+ reject(new Error(`Failed to get git remote: ${stderr}`));
92
+ }
93
+ });
94
+
95
+ gitProcess.on('error', (error) => {
96
+ reject(new Error(`Failed to execute git: ${error.message}`));
97
+ });
98
+ });
99
+ }
100
+
101
+ /**
102
+ * Normalize GitHub URLs for comparison
103
+ * @param {string} url - GitHub URL
104
+ * @returns {string} - Normalized URL
105
+ */
106
+ function normalizeGitHubUrl(url) {
107
+ // Remove .git suffix
108
+ let normalized = url.replace(/\.git$/, '');
109
+ // Convert SSH to HTTPS format for comparison
110
+ normalized = normalized.replace(/^git@github\.com:/, 'https://github.com/');
111
+ // Remove trailing slash
112
+ normalized = normalized.replace(/\/$/, '');
113
+ return normalized.toLowerCase();
114
+ }
115
+
116
+ /**
117
+ * Parse GitHub URL to extract owner and repo
118
+ * @param {string} url - GitHub URL (HTTPS or SSH)
119
+ * @returns {{owner: string, repo: string}} - Parsed owner and repo
120
+ */
121
+ function parseGitHubUrl(url) {
122
+ // Handle HTTPS URLs: https://github.com/owner/repo or https://github.com/owner/repo.git
123
+ // Handle SSH URLs: git@github.com:owner/repo or git@github.com:owner/repo.git
124
+ const match = url.match(/github\.com[:/]([^/]+)\/([^/]+?)(?:\.git)?$/);
125
+ if (!match) {
126
+ throw new Error('Invalid GitHub URL format');
127
+ }
128
+ return {
129
+ owner: match[1],
130
+ repo: match[2].replace(/\.git$/, '')
131
+ };
132
+ }
133
+
134
+ /**
135
+ * Auto-generate a branch name from a message
136
+ * @param {string} message - The agent message
137
+ * @returns {string} - Generated branch name
138
+ */
139
+ function autogenerateBranchName(message) {
140
+ // Convert to lowercase, replace spaces/special chars with hyphens
141
+ let branchName = message
142
+ .toLowerCase()
143
+ .replace(/[^a-z0-9\s-]/g, '') // Remove special characters
144
+ .replace(/\s+/g, '-') // Replace spaces with hyphens
145
+ .replace(/-+/g, '-') // Replace multiple hyphens with single
146
+ .replace(/^-|-$/g, ''); // Remove leading/trailing hyphens
147
+
148
+ // Ensure non-empty fallback
149
+ if (!branchName) {
150
+ branchName = 'task';
151
+ }
152
+
153
+ // Generate timestamp suffix (last 6 chars of base36 timestamp)
154
+ const timestamp = Date.now().toString(36).slice(-6);
155
+ const suffix = `-${timestamp}`;
156
+
157
+ // Limit length to ensure total length including suffix fits within 50 characters
158
+ const maxBaseLength = 50 - suffix.length;
159
+ if (branchName.length > maxBaseLength) {
160
+ branchName = branchName.substring(0, maxBaseLength);
161
+ }
162
+
163
+ // Remove any trailing hyphen after truncation and ensure no leading hyphen
164
+ branchName = branchName.replace(/-$/, '').replace(/^-+/, '');
165
+
166
+ // If still empty or starts with hyphen after cleanup, use fallback
167
+ if (!branchName || branchName.startsWith('-')) {
168
+ branchName = 'task';
169
+ }
170
+
171
+ // Combine base name with timestamp suffix
172
+ branchName = `${branchName}${suffix}`;
173
+
174
+ // Final validation: ensure it matches safe pattern
175
+ if (!/^[a-z0-9]+(?:-[a-z0-9]+)*$/.test(branchName)) {
176
+ // Fallback to deterministic safe name
177
+ return `branch-${timestamp}`;
178
+ }
179
+
180
+ return branchName;
181
+ }
182
+
183
+ /**
184
+ * Validate a Git branch name
185
+ * @param {string} branchName - Branch name to validate
186
+ * @returns {{valid: boolean, error?: string}} - Validation result
187
+ */
188
+ function validateBranchName(branchName) {
189
+ if (!branchName || branchName.trim() === '') {
190
+ return { valid: false, error: 'Branch name cannot be empty' };
191
+ }
192
+
193
+ // Git branch name rules
194
+ const invalidPatterns = [
195
+ { pattern: /^\./, message: 'Branch name cannot start with a dot' },
196
+ { pattern: /\.$/, message: 'Branch name cannot end with a dot' },
197
+ { pattern: /\.\./, message: 'Branch name cannot contain consecutive dots (..)' },
198
+ { pattern: /\s/, message: 'Branch name cannot contain spaces' },
199
+ { pattern: /[~^:?*\[\\]/, message: 'Branch name cannot contain special characters: ~ ^ : ? * [ \\' },
200
+ { pattern: /@{/, message: 'Branch name cannot contain @{' },
201
+ { pattern: /\/$/, message: 'Branch name cannot end with a slash' },
202
+ { pattern: /^\//, message: 'Branch name cannot start with a slash' },
203
+ { pattern: /\/\//, message: 'Branch name cannot contain consecutive slashes' },
204
+ { pattern: /\.lock$/, message: 'Branch name cannot end with .lock' }
205
+ ];
206
+
207
+ for (const { pattern, message } of invalidPatterns) {
208
+ if (pattern.test(branchName)) {
209
+ return { valid: false, error: message };
210
+ }
211
+ }
212
+
213
+ // Check for ASCII control characters
214
+ if (/[\x00-\x1F\x7F]/.test(branchName)) {
215
+ return { valid: false, error: 'Branch name cannot contain control characters' };
216
+ }
217
+
218
+ return { valid: true };
219
+ }
220
+
221
+ /**
222
+ * Get recent commit messages from a repository
223
+ * @param {string} projectPath - Path to the git repository
224
+ * @param {number} limit - Number of commits to retrieve (default: 5)
225
+ * @returns {Promise<string[]>} - Array of commit messages
226
+ */
227
+ async function getCommitMessages(projectPath, limit = 5) {
228
+ return new Promise((resolve, reject) => {
229
+ const gitProcess = spawn('git', ['log', `-${limit}`, '--pretty=format:%s'], {
230
+ cwd: projectPath,
231
+ stdio: ['pipe', 'pipe', 'pipe']
232
+ });
233
+
234
+ let stdout = '';
235
+ let stderr = '';
236
+
237
+ gitProcess.stdout.on('data', (data) => {
238
+ stdout += data.toString();
239
+ });
240
+
241
+ gitProcess.stderr.on('data', (data) => {
242
+ stderr += data.toString();
243
+ });
244
+
245
+ gitProcess.on('close', (code) => {
246
+ if (code === 0) {
247
+ const messages = stdout.trim().split('\n').filter(msg => msg.length > 0);
248
+ resolve(messages);
249
+ } else {
250
+ reject(new Error(`Failed to get commit messages: ${stderr}`));
251
+ }
252
+ });
253
+
254
+ gitProcess.on('error', (error) => {
255
+ reject(new Error(`Failed to execute git: ${error.message}`));
256
+ });
257
+ });
258
+ }
259
+
260
+ /**
261
+ * Create a new branch on GitHub using the API
262
+ * @param {Octokit} octokit - Octokit instance
263
+ * @param {string} owner - Repository owner
264
+ * @param {string} repo - Repository name
265
+ * @param {string} branchName - Name of the new branch
266
+ * @param {string} baseBranch - Base branch to branch from (default: 'main')
267
+ * @returns {Promise<void>}
268
+ */
269
+ async function createGitHubBranch(octokit, owner, repo, branchName, baseBranch = 'main') {
270
+ try {
271
+ // Get the SHA of the base branch
272
+ const { data: ref } = await octokit.git.getRef({
273
+ owner,
274
+ repo,
275
+ ref: `heads/${baseBranch}`
276
+ });
277
+
278
+ const baseSha = ref.object.sha;
279
+
280
+ // Create the new branch
281
+ await octokit.git.createRef({
282
+ owner,
283
+ repo,
284
+ ref: `refs/heads/${branchName}`,
285
+ sha: baseSha
286
+ });
287
+
288
+ console.log(`✅ Created branch '${branchName}' on GitHub`);
289
+ } catch (error) {
290
+ if (error.status === 422 && error.message.includes('Reference already exists')) {
291
+ console.log(`â„šī¸ Branch '${branchName}' already exists on GitHub`);
292
+ } else {
293
+ throw error;
294
+ }
295
+ }
296
+ }
297
+
298
+ /**
299
+ * Create a pull request on GitHub
300
+ * @param {Octokit} octokit - Octokit instance
301
+ * @param {string} owner - Repository owner
302
+ * @param {string} repo - Repository name
303
+ * @param {string} branchName - Head branch name
304
+ * @param {string} title - PR title
305
+ * @param {string} body - PR body/description
306
+ * @param {string} baseBranch - Base branch (default: 'main')
307
+ * @returns {Promise<{number: number, url: string}>} - PR number and URL
308
+ */
309
+ async function createGitHubPR(octokit, owner, repo, branchName, title, body, baseBranch = 'main') {
310
+ const { data: pr } = await octokit.pulls.create({
311
+ owner,
312
+ repo,
313
+ title,
314
+ head: branchName,
315
+ base: baseBranch,
316
+ body
317
+ });
318
+
319
+ console.log(`✅ Created pull request #${pr.number}: ${pr.html_url}`);
320
+
321
+ return {
322
+ number: pr.number,
323
+ url: pr.html_url
324
+ };
325
+ }
326
+
327
+ /**
328
+ * Clone a GitHub repository to a directory
329
+ * @param {string} githubUrl - GitHub repository URL
330
+ * @param {string} githubToken - Optional GitHub token for private repos
331
+ * @param {string} projectPath - Path for cloning the repository
332
+ * @returns {Promise<string>} - Path to the cloned repository
333
+ */
334
+ async function cloneGitHubRepo(githubUrl, githubToken = null, projectPath) {
335
+ return new Promise(async (resolve, reject) => {
336
+ try {
337
+ // Validate GitHub URL
338
+ if (!githubUrl || !githubUrl.includes('github.com')) {
339
+ throw new Error('Invalid GitHub URL');
340
+ }
341
+
342
+ const cloneDir = path.resolve(projectPath);
343
+
344
+ // Check if directory already exists
345
+ try {
346
+ await fs.access(cloneDir);
347
+ // Directory exists - check if it's a git repo with the same URL
348
+ try {
349
+ const existingUrl = await getGitRemoteUrl(cloneDir);
350
+ const normalizedExisting = normalizeGitHubUrl(existingUrl);
351
+ const normalizedRequested = normalizeGitHubUrl(githubUrl);
352
+
353
+ if (normalizedExisting === normalizedRequested) {
354
+ console.log('✅ Repository already exists at path with correct URL');
355
+ return resolve(cloneDir);
356
+ } else {
357
+ throw new Error(`Directory ${cloneDir} already exists with a different repository (${existingUrl}). Expected: ${githubUrl}`);
358
+ }
359
+ } catch (gitError) {
360
+ throw new Error(`Directory ${cloneDir} already exists but is not a valid git repository or git command failed`);
361
+ }
362
+ } catch (accessError) {
363
+ // Directory doesn't exist - proceed with clone
364
+ }
365
+
366
+ // Ensure parent directory exists
367
+ await fs.mkdir(path.dirname(cloneDir), { recursive: true });
368
+
369
+ // Prepare the git clone URL with authentication if token is provided
370
+ let cloneUrl = githubUrl;
371
+ if (githubToken) {
372
+ // Convert HTTPS URL to authenticated URL
373
+ // Example: https://github.com/user/repo -> https://token@github.com/user/repo
374
+ cloneUrl = githubUrl.replace('https://github.com', `https://${githubToken}@github.com`);
375
+ }
376
+
377
+ console.log('🔄 Cloning repository:', githubUrl);
378
+ console.log('📁 Destination:', cloneDir);
379
+
380
+ // Execute git clone
381
+ const gitProcess = spawn('git', ['clone', '--depth', '1', cloneUrl, cloneDir], {
382
+ stdio: ['pipe', 'pipe', 'pipe']
383
+ });
384
+
385
+ let stdout = '';
386
+ let stderr = '';
387
+
388
+ gitProcess.stdout.on('data', (data) => {
389
+ stdout += data.toString();
390
+ });
391
+
392
+ gitProcess.stderr.on('data', (data) => {
393
+ stderr += data.toString();
394
+ console.log('Git stderr:', data.toString());
395
+ });
396
+
397
+ gitProcess.on('close', (code) => {
398
+ if (code === 0) {
399
+ console.log('✅ Repository cloned successfully');
400
+ resolve(cloneDir);
401
+ } else {
402
+ console.error('❌ Git clone failed:', stderr);
403
+ reject(new Error(`Git clone failed: ${stderr}`));
404
+ }
405
+ });
406
+
407
+ gitProcess.on('error', (error) => {
408
+ reject(new Error(`Failed to execute git: ${error.message}`));
409
+ });
410
+ } catch (error) {
411
+ reject(error);
412
+ }
413
+ });
414
+ }
415
+
416
+ /**
417
+ * Clean up a temporary project directory and its Claude session
418
+ * @param {string} projectPath - Path to the project directory
419
+ * @param {string} sessionId - Session ID to clean up
420
+ */
421
+ async function cleanupProject(projectPath, sessionId = null) {
422
+ try {
423
+ // Only clean up projects in the external-projects directory
424
+ if (!projectPath.includes('.claude/external-projects')) {
425
+ console.warn('âš ī¸ Refusing to clean up non-external project:', projectPath);
426
+ return;
427
+ }
428
+
429
+ console.log('🧹 Cleaning up project:', projectPath);
430
+ await fs.rm(projectPath, { recursive: true, force: true });
431
+ console.log('✅ Project cleaned up');
432
+
433
+ // Also clean up the Claude session directory if sessionId provided
434
+ if (sessionId) {
435
+ try {
436
+ const sessionPath = path.join(os.homedir(), '.claude', 'sessions', sessionId);
437
+ console.log('🧹 Cleaning up session directory:', sessionPath);
438
+ await fs.rm(sessionPath, { recursive: true, force: true });
439
+ console.log('✅ Session directory cleaned up');
440
+ } catch (error) {
441
+ console.error('âš ī¸ Failed to clean up session directory:', error.message);
442
+ }
443
+ }
444
+ } catch (error) {
445
+ console.error('❌ Failed to clean up project:', error);
446
+ }
447
+ }
448
+
449
+ /**
450
+ * SSE Stream Writer - Adapts SDK/CLI output to Server-Sent Events
451
+ */
452
+ class SSEStreamWriter {
453
+ constructor(res, userId = null) {
454
+ this.res = res;
455
+ this.sessionId = null;
456
+ this.userId = userId;
457
+ this.isSSEStreamWriter = true; // Marker for transport detection
458
+ }
459
+
460
+ send(data) {
461
+ if (this.res.writableEnded) {
462
+ return;
463
+ }
464
+
465
+ // Format as SSE - providers send raw objects, we stringify
466
+ this.res.write(`data: ${JSON.stringify(data)}\n\n`);
467
+ }
468
+
469
+ end() {
470
+ if (!this.res.writableEnded) {
471
+ this.res.write('data: {"type":"done"}\n\n');
472
+ this.res.end();
473
+ }
474
+ }
475
+
476
+ setSessionId(sessionId) {
477
+ this.sessionId = sessionId;
478
+ this.send({ type: 'session-id', sessionId });
479
+ }
480
+
481
+ getSessionId() {
482
+ return this.sessionId;
483
+ }
484
+ }
485
+
486
+ /**
487
+ * Non-streaming response collector
488
+ */
489
+ class ResponseCollector {
490
+ constructor(userId = null) {
491
+ this.messages = [];
492
+ this.sessionId = null;
493
+ this.userId = userId;
494
+ }
495
+
496
+ send(data) {
497
+ // Store ALL messages for now - we'll filter when returning
498
+ this.messages.push(data);
499
+
500
+ // Extract sessionId if present
501
+ if (typeof data === 'string') {
502
+ try {
503
+ const parsed = JSON.parse(data);
504
+ if (parsed.sessionId) {
505
+ this.sessionId = parsed.sessionId;
506
+ }
507
+ } catch (e) {
508
+ // Not JSON, ignore
509
+ }
510
+ } else if (data && data.sessionId) {
511
+ this.sessionId = data.sessionId;
512
+ }
513
+ }
514
+
515
+ end() {
516
+ // Do nothing - we'll collect all messages
517
+ }
518
+
519
+ setSessionId(sessionId) {
520
+ this.sessionId = sessionId;
521
+ }
522
+
523
+ getSessionId() {
524
+ return this.sessionId;
525
+ }
526
+
527
+ getMessages() {
528
+ return this.messages;
529
+ }
530
+
531
+ /**
532
+ * Get filtered assistant messages only
533
+ */
534
+ getAssistantMessages() {
535
+ const assistantMessages = [];
536
+
537
+ for (const msg of this.messages) {
538
+ // Skip initial status message
539
+ if (msg && msg.type === 'status') {
540
+ continue;
541
+ }
542
+
543
+ // Handle JSON strings
544
+ if (typeof msg === 'string') {
545
+ try {
546
+ const parsed = JSON.parse(msg);
547
+ // Only include claude-response messages with assistant type
548
+ if (parsed.type === 'claude-response' && parsed.data && parsed.data.type === 'assistant') {
549
+ assistantMessages.push(parsed.data);
550
+ }
551
+ } catch (e) {
552
+ // Not JSON, skip
553
+ }
554
+ }
555
+ }
556
+
557
+ return assistantMessages;
558
+ }
559
+
560
+ /**
561
+ * Calculate total tokens from all messages
562
+ */
563
+ getTotalTokens() {
564
+ let totalInput = 0;
565
+ let totalOutput = 0;
566
+ let totalCacheRead = 0;
567
+ let totalCacheCreation = 0;
568
+
569
+ for (const msg of this.messages) {
570
+ let data = msg;
571
+
572
+ // Parse if string
573
+ if (typeof msg === 'string') {
574
+ try {
575
+ data = JSON.parse(msg);
576
+ } catch (e) {
577
+ continue;
578
+ }
579
+ }
580
+
581
+ // Extract usage from claude-response messages
582
+ if (data && data.type === 'claude-response' && data.data) {
583
+ const msgData = data.data;
584
+ if (msgData.message && msgData.message.usage) {
585
+ const usage = msgData.message.usage;
586
+ totalInput += usage.input_tokens || 0;
587
+ totalOutput += usage.output_tokens || 0;
588
+ totalCacheRead += usage.cache_read_input_tokens || 0;
589
+ totalCacheCreation += usage.cache_creation_input_tokens || 0;
590
+ }
591
+ }
592
+ }
593
+
594
+ return {
595
+ inputTokens: totalInput,
596
+ outputTokens: totalOutput,
597
+ cacheReadTokens: totalCacheRead,
598
+ cacheCreationTokens: totalCacheCreation,
599
+ totalTokens: totalInput + totalOutput + totalCacheRead + totalCacheCreation
600
+ };
601
+ }
602
+ }
603
+
604
+ // ===============================
605
+ // External API Endpoint
606
+ // ===============================
607
+
608
+ /**
609
+ * POST /api/agent
610
+ *
611
+ * Trigger an AI agent (Claude or Cursor) to work on a project.
612
+ * Supports automatic GitHub branch and pull request creation after successful completion.
613
+ *
614
+ * ================================================================================================
615
+ * REQUEST BODY PARAMETERS
616
+ * ================================================================================================
617
+ *
618
+ * @param {string} githubUrl - (Conditionally Required) GitHub repository URL to clone.
619
+ * Supported formats:
620
+ * - HTTPS: https://github.com/owner/repo
621
+ * - HTTPS with .git: https://github.com/owner/repo.git
622
+ * - SSH: git@github.com:owner/repo
623
+ * - SSH with .git: git@github.com:owner/repo.git
624
+ *
625
+ * @param {string} projectPath - (Conditionally Required) Path to existing project OR destination for cloning.
626
+ * Behavior depends on usage:
627
+ * - If used alone: Must point to existing project directory
628
+ * - If used with githubUrl: Target location for cloning
629
+ * - If omitted with githubUrl: Auto-generates temporary path in ~/.claude/external-projects/
630
+ *
631
+ * @param {string} message - (Required) Task description for the AI agent. Used as:
632
+ * - Instructions for the agent
633
+ * - Source for auto-generated branch names (if createBranch=true and no branchName)
634
+ * - Fallback for PR title if no commits are made
635
+ *
636
+ * @param {string} provider - (Optional) AI provider to use. Options: 'claude' | 'cursor' | 'codex' | 'gemini'
637
+ * Default: 'claude'
638
+ *
639
+ * @param {boolean} stream - (Optional) Enable Server-Sent Events (SSE) streaming for real-time updates.
640
+ * Default: true
641
+ * - true: Returns text/event-stream with incremental updates
642
+ * - false: Returns complete JSON response after completion
643
+ *
644
+ * @param {string} model - (Optional) Model identifier for providers.
645
+ *
646
+ * Claude models: 'sonnet' (default), 'opus', 'haiku', 'opusplan', 'sonnet[1m]'
647
+ * Cursor models: 'gpt-5' (default), 'gpt-5.2', 'gpt-5.2-high', 'sonnet-4.5', 'opus-4.5',
648
+ * 'gemini-3-pro', 'composer-1', 'auto', 'gpt-5.1', 'gpt-5.1-high',
649
+ * 'gpt-5.1-codex', 'gpt-5.1-codex-high', 'gpt-5.1-codex-max',
650
+ * 'gpt-5.1-codex-max-high', 'opus-4.1', 'grok', and thinking variants
651
+ * Codex models: 'gpt-5.2' (default), 'gpt-5.1-codex-max', 'o3', 'o4-mini'
652
+ *
653
+ * @param {boolean} cleanup - (Optional) Auto-cleanup project directory after completion.
654
+ * Default: true
655
+ * Behavior:
656
+ * - Only applies when cloning via githubUrl (not for existing projectPath)
657
+ * - Deletes cloned repository after 5 seconds
658
+ * - Also deletes associated Claude session directory
659
+ * - Remote branch and PR remain on GitHub if created
660
+ *
661
+ * @param {string} githubToken - (Optional) GitHub Personal Access Token for authentication.
662
+ * Overrides stored token from user settings.
663
+ * Required for:
664
+ * - Private repositories
665
+ * - Branch/PR creation features
666
+ * Token must have 'repo' scope for full functionality.
667
+ *
668
+ * @param {string} branchName - (Optional) Custom name for the Git branch.
669
+ * If provided, createBranch is automatically set to true.
670
+ * Validation rules (errors returned if violated):
671
+ * - Cannot be empty or whitespace only
672
+ * - Cannot start or end with dot (.)
673
+ * - Cannot contain consecutive dots (..)
674
+ * - Cannot contain spaces
675
+ * - Cannot contain special characters: ~ ^ : ? * [ \
676
+ * - Cannot contain @{
677
+ * - Cannot start or end with forward slash (/)
678
+ * - Cannot contain consecutive slashes (//)
679
+ * - Cannot end with .lock
680
+ * - Cannot contain ASCII control characters
681
+ * Examples: 'feature/user-auth', 'bugfix/login-error', 'refactor/db-optimization'
682
+ *
683
+ * @param {boolean} createBranch - (Optional) Create a new Git branch after successful agent completion.
684
+ * Default: false (or true if branchName is provided)
685
+ * Behavior:
686
+ * - Creates branch locally and pushes to remote
687
+ * - If branch exists locally: Checks out existing branch (no error)
688
+ * - If branch exists on remote: Uses existing branch (no error)
689
+ * - Branch name: Custom (if branchName provided) or auto-generated from message
690
+ * - Requires either githubUrl OR projectPath with GitHub remote
691
+ *
692
+ * @param {boolean} createPR - (Optional) Create a GitHub Pull Request after successful completion.
693
+ * Default: false
694
+ * Behavior:
695
+ * - PR title: First commit message (or fallback to message parameter)
696
+ * - PR description: Auto-generated from all commit messages
697
+ * - Base branch: Always 'main' (currently hardcoded)
698
+ * - If PR already exists: GitHub returns error with details
699
+ * - Requires either githubUrl OR projectPath with GitHub remote
700
+ *
701
+ * ================================================================================================
702
+ * PATH HANDLING BEHAVIOR
703
+ * ================================================================================================
704
+ *
705
+ * Scenario 1: Only githubUrl provided
706
+ * Input: { githubUrl: "https://github.com/owner/repo" }
707
+ * Action: Clones to auto-generated temporary path: ~/.claude/external-projects/<hash>/
708
+ * Cleanup: Yes (if cleanup=true)
709
+ *
710
+ * Scenario 2: Only projectPath provided
711
+ * Input: { projectPath: "/home/user/my-project" }
712
+ * Action: Uses existing project at specified path
713
+ * Validation: Path must exist and be accessible
714
+ * Cleanup: No (never cleanup existing projects)
715
+ *
716
+ * Scenario 3: Both githubUrl and projectPath provided
717
+ * Input: { githubUrl: "https://github.com/owner/repo", projectPath: "/custom/path" }
718
+ * Action: Clones githubUrl to projectPath location
719
+ * Validation:
720
+ * - If projectPath exists with git repo:
721
+ * - Compares remote URL with githubUrl
722
+ * - If URLs match: Reuses existing repo
723
+ * - If URLs differ: Returns error
724
+ * Cleanup: Yes (if cleanup=true)
725
+ *
726
+ * ================================================================================================
727
+ * GITHUB BRANCH/PR CREATION REQUIREMENTS
728
+ * ================================================================================================
729
+ *
730
+ * For createBranch or createPR to work, one of the following must be true:
731
+ *
732
+ * Option A: githubUrl provided
733
+ * - Repository URL directly specified
734
+ * - Works with both cloning and existing paths
735
+ *
736
+ * Option B: projectPath with GitHub remote
737
+ * - Project must be a Git repository
738
+ * - Must have 'origin' remote configured
739
+ * - Remote URL must point to github.com
740
+ * - System auto-detects GitHub URL via: git remote get-url origin
741
+ *
742
+ * Additional Requirements:
743
+ * - Valid GitHub token (from settings or githubToken parameter)
744
+ * - Token must have 'repo' scope for private repos
745
+ * - Project must have commits (for PR creation)
746
+ *
747
+ * ================================================================================================
748
+ * VALIDATION & ERROR HANDLING
749
+ * ================================================================================================
750
+ *
751
+ * Input Validations (400 Bad Request):
752
+ * - Either githubUrl OR projectPath must be provided (not neither)
753
+ * - message must be non-empty string
754
+ * - provider must be 'claude', 'cursor', 'codex', or 'gemini'
755
+ * - createBranch/createPR requires githubUrl OR projectPath (not neither)
756
+ * - branchName must pass Git naming rules (if provided)
757
+ *
758
+ * Runtime Validations (500 Internal Server Error or specific error in response):
759
+ * - projectPath must exist (if used alone)
760
+ * - GitHub URL format must be valid
761
+ * - Git remote URL must include github.com (for projectPath + branch/PR)
762
+ * - GitHub token must be available (for private repos and branch/PR)
763
+ * - Directory conflicts handled (existing path with different repo)
764
+ *
765
+ * Branch Name Validation Errors (returned in response, not HTTP error):
766
+ * Invalid names return: { branch: { error: "Invalid branch name: <reason>" } }
767
+ * Examples:
768
+ * - "my branch" → "Branch name cannot contain spaces"
769
+ * - ".feature" → "Branch name cannot start with a dot"
770
+ * - "feature.lock" → "Branch name cannot end with .lock"
771
+ *
772
+ * ================================================================================================
773
+ * RESPONSE FORMATS
774
+ * ================================================================================================
775
+ *
776
+ * Streaming Response (stream=true):
777
+ * Content-Type: text/event-stream
778
+ * Events:
779
+ * - { type: "status", message: "...", projectPath: "..." }
780
+ * - { type: "claude-response", data: {...} }
781
+ * - { type: "github-branch", branch: { name: "...", url: "..." } }
782
+ * - { type: "github-pr", pullRequest: { number: 42, url: "..." } }
783
+ * - { type: "github-error", error: "..." }
784
+ * - { type: "done" }
785
+ *
786
+ * Non-Streaming Response (stream=false):
787
+ * Content-Type: application/json
788
+ * {
789
+ * success: true,
790
+ * sessionId: "session-123",
791
+ * messages: [...], // Assistant messages only (filtered)
792
+ * tokens: {
793
+ * inputTokens: 150,
794
+ * outputTokens: 50,
795
+ * cacheReadTokens: 0,
796
+ * cacheCreationTokens: 0,
797
+ * totalTokens: 200
798
+ * },
799
+ * projectPath: "/path/to/project",
800
+ * branch: { // Only if createBranch=true
801
+ * name: "feature/xyz",
802
+ * url: "https://github.com/owner/repo/tree/feature/xyz"
803
+ * } | { error: "..." },
804
+ * pullRequest: { // Only if createPR=true
805
+ * number: 42,
806
+ * url: "https://github.com/owner/repo/pull/42"
807
+ * } | { error: "..." }
808
+ * }
809
+ *
810
+ * Error Response:
811
+ * HTTP Status: 400, 401, 500
812
+ * Content-Type: application/json
813
+ * { success: false, error: "Error description" }
814
+ *
815
+ * ================================================================================================
816
+ * EXAMPLES
817
+ * ================================================================================================
818
+ *
819
+ * Example 1: Clone and process with auto-cleanup
820
+ * POST /api/agent
821
+ * { "githubUrl": "https://github.com/user/repo", "message": "Fix bug" }
822
+ *
823
+ * Example 2: Use existing project with custom branch and PR
824
+ * POST /api/agent
825
+ * {
826
+ * "projectPath": "/home/user/project",
827
+ * "message": "Add feature",
828
+ * "branchName": "feature/new-feature",
829
+ * "createPR": true
830
+ * }
831
+ *
832
+ * Example 3: Clone to specific path with auto-generated branch
833
+ * POST /api/agent
834
+ * {
835
+ * "githubUrl": "https://github.com/user/repo",
836
+ * "projectPath": "/tmp/work",
837
+ * "message": "Refactor code",
838
+ * "createBranch": true,
839
+ * "cleanup": false
840
+ * }
841
+ */
842
+ router.post('/', validateExternalApiKey, async (req, res) => {
843
+ const { githubUrl, projectPath, message, provider = 'claude', model, githubToken, branchName, sessionId } = req.body;
844
+
845
+ // Parse stream and cleanup as booleans (handle string "true"/"false" from curl)
846
+ const stream = req.body.stream === undefined ? true : (req.body.stream === true || req.body.stream === 'true');
847
+ const cleanup = req.body.cleanup === undefined ? true : (req.body.cleanup === true || req.body.cleanup === 'true');
848
+
849
+ // If branchName is provided, automatically enable createBranch
850
+ const createBranch = branchName ? true : (req.body.createBranch === true || req.body.createBranch === 'true');
851
+ const createPR = req.body.createPR === true || req.body.createPR === 'true';
852
+
853
+ // Validate inputs
854
+ if (!githubUrl && !projectPath) {
855
+ return res.status(400).json({ error: 'Either githubUrl or projectPath is required' });
856
+ }
857
+
858
+ if (!message || !message.trim()) {
859
+ return res.status(400).json({ error: 'message is required' });
860
+ }
861
+
862
+ if (!['claude', 'cursor', 'codex', 'gemini'].includes(provider)) {
863
+ return res.status(400).json({ error: 'provider must be "claude", "cursor", "codex", or "gemini"' });
864
+ }
865
+
866
+ // Validate GitHub branch/PR creation requirements
867
+ // Allow branch/PR creation with projectPath as long as it has a GitHub remote
868
+ if ((createBranch || createPR) && !githubUrl && !projectPath) {
869
+ return res.status(400).json({ error: 'createBranch and createPR require either githubUrl or projectPath with a GitHub remote' });
870
+ }
871
+
872
+ let finalProjectPath = null;
873
+ let writer = null;
874
+
875
+ try {
876
+ // Determine the final project path
877
+ if (githubUrl) {
878
+ // Clone repository (to projectPath if provided, otherwise generate path)
879
+ const tokenToUse = githubToken || githubTokensDb.getActiveGithubToken(req.user.id);
880
+
881
+ let targetPath;
882
+ if (projectPath) {
883
+ targetPath = projectPath;
884
+ } else {
885
+ // Generate a unique path for cloning
886
+ const repoHash = crypto.createHash('md5').update(githubUrl + Date.now()).digest('hex');
887
+ targetPath = path.join(os.homedir(), '.claude', 'external-projects', repoHash);
888
+ }
889
+
890
+ finalProjectPath = await cloneGitHubRepo(githubUrl.trim(), tokenToUse, targetPath);
891
+ } else {
892
+ // Use existing project path
893
+ finalProjectPath = normalizeProjectPath(path.resolve(projectPath));
894
+
895
+ // Verify the path exists
896
+ try {
897
+ await fs.access(finalProjectPath);
898
+ } catch (error) {
899
+ throw new Error(`Project path does not exist: ${finalProjectPath}`);
900
+ }
901
+ }
902
+
903
+ finalProjectPath = normalizeProjectPath(finalProjectPath);
904
+
905
+ // Warn if the same project path is actively used by another user
906
+ const sharedPathCheck = projectsDb.isProjectPathUsedByOthers(req.user.id, finalProjectPath);
907
+ if (sharedPathCheck.used) {
908
+ console.warn(`[WARN] Project path "${finalProjectPath}" is shared with user(s): ${sharedPathCheck.usernames.join(', ')}`);
909
+ }
910
+
911
+ // Register project path in DB (or reuse existing active registration)
912
+ const registrationResult = projectsDb.createProjectPath(req.user.id, finalProjectPath, null);
913
+ if (registrationResult.outcome === 'active_conflict') {
914
+ console.log('Project registration already exists for:', finalProjectPath);
915
+ } else {
916
+ console.log('Project registered:', registrationResult.project);
917
+ }
918
+
919
+ // Set up writer based on streaming mode
920
+ if (stream) {
921
+ // Set up SSE headers for streaming
922
+ res.setHeader('Content-Type', 'text/event-stream');
923
+ res.setHeader('Cache-Control', 'no-cache');
924
+ res.setHeader('Connection', 'keep-alive');
925
+ res.setHeader('X-Accel-Buffering', 'no'); // Disable nginx buffering
926
+
927
+ writer = new SSEStreamWriter(res, req.user.id);
928
+
929
+ // Send initial status
930
+ writer.send({
931
+ type: 'status',
932
+ message: githubUrl ? 'Repository cloned and session started' : 'Session started',
933
+ projectPath: finalProjectPath
934
+ });
935
+ } else {
936
+ // Non-streaming mode: collect messages
937
+ writer = new ResponseCollector(req.user.id);
938
+
939
+ // Collect initial status message
940
+ writer.send({
941
+ type: 'status',
942
+ message: githubUrl ? 'Repository cloned and session started' : 'Session started',
943
+ projectPath: finalProjectPath
944
+ });
945
+ }
946
+
947
+ // Start the appropriate session
948
+ if (provider === 'claude') {
949
+ console.log('🤖 Starting Claude SDK session');
950
+
951
+ await queryClaudeSDK(message.trim(), {
952
+ projectPath: finalProjectPath,
953
+ cwd: finalProjectPath,
954
+ sessionId: sessionId || null,
955
+ model: model,
956
+ permissionMode: 'bypassPermissions' // Bypass all permissions for API calls
957
+ }, writer);
958
+
959
+ } else if (provider === 'cursor') {
960
+ console.log('đŸ–ąī¸ Starting Cursor CLI session');
961
+
962
+ await spawnCursor(message.trim(), {
963
+ projectPath: finalProjectPath,
964
+ cwd: finalProjectPath,
965
+ sessionId: sessionId || null,
966
+ model: model || undefined,
967
+ skipPermissions: true // Bypass permissions for Cursor
968
+ }, writer);
969
+ } else if (provider === 'codex') {
970
+ console.log('🤖 Starting Codex SDK session');
971
+
972
+ await queryCodex(message.trim(), {
973
+ projectPath: finalProjectPath,
974
+ cwd: finalProjectPath,
975
+ sessionId: sessionId || null,
976
+ model: model || CODEX_MODELS.DEFAULT,
977
+ permissionMode: 'bypassPermissions'
978
+ }, writer);
979
+ } else if (provider === 'gemini') {
980
+ console.log('✨ Starting Gemini CLI session');
981
+
982
+ await spawnGemini(message.trim(), {
983
+ projectPath: finalProjectPath,
984
+ cwd: finalProjectPath,
985
+ sessionId: sessionId || null,
986
+ model: model,
987
+ skipPermissions: true // CLI mode bypasses permissions
988
+ }, writer);
989
+ }
990
+
991
+ // Handle GitHub branch and PR creation after successful agent completion
992
+ let branchInfo = null;
993
+ let prInfo = null;
994
+
995
+ if (createBranch || createPR) {
996
+ try {
997
+ console.log('🔄 Starting GitHub branch/PR creation workflow...');
998
+
999
+ // Get GitHub token
1000
+ const tokenToUse = githubToken || githubTokensDb.getActiveGithubToken(req.user.id);
1001
+
1002
+ if (!tokenToUse) {
1003
+ throw new Error('GitHub token required for branch/PR creation. Please configure a GitHub token in settings.');
1004
+ }
1005
+
1006
+ // Initialize Octokit
1007
+ const octokit = new Octokit({ auth: tokenToUse });
1008
+
1009
+ // Get GitHub URL - either from parameter or from git remote
1010
+ let repoUrl = githubUrl;
1011
+ if (!repoUrl) {
1012
+ console.log('🔍 Getting GitHub URL from git remote...');
1013
+ try {
1014
+ repoUrl = await getGitRemoteUrl(finalProjectPath);
1015
+ if (!repoUrl.includes('github.com')) {
1016
+ throw new Error('Project does not have a GitHub remote configured');
1017
+ }
1018
+ console.log(`✅ Found GitHub remote: ${repoUrl}`);
1019
+ } catch (error) {
1020
+ throw new Error(`Failed to get GitHub remote URL: ${error.message}`);
1021
+ }
1022
+ }
1023
+
1024
+ // Parse GitHub URL to get owner and repo
1025
+ const { owner, repo } = parseGitHubUrl(repoUrl);
1026
+ console.log(`đŸ“Ļ Repository: ${owner}/${repo}`);
1027
+
1028
+ // Use provided branch name or auto-generate from message
1029
+ const finalBranchName = branchName || autogenerateBranchName(message);
1030
+ if (branchName) {
1031
+ console.log(`đŸŒŋ Using provided branch name: ${finalBranchName}`);
1032
+
1033
+ // Validate custom branch name
1034
+ const validation = validateBranchName(finalBranchName);
1035
+ if (!validation.valid) {
1036
+ throw new Error(`Invalid branch name: ${validation.error}`);
1037
+ }
1038
+ } else {
1039
+ console.log(`đŸŒŋ Auto-generated branch name: ${finalBranchName}`);
1040
+ }
1041
+
1042
+ if (createBranch) {
1043
+ // Create and checkout the new branch locally
1044
+ console.log('🔄 Creating local branch...');
1045
+ const checkoutProcess = spawn('git', ['checkout', '-b', finalBranchName], {
1046
+ cwd: finalProjectPath,
1047
+ stdio: 'pipe'
1048
+ });
1049
+
1050
+ await new Promise((resolve, reject) => {
1051
+ let stderr = '';
1052
+ checkoutProcess.stderr.on('data', (data) => { stderr += data.toString(); });
1053
+ checkoutProcess.on('close', (code) => {
1054
+ if (code === 0) {
1055
+ console.log(`✅ Created and checked out local branch '${finalBranchName}'`);
1056
+ resolve();
1057
+ } else {
1058
+ // Branch might already exist locally, try to checkout
1059
+ if (stderr.includes('already exists')) {
1060
+ console.log(`â„šī¸ Branch '${finalBranchName}' already exists locally, checking out...`);
1061
+ const checkoutExisting = spawn('git', ['checkout', finalBranchName], {
1062
+ cwd: finalProjectPath,
1063
+ stdio: 'pipe'
1064
+ });
1065
+ checkoutExisting.on('close', (checkoutCode) => {
1066
+ if (checkoutCode === 0) {
1067
+ console.log(`✅ Checked out existing branch '${finalBranchName}'`);
1068
+ resolve();
1069
+ } else {
1070
+ reject(new Error(`Failed to checkout existing branch: ${stderr}`));
1071
+ }
1072
+ });
1073
+ } else {
1074
+ reject(new Error(`Failed to create branch: ${stderr}`));
1075
+ }
1076
+ }
1077
+ });
1078
+ });
1079
+
1080
+ // Push the branch to remote
1081
+ console.log('🔄 Pushing branch to remote...');
1082
+ const pushProcess = spawn('git', ['push', '-u', 'origin', finalBranchName], {
1083
+ cwd: finalProjectPath,
1084
+ stdio: 'pipe'
1085
+ });
1086
+
1087
+ await new Promise((resolve, reject) => {
1088
+ let stderr = '';
1089
+ let stdout = '';
1090
+ pushProcess.stdout.on('data', (data) => { stdout += data.toString(); });
1091
+ pushProcess.stderr.on('data', (data) => { stderr += data.toString(); });
1092
+ pushProcess.on('close', (code) => {
1093
+ if (code === 0) {
1094
+ console.log(`✅ Pushed branch '${finalBranchName}' to remote`);
1095
+ resolve();
1096
+ } else {
1097
+ // Check if branch exists on remote but has different commits
1098
+ if (stderr.includes('already exists') || stderr.includes('up-to-date')) {
1099
+ console.log(`â„šī¸ Branch '${finalBranchName}' already exists on remote, using existing branch`);
1100
+ resolve();
1101
+ } else {
1102
+ reject(new Error(`Failed to push branch: ${stderr}`));
1103
+ }
1104
+ }
1105
+ });
1106
+ });
1107
+
1108
+ branchInfo = {
1109
+ name: finalBranchName,
1110
+ url: `https://github.com/${owner}/${repo}/tree/${finalBranchName}`
1111
+ };
1112
+ }
1113
+
1114
+ if (createPR) {
1115
+ // Get commit messages to generate PR description
1116
+ console.log('🔄 Generating PR title and description...');
1117
+ const commitMessages = await getCommitMessages(finalProjectPath, 5);
1118
+
1119
+ // Use the first commit message as the PR title, or fallback to the agent message
1120
+ const prTitle = commitMessages.length > 0 ? commitMessages[0] : message;
1121
+
1122
+ // Generate PR body from commit messages
1123
+ let prBody = '## Changes\n\n';
1124
+ if (commitMessages.length > 0) {
1125
+ prBody += commitMessages.map(msg => `- ${msg}`).join('\n');
1126
+ } else {
1127
+ prBody += `Agent task: ${message}`;
1128
+ }
1129
+ prBody += '\n\n---\n*This pull request was automatically created by CloudCLI.ai Agent.*';
1130
+
1131
+ console.log(`📝 PR Title: ${prTitle}`);
1132
+
1133
+ // Create the pull request
1134
+ console.log('🔄 Creating pull request...');
1135
+ prInfo = await createGitHubPR(octokit, owner, repo, finalBranchName, prTitle, prBody, 'main');
1136
+ }
1137
+
1138
+ // Send branch/PR info in response
1139
+ if (stream) {
1140
+ if (branchInfo) {
1141
+ writer.send({
1142
+ type: 'github-branch',
1143
+ branch: branchInfo
1144
+ });
1145
+ }
1146
+ if (prInfo) {
1147
+ writer.send({
1148
+ type: 'github-pr',
1149
+ pullRequest: prInfo
1150
+ });
1151
+ }
1152
+ }
1153
+
1154
+ } catch (error) {
1155
+ console.error('❌ GitHub branch/PR creation error:', error);
1156
+
1157
+ // Send error but don't fail the entire request
1158
+ if (stream) {
1159
+ writer.send({
1160
+ type: 'github-error',
1161
+ error: error.message
1162
+ });
1163
+ }
1164
+ // Store error info for non-streaming response
1165
+ if (!stream) {
1166
+ branchInfo = { error: error.message };
1167
+ prInfo = { error: error.message };
1168
+ }
1169
+ }
1170
+ }
1171
+
1172
+ // Handle response based on streaming mode
1173
+ if (stream) {
1174
+ // Streaming mode: end the SSE stream
1175
+ writer.end();
1176
+ } else {
1177
+ // Non-streaming mode: send filtered messages and token summary as JSON
1178
+ const assistantMessages = writer.getAssistantMessages();
1179
+ const tokenSummary = writer.getTotalTokens();
1180
+
1181
+ const response = {
1182
+ success: true,
1183
+ sessionId: writer.getSessionId(),
1184
+ messages: assistantMessages,
1185
+ tokens: tokenSummary,
1186
+ projectPath: finalProjectPath
1187
+ };
1188
+
1189
+ // Add branch/PR info if created
1190
+ if (branchInfo) {
1191
+ response.branch = branchInfo;
1192
+ }
1193
+ if (prInfo) {
1194
+ response.pullRequest = prInfo;
1195
+ }
1196
+
1197
+ res.json(response);
1198
+ }
1199
+
1200
+ // Clean up if requested
1201
+ if (cleanup && githubUrl) {
1202
+ // Only cleanup if we cloned a repo (not for existing project paths)
1203
+ const sessionIdForCleanup = writer.getSessionId();
1204
+ setTimeout(() => {
1205
+ cleanupProject(finalProjectPath, sessionIdForCleanup);
1206
+ }, 5000);
1207
+ }
1208
+
1209
+ } catch (error) {
1210
+ console.error('❌ External session error:', error);
1211
+
1212
+ // Clean up on error
1213
+ if (finalProjectPath && cleanup && githubUrl) {
1214
+ const sessionIdForCleanup = writer ? writer.getSessionId() : null;
1215
+ cleanupProject(finalProjectPath, sessionIdForCleanup);
1216
+ }
1217
+
1218
+ if (stream) {
1219
+ // For streaming, send error event and stop
1220
+ if (!writer) {
1221
+ // Set up SSE headers if not already done
1222
+ res.setHeader('Content-Type', 'text/event-stream');
1223
+ res.setHeader('Cache-Control', 'no-cache');
1224
+ res.setHeader('Connection', 'keep-alive');
1225
+ res.setHeader('X-Accel-Buffering', 'no');
1226
+ writer = new SSEStreamWriter(res, req.user.id);
1227
+ }
1228
+
1229
+ if (!res.writableEnded) {
1230
+ writer.send({
1231
+ type: 'error',
1232
+ error: error.message,
1233
+ message: `Failed: ${error.message}`
1234
+ });
1235
+ writer.end();
1236
+ }
1237
+ } else if (!res.headersSent) {
1238
+ res.status(500).json({
1239
+ success: false,
1240
+ error: error.message
1241
+ });
1242
+ }
1243
+ }
1244
+ });
1245
+
1246
+ export default router;