@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,54 @@
1
+ /**
2
+ * GitHub tokens repository.
3
+ *
4
+ * Backward-compatible helper layer over generic credentials storage.
5
+ * Tokens are stored in `user_credentials` with `credential_type = 'github_token'`.
6
+ */
7
+ import { getConnection } from '../../../modules/database/connection.js';
8
+ import { credentialsDb } from '../../../modules/database/repositories/credentials.js';
9
+ const GITHUB_TOKEN_TYPE = 'github_token';
10
+ export const githubTokensDb = {
11
+ /** Creates a GitHub token credential entry. */
12
+ createGithubToken(userId, tokenName, githubToken, description = null) {
13
+ return credentialsDb.createCredential(userId, tokenName, GITHUB_TOKEN_TYPE, githubToken, description);
14
+ },
15
+ /** Returns all GitHub tokens (safe shape: no credential value). */
16
+ getGithubTokens(userId) {
17
+ return credentialsDb.getCredentials(userId, GITHUB_TOKEN_TYPE);
18
+ },
19
+ /** Returns the most recent active GitHub token value for a user. */
20
+ getActiveGithubToken(userId) {
21
+ return credentialsDb.getActiveCredential(userId, GITHUB_TOKEN_TYPE);
22
+ },
23
+ /**
24
+ * Returns a specific active GitHub token row by id/user, including
25
+ * a `github_token` compatibility field.
26
+ */
27
+ getGithubTokenById(userId, tokenId) {
28
+ const db = getConnection();
29
+ const row = db
30
+ .prepare(`SELECT *
31
+ FROM user_credentials
32
+ WHERE id = ? AND user_id = ? AND credential_type = ? AND is_active = 1`)
33
+ .get(tokenId, userId, GITHUB_TOKEN_TYPE);
34
+ if (!row)
35
+ return null;
36
+ return {
37
+ ...row,
38
+ github_token: row.credential_value,
39
+ };
40
+ },
41
+ /** Updates active state for a GitHub token. */
42
+ updateGithubToken(userId, tokenId, isActive) {
43
+ return credentialsDb.toggleCredential(userId, tokenId, isActive);
44
+ },
45
+ /** Deletes a GitHub token. */
46
+ deleteGithubToken(userId, tokenId) {
47
+ return credentialsDb.deleteCredential(userId, tokenId);
48
+ },
49
+ // Legacy alias used by existing routes
50
+ toggleGithubToken(userId, tokenId, isActive) {
51
+ return githubTokensDb.updateGithubToken(userId, tokenId, isActive);
52
+ },
53
+ };
54
+ //# sourceMappingURL=github-tokens.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"github-tokens.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/github-tokens.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,kCAAkC,CAAC;AACjE,OAAO,EAAE,aAAa,EAAE,MAAM,gDAAgD,CAAC;AAM/E,MAAM,iBAAiB,GAAG,cAAc,CAAC;AAiBzC,MAAM,CAAC,MAAM,cAAc,GAAG;IAC5B,+CAA+C;IAC/C,iBAAiB,CACf,MAAc,EACd,SAAiB,EACjB,WAAmB,EACnB,cAA6B,IAAI;QAEjC,OAAO,aAAa,CAAC,gBAAgB,CACnC,MAAM,EACN,SAAS,EACT,iBAAiB,EACjB,WAAW,EACX,WAAW,CACZ,CAAC;IACJ,CAAC;IAED,mEAAmE;IACnE,eAAe,CAAC,MAAc;QAC5B,OAAO,aAAa,CAAC,cAAc,CAAC,MAAM,EAAE,iBAAiB,CAAC,CAAC;IACjE,CAAC;IAED,oEAAoE;IACpE,oBAAoB,CAAC,MAAc;QACjC,OAAO,aAAa,CAAC,mBAAmB,CAAC,MAAM,EAAE,iBAAiB,CAAC,CAAC;IACtE,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAC,MAAc,EAAE,OAAe;QAChD,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,MAAM,GAAG,GAAG,EAAE;aACX,OAAO,CACN;;gFAEwE,CACzE;aACA,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,iBAAiB,CAA8B,CAAC;QAExE,IAAI,CAAC,GAAG;YAAE,OAAO,IAAI,CAAC;QAEtB,OAAO;YACL,GAAG,GAAG;YACN,YAAY,EAAE,GAAG,CAAC,gBAAgB;SACnC,CAAC;IACJ,CAAC;IAED,+CAA+C;IAC/C,iBAAiB,CACf,MAAc,EACd,OAAe,EACf,QAAiB;QAEjB,OAAO,aAAa,CAAC,gBAAgB,CAAC,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACnE,CAAC;IAED,8BAA8B;IAC9B,iBAAiB,CAAC,MAAc,EAAE,OAAe;QAC/C,OAAO,aAAa,CAAC,gBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACzD,CAAC;IAED,uCAAuC;IACvC,iBAAiB,CAAC,MAAc,EAAE,OAAe,EAAE,QAAiB;QAClE,OAAO,cAAc,CAAC,iBAAiB,CAAC,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACrE,CAAC;CACF,CAAC"}
@@ -0,0 +1,72 @@
1
+ /**
2
+ * Notification preferences repository.
3
+ *
4
+ * Stores per-user notification channel/event preferences as JSON.
5
+ */
6
+ import { getConnection } from '../../../modules/database/connection.js';
7
+ const DEFAULT_NOTIFICATION_PREFERENCES = {
8
+ channels: {
9
+ inApp: false,
10
+ webPush: false,
11
+ },
12
+ events: {
13
+ actionRequired: true,
14
+ stop: true,
15
+ error: true,
16
+ },
17
+ };
18
+ function normalizeNotificationPreferences(value) {
19
+ const source = value && typeof value === 'object' ? value : {};
20
+ return {
21
+ channels: {
22
+ inApp: source.channels?.inApp === true,
23
+ webPush: source.channels?.webPush === true,
24
+ },
25
+ events: {
26
+ actionRequired: source.events?.actionRequired !== false,
27
+ stop: source.events?.stop !== false,
28
+ error: source.events?.error !== false,
29
+ },
30
+ };
31
+ }
32
+ export const notificationPreferencesDb = {
33
+ /** Returns the normalized preferences for a user, creating defaults on first read. */
34
+ getNotificationPreferences(userId) {
35
+ const db = getConnection();
36
+ const row = db
37
+ .prepare('SELECT preferences_json FROM user_notification_preferences WHERE user_id = ?')
38
+ .get(userId);
39
+ if (!row) {
40
+ const defaults = normalizeNotificationPreferences(DEFAULT_NOTIFICATION_PREFERENCES);
41
+ db.prepare('INSERT INTO user_notification_preferences (user_id, preferences_json, updated_at) VALUES (?, ?, CURRENT_TIMESTAMP)').run(userId, JSON.stringify(defaults));
42
+ return defaults;
43
+ }
44
+ let parsed;
45
+ try {
46
+ parsed = JSON.parse(row.preferences_json);
47
+ }
48
+ catch {
49
+ parsed = DEFAULT_NOTIFICATION_PREFERENCES;
50
+ }
51
+ return normalizeNotificationPreferences(parsed);
52
+ },
53
+ /** Upserts normalized preferences for a user and returns the stored value. */
54
+ updateNotificationPreferences(userId, preferences) {
55
+ const normalized = normalizeNotificationPreferences(preferences);
56
+ const db = getConnection();
57
+ db.prepare(`INSERT INTO user_notification_preferences (user_id, preferences_json, updated_at)
58
+ VALUES (?, ?, CURRENT_TIMESTAMP)
59
+ ON CONFLICT(user_id) DO UPDATE SET
60
+ preferences_json = excluded.preferences_json,
61
+ updated_at = CURRENT_TIMESTAMP`).run(userId, JSON.stringify(normalized));
62
+ return normalized;
63
+ },
64
+ // Legacy aliases used by existing services/routes
65
+ getPreferences(userId) {
66
+ return notificationPreferencesDb.getNotificationPreferences(userId);
67
+ },
68
+ updatePreferences(userId, preferences) {
69
+ return notificationPreferencesDb.updateNotificationPreferences(userId, preferences);
70
+ },
71
+ };
72
+ //# sourceMappingURL=notification-preferences.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"notification-preferences.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/notification-preferences.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,kCAAkC,CAAC;AAcjE,MAAM,gCAAgC,GAA4B;IAChE,QAAQ,EAAE;QACR,KAAK,EAAE,KAAK;QACZ,OAAO,EAAE,KAAK;KACf;IACD,MAAM,EAAE;QACN,cAAc,EAAE,IAAI;QACpB,IAAI,EAAE,IAAI;QACV,KAAK,EAAE,IAAI;KACZ;CACF,CAAC;AAEF,SAAS,gCAAgC,CAAC,KAAc;IACtD,MAAM,MAAM,GAAG,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAE,KAA6B,CAAC,CAAC,CAAC,EAAE,CAAC;IAExF,OAAO;QACL,QAAQ,EAAE;YACR,KAAK,EAAE,MAAM,CAAC,QAAQ,EAAE,KAAK,KAAK,IAAI;YACtC,OAAO,EAAE,MAAM,CAAC,QAAQ,EAAE,OAAO,KAAK,IAAI;SAC3C;QACD,MAAM,EAAE;YACN,cAAc,EAAE,MAAM,CAAC,MAAM,EAAE,cAAc,KAAK,KAAK;YACvD,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,IAAI,KAAK,KAAK;YACnC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,KAAK,KAAK,KAAK;SACtC;KACF,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,MAAM,yBAAyB,GAAG;IACvC,sFAAsF;IACtF,0BAA0B,CAAC,MAAc;QACvC,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,MAAM,GAAG,GAAG,EAAE;aACX,OAAO,CACN,8EAA8E,CAC/E;aACA,GAAG,CAAC,MAAM,CAA6C,CAAC;QAE3D,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,MAAM,QAAQ,GAAG,gCAAgC,CAAC,gCAAgC,CAAC,CAAC;YACpF,EAAE,CAAC,OAAO,CACR,oHAAoH,CACrH,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC;YACxC,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,IAAI,MAAe,CAAC;QACpB,IAAI,CAAC;YACH,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAC5C,CAAC;QAAC,MAAM,CAAC;YACP,MAAM,GAAG,gCAAgC,CAAC;QAC5C,CAAC;QACD,OAAO,gCAAgC,CAAC,MAAM,CAAC,CAAC;IAClD,CAAC;IAED,8EAA8E;IAC9E,6BAA6B,CAC3B,MAAc,EACd,WAAoB;QAEpB,MAAM,UAAU,GAAG,gCAAgC,CAAC,WAAW,CAAC,CAAC;QACjE,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAE3B,EAAE,CAAC,OAAO,CACR;;;;wCAIkC,CACnC,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC,CAAC;QAE1C,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,kDAAkD;IAClD,cAAc,CAAC,MAAc;QAC3B,OAAO,yBAAyB,CAAC,0BAA0B,CAAC,MAAM,CAAC,CAAC;IACtE,CAAC;IACD,iBAAiB,CAAC,MAAc,EAAE,WAAoB;QACpD,OAAO,yBAAyB,CAAC,6BAA6B,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;IACtF,CAAC;CACF,CAAC"}
@@ -0,0 +1,67 @@
1
+ import assert from 'node:assert/strict';
2
+ import { mkdtemp, rm } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import path from 'node:path';
5
+ import test from 'node:test';
6
+ import { closeConnection } from '../../../modules/database/connection.js';
7
+ import { initializeDatabase } from '../../../modules/database/init-db.js';
8
+ import { projectsDb } from '../../../modules/database/repositories/projects.db.js';
9
+ import { userDb } from '../../../modules/database/repositories/users.js';
10
+ async function withIsolatedDatabase(runTest) {
11
+ const previousDatabasePath = process.env.DATABASE_PATH;
12
+ const tempDirectory = await mkdtemp(path.join(tmpdir(), 'projects-db-'));
13
+ const databasePath = path.join(tempDirectory, 'auth.db');
14
+ closeConnection();
15
+ process.env.DATABASE_PATH = databasePath;
16
+ await initializeDatabase();
17
+ const created = userDb.createUser('test-user', 'hash');
18
+ const userId = Number(created.id);
19
+ try {
20
+ await runTest(userId);
21
+ }
22
+ finally {
23
+ closeConnection();
24
+ if (previousDatabasePath === undefined) {
25
+ delete process.env.DATABASE_PATH;
26
+ }
27
+ else {
28
+ process.env.DATABASE_PATH = previousDatabasePath;
29
+ }
30
+ await rm(tempDirectory, { recursive: true, force: true });
31
+ }
32
+ }
33
+ test('projectsDb.createProjectPath returns created for fresh paths', async () => {
34
+ await withIsolatedDatabase((userId) => {
35
+ const created = projectsDb.createProjectPath(userId, '/workspace/new-project');
36
+ assert.equal(created.outcome, 'created');
37
+ assert.ok(created.project);
38
+ assert.equal(created.project?.project_path, '/workspace/new-project');
39
+ assert.equal(created.project?.isArchived, 0);
40
+ });
41
+ });
42
+ test('projectsDb.createProjectPath returns reactivated_archived for archived duplicates', async () => {
43
+ await withIsolatedDatabase((userId) => {
44
+ const initial = projectsDb.createProjectPath(userId, '/workspace/archived-project', 'Archived Project');
45
+ assert.equal(initial.outcome, 'created');
46
+ assert.ok(initial.project);
47
+ projectsDb.updateProjectIsArchived(userId, '/workspace/archived-project', true);
48
+ const reused = projectsDb.createProjectPath(userId, '/workspace/archived-project', 'Renamed Project');
49
+ assert.equal(reused.outcome, 'reactivated_archived');
50
+ assert.ok(reused.project);
51
+ assert.equal(reused.project?.project_id, initial.project?.project_id);
52
+ assert.equal(reused.project?.isArchived, 0);
53
+ });
54
+ });
55
+ test('projectsDb.createProjectPath returns active_conflict for active duplicates', async () => {
56
+ await withIsolatedDatabase((userId) => {
57
+ const initial = projectsDb.createProjectPath(userId, '/workspace/active-project');
58
+ assert.equal(initial.outcome, 'created');
59
+ assert.ok(initial.project);
60
+ const conflict = projectsDb.createProjectPath(userId, '/workspace/active-project');
61
+ assert.equal(conflict.outcome, 'active_conflict');
62
+ assert.ok(conflict.project);
63
+ assert.equal(conflict.project?.project_id, initial.project?.project_id);
64
+ assert.equal(conflict.project?.isArchived, 0);
65
+ });
66
+ });
67
+ //# sourceMappingURL=projects.db.integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.db.integration.test.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/projects.db.integration.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,185 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import path from 'node:path';
3
+ import { getConnection } from '../../../modules/database/connection.js';
4
+ import { normalizeProjectPath } from '../../../shared/utils.js';
5
+ function normalizeProjectDisplayName(projectPath, customProjectName) {
6
+ const trimmedCustomName = typeof customProjectName === 'string' ? customProjectName.trim() : '';
7
+ if (trimmedCustomName.length > 0) {
8
+ return trimmedCustomName;
9
+ }
10
+ const directoryName = path.basename(projectPath);
11
+ return directoryName || projectPath;
12
+ }
13
+ export const projectsDb = {
14
+ createProjectPath(userId, projectPath, customProjectName = null) {
15
+ const db = getConnection();
16
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
17
+ const normalizedProjectName = normalizeProjectDisplayName(normalizedProjectPath, customProjectName);
18
+ const attemptedId = randomUUID();
19
+ const row = db.prepare(`
20
+ INSERT INTO projects (project_id, user_id, project_path, custom_project_name, isArchived)
21
+ VALUES (?, ?, ?, ?, 0)
22
+ ON CONFLICT(user_id, project_path) DO UPDATE SET
23
+ isArchived = 0
24
+ WHERE projects.isArchived = 1
25
+ RETURNING project_id, project_path, custom_project_name, isStarred, isArchived
26
+ `).get(attemptedId, userId, normalizedProjectPath, normalizedProjectName);
27
+ if (row) {
28
+ return {
29
+ outcome: row.project_id === attemptedId ? 'created' : 'reactivated_archived',
30
+ project: row,
31
+ };
32
+ }
33
+ const existingProject = projectsDb.getProjectPath(userId, normalizedProjectPath);
34
+ return {
35
+ outcome: 'active_conflict',
36
+ project: existingProject,
37
+ };
38
+ },
39
+ getProjectPath(userId, projectPath) {
40
+ const db = getConnection();
41
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
42
+ const row = db.prepare(`
43
+ SELECT project_id, project_path, custom_project_name, isStarred, isArchived
44
+ FROM projects
45
+ WHERE user_id = ? AND project_path = ?
46
+ `).get(userId, normalizedProjectPath);
47
+ return row ?? null;
48
+ },
49
+ getProjectById(userId, projectId) {
50
+ const db = getConnection();
51
+ const row = db.prepare(`
52
+ SELECT project_id, project_path, custom_project_name, isStarred, isArchived
53
+ FROM projects
54
+ WHERE user_id = ? AND project_id = ?
55
+ `).get(userId, projectId);
56
+ return row ?? null;
57
+ },
58
+ /**
59
+ * Resolve the absolute project directory from a database project_id.
60
+ *
61
+ * This is the canonical lookup used after the projectName → projectId migration:
62
+ * API routes receive the DB-assigned `projectId` and must resolve the real folder
63
+ * path through this helper before touching the filesystem. Returns `null` when the
64
+ * project row does not exist so callers can respond with a 404.
65
+ */
66
+ getProjectPathById(userId, projectId) {
67
+ const db = getConnection();
68
+ const row = db.prepare(`
69
+ SELECT project_path
70
+ FROM projects
71
+ WHERE user_id = ? AND project_id = ?
72
+ `).get(userId, projectId);
73
+ return row?.project_path ?? null;
74
+ },
75
+ getProjectPaths(userId) {
76
+ const db = getConnection();
77
+ return db.prepare(`
78
+ SELECT project_id, project_path, custom_project_name, isStarred, isArchived
79
+ FROM projects
80
+ WHERE user_id = ? AND isArchived = 0
81
+ `).all(userId);
82
+ },
83
+ /**
84
+ * Archived rows are queried separately so archive-focused UIs can present
85
+ * hidden workspaces without reintroducing them into the active sidebar list.
86
+ */
87
+ getArchivedProjectPaths(userId) {
88
+ const db = getConnection();
89
+ return db.prepare(`
90
+ SELECT project_id, project_path, custom_project_name, isStarred, isArchived
91
+ FROM projects
92
+ WHERE user_id = ? AND isArchived = 1
93
+ `).all(userId);
94
+ },
95
+ getCustomProjectName(userId, projectPath) {
96
+ const db = getConnection();
97
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
98
+ const row = db.prepare(`
99
+ SELECT custom_project_name
100
+ FROM projects
101
+ WHERE user_id = ? AND project_path = ?
102
+ `).get(userId, normalizedProjectPath);
103
+ return row?.custom_project_name ?? null;
104
+ },
105
+ updateCustomProjectName(userId, projectPath, customProjectName) {
106
+ const db = getConnection();
107
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
108
+ db.prepare(`
109
+ INSERT INTO projects (project_id, user_id, project_path, custom_project_name)
110
+ VALUES (?, ?, ?, ?)
111
+ ON CONFLICT(user_id, project_path) DO UPDATE SET custom_project_name = excluded.custom_project_name
112
+ `).run(randomUUID(), userId, normalizedProjectPath, customProjectName);
113
+ },
114
+ updateCustomProjectNameById(userId, projectId, customProjectName) {
115
+ const db = getConnection();
116
+ db.prepare(`
117
+ UPDATE projects
118
+ SET custom_project_name = ?
119
+ WHERE user_id = ? AND project_id = ?
120
+ `).run(customProjectName, userId, projectId);
121
+ },
122
+ updateProjectIsStarred(userId, projectPath, isStarred) {
123
+ const db = getConnection();
124
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
125
+ db.prepare(`
126
+ UPDATE projects
127
+ SET isStarred = ?
128
+ WHERE user_id = ? AND project_path = ?
129
+ `).run(isStarred ? 1 : 0, userId, normalizedProjectPath);
130
+ },
131
+ updateProjectIsStarredById(userId, projectId, isStarred) {
132
+ const db = getConnection();
133
+ db.prepare(`
134
+ UPDATE projects
135
+ SET isStarred = ?
136
+ WHERE user_id = ? AND project_id = ?
137
+ `).run(isStarred ? 1 : 0, userId, projectId);
138
+ },
139
+ updateProjectIsArchived(userId, projectPath, isArchived) {
140
+ const db = getConnection();
141
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
142
+ db.prepare(`
143
+ UPDATE projects
144
+ SET isArchived = ?
145
+ WHERE user_id = ? AND project_path = ?
146
+ `).run(isArchived ? 1 : 0, userId, normalizedProjectPath);
147
+ },
148
+ updateProjectIsArchivedById(userId, projectId, isArchived) {
149
+ const db = getConnection();
150
+ db.prepare(`
151
+ UPDATE projects
152
+ SET isArchived = ?
153
+ WHERE user_id = ? AND project_id = ?
154
+ `).run(isArchived ? 1 : 0, userId, projectId);
155
+ },
156
+ deleteProjectPath(userId, projectPath) {
157
+ const db = getConnection();
158
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
159
+ db.prepare(`
160
+ DELETE FROM projects
161
+ WHERE user_id = ? AND project_path = ?
162
+ `).run(userId, normalizedProjectPath);
163
+ },
164
+ deleteProjectById(userId, projectId) {
165
+ const db = getConnection();
166
+ db.prepare(`
167
+ DELETE FROM projects
168
+ WHERE user_id = ? AND project_id = ?
169
+ `).run(userId, projectId);
170
+ },
171
+ isProjectPathUsedByOthers(userId, projectPath) {
172
+ const db = getConnection();
173
+ const normalizedProjectPath = normalizeProjectPath(projectPath);
174
+ const rows = db.prepare(`
175
+ SELECT u.username FROM projects p
176
+ JOIN users u ON u.id = p.user_id
177
+ WHERE p.project_path = ? AND p.user_id != ? AND p.isArchived = 0 AND u.is_active = 1
178
+ `).all(normalizedProjectPath, userId);
179
+ return {
180
+ used: rows.length > 0,
181
+ usernames: rows.map(r => r.username),
182
+ };
183
+ },
184
+ };
185
+ //# sourceMappingURL=projects.db.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.db.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/projects.db.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Push subscriptions repository.
3
+ *
4
+ * Persists browser push subscription endpoints and keys per user.
5
+ */
6
+ import { getConnection } from '../../../modules/database/connection.js';
7
+ export const pushSubscriptionsDb = {
8
+ /** Upserts a push subscription endpoint for a user. */
9
+ createPushSubscription(userId, endpoint, keysP256dh, keysAuth) {
10
+ const db = getConnection();
11
+ db.prepare(`INSERT INTO push_subscriptions (user_id, endpoint, keys_p256dh, keys_auth)
12
+ VALUES (?, ?, ?, ?)
13
+ ON CONFLICT(endpoint) DO UPDATE SET
14
+ user_id = excluded.user_id,
15
+ keys_p256dh = excluded.keys_p256dh,
16
+ keys_auth = excluded.keys_auth`).run(userId, endpoint, keysP256dh, keysAuth);
17
+ },
18
+ /** Returns all subscriptions for a user. */
19
+ getPushSubscriptions(userId) {
20
+ const db = getConnection();
21
+ return db
22
+ .prepare('SELECT endpoint, keys_p256dh, keys_auth FROM push_subscriptions WHERE user_id = ?')
23
+ .all(userId);
24
+ },
25
+ /** Deletes one subscription by endpoint. */
26
+ deletePushSubscription(endpoint) {
27
+ const db = getConnection();
28
+ db.prepare('DELETE FROM push_subscriptions WHERE endpoint = ?').run(endpoint);
29
+ },
30
+ /** Deletes all subscriptions for a user. */
31
+ deletePushSubscriptionsForUser(userId) {
32
+ const db = getConnection();
33
+ db.prepare('DELETE FROM push_subscriptions WHERE user_id = ?').run(userId);
34
+ },
35
+ // Legacy aliases used by existing services/routes
36
+ saveSubscription(userId, endpoint, keysP256dh, keysAuth) {
37
+ pushSubscriptionsDb.createPushSubscription(userId, endpoint, keysP256dh, keysAuth);
38
+ },
39
+ getSubscriptions(userId) {
40
+ return pushSubscriptionsDb.getPushSubscriptions(userId);
41
+ },
42
+ removeSubscription(endpoint) {
43
+ pushSubscriptionsDb.deletePushSubscription(endpoint);
44
+ },
45
+ removeAllForUser(userId) {
46
+ pushSubscriptionsDb.deletePushSubscriptionsForUser(userId);
47
+ },
48
+ };
49
+ //# sourceMappingURL=push-subscriptions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"push-subscriptions.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/push-subscriptions.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,kCAAkC,CAAC;AAQjE,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,uDAAuD;IACvD,sBAAsB,CACpB,MAAc,EACd,QAAgB,EAChB,UAAkB,EAClB,QAAgB;QAEhB,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,EAAE,CAAC,OAAO,CACR;;;;;wCAKkC,CACnC,CAAC,GAAG,CAAC,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,QAAQ,CAAC,CAAC;IAChD,CAAC;IAED,4CAA4C;IAC5C,oBAAoB,CAAC,MAAc;QACjC,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,OAAO,EAAE;aACN,OAAO,CACN,mFAAmF,CACpF;aACA,GAAG,CAAC,MAAM,CAAgC,CAAC;IAChD,CAAC;IAED,4CAA4C;IAC5C,sBAAsB,CAAC,QAAgB;QACrC,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,EAAE,CAAC,OAAO,CAAC,mDAAmD,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAChF,CAAC;IAED,4CAA4C;IAC5C,8BAA8B,CAAC,MAAc;QAC3C,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,EAAE,CAAC,OAAO,CAAC,kDAAkD,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC7E,CAAC;IAED,kDAAkD;IAClD,gBAAgB,CACd,MAAc,EACd,QAAgB,EAChB,UAAkB,EAClB,QAAgB;QAEhB,mBAAmB,CAAC,sBAAsB,CACxC,MAAM,EACN,QAAQ,EACR,UAAU,EACV,QAAQ,CACT,CAAC;IACJ,CAAC;IACD,gBAAgB,CAAC,MAAc;QAC7B,OAAO,mBAAmB,CAAC,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAC1D,CAAC;IACD,kBAAkB,CAAC,QAAgB;QACjC,mBAAmB,CAAC,sBAAsB,CAAC,QAAQ,CAAC,CAAC;IACvD,CAAC;IACD,gBAAgB,CAAC,MAAc;QAC7B,mBAAmB,CAAC,8BAA8B,CAAC,MAAM,CAAC,CAAC;IAC7D,CAAC;CACF,CAAC"}
@@ -0,0 +1,31 @@
1
+ import { getConnection } from '../../../modules/database/connection.js';
2
+ export const scanStateDb = {
3
+ getLastScannedAt() {
4
+ const db = getConnection();
5
+ const row = db
6
+ .prepare(`SELECT last_scanned_at FROM scan_state WHERE id = 1`)
7
+ .get();
8
+ if (!row) {
9
+ return null; // Before any scan, the row is undefined.
10
+ }
11
+ let lastScannedDate = null;
12
+ const lastScannedStr = row.last_scanned_at;
13
+ if (lastScannedStr) {
14
+ // SQLite CURRENT_TIMESTAMP returns UTC in "YYYY-MM-DD HH:MM:SS" format.
15
+ // Replace space with 'T' and append 'Z' to parse reliably in JS across all platforms.
16
+ lastScannedDate = new Date(lastScannedStr.replace(' ', 'T') + 'Z');
17
+ }
18
+ return lastScannedDate;
19
+ },
20
+ updateLastScannedAt(scannedAt = new Date()) {
21
+ const db = getConnection();
22
+ const sqliteTimestamp = scannedAt.toISOString().slice(0, 19).replace('T', ' ');
23
+ db.prepare(`
24
+ INSERT INTO scan_state (id, last_scanned_at)
25
+ VALUES (1, ?)
26
+ ON CONFLICT (id)
27
+ DO UPDATE SET last_scanned_at = excluded.last_scanned_at
28
+ `).run(sqliteTimestamp);
29
+ }
30
+ };
31
+ //# sourceMappingURL=scan-state.db.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scan-state.db.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/scan-state.db.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,kCAAkC,CAAC;AAMjE,MAAM,CAAC,MAAM,WAAW,GAAG;IACvB,gBAAgB;QACZ,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAE3B,MAAM,GAAG,GAAG,EAAE;aACT,OAAO,CAAC,qDAAqD,CAAC;aAC9D,GAAG,EAAkB,CAAC;QAE3B,IAAI,CAAC,GAAG,EAAE,CAAC;YACP,OAAO,IAAI,CAAC,CAAC,yCAAyC;QAC1D,CAAC;QAED,IAAI,eAAe,GAAgB,IAAI,CAAC;QACxC,MAAM,cAAc,GAAG,GAAG,CAAC,eAAe,CAAC;QAE3C,IAAI,cAAc,EAAE,CAAC;YACjB,wEAAwE;YACxE,sFAAsF;YACtF,eAAe,GAAG,IAAI,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,GAAG,GAAG,CAAC,CAAC;QACvE,CAAC;QAED,OAAO,eAAe,CAAC;IAC3B,CAAC;IAED,mBAAmB,CAAC,YAAkB,IAAI,IAAI,EAAE;QAC5C,MAAM,EAAE,GAAG,aAAa,EAAE,CAAC;QAC3B,MAAM,eAAe,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QAE/E,EAAE,CAAC,OAAO,CAAC;;;;;SAKV,CAAC,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IAC5B,CAAC;CACJ,CAAC"}
@@ -0,0 +1,64 @@
1
+ import assert from 'node:assert/strict';
2
+ import { mkdtemp, rm } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import path from 'node:path';
5
+ import test from 'node:test';
6
+ import { closeConnection } from '../../../modules/database/connection.js';
7
+ import { initializeDatabase } from '../../../modules/database/init-db.js';
8
+ import { sessionsDb } from '../../../modules/database/repositories/sessions.db.js';
9
+ import { userDb } from '../../../modules/database/repositories/users.js';
10
+ async function withIsolatedDatabase(runTest) {
11
+ const previousDatabasePath = process.env.DATABASE_PATH;
12
+ const tempDirectory = await mkdtemp(path.join(tmpdir(), 'sessions-db-'));
13
+ const databasePath = path.join(tempDirectory, 'auth.db');
14
+ closeConnection();
15
+ process.env.DATABASE_PATH = databasePath;
16
+ await initializeDatabase();
17
+ const created = userDb.createUser('test-user', 'hash');
18
+ const userId = Number(created.id);
19
+ try {
20
+ await runTest(userId);
21
+ }
22
+ finally {
23
+ closeConnection();
24
+ if (previousDatabasePath === undefined) {
25
+ delete process.env.DATABASE_PATH;
26
+ }
27
+ else {
28
+ process.env.DATABASE_PATH = previousDatabasePath;
29
+ }
30
+ await rm(tempDirectory, { recursive: true, force: true });
31
+ }
32
+ }
33
+ test('session archive queries hide archived rows from active project views', async () => {
34
+ await withIsolatedDatabase((userId) => {
35
+ sessionsDb.createSession(userId, 'session-active', 'claude', '/workspace/demo-project', 'Active Session');
36
+ sessionsDb.createSession(userId, 'session-archived', 'claude', '/workspace/demo-project', 'Archived Session');
37
+ sessionsDb.updateSessionIsArchived(userId, 'session-archived', true);
38
+ const activeSessions = sessionsDb.getAllSessions(userId);
39
+ const archivedSessions = sessionsDb.getArchivedSessions(userId);
40
+ const activeProjectSessions = sessionsDb.getSessionsByProjectPath(userId, '/workspace/demo-project');
41
+ const allProjectSessions = sessionsDb.getSessionsByProjectPathIncludingArchived(userId, '/workspace/demo-project');
42
+ assert.deepEqual(activeSessions.map((session) => session.session_id), ['session-active']);
43
+ assert.deepEqual(archivedSessions.map((session) => session.session_id), ['session-archived']);
44
+ assert.deepEqual(activeProjectSessions.map((session) => session.session_id), ['session-active']);
45
+ assert.deepEqual(allProjectSessions.map((session) => session.session_id).sort(), ['session-active', 'session-archived']);
46
+ assert.equal(sessionsDb.countSessionsByProjectPath(userId, '/workspace/demo-project'), 1);
47
+ });
48
+ });
49
+ test('createSession reactivates archived rows when the session becomes active again', async () => {
50
+ await withIsolatedDatabase((userId) => {
51
+ sessionsDb.createSession(userId, 'session-reused', 'claude', '/workspace/demo-project', 'First Name');
52
+ sessionsDb.updateSessionIsArchived(userId, 'session-reused', true);
53
+ sessionsDb.createSession(userId, 'session-reused', 'claude', '/workspace/demo-project', 'Updated Name');
54
+ const activeSessions = sessionsDb.getAllSessions(userId);
55
+ const archivedSessions = sessionsDb.getArchivedSessions(userId);
56
+ const restoredSession = sessionsDb.getSessionById(userId, 'session-reused');
57
+ assert.equal(activeSessions.length, 1);
58
+ assert.equal(activeSessions[0]?.session_id, 'session-reused');
59
+ assert.equal(activeSessions[0]?.custom_name, 'Updated Name');
60
+ assert.equal(archivedSessions.length, 0);
61
+ assert.equal(restoredSession?.isArchived, 0);
62
+ });
63
+ });
64
+ //# sourceMappingURL=sessions.db.integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sessions.db.integration.test.js","sourceRoot":"","sources":["../../../../../server/modules/database/repositories/sessions.db.integration.test.ts"],"names":[],"mappings":"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"}