@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,327 @@
1
+ import { spawn } from 'node:child_process';
2
+ import { access, mkdir, rm } from 'node:fs/promises';
3
+ import path from 'node:path';
4
+
5
+ import { githubTokensDb } from '@/modules/database/index.js';
6
+ import { createProject } from '@/modules/projects/services/project-management.service.js';
7
+ import type { WorkspacePathValidationResult } from '@/shared/types.js';
8
+ import { AppError, validateWorkspacePath } from '@/shared/utils.js';
9
+
10
+ type CloneProjectInput = {
11
+ workspacePath: string;
12
+ githubUrl: string;
13
+ githubTokenId?: number | null;
14
+ newGithubToken?: string | null;
15
+ userId: number | string;
16
+ };
17
+
18
+ type CloneCompletePayload = {
19
+ project: Record<string, unknown>;
20
+ message: string;
21
+ };
22
+
23
+ type CloneProjectEventHandlers = {
24
+ onProgress: (message: string) => void;
25
+ onComplete: (payload: CloneCompletePayload) => void;
26
+ };
27
+
28
+ type GitCloneProcess = {
29
+ stdout: NodeJS.ReadableStream | null;
30
+ stderr: NodeJS.ReadableStream | null;
31
+ on(event: 'close', listener: (code: number | null) => void): void;
32
+ on(event: 'error', listener: (error: NodeJS.ErrnoException) => void): void;
33
+ kill(): void;
34
+ };
35
+
36
+ type CloneProjectDependencies = {
37
+ validatePath: (requestedPath: string) => Promise<WorkspacePathValidationResult>;
38
+ ensureDirectory: (directoryPath: string) => Promise<void>;
39
+ pathExists: (targetPath: string) => Promise<boolean>;
40
+ removePath: (targetPath: string) => Promise<void>;
41
+ getGithubTokenById: (
42
+ tokenId: number,
43
+ userId: number,
44
+ ) => Promise<{ github_token: string } | null>;
45
+ spawnGitClone: (cloneUrl: string, clonePath: string) => GitCloneProcess;
46
+ registerProject: (
47
+ userId: number,
48
+ projectPath: string,
49
+ customName: string,
50
+ ) => Promise<{ project: Record<string, unknown> }>;
51
+ logError: (message: string, error: unknown) => void;
52
+ };
53
+
54
+ export type CloneProjectOperation = {
55
+ waitForCompletion: Promise<void>;
56
+ cancel: () => void;
57
+ };
58
+
59
+ async function defaultPathExists(targetPath: string): Promise<boolean> {
60
+ try {
61
+ await access(targetPath);
62
+ return true;
63
+ } catch (error) {
64
+ if ((error as NodeJS.ErrnoException).code === 'ENOENT') {
65
+ return false;
66
+ }
67
+
68
+ throw error;
69
+ }
70
+ }
71
+
72
+ function sanitizeGitError(message: string, token: string | null): string {
73
+ if (!message || !token) {
74
+ return message;
75
+ }
76
+
77
+ const escapedToken = token.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
78
+ return message.replace(new RegExp(escapedToken, 'g'), '***');
79
+ }
80
+
81
+ function resolveCloneFailureMessage(lastError: string, sanitizedError: string): string {
82
+ if (lastError.includes('Authentication failed') || lastError.includes('could not read Username')) {
83
+ return 'Authentication failed. Please check your credentials.';
84
+ }
85
+
86
+ if (lastError.includes('Repository not found')) {
87
+ return 'Repository not found. Please check the URL and ensure you have access.';
88
+ }
89
+
90
+ if (lastError.includes('already exists')) {
91
+ return 'Directory already exists';
92
+ }
93
+
94
+ if (sanitizedError) {
95
+ return sanitizedError;
96
+ }
97
+
98
+ return 'Git clone failed';
99
+ }
100
+
101
+ function resolveErrorMessage(error: unknown): string {
102
+ if (error instanceof AppError) {
103
+ return error.message;
104
+ }
105
+
106
+ if (error instanceof Error && error.message) {
107
+ return error.message;
108
+ }
109
+
110
+ return 'Unexpected error';
111
+ }
112
+
113
+ const defaultDependencies: CloneProjectDependencies = {
114
+ validatePath: validateWorkspacePath,
115
+ ensureDirectory: async (directoryPath: string): Promise<void> => {
116
+ await mkdir(directoryPath, { recursive: true });
117
+ },
118
+ pathExists: defaultPathExists,
119
+ removePath: async (targetPath: string): Promise<void> => {
120
+ await rm(targetPath, { recursive: true, force: true });
121
+ },
122
+ getGithubTokenById: async (
123
+ tokenId: number,
124
+ userId: number,
125
+ ): Promise<{ github_token: string } | null> => {
126
+ const tokenRow = githubTokensDb.getGithubTokenById(userId, tokenId) as
127
+ | { github_token: string }
128
+ | null;
129
+ return tokenRow;
130
+ },
131
+ spawnGitClone: (cloneUrl: string, clonePath: string): GitCloneProcess =>
132
+ spawn('git', ['clone', '--progress', '--', cloneUrl, clonePath], {
133
+ stdio: ['ignore', 'pipe', 'pipe'],
134
+ env: {
135
+ ...process.env,
136
+ GIT_TERMINAL_PROMPT: '0',
137
+ },
138
+ }) as unknown as GitCloneProcess,
139
+ registerProject: async (
140
+ userId: number,
141
+ projectPath: string,
142
+ customName: string,
143
+ ): Promise<{ project: Record<string, unknown> }> =>
144
+ createProject({
145
+ userId,
146
+ projectPath,
147
+ customName,
148
+ }) as Promise<{ project: Record<string, unknown> }>,
149
+ logError: (message: string, error: unknown): void => {
150
+ console.error(message, error);
151
+ },
152
+ };
153
+
154
+ export async function startCloneProject(
155
+ input: CloneProjectInput,
156
+ handlers: CloneProjectEventHandlers,
157
+ dependencies: CloneProjectDependencies = defaultDependencies,
158
+ ): Promise<CloneProjectOperation> {
159
+ const normalizedWorkspacePath = input.workspacePath.trim();
160
+ const normalizedGithubUrl = input.githubUrl.trim();
161
+
162
+ if (!normalizedWorkspacePath) {
163
+ throw new AppError('workspacePath and githubUrl are required', {
164
+ code: 'WORKSPACE_PATH_REQUIRED',
165
+ statusCode: 400,
166
+ });
167
+ }
168
+
169
+ if (!normalizedGithubUrl) {
170
+ throw new AppError('workspacePath and githubUrl are required', {
171
+ code: 'GITHUB_URL_REQUIRED',
172
+ statusCode: 400,
173
+ });
174
+ }
175
+
176
+ if (normalizedGithubUrl.startsWith('-')) {
177
+ throw new AppError('Invalid githubUrl', {
178
+ code: 'INVALID_GITHUB_URL',
179
+ statusCode: 400,
180
+ });
181
+ }
182
+
183
+ const pathValidation = await dependencies.validatePath(normalizedWorkspacePath);
184
+ if (!pathValidation.valid || !pathValidation.resolvedPath) {
185
+ throw new AppError(pathValidation.error || 'Invalid workspace path', {
186
+ code: 'INVALID_PROJECT_PATH',
187
+ statusCode: 400,
188
+ });
189
+ }
190
+
191
+ const absolutePath = pathValidation.resolvedPath;
192
+ await dependencies.ensureDirectory(absolutePath);
193
+
194
+ const numericUserId =
195
+ typeof input.userId === 'number' ? input.userId : Number.parseInt(String(input.userId), 10);
196
+ if (Number.isNaN(numericUserId)) {
197
+ throw new AppError('Authenticated user is required', {
198
+ code: 'AUTHENTICATION_REQUIRED',
199
+ statusCode: 401,
200
+ });
201
+ }
202
+
203
+ let githubToken: string | null = null;
204
+ if (typeof input.githubTokenId === 'number') {
205
+ const token = await dependencies.getGithubTokenById(input.githubTokenId, numericUserId);
206
+ if (!token) {
207
+ throw new AppError('GitHub token not found', {
208
+ code: 'GITHUB_TOKEN_NOT_FOUND',
209
+ statusCode: 404,
210
+ });
211
+ }
212
+
213
+ githubToken = token.github_token;
214
+ } else if (input.newGithubToken && input.newGithubToken.trim().length > 0) {
215
+ githubToken = input.newGithubToken.trim();
216
+ }
217
+
218
+ const sanitizedGithubUrl = normalizedGithubUrl.replace(/\/+$/, '').replace(/\.git$/, '');
219
+ const repoName = sanitizedGithubUrl.split('/').pop() || 'repository';
220
+ const clonePath = path.join(absolutePath, repoName);
221
+
222
+ if (await dependencies.pathExists(clonePath)) {
223
+ throw new AppError(
224
+ `Directory "${repoName}" already exists. Please choose a different location or remove the existing directory.`,
225
+ {
226
+ code: 'CLONE_TARGET_ALREADY_EXISTS',
227
+ statusCode: 409,
228
+ },
229
+ );
230
+ }
231
+
232
+ let cloneUrl = normalizedGithubUrl;
233
+ if (githubToken) {
234
+ try {
235
+ const url = new URL(normalizedGithubUrl);
236
+ url.username = githubToken;
237
+ url.password = '';
238
+ cloneUrl = url.toString();
239
+ } catch {
240
+ // SSH URLs cannot be represented by URL constructor and are used as-is.
241
+ }
242
+ }
243
+
244
+ handlers.onProgress(`Cloning into '${repoName}'...`);
245
+ const gitProcess = dependencies.spawnGitClone(cloneUrl, clonePath);
246
+ let lastError = '';
247
+
248
+ gitProcess.stdout?.on('data', (data: Buffer | string) => {
249
+ const message = data.toString().trim();
250
+ if (message) {
251
+ handlers.onProgress(message);
252
+ }
253
+ });
254
+
255
+ gitProcess.stderr?.on('data', (data: Buffer | string) => {
256
+ const message = data.toString().trim();
257
+ lastError = message;
258
+ if (message) {
259
+ handlers.onProgress(message);
260
+ }
261
+ });
262
+
263
+ const waitForCompletion = new Promise<void>((resolve, reject) => {
264
+ gitProcess.on('close', async (code) => {
265
+ if (code === 0) {
266
+ try {
267
+ const createdProject = await dependencies.registerProject(numericUserId, clonePath, repoName);
268
+ handlers.onComplete({
269
+ project: createdProject.project,
270
+ message: 'Repository cloned successfully',
271
+ });
272
+ resolve();
273
+ } catch (error) {
274
+ reject(
275
+ new AppError(`Clone succeeded but failed to add project: ${resolveErrorMessage(error)}`, {
276
+ code: 'CLONE_PROJECT_REGISTRATION_FAILED',
277
+ statusCode: 500,
278
+ }),
279
+ );
280
+ }
281
+ return;
282
+ }
283
+
284
+ const sanitizedError = sanitizeGitError(lastError, githubToken);
285
+ const errorMessage = resolveCloneFailureMessage(lastError, sanitizedError);
286
+
287
+ try {
288
+ await dependencies.removePath(clonePath);
289
+ } catch (cleanupError) {
290
+ dependencies.logError('Failed to clean up after clone failure:', cleanupError);
291
+ }
292
+
293
+ reject(
294
+ new AppError(errorMessage, {
295
+ code: 'GIT_CLONE_FAILED',
296
+ statusCode: 500,
297
+ }),
298
+ );
299
+ });
300
+
301
+ gitProcess.on('error', (error) => {
302
+ if (error.code === 'ENOENT') {
303
+ reject(
304
+ new AppError('Git is not installed or not in PATH', {
305
+ code: 'GIT_NOT_FOUND',
306
+ statusCode: 500,
307
+ }),
308
+ );
309
+ return;
310
+ }
311
+
312
+ reject(
313
+ new AppError(error.message, {
314
+ code: 'GIT_EXECUTION_FAILED',
315
+ statusCode: 500,
316
+ }),
317
+ );
318
+ });
319
+ });
320
+
321
+ return {
322
+ waitForCompletion,
323
+ cancel: () => {
324
+ gitProcess.kill();
325
+ },
326
+ };
327
+ }
@@ -0,0 +1,95 @@
1
+ import { promises as fs } from 'node:fs';
2
+ import path from 'node:path';
3
+
4
+ import { projectsDb, sessionsDb } from '@/modules/database/index.js';
5
+ import { AppError } from '@/shared/utils.js';
6
+
7
+ function uniqueJsonlPathsFromSessions(
8
+ sessions: Array<{ jsonl_path: string | null }>,
9
+ ): string[] {
10
+ const seen = new Set<string>();
11
+ const result: string[] = [];
12
+
13
+ for (const row of sessions) {
14
+ const raw = row.jsonl_path?.trim();
15
+ if (!raw) {
16
+ continue;
17
+ }
18
+ const absolute = path.isAbsolute(raw) ? path.normalize(raw) : path.resolve(raw);
19
+ if (seen.has(absolute)) {
20
+ continue;
21
+ }
22
+ seen.add(absolute);
23
+ result.push(absolute);
24
+ }
25
+
26
+ return result;
27
+ }
28
+
29
+ async function unlinkJsonlIfExists(filePath: string): Promise<void> {
30
+ try {
31
+ await fs.unlink(filePath);
32
+ } catch (error) {
33
+ const code = (error as NodeJS.ErrnoException).code;
34
+ if (code === 'ENOENT') {
35
+ return;
36
+ }
37
+ console.warn(`[project-delete] Failed to remove ${filePath}:`, (error as Error).message);
38
+ }
39
+ }
40
+
41
+ /**
42
+ * Loads all session rows for the project path (scoped to the user) and
43
+ * removes each distinct `jsonl_path` file on disk.
44
+ */
45
+ export async function deleteSessionJsonlFilesForProjectPath(userId: number, projectPath: string): Promise<void> {
46
+ const sessions = sessionsDb.getSessionsByProjectPathIncludingArchived(userId, projectPath);
47
+ const paths = uniqueJsonlPathsFromSessions(sessions);
48
+
49
+ for (const filePath of paths) {
50
+ await unlinkJsonlIfExists(filePath);
51
+ }
52
+ }
53
+
54
+ /**
55
+ * - **Soft delete** (`force` false): set `isArchived` on the `projects` row (hide from the active list; DB only).
56
+ * - **Force** (`force` true): for each session row for that `project_path`, delete the file at `jsonl_path`
57
+ * (when set), then remove session rows and the `projects` row.
58
+ *
59
+ * All operations are scoped to the authenticated `userId` so users only ever
60
+ * mutate rows they own.
61
+ */
62
+ export async function deleteOrArchiveProject(userId: number, projectId: string, force: boolean): Promise<void> {
63
+ const row = projectsDb.getProjectById(userId, projectId);
64
+ if (!row) {
65
+ throw new AppError(`Unknown projectId: ${projectId}`, {
66
+ code: 'PROJECT_NOT_FOUND',
67
+ statusCode: 404,
68
+ });
69
+ }
70
+
71
+ if (!force) {
72
+ projectsDb.updateProjectIsArchivedById(userId, projectId, true);
73
+ return;
74
+ }
75
+
76
+ await deleteSessionJsonlFilesForProjectPath(userId, row.project_path);
77
+ sessionsDb.deleteSessionsByProjectPath(userId, row.project_path);
78
+ projectsDb.deleteProjectById(userId, projectId);
79
+ }
80
+
81
+ /**
82
+ * Restores one archived project row back into the active project list (scoped
83
+ * to the requesting user).
84
+ */
85
+ export function restoreArchivedProject(userId: number, projectId: string): void {
86
+ const row = projectsDb.getProjectById(userId, projectId);
87
+ if (!row) {
88
+ throw new AppError(`Unknown projectId: ${projectId}`, {
89
+ code: 'PROJECT_NOT_FOUND',
90
+ statusCode: 404,
91
+ });
92
+ }
93
+
94
+ projectsDb.updateProjectIsArchivedById(userId, projectId, false);
95
+ }
@@ -0,0 +1,158 @@
1
+ import fs from 'node:fs/promises';
2
+ import path from 'node:path';
3
+
4
+ import { projectsDb } from '@/modules/database/index.js';
5
+ import type {
6
+ CreateProjectPathResult,
7
+ ProjectRepositoryRow,
8
+ WorkspacePathValidationResult,
9
+ } from '@/shared/types.js';
10
+ import { AppError, normalizeProjectPath, validateWorkspacePath } from '@/shared/utils.js';
11
+
12
+ type CreateProjectInput = {
13
+ userId: number;
14
+ projectPath: string;
15
+ customName?: string | null;
16
+ };
17
+
18
+ type CreateProjectDependencies = {
19
+ validatePath: (projectPath: string) => Promise<WorkspacePathValidationResult>;
20
+ ensureWorkspaceDirectory: (projectPath: string) => Promise<void>;
21
+ persistProjectPath: (userId: number, projectPath: string, customName: string | null) => CreateProjectPathResult;
22
+ getProjectByPath: (userId: number, projectPath: string) => ProjectRepositoryRow | null;
23
+ };
24
+
25
+ type ProjectApiView = {
26
+ projectId: string;
27
+ path: string;
28
+ fullPath: string;
29
+ displayName: string;
30
+ customName: string | null;
31
+ isArchived: boolean;
32
+ isStarred: boolean;
33
+ sessions: [];
34
+ cursorSessions: [];
35
+ codexSessions: [];
36
+ geminiSessions: [];
37
+ sessionMeta: {
38
+ hasMore: false;
39
+ total: 0;
40
+ };
41
+ };
42
+
43
+ type CreateProjectServiceResult = {
44
+ outcome: 'created' | 'reactivated_archived';
45
+ project: ProjectApiView;
46
+ };
47
+
48
+ const defaultDependencies: CreateProjectDependencies = {
49
+ validatePath: validateWorkspacePath,
50
+ ensureWorkspaceDirectory: async (projectPath: string): Promise<void> => {
51
+ await fs.mkdir(projectPath, { recursive: true });
52
+ const directoryStats = await fs.stat(projectPath);
53
+ if (!directoryStats.isDirectory()) {
54
+ throw new AppError('Path exists but is not a directory', {
55
+ code: 'PROJECT_PATH_NOT_DIRECTORY',
56
+ statusCode: 400,
57
+ });
58
+ }
59
+ },
60
+ persistProjectPath: (userId: number, projectPath: string, customName: string | null): CreateProjectPathResult =>
61
+ projectsDb.createProjectPath(userId, projectPath, customName),
62
+ getProjectByPath: (userId: number, projectPath: string): ProjectRepositoryRow | null =>
63
+ projectsDb.getProjectPath(userId, projectPath),
64
+ };
65
+
66
+ function resolveDisplayName(customName: string | null | undefined, projectPath: string): string {
67
+ const trimmedCustomName = typeof customName === 'string' ? customName.trim() : '';
68
+ if (trimmedCustomName.length > 0) {
69
+ return trimmedCustomName;
70
+ }
71
+
72
+ return path.basename(projectPath) || projectPath;
73
+ }
74
+
75
+ function mapProjectRowToApiView(projectRow: ProjectRepositoryRow): ProjectApiView {
76
+ return {
77
+ projectId: projectRow.project_id,
78
+ path: projectRow.project_path,
79
+ fullPath: projectRow.project_path,
80
+ displayName: resolveDisplayName(projectRow.custom_project_name, projectRow.project_path),
81
+ customName: projectRow.custom_project_name,
82
+ isArchived: Boolean(projectRow.isArchived),
83
+ isStarred: Boolean(projectRow.isStarred),
84
+ sessions: [],
85
+ cursorSessions: [],
86
+ codexSessions: [],
87
+ geminiSessions: [],
88
+ sessionMeta: {
89
+ hasMore: false,
90
+ total: 0,
91
+ },
92
+ };
93
+ }
94
+
95
+ export async function createProject(
96
+ input: CreateProjectInput,
97
+ dependencies: CreateProjectDependencies = defaultDependencies,
98
+ ): Promise<CreateProjectServiceResult> {
99
+ if (!Number.isFinite(input.userId)) {
100
+ throw new AppError('userId is required', {
101
+ code: 'AUTHENTICATION_REQUIRED',
102
+ statusCode: 401,
103
+ });
104
+ }
105
+
106
+ const normalizedPath = normalizeProjectPath(input.projectPath || '');
107
+ if (!normalizedPath) {
108
+ throw new AppError('path is required', {
109
+ code: 'PROJECT_PATH_REQUIRED',
110
+ statusCode: 400,
111
+ });
112
+ }
113
+
114
+ const pathValidation = await dependencies.validatePath(normalizedPath);
115
+ if (!pathValidation.valid || !pathValidation.resolvedPath) {
116
+ throw new AppError('Invalid project path', {
117
+ code: 'INVALID_PROJECT_PATH',
118
+ statusCode: 400,
119
+ details: pathValidation.error ?? 'Path validation failed',
120
+ });
121
+ }
122
+
123
+ const resolvedProjectPath = normalizeProjectPath(pathValidation.resolvedPath);
124
+ await dependencies.ensureWorkspaceDirectory(resolvedProjectPath);
125
+
126
+ const normalizedCustomName = resolveDisplayName(input.customName ?? null, resolvedProjectPath);
127
+ const persistedProject = dependencies.persistProjectPath(input.userId, resolvedProjectPath, normalizedCustomName);
128
+
129
+ if (persistedProject.outcome === 'active_conflict') {
130
+ throw new AppError('Project path already exists and is active', {
131
+ code: 'PROJECT_ALREADY_EXISTS',
132
+ statusCode: 409,
133
+ details: `Project path already exists: ${resolvedProjectPath}`,
134
+ });
135
+ }
136
+
137
+ const projectRow = persistedProject.project ?? dependencies.getProjectByPath(input.userId, resolvedProjectPath);
138
+ if (!projectRow) {
139
+ throw new AppError('Failed to resolve project after creation', {
140
+ code: 'PROJECT_CREATE_FAILED',
141
+ statusCode: 500,
142
+ });
143
+ }
144
+
145
+ // Archived rows intentionally remain archived when reused, as requested.
146
+ return {
147
+ outcome: persistedProject.outcome,
148
+ project: mapProjectRowToApiView(projectRow),
149
+ };
150
+ }
151
+
152
+ /**
153
+ * Sets `projects.custom_project_name` for the given user + `projectId` (or clears it when empty).
154
+ */
155
+ export function updateProjectDisplayName(userId: number, projectId: string, newDisplayName: unknown): void {
156
+ const trimmed = typeof newDisplayName === 'string' ? newDisplayName.trim() : '';
157
+ projectsDb.updateCustomProjectNameById(userId, projectId, trimmed.length > 0 ? trimmed : null);
158
+ }
@@ -0,0 +1,78 @@
1
+ import { projectsDb } from '@/modules/database/index.js';
2
+ import { AppError } from '@/shared/utils.js';
3
+
4
+ type ToggleProjectStarResult = {
5
+ isStarred: boolean;
6
+ };
7
+
8
+ type ApplyLegacyStarredProjectIdsResult = {
9
+ updated: number;
10
+ };
11
+
12
+ function normalizeProjectId(projectId: string): string {
13
+ return projectId.trim();
14
+ }
15
+
16
+ function uniqueProjectIds(projectIds: string[]): string[] {
17
+ const uniqueIds = new Set<string>();
18
+ for (const projectId of projectIds) {
19
+ const normalizedProjectId = normalizeProjectId(projectId);
20
+ if (!normalizedProjectId) {
21
+ continue;
22
+ }
23
+ uniqueIds.add(normalizedProjectId);
24
+ }
25
+ return [...uniqueIds];
26
+ }
27
+
28
+ /**
29
+ * Applies legacy `localStorage` stars keyed by DB `projectId` onto `projects.isStarred`.
30
+ *
31
+ * The operation is idempotent: already-starred projects are ignored, unknown ids are skipped.
32
+ */
33
+ export function applyLegacyStarredProjectIds(userId: number, projectIds: string[]): ApplyLegacyStarredProjectIdsResult {
34
+ const normalizedProjectIds = uniqueProjectIds(projectIds);
35
+ let updated = 0;
36
+
37
+ for (const projectId of normalizedProjectIds) {
38
+ const project = projectsDb.getProjectById(userId, projectId);
39
+ if (!project) {
40
+ continue;
41
+ }
42
+
43
+ if (Boolean(project.isStarred)) {
44
+ continue;
45
+ }
46
+
47
+ projectsDb.updateProjectIsStarredById(userId, projectId, true);
48
+ updated += 1;
49
+ }
50
+
51
+ return { updated };
52
+ }
53
+
54
+ /**
55
+ * Flips `projects.isStarred` for one project and returns the new state.
56
+ */
57
+ export function toggleProjectStar(userId: number, projectId: string): ToggleProjectStarResult {
58
+ const normalizedProjectId = normalizeProjectId(projectId);
59
+ if (!normalizedProjectId) {
60
+ throw new AppError('projectId is required', {
61
+ code: 'PROJECT_ID_REQUIRED',
62
+ statusCode: 400,
63
+ });
64
+ }
65
+
66
+ const project = projectsDb.getProjectById(userId, normalizedProjectId);
67
+ if (!project) {
68
+ throw new AppError('Project not found', {
69
+ code: 'PROJECT_NOT_FOUND',
70
+ statusCode: 404,
71
+ });
72
+ }
73
+
74
+ const nextStarredState = !Boolean(project.isStarred);
75
+ projectsDb.updateProjectIsStarredById(userId, normalizedProjectId, nextStarredState);
76
+
77
+ return { isStarred: nextStarredState };
78
+ }