@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,1263 @@
1
+ import express from 'express';
2
+ import { spawn } from 'child_process';
3
+ import path from 'path';
4
+ import { promises as fs } from 'fs';
5
+ import { projectsDb } from '../modules/database/index.js';
6
+ import { queryClaudeSDK } from '../claude-sdk.js';
7
+ import { spawnCursor } from '../cursor-cli.js';
8
+ const router = express.Router();
9
+ const COMMIT_DIFF_CHARACTER_LIMIT = 500_000;
10
+ function spawnAsync(command, args, options = {}) {
11
+ return new Promise((resolve, reject) => {
12
+ const child = spawn(command, args, {
13
+ ...options,
14
+ shell: false,
15
+ });
16
+ let stdout = '';
17
+ let stderr = '';
18
+ child.stdout.on('data', (data) => {
19
+ stdout += data.toString();
20
+ });
21
+ child.stderr.on('data', (data) => {
22
+ stderr += data.toString();
23
+ });
24
+ child.on('error', (error) => {
25
+ reject(error);
26
+ });
27
+ child.on('close', (code) => {
28
+ if (code === 0) {
29
+ resolve({ stdout, stderr });
30
+ return;
31
+ }
32
+ const error = new Error(`Command failed: ${command} ${args.join(' ')}`);
33
+ error.code = code;
34
+ error.stdout = stdout;
35
+ error.stderr = stderr;
36
+ reject(error);
37
+ });
38
+ });
39
+ }
40
+ // Input validation helpers (defense-in-depth)
41
+ function validateCommitRef(commit) {
42
+ // Allow hex hashes, HEAD, HEAD~N, HEAD^N, tag names, branch names
43
+ if (!/^[a-zA-Z0-9._~^{}@\/-]+$/.test(commit)) {
44
+ throw new Error('Invalid commit reference');
45
+ }
46
+ return commit;
47
+ }
48
+ function validateBranchName(branch) {
49
+ if (!/^[a-zA-Z0-9._\/-]+$/.test(branch)) {
50
+ throw new Error('Invalid branch name');
51
+ }
52
+ return branch;
53
+ }
54
+ function validateFilePath(file, projectPath) {
55
+ if (!file || file.includes('\0')) {
56
+ throw new Error('Invalid file path');
57
+ }
58
+ // Prevent path traversal: resolve the file relative to the project root
59
+ // and ensure the result stays within the project directory
60
+ if (projectPath) {
61
+ const resolved = path.resolve(projectPath, file);
62
+ const normalizedRoot = path.resolve(projectPath) + path.sep;
63
+ if (!resolved.startsWith(normalizedRoot) && resolved !== path.resolve(projectPath)) {
64
+ throw new Error('Invalid file path: path traversal detected');
65
+ }
66
+ }
67
+ return file;
68
+ }
69
+ function validateRemoteName(remote) {
70
+ if (!/^[a-zA-Z0-9._-]+$/.test(remote)) {
71
+ throw new Error('Invalid remote name');
72
+ }
73
+ return remote;
74
+ }
75
+ function validateProjectPath(projectPath) {
76
+ if (!projectPath || projectPath.includes('\0')) {
77
+ throw new Error('Invalid project path');
78
+ }
79
+ const resolved = path.resolve(projectPath);
80
+ // Must be an absolute path after resolution
81
+ if (!path.isAbsolute(resolved)) {
82
+ throw new Error('Invalid project path: must be absolute');
83
+ }
84
+ // Block obviously dangerous paths
85
+ if (resolved === '/' || resolved === path.sep) {
86
+ throw new Error('Invalid project path: root directory not allowed');
87
+ }
88
+ return resolved;
89
+ }
90
+ /**
91
+ * Resolve the absolute project directory for a given DB `projectId`.
92
+ *
93
+ * After the projectName → projectId migration, every git endpoint receives
94
+ * the DB primary key (`project` query/body param). The legacy filesystem
95
+ * resolver that walked Claude's JSONL history is no longer used here; the
96
+ * path comes straight from the `projects` table and is then sanity-checked
97
+ * by `validateProjectPath` before any `git` command runs against it.
98
+ */
99
+ async function getActualProjectPath(userId, projectId) {
100
+ const projectPath = await projectsDb.getProjectPathById(userId, projectId);
101
+ if (!projectPath) {
102
+ throw new Error(`Unable to resolve project path for "${projectId}"`);
103
+ }
104
+ return validateProjectPath(projectPath);
105
+ }
106
+ // Helper function to strip git diff headers
107
+ function stripDiffHeaders(diff) {
108
+ if (!diff)
109
+ return '';
110
+ const lines = diff.split('\n');
111
+ const filteredLines = [];
112
+ let startIncluding = false;
113
+ for (const line of lines) {
114
+ // Skip all header lines including diff --git, index, file mode, and --- / +++ file paths
115
+ if (line.startsWith('diff --git') ||
116
+ line.startsWith('index ') ||
117
+ line.startsWith('new file mode') ||
118
+ line.startsWith('deleted file mode') ||
119
+ line.startsWith('---') ||
120
+ line.startsWith('+++')) {
121
+ continue;
122
+ }
123
+ // Start including lines from @@ hunk headers onwards
124
+ if (line.startsWith('@@') || startIncluding) {
125
+ startIncluding = true;
126
+ filteredLines.push(line);
127
+ }
128
+ }
129
+ return filteredLines.join('\n');
130
+ }
131
+ // Helper function to validate git repository
132
+ async function validateGitRepository(projectPath) {
133
+ try {
134
+ // Check if directory exists
135
+ await fs.access(projectPath);
136
+ }
137
+ catch {
138
+ throw new Error(`Project path not found: ${projectPath}`);
139
+ }
140
+ try {
141
+ // Allow any directory that is inside a work tree (repo root or nested folder).
142
+ const { stdout: insideWorkTreeOutput } = await spawnAsync('git', ['rev-parse', '--is-inside-work-tree'], { cwd: projectPath });
143
+ const isInsideWorkTree = insideWorkTreeOutput.trim() === 'true';
144
+ if (!isInsideWorkTree) {
145
+ throw new Error('Not inside a git work tree');
146
+ }
147
+ // Ensure git can resolve the repository root for this directory.
148
+ await spawnAsync('git', ['rev-parse', '--show-toplevel'], { cwd: projectPath });
149
+ }
150
+ catch {
151
+ throw new Error('Not a git repository. This directory does not contain a .git folder. Initialize a git repository with "git init" to use source control features.');
152
+ }
153
+ }
154
+ function getGitErrorDetails(error) {
155
+ return `${error?.message || ''} ${error?.stderr || ''} ${error?.stdout || ''}`;
156
+ }
157
+ function isMissingHeadRevisionError(error) {
158
+ const errorDetails = getGitErrorDetails(error).toLowerCase();
159
+ return errorDetails.includes('unknown revision')
160
+ || errorDetails.includes('ambiguous argument')
161
+ || errorDetails.includes('needed a single revision')
162
+ || errorDetails.includes('bad revision');
163
+ }
164
+ async function getCurrentBranchName(projectPath) {
165
+ try {
166
+ // symbolic-ref works even when the repository has no commits.
167
+ const { stdout } = await spawnAsync('git', ['symbolic-ref', '--short', 'HEAD'], { cwd: projectPath });
168
+ const branchName = stdout.trim();
169
+ if (branchName) {
170
+ return branchName;
171
+ }
172
+ }
173
+ catch (error) {
174
+ // Fall back to rev-parse for detached HEAD and older git edge cases.
175
+ }
176
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], { cwd: projectPath });
177
+ return stdout.trim();
178
+ }
179
+ async function repositoryHasCommits(projectPath) {
180
+ try {
181
+ await spawnAsync('git', ['rev-parse', '--verify', 'HEAD'], { cwd: projectPath });
182
+ return true;
183
+ }
184
+ catch (error) {
185
+ if (isMissingHeadRevisionError(error)) {
186
+ return false;
187
+ }
188
+ throw error;
189
+ }
190
+ }
191
+ async function getRepositoryRootPath(projectPath) {
192
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--show-toplevel'], { cwd: projectPath });
193
+ return stdout.trim();
194
+ }
195
+ function normalizeRepositoryRelativeFilePath(filePath) {
196
+ return String(filePath)
197
+ .replace(/\\/g, '/')
198
+ .replace(/^\.\/+/, '')
199
+ .replace(/^\/+/, '')
200
+ .trim();
201
+ }
202
+ function parseStatusFilePaths(statusOutput) {
203
+ return statusOutput
204
+ .split('\n')
205
+ .map((line) => line.trimEnd())
206
+ .filter((line) => line.trim())
207
+ .map((line) => {
208
+ const statusPath = line.substring(3);
209
+ const renamedFilePath = statusPath.split(' -> ')[1];
210
+ return normalizeRepositoryRelativeFilePath(renamedFilePath || statusPath);
211
+ })
212
+ .filter(Boolean);
213
+ }
214
+ function buildFilePathCandidates(projectPath, repositoryRootPath, filePath) {
215
+ const normalizedFilePath = normalizeRepositoryRelativeFilePath(filePath);
216
+ const projectRelativePath = normalizeRepositoryRelativeFilePath(path.relative(repositoryRootPath, projectPath));
217
+ const candidates = [normalizedFilePath];
218
+ if (projectRelativePath
219
+ && projectRelativePath !== '.'
220
+ && !normalizedFilePath.startsWith(`${projectRelativePath}/`)) {
221
+ candidates.push(`${projectRelativePath}/${normalizedFilePath}`);
222
+ }
223
+ return Array.from(new Set(candidates.filter(Boolean)));
224
+ }
225
+ async function resolveRepositoryFilePath(projectPath, filePath) {
226
+ validateFilePath(filePath);
227
+ const repositoryRootPath = await getRepositoryRootPath(projectPath);
228
+ const candidateFilePaths = buildFilePathCandidates(projectPath, repositoryRootPath, filePath);
229
+ for (const candidateFilePath of candidateFilePaths) {
230
+ const { stdout } = await spawnAsync('git', ['status', '--porcelain', '--', candidateFilePath], { cwd: repositoryRootPath });
231
+ if (stdout.trim()) {
232
+ return {
233
+ repositoryRootPath,
234
+ repositoryRelativeFilePath: candidateFilePath,
235
+ };
236
+ }
237
+ }
238
+ // If the caller sent a bare filename (e.g. "hello.ts"), recover it from changed files.
239
+ const normalizedFilePath = normalizeRepositoryRelativeFilePath(filePath);
240
+ if (!normalizedFilePath.includes('/')) {
241
+ const { stdout: repositoryStatusOutput } = await spawnAsync('git', ['status', '--porcelain'], { cwd: repositoryRootPath });
242
+ const changedFilePaths = parseStatusFilePaths(repositoryStatusOutput);
243
+ const suffixMatches = changedFilePaths.filter((changedFilePath) => changedFilePath === normalizedFilePath || changedFilePath.endsWith(`/${normalizedFilePath}`));
244
+ if (suffixMatches.length === 1) {
245
+ return {
246
+ repositoryRootPath,
247
+ repositoryRelativeFilePath: suffixMatches[0],
248
+ };
249
+ }
250
+ }
251
+ return {
252
+ repositoryRootPath,
253
+ repositoryRelativeFilePath: candidateFilePaths[0],
254
+ };
255
+ }
256
+ // Get git status for a project
257
+ router.get('/status', async (req, res) => {
258
+ const { project } = req.query;
259
+ if (!project) {
260
+ return res.status(400).json({ error: 'Project id is required' });
261
+ }
262
+ try {
263
+ const projectPath = await getActualProjectPath(req.user.id, project);
264
+ // Validate git repository
265
+ await validateGitRepository(projectPath);
266
+ const branch = await getCurrentBranchName(projectPath);
267
+ const hasCommits = await repositoryHasCommits(projectPath);
268
+ // Get git status
269
+ const { stdout: statusOutput } = await spawnAsync('git', ['status', '--porcelain'], { cwd: projectPath });
270
+ const modified = [];
271
+ const added = [];
272
+ const deleted = [];
273
+ const untracked = [];
274
+ statusOutput.split('\n').forEach(line => {
275
+ if (!line.trim())
276
+ return;
277
+ const status = line.substring(0, 2);
278
+ const file = line.substring(3);
279
+ if (status === 'M ' || status === ' M' || status === 'MM') {
280
+ modified.push(file);
281
+ }
282
+ else if (status === 'A ' || status === 'AM') {
283
+ added.push(file);
284
+ }
285
+ else if (status === 'D ' || status === ' D') {
286
+ deleted.push(file);
287
+ }
288
+ else if (status === '??') {
289
+ untracked.push(file);
290
+ }
291
+ });
292
+ res.json({
293
+ branch,
294
+ hasCommits,
295
+ modified,
296
+ added,
297
+ deleted,
298
+ untracked
299
+ });
300
+ }
301
+ catch (error) {
302
+ console.error('Git status error:', error);
303
+ res.json({
304
+ error: error.message.includes('not a git repository') || error.message.includes('Project directory is not a git repository')
305
+ ? error.message
306
+ : 'Git operation failed',
307
+ details: error.message.includes('not a git repository') || error.message.includes('Project directory is not a git repository')
308
+ ? error.message
309
+ : `Failed to get git status: ${error.message}`
310
+ });
311
+ }
312
+ });
313
+ // Get diff for a specific file
314
+ router.get('/diff', async (req, res) => {
315
+ const { project, file } = req.query;
316
+ if (!project || !file) {
317
+ return res.status(400).json({ error: 'Project id and file path are required' });
318
+ }
319
+ try {
320
+ const projectPath = await getActualProjectPath(req.user.id, project);
321
+ // Validate git repository
322
+ await validateGitRepository(projectPath);
323
+ const { repositoryRootPath, repositoryRelativeFilePath, } = await resolveRepositoryFilePath(projectPath, file);
324
+ // Check if file is untracked or deleted
325
+ const { stdout: statusOutput } = await spawnAsync('git', ['status', '--porcelain', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
326
+ const isUntracked = statusOutput.startsWith('??');
327
+ const isDeleted = statusOutput.trim().startsWith('D ') || statusOutput.trim().startsWith(' D');
328
+ let diff;
329
+ if (isUntracked) {
330
+ // For untracked files, show the entire file content as additions
331
+ const filePath = path.join(repositoryRootPath, repositoryRelativeFilePath);
332
+ const stats = await fs.stat(filePath);
333
+ if (stats.isDirectory()) {
334
+ // For directories, show a simple message
335
+ diff = `Directory: ${repositoryRelativeFilePath}\n(Cannot show diff for directories)`;
336
+ }
337
+ else {
338
+ const fileContent = await fs.readFile(filePath, 'utf-8');
339
+ const lines = fileContent.split('\n');
340
+ diff = `--- /dev/null\n+++ b/${repositoryRelativeFilePath}\n@@ -0,0 +1,${lines.length} @@\n` +
341
+ lines.map(line => `+${line}`).join('\n');
342
+ }
343
+ }
344
+ else if (isDeleted) {
345
+ // For deleted files, show the entire file content from HEAD as deletions
346
+ const { stdout: fileContent } = await spawnAsync('git', ['show', `HEAD:${repositoryRelativeFilePath}`], { cwd: repositoryRootPath });
347
+ const lines = fileContent.split('\n');
348
+ diff = `--- a/${repositoryRelativeFilePath}\n+++ /dev/null\n@@ -1,${lines.length} +0,0 @@\n` +
349
+ lines.map(line => `-${line}`).join('\n');
350
+ }
351
+ else {
352
+ // Get diff for tracked files
353
+ // First check for unstaged changes (working tree vs index)
354
+ const { stdout: unstagedDiff } = await spawnAsync('git', ['diff', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
355
+ if (unstagedDiff) {
356
+ // Show unstaged changes if they exist
357
+ diff = stripDiffHeaders(unstagedDiff);
358
+ }
359
+ else {
360
+ // If no unstaged changes, check for staged changes (index vs HEAD)
361
+ const { stdout: stagedDiff } = await spawnAsync('git', ['diff', '--cached', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
362
+ diff = stripDiffHeaders(stagedDiff) || '';
363
+ }
364
+ }
365
+ res.json({ diff });
366
+ }
367
+ catch (error) {
368
+ console.error('Git diff error:', error);
369
+ res.json({ error: error.message });
370
+ }
371
+ });
372
+ // Get file content with diff information for CodeEditor
373
+ router.get('/file-with-diff', async (req, res) => {
374
+ const { project, file } = req.query;
375
+ if (!project || !file) {
376
+ return res.status(400).json({ error: 'Project id and file path are required' });
377
+ }
378
+ try {
379
+ const projectPath = await getActualProjectPath(req.user.id, project);
380
+ // Validate git repository
381
+ await validateGitRepository(projectPath);
382
+ const { repositoryRootPath, repositoryRelativeFilePath, } = await resolveRepositoryFilePath(projectPath, file);
383
+ // Check file status
384
+ const { stdout: statusOutput } = await spawnAsync('git', ['status', '--porcelain', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
385
+ const isUntracked = statusOutput.startsWith('??');
386
+ const isDeleted = statusOutput.trim().startsWith('D ') || statusOutput.trim().startsWith(' D');
387
+ let currentContent = '';
388
+ let oldContent = '';
389
+ if (isDeleted) {
390
+ // For deleted files, get content from HEAD
391
+ const { stdout: headContent } = await spawnAsync('git', ['show', `HEAD:${repositoryRelativeFilePath}`], { cwd: repositoryRootPath });
392
+ oldContent = headContent;
393
+ currentContent = headContent; // Show the deleted content in editor
394
+ }
395
+ else {
396
+ // Get current file content
397
+ const filePath = path.join(repositoryRootPath, repositoryRelativeFilePath);
398
+ const stats = await fs.stat(filePath);
399
+ if (stats.isDirectory()) {
400
+ // Cannot show content for directories
401
+ return res.status(400).json({ error: 'Cannot show diff for directories' });
402
+ }
403
+ currentContent = await fs.readFile(filePath, 'utf-8');
404
+ if (!isUntracked) {
405
+ // Get the old content from HEAD for tracked files
406
+ try {
407
+ const { stdout: headContent } = await spawnAsync('git', ['show', `HEAD:${repositoryRelativeFilePath}`], { cwd: repositoryRootPath });
408
+ oldContent = headContent;
409
+ }
410
+ catch (error) {
411
+ // File might be newly added to git (staged but not committed)
412
+ oldContent = '';
413
+ }
414
+ }
415
+ }
416
+ res.json({
417
+ currentContent,
418
+ oldContent,
419
+ isDeleted,
420
+ isUntracked
421
+ });
422
+ }
423
+ catch (error) {
424
+ console.error('Git file-with-diff error:', error);
425
+ res.json({ error: error.message });
426
+ }
427
+ });
428
+ // Create initial commit
429
+ router.post('/initial-commit', async (req, res) => {
430
+ const { project } = req.body;
431
+ if (!project) {
432
+ return res.status(400).json({ error: 'Project id is required' });
433
+ }
434
+ try {
435
+ const projectPath = await getActualProjectPath(req.user.id, project);
436
+ // Validate git repository
437
+ await validateGitRepository(projectPath);
438
+ // Check if there are already commits
439
+ try {
440
+ await spawnAsync('git', ['rev-parse', 'HEAD'], { cwd: projectPath });
441
+ return res.status(400).json({ error: 'Repository already has commits. Use regular commit instead.' });
442
+ }
443
+ catch (error) {
444
+ // No HEAD - this is good, we can create initial commit
445
+ }
446
+ // Add all files
447
+ await spawnAsync('git', ['add', '.'], { cwd: projectPath });
448
+ // Create initial commit
449
+ const { stdout } = await spawnAsync('git', ['commit', '-m', 'Initial commit'], { cwd: projectPath });
450
+ res.json({ success: true, output: stdout, message: 'Initial commit created successfully' });
451
+ }
452
+ catch (error) {
453
+ console.error('Git initial commit error:', error);
454
+ // Handle the case where there's nothing to commit
455
+ if (error.message.includes('nothing to commit')) {
456
+ return res.status(400).json({
457
+ error: 'Nothing to commit',
458
+ details: 'No files found in the repository. Add some files first.'
459
+ });
460
+ }
461
+ res.status(500).json({ error: error.message });
462
+ }
463
+ });
464
+ // Commit changes
465
+ router.post('/commit', async (req, res) => {
466
+ const { project, message, files } = req.body;
467
+ if (!project || !message || !files || files.length === 0) {
468
+ return res.status(400).json({ error: 'Project name, commit message, and files are required' });
469
+ }
470
+ try {
471
+ const projectPath = await getActualProjectPath(req.user.id, project);
472
+ // Validate git repository
473
+ await validateGitRepository(projectPath);
474
+ const repositoryRootPath = await getRepositoryRootPath(projectPath);
475
+ // Stage selected files
476
+ for (const file of files) {
477
+ const { repositoryRelativeFilePath } = await resolveRepositoryFilePath(projectPath, file);
478
+ await spawnAsync('git', ['add', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
479
+ }
480
+ // Commit with message
481
+ const { stdout } = await spawnAsync('git', ['commit', '-m', message], { cwd: repositoryRootPath });
482
+ res.json({ success: true, output: stdout });
483
+ }
484
+ catch (error) {
485
+ console.error('Git commit error:', error);
486
+ res.status(500).json({ error: error.message });
487
+ }
488
+ });
489
+ // Revert latest local commit (keeps changes staged)
490
+ router.post('/revert-local-commit', async (req, res) => {
491
+ const { project } = req.body;
492
+ if (!project) {
493
+ return res.status(400).json({ error: 'Project id is required' });
494
+ }
495
+ try {
496
+ const projectPath = await getActualProjectPath(req.user.id, project);
497
+ await validateGitRepository(projectPath);
498
+ try {
499
+ await spawnAsync('git', ['rev-parse', '--verify', 'HEAD'], { cwd: projectPath });
500
+ }
501
+ catch (error) {
502
+ return res.status(400).json({
503
+ error: 'No local commit to revert',
504
+ details: 'This repository has no commit yet.',
505
+ });
506
+ }
507
+ try {
508
+ // Soft reset rewinds one commit while preserving all file changes in the index.
509
+ await spawnAsync('git', ['reset', '--soft', 'HEAD~1'], { cwd: projectPath });
510
+ }
511
+ catch (error) {
512
+ const errorDetails = `${error.stderr || ''} ${error.message || ''}`;
513
+ const isInitialCommit = errorDetails.includes('HEAD~1') &&
514
+ (errorDetails.includes('unknown revision') || errorDetails.includes('ambiguous argument'));
515
+ if (!isInitialCommit) {
516
+ throw error;
517
+ }
518
+ // Initial commit has no parent; deleting HEAD uncommits it and keeps files staged.
519
+ await spawnAsync('git', ['update-ref', '-d', 'HEAD'], { cwd: projectPath });
520
+ }
521
+ res.json({
522
+ success: true,
523
+ output: 'Latest local commit reverted successfully. Changes were kept staged.',
524
+ });
525
+ }
526
+ catch (error) {
527
+ console.error('Git revert local commit error:', error);
528
+ res.status(500).json({ error: error.message });
529
+ }
530
+ });
531
+ // Get list of branches
532
+ router.get('/branches', async (req, res) => {
533
+ const { project } = req.query;
534
+ if (!project) {
535
+ return res.status(400).json({ error: 'Project id is required' });
536
+ }
537
+ try {
538
+ const projectPath = await getActualProjectPath(req.user.id, project);
539
+ // Validate git repository
540
+ await validateGitRepository(projectPath);
541
+ // Get all branches
542
+ const { stdout } = await spawnAsync('git', ['branch', '-a'], { cwd: projectPath });
543
+ const rawLines = stdout
544
+ .split('\n')
545
+ .map(b => b.trim())
546
+ .filter(b => b && !b.includes('->'));
547
+ // Local branches (may start with '* ' for current)
548
+ const localBranches = rawLines
549
+ .filter(b => !b.startsWith('remotes/'))
550
+ .map(b => (b.startsWith('* ') ? b.substring(2) : b));
551
+ // Remote branches — strip 'remotes/<remote>/' prefix
552
+ const remoteBranches = rawLines
553
+ .filter(b => b.startsWith('remotes/'))
554
+ .map(b => b.replace(/^remotes\/[^/]+\//, ''))
555
+ .filter(name => !localBranches.includes(name)); // skip if already a local branch
556
+ // Backward-compat flat list (local + unique remotes, deduplicated)
557
+ const branches = [...localBranches, ...remoteBranches]
558
+ .filter((b, i, arr) => arr.indexOf(b) === i);
559
+ res.json({ branches, localBranches, remoteBranches });
560
+ }
561
+ catch (error) {
562
+ console.error('Git branches error:', error);
563
+ res.json({ error: error.message });
564
+ }
565
+ });
566
+ // Checkout branch
567
+ router.post('/checkout', async (req, res) => {
568
+ const { project, branch } = req.body;
569
+ if (!project || !branch) {
570
+ return res.status(400).json({ error: 'Project id and branch are required' });
571
+ }
572
+ try {
573
+ const projectPath = await getActualProjectPath(req.user.id, project);
574
+ // Checkout the branch
575
+ validateBranchName(branch);
576
+ const { stdout } = await spawnAsync('git', ['checkout', branch], { cwd: projectPath });
577
+ res.json({ success: true, output: stdout });
578
+ }
579
+ catch (error) {
580
+ console.error('Git checkout error:', error);
581
+ res.status(500).json({ error: error.message });
582
+ }
583
+ });
584
+ // Create new branch
585
+ router.post('/create-branch', async (req, res) => {
586
+ const { project, branch } = req.body;
587
+ if (!project || !branch) {
588
+ return res.status(400).json({ error: 'Project id and branch name are required' });
589
+ }
590
+ try {
591
+ const projectPath = await getActualProjectPath(req.user.id, project);
592
+ // Create and checkout new branch
593
+ validateBranchName(branch);
594
+ const { stdout } = await spawnAsync('git', ['checkout', '-b', branch], { cwd: projectPath });
595
+ res.json({ success: true, output: stdout });
596
+ }
597
+ catch (error) {
598
+ console.error('Git create branch error:', error);
599
+ res.status(500).json({ error: error.message });
600
+ }
601
+ });
602
+ // Delete a local branch
603
+ router.post('/delete-branch', async (req, res) => {
604
+ const { project, branch } = req.body;
605
+ if (!project || !branch) {
606
+ return res.status(400).json({ error: 'Project id and branch name are required' });
607
+ }
608
+ try {
609
+ const projectPath = await getActualProjectPath(req.user.id, project);
610
+ await validateGitRepository(projectPath);
611
+ // Safety: cannot delete the currently checked-out branch
612
+ const { stdout: currentBranch } = await spawnAsync('git', ['branch', '--show-current'], { cwd: projectPath });
613
+ if (currentBranch.trim() === branch) {
614
+ return res.status(400).json({ error: 'Cannot delete the currently checked-out branch' });
615
+ }
616
+ const { stdout } = await spawnAsync('git', ['branch', '-d', branch], { cwd: projectPath });
617
+ res.json({ success: true, output: stdout });
618
+ }
619
+ catch (error) {
620
+ console.error('Git delete branch error:', error);
621
+ res.status(500).json({ error: error.message });
622
+ }
623
+ });
624
+ // Get recent commits
625
+ router.get('/commits', async (req, res) => {
626
+ const { project, limit = 10 } = req.query;
627
+ if (!project) {
628
+ return res.status(400).json({ error: 'Project id is required' });
629
+ }
630
+ try {
631
+ const projectPath = await getActualProjectPath(req.user.id, project);
632
+ await validateGitRepository(projectPath);
633
+ const parsedLimit = Number.parseInt(String(limit), 10);
634
+ const safeLimit = Number.isFinite(parsedLimit) && parsedLimit > 0
635
+ ? Math.min(parsedLimit, 100)
636
+ : 10;
637
+ // Get commit log with stats
638
+ const { stdout } = await spawnAsync('git', ['log', '--pretty=format:%H|%an|%ae|%ad|%s', '--date=iso-strict', '-n', String(safeLimit)], { cwd: projectPath });
639
+ const commits = stdout
640
+ .split('\n')
641
+ .filter(line => line.trim())
642
+ .map(line => {
643
+ const [hash, author, email, date, ...messageParts] = line.split('|');
644
+ return {
645
+ hash,
646
+ author,
647
+ email,
648
+ date,
649
+ message: messageParts.join('|')
650
+ };
651
+ });
652
+ // Get stats for each commit
653
+ for (const commit of commits) {
654
+ try {
655
+ const { stdout: stats } = await spawnAsync('git', ['show', '--stat', '--format=', commit.hash], { cwd: projectPath });
656
+ commit.stats = stats.trim().split('\n').pop(); // Get the summary line
657
+ }
658
+ catch (error) {
659
+ commit.stats = '';
660
+ }
661
+ }
662
+ res.json({ commits });
663
+ }
664
+ catch (error) {
665
+ console.error('Git commits error:', error);
666
+ res.json({ error: error.message });
667
+ }
668
+ });
669
+ // Get diff for a specific commit
670
+ router.get('/commit-diff', async (req, res) => {
671
+ const { project, commit } = req.query;
672
+ if (!project || !commit) {
673
+ return res.status(400).json({ error: 'Project id and commit hash are required' });
674
+ }
675
+ try {
676
+ const projectPath = await getActualProjectPath(req.user.id, project);
677
+ // Validate commit reference (defense-in-depth)
678
+ validateCommitRef(commit);
679
+ // Get diff for the commit
680
+ const { stdout } = await spawnAsync('git', ['show', commit], { cwd: projectPath });
681
+ const isTruncated = stdout.length > COMMIT_DIFF_CHARACTER_LIMIT;
682
+ const diff = isTruncated
683
+ ? `${stdout.slice(0, COMMIT_DIFF_CHARACTER_LIMIT)}\n\n... Diff truncated to keep the UI responsive ...`
684
+ : stdout;
685
+ res.json({ diff, isTruncated });
686
+ }
687
+ catch (error) {
688
+ console.error('Git commit diff error:', error);
689
+ res.json({ error: error.message });
690
+ }
691
+ });
692
+ // Generate commit message based on staged changes using AI
693
+ router.post('/generate-commit-message', async (req, res) => {
694
+ const { project, files, provider = 'claude' } = req.body;
695
+ if (!project || !files || files.length === 0) {
696
+ return res.status(400).json({ error: 'Project id and files are required' });
697
+ }
698
+ // Validate provider
699
+ if (!['claude', 'cursor'].includes(provider)) {
700
+ return res.status(400).json({ error: 'provider must be "claude" or "cursor"' });
701
+ }
702
+ try {
703
+ const projectPath = await getActualProjectPath(req.user.id, project);
704
+ await validateGitRepository(projectPath);
705
+ const repositoryRootPath = await getRepositoryRootPath(projectPath);
706
+ // Get diff for selected files
707
+ let diffContext = '';
708
+ for (const file of files) {
709
+ try {
710
+ const { repositoryRelativeFilePath } = await resolveRepositoryFilePath(projectPath, file);
711
+ const { stdout } = await spawnAsync('git', ['diff', 'HEAD', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
712
+ if (stdout) {
713
+ diffContext += `\n--- ${repositoryRelativeFilePath} ---\n${stdout}`;
714
+ }
715
+ }
716
+ catch (error) {
717
+ console.error(`Error getting diff for ${file}:`, error);
718
+ }
719
+ }
720
+ // If no diff found, might be untracked files
721
+ if (!diffContext.trim()) {
722
+ // Try to get content of untracked files
723
+ for (const file of files) {
724
+ try {
725
+ const { repositoryRelativeFilePath } = await resolveRepositoryFilePath(projectPath, file);
726
+ const filePath = path.join(repositoryRootPath, repositoryRelativeFilePath);
727
+ const stats = await fs.stat(filePath);
728
+ if (!stats.isDirectory()) {
729
+ const content = await fs.readFile(filePath, 'utf-8');
730
+ diffContext += `\n--- ${repositoryRelativeFilePath} (new file) ---\n${content.substring(0, 1000)}\n`;
731
+ }
732
+ else {
733
+ diffContext += `\n--- ${repositoryRelativeFilePath} (new directory) ---\n`;
734
+ }
735
+ }
736
+ catch (error) {
737
+ console.error(`Error reading file ${file}:`, error);
738
+ }
739
+ }
740
+ }
741
+ // Generate commit message using AI
742
+ const message = await generateCommitMessageWithAI(files, diffContext, provider, projectPath);
743
+ res.json({ message });
744
+ }
745
+ catch (error) {
746
+ console.error('Generate commit message error:', error);
747
+ res.status(500).json({ error: error.message });
748
+ }
749
+ });
750
+ /**
751
+ * Generates a commit message using AI (Claude SDK or Cursor CLI)
752
+ * @param {Array<string>} files - List of changed files
753
+ * @param {string} diffContext - Git diff content
754
+ * @param {string} provider - 'claude' or 'cursor'
755
+ * @param {string} projectPath - Project directory path
756
+ * @returns {Promise<string>} Generated commit message
757
+ */
758
+ async function generateCommitMessageWithAI(files, diffContext, provider, projectPath) {
759
+ // Create the prompt
760
+ const prompt = `Generate a conventional commit message for these changes.
761
+
762
+ REQUIREMENTS:
763
+ - Format: type(scope): subject
764
+ - Include body explaining what changed and why
765
+ - Types: feat, fix, docs, style, refactor, perf, test, build, ci, chore
766
+ - Subject under 50 chars, body wrapped at 72 chars
767
+ - Focus on user-facing changes, not implementation details
768
+ - Consider what's being added AND removed
769
+ - Return ONLY the commit message (no markdown, explanations, or code blocks)
770
+
771
+ FILES CHANGED:
772
+ ${files.map(f => `- ${f}`).join('\n')}
773
+
774
+ DIFFS:
775
+ ${diffContext.substring(0, 4000)}
776
+
777
+ Generate the commit message:`;
778
+ try {
779
+ // Create a simple writer that collects the response
780
+ let responseText = '';
781
+ const writer = {
782
+ send: (data) => {
783
+ try {
784
+ const parsed = typeof data === 'string' ? JSON.parse(data) : data;
785
+ console.log('🔍 Writer received message type:', parsed.type);
786
+ // Handle different message formats from Claude SDK and Cursor CLI
787
+ // Claude SDK sends: {type: 'claude-response', data: {message: {content: [...]}}}
788
+ if (parsed.type === 'claude-response' && parsed.data) {
789
+ const message = parsed.data.message || parsed.data;
790
+ console.log('📦 Claude response message:', JSON.stringify(message, null, 2).substring(0, 500));
791
+ if (message.content && Array.isArray(message.content)) {
792
+ // Extract text from content array
793
+ for (const item of message.content) {
794
+ if (item.type === 'text' && item.text) {
795
+ console.log('✅ Extracted text chunk:', item.text.substring(0, 100));
796
+ responseText += item.text;
797
+ }
798
+ }
799
+ }
800
+ }
801
+ // Cursor CLI sends: {type: 'cursor-output', output: '...'}
802
+ else if (parsed.type === 'cursor-output' && parsed.output) {
803
+ console.log('✅ Cursor output:', parsed.output.substring(0, 100));
804
+ responseText += parsed.output;
805
+ }
806
+ // Also handle direct text messages
807
+ else if (parsed.type === 'text' && parsed.text) {
808
+ console.log('✅ Direct text:', parsed.text.substring(0, 100));
809
+ responseText += parsed.text;
810
+ }
811
+ }
812
+ catch (e) {
813
+ // Ignore parse errors
814
+ console.error('Error parsing writer data:', e);
815
+ }
816
+ },
817
+ setSessionId: () => { }, // No-op for this use case
818
+ };
819
+ console.log('🚀 Calling AI agent with provider:', provider);
820
+ console.log('📝 Prompt length:', prompt.length);
821
+ // Call the appropriate agent
822
+ if (provider === 'claude') {
823
+ await queryClaudeSDK(prompt, {
824
+ cwd: projectPath,
825
+ permissionMode: 'bypassPermissions',
826
+ model: 'sonnet'
827
+ }, writer);
828
+ }
829
+ else if (provider === 'cursor') {
830
+ await spawnCursor(prompt, {
831
+ cwd: projectPath,
832
+ skipPermissions: true
833
+ }, writer);
834
+ }
835
+ console.log('📊 Total response text collected:', responseText.length, 'characters');
836
+ console.log('📄 Response preview:', responseText.substring(0, 200));
837
+ // Clean up the response
838
+ const cleanedMessage = cleanCommitMessage(responseText);
839
+ console.log('🧹 Cleaned message:', cleanedMessage.substring(0, 200));
840
+ return cleanedMessage || 'chore: update files';
841
+ }
842
+ catch (error) {
843
+ console.error('Error generating commit message with AI:', error);
844
+ // Fallback to simple message
845
+ return `chore: update ${files.length} file${files.length !== 1 ? 's' : ''}`;
846
+ }
847
+ }
848
+ /**
849
+ * Cleans the AI-generated commit message by removing markdown, code blocks, and extra formatting
850
+ * @param {string} text - Raw AI response
851
+ * @returns {string} Clean commit message
852
+ */
853
+ function cleanCommitMessage(text) {
854
+ if (!text || !text.trim()) {
855
+ return '';
856
+ }
857
+ let cleaned = text.trim();
858
+ // Remove markdown code blocks
859
+ cleaned = cleaned.replace(/```[a-z]*\n/g, '');
860
+ cleaned = cleaned.replace(/```/g, '');
861
+ // Remove markdown headers
862
+ cleaned = cleaned.replace(/^#+\s*/gm, '');
863
+ // Remove leading/trailing quotes
864
+ cleaned = cleaned.replace(/^["']|["']$/g, '');
865
+ // If there are multiple lines, take everything (subject + body)
866
+ // Just clean up extra blank lines
867
+ cleaned = cleaned.replace(/\n{3,}/g, '\n\n');
868
+ // Remove any explanatory text before the actual commit message
869
+ // Look for conventional commit pattern and start from there
870
+ const conventionalCommitMatch = cleaned.match(/(feat|fix|docs|style|refactor|perf|test|build|ci|chore)(\(.+?\))?:.+/s);
871
+ if (conventionalCommitMatch) {
872
+ cleaned = cleaned.substring(cleaned.indexOf(conventionalCommitMatch[0]));
873
+ }
874
+ return cleaned.trim();
875
+ }
876
+ // Get remote status (ahead/behind commits with smart remote detection)
877
+ router.get('/remote-status', async (req, res) => {
878
+ const { project } = req.query;
879
+ if (!project) {
880
+ return res.status(400).json({ error: 'Project id is required' });
881
+ }
882
+ try {
883
+ const projectPath = await getActualProjectPath(req.user.id, project);
884
+ await validateGitRepository(projectPath);
885
+ const branch = await getCurrentBranchName(projectPath);
886
+ const hasCommits = await repositoryHasCommits(projectPath);
887
+ const { stdout: remoteOutput } = await spawnAsync('git', ['remote'], { cwd: projectPath });
888
+ const remotes = remoteOutput.trim().split('\n').filter(r => r.trim());
889
+ const hasRemote = remotes.length > 0;
890
+ const fallbackRemoteName = hasRemote
891
+ ? (remotes.includes('origin') ? 'origin' : remotes[0])
892
+ : null;
893
+ // Repositories initialized with `git init` can have a branch but no commits.
894
+ // Return a non-error state so the UI can show the initial-commit workflow.
895
+ if (!hasCommits) {
896
+ return res.json({
897
+ hasRemote,
898
+ hasUpstream: false,
899
+ branch,
900
+ remoteName: fallbackRemoteName,
901
+ ahead: 0,
902
+ behind: 0,
903
+ isUpToDate: false,
904
+ message: 'Repository has no commits yet'
905
+ });
906
+ }
907
+ // Check if there's a remote tracking branch (smart detection)
908
+ let trackingBranch;
909
+ let remoteName;
910
+ try {
911
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--abbrev-ref', `${branch}@{upstream}`], { cwd: projectPath });
912
+ trackingBranch = stdout.trim();
913
+ remoteName = trackingBranch.split('/')[0]; // Extract remote name (e.g., "origin/main" -> "origin")
914
+ }
915
+ catch (error) {
916
+ return res.json({
917
+ hasRemote,
918
+ hasUpstream: false,
919
+ branch,
920
+ remoteName: fallbackRemoteName,
921
+ message: 'No remote tracking branch configured'
922
+ });
923
+ }
924
+ // Get ahead/behind counts
925
+ const { stdout: countOutput } = await spawnAsync('git', ['rev-list', '--count', '--left-right', `${trackingBranch}...HEAD`], { cwd: projectPath });
926
+ const [behind, ahead] = countOutput.trim().split('\t').map(Number);
927
+ res.json({
928
+ hasRemote: true,
929
+ hasUpstream: true,
930
+ branch,
931
+ remoteBranch: trackingBranch,
932
+ remoteName,
933
+ ahead: ahead || 0,
934
+ behind: behind || 0,
935
+ isUpToDate: ahead === 0 && behind === 0
936
+ });
937
+ }
938
+ catch (error) {
939
+ console.error('Git remote status error:', error);
940
+ res.json({ error: error.message });
941
+ }
942
+ });
943
+ // Fetch from remote (using smart remote detection)
944
+ router.post('/fetch', async (req, res) => {
945
+ const { project } = req.body;
946
+ if (!project) {
947
+ return res.status(400).json({ error: 'Project id is required' });
948
+ }
949
+ try {
950
+ const projectPath = await getActualProjectPath(req.user.id, project);
951
+ await validateGitRepository(projectPath);
952
+ // Get current branch and its upstream remote
953
+ const branch = await getCurrentBranchName(projectPath);
954
+ let remoteName = 'origin'; // fallback
955
+ try {
956
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--abbrev-ref', `${branch}@{upstream}`], { cwd: projectPath });
957
+ remoteName = stdout.trim().split('/')[0]; // Extract remote name
958
+ }
959
+ catch (error) {
960
+ // No upstream, try to fetch from origin anyway
961
+ console.log('No upstream configured, using origin as fallback');
962
+ }
963
+ validateRemoteName(remoteName);
964
+ const { stdout } = await spawnAsync('git', ['fetch', remoteName], { cwd: projectPath });
965
+ res.json({ success: true, output: stdout || 'Fetch completed successfully', remoteName });
966
+ }
967
+ catch (error) {
968
+ console.error('Git fetch error:', error);
969
+ res.status(500).json({
970
+ error: 'Fetch failed',
971
+ details: error.message.includes('Could not resolve hostname')
972
+ ? 'Unable to connect to remote repository. Check your internet connection.'
973
+ : error.message.includes('fatal: \'origin\' does not appear to be a git repository')
974
+ ? 'No remote repository configured. Add a remote with: git remote add origin <url>'
975
+ : error.message
976
+ });
977
+ }
978
+ });
979
+ // Pull from remote (fetch + merge using smart remote detection)
980
+ router.post('/pull', async (req, res) => {
981
+ const { project } = req.body;
982
+ if (!project) {
983
+ return res.status(400).json({ error: 'Project id is required' });
984
+ }
985
+ try {
986
+ const projectPath = await getActualProjectPath(req.user.id, project);
987
+ await validateGitRepository(projectPath);
988
+ // Get current branch and its upstream remote
989
+ const branch = await getCurrentBranchName(projectPath);
990
+ let remoteName = 'origin'; // fallback
991
+ let remoteBranch = branch; // fallback
992
+ try {
993
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--abbrev-ref', `${branch}@{upstream}`], { cwd: projectPath });
994
+ const tracking = stdout.trim();
995
+ remoteName = tracking.split('/')[0]; // Extract remote name
996
+ remoteBranch = tracking.split('/').slice(1).join('/'); // Extract branch name
997
+ }
998
+ catch (error) {
999
+ // No upstream, use fallback
1000
+ console.log('No upstream configured, using origin/branch as fallback');
1001
+ }
1002
+ validateRemoteName(remoteName);
1003
+ validateBranchName(remoteBranch);
1004
+ const { stdout } = await spawnAsync('git', ['pull', remoteName, remoteBranch], { cwd: projectPath });
1005
+ res.json({
1006
+ success: true,
1007
+ output: stdout || 'Pull completed successfully',
1008
+ remoteName,
1009
+ remoteBranch
1010
+ });
1011
+ }
1012
+ catch (error) {
1013
+ console.error('Git pull error:', error);
1014
+ // Enhanced error handling for common pull scenarios
1015
+ let errorMessage = 'Pull failed';
1016
+ let details = error.message;
1017
+ if (error.message.includes('CONFLICT')) {
1018
+ errorMessage = 'Merge conflicts detected';
1019
+ details = 'Pull created merge conflicts. Please resolve conflicts manually in the editor, then commit the changes.';
1020
+ }
1021
+ else if (error.message.includes('Please commit your changes or stash them')) {
1022
+ errorMessage = 'Uncommitted changes detected';
1023
+ details = 'Please commit or stash your local changes before pulling.';
1024
+ }
1025
+ else if (error.message.includes('Could not resolve hostname')) {
1026
+ errorMessage = 'Network error';
1027
+ details = 'Unable to connect to remote repository. Check your internet connection.';
1028
+ }
1029
+ else if (error.message.includes('fatal: \'origin\' does not appear to be a git repository')) {
1030
+ errorMessage = 'Remote not configured';
1031
+ details = 'No remote repository configured. Add a remote with: git remote add origin <url>';
1032
+ }
1033
+ else if (error.message.includes('diverged')) {
1034
+ errorMessage = 'Branches have diverged';
1035
+ details = 'Your local branch and remote branch have diverged. Consider fetching first to review changes.';
1036
+ }
1037
+ res.status(500).json({
1038
+ error: errorMessage,
1039
+ details: details
1040
+ });
1041
+ }
1042
+ });
1043
+ // Push commits to remote repository
1044
+ router.post('/push', async (req, res) => {
1045
+ const { project } = req.body;
1046
+ if (!project) {
1047
+ return res.status(400).json({ error: 'Project id is required' });
1048
+ }
1049
+ try {
1050
+ const projectPath = await getActualProjectPath(req.user.id, project);
1051
+ await validateGitRepository(projectPath);
1052
+ // Get current branch and its upstream remote
1053
+ const branch = await getCurrentBranchName(projectPath);
1054
+ let remoteName = 'origin'; // fallback
1055
+ let remoteBranch = branch; // fallback
1056
+ try {
1057
+ const { stdout } = await spawnAsync('git', ['rev-parse', '--abbrev-ref', `${branch}@{upstream}`], { cwd: projectPath });
1058
+ const tracking = stdout.trim();
1059
+ remoteName = tracking.split('/')[0]; // Extract remote name
1060
+ remoteBranch = tracking.split('/').slice(1).join('/'); // Extract branch name
1061
+ }
1062
+ catch (error) {
1063
+ // No upstream, use fallback
1064
+ console.log('No upstream configured, using origin/branch as fallback');
1065
+ }
1066
+ validateRemoteName(remoteName);
1067
+ validateBranchName(remoteBranch);
1068
+ const { stdout } = await spawnAsync('git', ['push', remoteName, remoteBranch], { cwd: projectPath });
1069
+ res.json({
1070
+ success: true,
1071
+ output: stdout || 'Push completed successfully',
1072
+ remoteName,
1073
+ remoteBranch
1074
+ });
1075
+ }
1076
+ catch (error) {
1077
+ console.error('Git push error:', error);
1078
+ // Enhanced error handling for common push scenarios
1079
+ let errorMessage = 'Push failed';
1080
+ let details = error.message;
1081
+ if (error.message.includes('rejected')) {
1082
+ errorMessage = 'Push rejected';
1083
+ details = 'The remote has newer commits. Pull first to merge changes before pushing.';
1084
+ }
1085
+ else if (error.message.includes('non-fast-forward')) {
1086
+ errorMessage = 'Non-fast-forward push';
1087
+ details = 'Your branch is behind the remote. Pull the latest changes first.';
1088
+ }
1089
+ else if (error.message.includes('Could not resolve hostname')) {
1090
+ errorMessage = 'Network error';
1091
+ details = 'Unable to connect to remote repository. Check your internet connection.';
1092
+ }
1093
+ else if (error.message.includes('fatal: \'origin\' does not appear to be a git repository')) {
1094
+ errorMessage = 'Remote not configured';
1095
+ details = 'No remote repository configured. Add a remote with: git remote add origin <url>';
1096
+ }
1097
+ else if (error.message.includes('Permission denied')) {
1098
+ errorMessage = 'Authentication failed';
1099
+ details = 'Permission denied. Check your credentials or SSH keys.';
1100
+ }
1101
+ else if (error.message.includes('no upstream branch')) {
1102
+ errorMessage = 'No upstream branch';
1103
+ details = 'No upstream branch configured. Use: git push --set-upstream origin <branch>';
1104
+ }
1105
+ res.status(500).json({
1106
+ error: errorMessage,
1107
+ details: details
1108
+ });
1109
+ }
1110
+ });
1111
+ // Publish branch to remote (set upstream and push)
1112
+ router.post('/publish', async (req, res) => {
1113
+ const { project, branch } = req.body;
1114
+ if (!project || !branch) {
1115
+ return res.status(400).json({ error: 'Project id and branch are required' });
1116
+ }
1117
+ try {
1118
+ const projectPath = await getActualProjectPath(req.user.id, project);
1119
+ await validateGitRepository(projectPath);
1120
+ // Validate branch name
1121
+ validateBranchName(branch);
1122
+ // Get current branch to verify it matches the requested branch
1123
+ const currentBranchName = await getCurrentBranchName(projectPath);
1124
+ if (currentBranchName !== branch) {
1125
+ return res.status(400).json({
1126
+ error: `Branch mismatch. Current branch is ${currentBranchName}, but trying to publish ${branch}`
1127
+ });
1128
+ }
1129
+ // Check if remote exists
1130
+ let remoteName = 'origin';
1131
+ try {
1132
+ const { stdout } = await spawnAsync('git', ['remote'], { cwd: projectPath });
1133
+ const remotes = stdout.trim().split('\n').filter(r => r.trim());
1134
+ if (remotes.length === 0) {
1135
+ return res.status(400).json({
1136
+ error: 'No remote repository configured. Add a remote with: git remote add origin <url>'
1137
+ });
1138
+ }
1139
+ remoteName = remotes.includes('origin') ? 'origin' : remotes[0];
1140
+ }
1141
+ catch (error) {
1142
+ return res.status(400).json({
1143
+ error: 'No remote repository configured. Add a remote with: git remote add origin <url>'
1144
+ });
1145
+ }
1146
+ // Publish the branch (set upstream and push)
1147
+ validateRemoteName(remoteName);
1148
+ const { stdout } = await spawnAsync('git', ['push', '--set-upstream', remoteName, branch], { cwd: projectPath });
1149
+ res.json({
1150
+ success: true,
1151
+ output: stdout || 'Branch published successfully',
1152
+ remoteName,
1153
+ branch
1154
+ });
1155
+ }
1156
+ catch (error) {
1157
+ console.error('Git publish error:', error);
1158
+ // Enhanced error handling for common publish scenarios
1159
+ let errorMessage = 'Publish failed';
1160
+ let details = error.message;
1161
+ if (error.message.includes('rejected')) {
1162
+ errorMessage = 'Publish rejected';
1163
+ details = 'The remote branch already exists and has different commits. Use push instead.';
1164
+ }
1165
+ else if (error.message.includes('Could not resolve hostname')) {
1166
+ errorMessage = 'Network error';
1167
+ details = 'Unable to connect to remote repository. Check your internet connection.';
1168
+ }
1169
+ else if (error.message.includes('Permission denied')) {
1170
+ errorMessage = 'Authentication failed';
1171
+ details = 'Permission denied. Check your credentials or SSH keys.';
1172
+ }
1173
+ else if (error.message.includes('fatal:') && error.message.includes('does not appear to be a git repository')) {
1174
+ errorMessage = 'Remote not configured';
1175
+ details = 'Remote repository not properly configured. Check your remote URL.';
1176
+ }
1177
+ res.status(500).json({
1178
+ error: errorMessage,
1179
+ details: details
1180
+ });
1181
+ }
1182
+ });
1183
+ // Discard changes for a specific file
1184
+ router.post('/discard', async (req, res) => {
1185
+ const { project, file } = req.body;
1186
+ if (!project || !file) {
1187
+ return res.status(400).json({ error: 'Project id and file path are required' });
1188
+ }
1189
+ try {
1190
+ const projectPath = await getActualProjectPath(req.user.id, project);
1191
+ await validateGitRepository(projectPath);
1192
+ const { repositoryRootPath, repositoryRelativeFilePath, } = await resolveRepositoryFilePath(projectPath, file);
1193
+ // Check file status to determine correct discard command
1194
+ const { stdout: statusOutput } = await spawnAsync('git', ['status', '--porcelain', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
1195
+ if (!statusOutput.trim()) {
1196
+ return res.status(400).json({ error: 'No changes to discard for this file' });
1197
+ }
1198
+ const status = statusOutput.substring(0, 2);
1199
+ if (status === '??') {
1200
+ // Untracked file or directory - delete it
1201
+ const filePath = path.join(repositoryRootPath, repositoryRelativeFilePath);
1202
+ const stats = await fs.stat(filePath);
1203
+ if (stats.isDirectory()) {
1204
+ await fs.rm(filePath, { recursive: true, force: true });
1205
+ }
1206
+ else {
1207
+ await fs.unlink(filePath);
1208
+ }
1209
+ }
1210
+ else if (status.includes('M') || status.includes('D')) {
1211
+ // Modified or deleted file - restore from HEAD
1212
+ await spawnAsync('git', ['restore', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
1213
+ }
1214
+ else if (status.includes('A')) {
1215
+ // Added file - unstage it
1216
+ await spawnAsync('git', ['reset', 'HEAD', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
1217
+ }
1218
+ res.json({ success: true, message: `Changes discarded for ${repositoryRelativeFilePath}` });
1219
+ }
1220
+ catch (error) {
1221
+ console.error('Git discard error:', error);
1222
+ res.status(500).json({ error: error.message });
1223
+ }
1224
+ });
1225
+ // Delete untracked file
1226
+ router.post('/delete-untracked', async (req, res) => {
1227
+ const { project, file } = req.body;
1228
+ if (!project || !file) {
1229
+ return res.status(400).json({ error: 'Project id and file path are required' });
1230
+ }
1231
+ try {
1232
+ const projectPath = await getActualProjectPath(req.user.id, project);
1233
+ await validateGitRepository(projectPath);
1234
+ const { repositoryRootPath, repositoryRelativeFilePath, } = await resolveRepositoryFilePath(projectPath, file);
1235
+ // Check if file is actually untracked
1236
+ const { stdout: statusOutput } = await spawnAsync('git', ['status', '--porcelain', '--', repositoryRelativeFilePath], { cwd: repositoryRootPath });
1237
+ if (!statusOutput.trim()) {
1238
+ return res.status(400).json({ error: 'File is not untracked or does not exist' });
1239
+ }
1240
+ const status = statusOutput.substring(0, 2);
1241
+ if (status !== '??') {
1242
+ return res.status(400).json({ error: 'File is not untracked. Use discard for tracked files.' });
1243
+ }
1244
+ // Delete the untracked file or directory
1245
+ const filePath = path.join(repositoryRootPath, repositoryRelativeFilePath);
1246
+ const stats = await fs.stat(filePath);
1247
+ if (stats.isDirectory()) {
1248
+ // Use rm with recursive option for directories
1249
+ await fs.rm(filePath, { recursive: true, force: true });
1250
+ res.json({ success: true, message: `Untracked directory ${repositoryRelativeFilePath} deleted successfully` });
1251
+ }
1252
+ else {
1253
+ await fs.unlink(filePath);
1254
+ res.json({ success: true, message: `Untracked file ${repositoryRelativeFilePath} deleted successfully` });
1255
+ }
1256
+ }
1257
+ catch (error) {
1258
+ console.error('Git delete untracked error:', error);
1259
+ res.status(500).json({ error: error.message });
1260
+ }
1261
+ });
1262
+ export default router;
1263
+ //# sourceMappingURL=git.js.map