@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,984 @@
1
+ import fsSync, { promises as fs } from 'node:fs';
2
+ import path from 'node:path';
3
+ import readline from 'node:readline';
4
+ import { spawn } from 'cross-spawn';
5
+ import { rgPath } from '@vscode/ripgrep';
6
+ import { projectsDb, sessionsDb } from '../../../modules/database/index.js';
7
+ const SUPPORTED_PROVIDERS = new Set(['claude', 'codex', 'gemini']);
8
+ const MAX_MATCHES_PER_SESSION = 2;
9
+ const RIPGREP_FILE_CHUNK_SIZE = 40;
10
+ const RIPGREP_CHUNK_CONCURRENCY = 6;
11
+ const UNKNOWN_PROJECT_KEY = '__unknown_project__';
12
+ const INTERNAL_CONTENT_PREFIXES = [
13
+ '<system-reminder>',
14
+ 'Caveat:',
15
+ 'Invalid API key',
16
+ '[Request interrupted',
17
+ ];
18
+ /**
19
+ * Codex includes extra internal metadata tags that should not surface as
20
+ * user-facing searchable conversation content.
21
+ */
22
+ const CODEX_INTERNAL_CONTENT_PREFIXES = [
23
+ '<environment_context>',
24
+ '<cwd>',
25
+ ];
26
+ function normalizeComparablePath(inputPath) {
27
+ if (!inputPath || typeof inputPath !== 'string') {
28
+ return '';
29
+ }
30
+ const withoutLongPathPrefix = inputPath.startsWith('\\\\?\\')
31
+ ? inputPath.slice(4)
32
+ : inputPath;
33
+ const normalized = path.normalize(withoutLongPathPrefix.trim());
34
+ if (!normalized) {
35
+ return '';
36
+ }
37
+ const resolved = path.resolve(normalized);
38
+ return process.platform === 'win32' ? resolved.toLowerCase() : resolved;
39
+ }
40
+ function chunkArray(items, size) {
41
+ if (size <= 0) {
42
+ return [items];
43
+ }
44
+ const chunks = [];
45
+ for (let idx = 0; idx < items.length; idx += size) {
46
+ chunks.push(items.slice(idx, idx + size));
47
+ }
48
+ return chunks;
49
+ }
50
+ function getSessionKey(session) {
51
+ return `${session.provider}:${session.session_id}`;
52
+ }
53
+ function makeProjectKey(projectPath) {
54
+ const normalized = typeof projectPath === 'string' ? projectPath.trim() : '';
55
+ return normalized.length > 0 ? normalized : UNKNOWN_PROJECT_KEY;
56
+ }
57
+ function toSummaryText(customName, fallback, emptyLabel) {
58
+ const trimmedCustomName = typeof customName === 'string' ? customName.trim() : '';
59
+ if (trimmedCustomName) {
60
+ return trimmedCustomName;
61
+ }
62
+ const trimmedFallback = typeof fallback === 'string' ? fallback.trim() : '';
63
+ if (!trimmedFallback) {
64
+ return emptyLabel;
65
+ }
66
+ return trimmedFallback.length > 50 ? `${trimmedFallback.slice(0, 50)}...` : trimmedFallback;
67
+ }
68
+ function isInternalContent(content) {
69
+ return INTERNAL_CONTENT_PREFIXES.some((prefix) => content.startsWith(prefix));
70
+ }
71
+ function isInternalCodexContent(content) {
72
+ const normalized = content.trimStart();
73
+ return CODEX_INTERNAL_CONTENT_PREFIXES.some((prefix) => normalized.startsWith(prefix));
74
+ }
75
+ function escapeRegex(value) {
76
+ return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
77
+ }
78
+ function createWordMatcher(rawQuery, words) {
79
+ const normalizedQuery = rawQuery.trim().replace(/\s+/g, ' ');
80
+ const requireExactPhrase = words.length > 1 && normalizedQuery.length > 0;
81
+ const wordPatterns = words.map((word) => new RegExp(`(?<!\\p{L})${escapeRegex(word)}(?!\\p{L})`, 'u'));
82
+ const phrasePattern = words.map((word) => escapeRegex(word)).join('\\s+');
83
+ const phraseRegex = new RegExp(phrasePattern, 'iu');
84
+ const allWordsMatch = (textLower) => wordPatterns.every((pattern) => pattern.test(textLower));
85
+ const matchesQuery = (text) => {
86
+ if (typeof text !== 'string' || text.length === 0) {
87
+ return false;
88
+ }
89
+ if (requireExactPhrase) {
90
+ return phraseRegex.test(text);
91
+ }
92
+ if (phraseRegex.test(text)) {
93
+ return true;
94
+ }
95
+ if (words.length === 1) {
96
+ return allWordsMatch(text.toLowerCase());
97
+ }
98
+ return allWordsMatch(text.toLowerCase());
99
+ };
100
+ const buildSnippet = (text, snippetLen = 150) => {
101
+ const textLower = text.toLowerCase();
102
+ let firstIndex = -1;
103
+ let firstWordLen = 0;
104
+ let phraseStart = -1;
105
+ let phraseLength = 0;
106
+ const phraseMatch = phraseRegex.exec(text);
107
+ if (phraseMatch) {
108
+ phraseStart = phraseMatch.index;
109
+ phraseLength = phraseMatch[0].length;
110
+ firstIndex = phraseStart;
111
+ firstWordLen = phraseLength;
112
+ }
113
+ if (firstIndex === -1) {
114
+ for (const word of words) {
115
+ const regex = new RegExp(`(?<!\\p{L})${escapeRegex(word)}(?!\\p{L})`, 'u');
116
+ const match = regex.exec(textLower);
117
+ if (match && (firstIndex === -1 || match.index < firstIndex)) {
118
+ firstIndex = match.index;
119
+ firstWordLen = word.length;
120
+ }
121
+ }
122
+ }
123
+ if (firstIndex === -1) {
124
+ firstIndex = 0;
125
+ }
126
+ const halfLen = Math.floor(snippetLen / 2);
127
+ const start = Math.max(0, firstIndex - halfLen);
128
+ const end = Math.min(text.length, firstIndex + halfLen + firstWordLen);
129
+ const prefix = start > 0 ? '...' : '';
130
+ const suffix = end < text.length ? '...' : '';
131
+ const snippetBody = text.slice(start, end).replace(/\n/g, ' ');
132
+ const snippet = `${prefix}${snippetBody}${suffix}`;
133
+ const snippetLower = snippet.toLowerCase();
134
+ const highlights = [];
135
+ if (phraseStart >= start && phraseStart + phraseLength <= end) {
136
+ const phraseOffset = prefix.length + (phraseStart - start);
137
+ highlights.push({
138
+ start: phraseOffset,
139
+ end: phraseOffset + phraseLength,
140
+ });
141
+ }
142
+ if (!requireExactPhrase) {
143
+ for (const word of words) {
144
+ const regex = new RegExp(`(?<!\\p{L})${escapeRegex(word)}(?!\\p{L})`, 'gu');
145
+ let match = regex.exec(snippetLower);
146
+ while (match) {
147
+ highlights.push({ start: match.index, end: match.index + word.length });
148
+ match = regex.exec(snippetLower);
149
+ }
150
+ }
151
+ }
152
+ highlights.sort((left, right) => left.start - right.start);
153
+ const merged = [];
154
+ for (const highlight of highlights) {
155
+ const previous = merged[merged.length - 1];
156
+ if (previous && highlight.start <= previous.end) {
157
+ previous.end = Math.max(previous.end, highlight.end);
158
+ }
159
+ else {
160
+ merged.push({ ...highlight });
161
+ }
162
+ }
163
+ return { snippet, highlights: merged };
164
+ };
165
+ return { matchesQuery, buildSnippet };
166
+ }
167
+ function extractClaudeText(content) {
168
+ if (typeof content === 'string') {
169
+ return content;
170
+ }
171
+ if (!Array.isArray(content)) {
172
+ return '';
173
+ }
174
+ return content
175
+ .filter((part) => part?.type === 'text' && typeof part?.text === 'string')
176
+ .map((part) => String(part.text))
177
+ .join(' ');
178
+ }
179
+ function extractTaggedContent(content, tagName) {
180
+ const escapedTagName = tagName.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
181
+ const match = new RegExp(`<${escapedTagName}>([\\s\\S]*?)<\\/${escapedTagName}>`).exec(content);
182
+ return match ? match[1] : null;
183
+ }
184
+ function parseClaudeLocalCommandPayload(content) {
185
+ const commandName = extractTaggedContent(content, 'command-name');
186
+ const commandMessage = extractTaggedContent(content, 'command-message');
187
+ const commandArgs = extractTaggedContent(content, 'command-args');
188
+ if (commandName === null && commandMessage === null && commandArgs === null) {
189
+ return null;
190
+ }
191
+ return {
192
+ commandName: commandName ?? '',
193
+ commandMessage: commandMessage ?? '',
194
+ commandArgs: commandArgs ?? '',
195
+ };
196
+ }
197
+ function buildClaudeLocalCommandDisplayText(payload) {
198
+ const commandName = payload.commandName.trim();
199
+ const commandMessage = payload.commandMessage.trim();
200
+ const commandArgs = payload.commandArgs.trim();
201
+ const baseCommand = commandName || commandMessage;
202
+ if (!baseCommand) {
203
+ return '';
204
+ }
205
+ return commandArgs ? `${baseCommand} ${commandArgs}` : baseCommand;
206
+ }
207
+ function stripAnsiFormatting(text) {
208
+ return text.replace(/\u001B\[[0-9;?]*[ -/]*[@-~]/g, '');
209
+ }
210
+ /**
211
+ * Claude mixes visible chat, compact summaries, and local command wrappers into
212
+ * the same transcript stream. Search should operate on the user-visible meaning
213
+ * of those rows rather than the raw wrapper syntax.
214
+ */
215
+ function extractClaudeSearchableMessage(entry) {
216
+ if (!entry.message?.content || entry.isApiErrorMessage) {
217
+ return null;
218
+ }
219
+ const rawRole = entry.message.role;
220
+ if (rawRole !== 'user' && rawRole !== 'assistant') {
221
+ return null;
222
+ }
223
+ if (typeof entry.message.content === 'string') {
224
+ const content = String(entry.message.content);
225
+ if (entry.isCompactSummary === true && content.trim()) {
226
+ return {
227
+ text: content,
228
+ role: 'assistant',
229
+ };
230
+ }
231
+ const localCommand = parseClaudeLocalCommandPayload(content);
232
+ if (localCommand) {
233
+ const displayText = buildClaudeLocalCommandDisplayText(localCommand);
234
+ return displayText
235
+ ? {
236
+ text: displayText,
237
+ role: 'user',
238
+ }
239
+ : null;
240
+ }
241
+ const localCommandStdout = extractTaggedContent(content, 'local-command-stdout');
242
+ if (localCommandStdout !== null) {
243
+ const stdoutText = stripAnsiFormatting(localCommandStdout).trim();
244
+ return stdoutText
245
+ ? {
246
+ text: stdoutText,
247
+ role: 'assistant',
248
+ }
249
+ : null;
250
+ }
251
+ if (!content || isInternalContent(content)) {
252
+ return null;
253
+ }
254
+ return {
255
+ text: content,
256
+ role: rawRole,
257
+ };
258
+ }
259
+ const text = extractClaudeText(entry.message.content);
260
+ if (!text) {
261
+ return null;
262
+ }
263
+ if (entry.isCompactSummary === true) {
264
+ return {
265
+ text,
266
+ role: 'assistant',
267
+ };
268
+ }
269
+ if (isInternalContent(text)) {
270
+ return null;
271
+ }
272
+ return {
273
+ text,
274
+ role: rawRole,
275
+ };
276
+ }
277
+ function extractCodexText(content) {
278
+ if (typeof content === 'string') {
279
+ return content;
280
+ }
281
+ if (!Array.isArray(content)) {
282
+ return '';
283
+ }
284
+ return content
285
+ .map((item) => {
286
+ if (!item || typeof item !== 'object') {
287
+ return '';
288
+ }
289
+ const record = item;
290
+ if ((record.type === 'input_text' || record.type === 'output_text' || record.type === 'text')
291
+ && typeof record.text === 'string') {
292
+ return record.text;
293
+ }
294
+ return '';
295
+ })
296
+ .filter(Boolean)
297
+ .join(' ');
298
+ }
299
+ function extractGeminiText(content) {
300
+ if (typeof content === 'string') {
301
+ return content;
302
+ }
303
+ if (!Array.isArray(content)) {
304
+ return '';
305
+ }
306
+ return content
307
+ .filter((part) => typeof part?.text === 'string')
308
+ .map((part) => String(part.text))
309
+ .join(' ');
310
+ }
311
+ function normalizeSearchableSessions(userId, rows) {
312
+ const normalizedRows = [];
313
+ const projectArchiveStateByPath = new Map();
314
+ for (const row of rows) {
315
+ const provider = row.provider;
316
+ if (!SUPPORTED_PROVIDERS.has(provider)) {
317
+ continue;
318
+ }
319
+ const rawJsonlPath = typeof row.jsonl_path === 'string' ? row.jsonl_path.trim() : '';
320
+ if (!rawJsonlPath) {
321
+ continue;
322
+ }
323
+ const absoluteJsonlPath = path.resolve(rawJsonlPath);
324
+ if (!fsSync.existsSync(absoluteJsonlPath)) {
325
+ continue;
326
+ }
327
+ /**
328
+ * Active session rows can still belong to an archived project because
329
+ * project archiving intentionally preserves the underlying session data.
330
+ * Global conversation search should follow the visible workspace model,
331
+ * which means excluding any session whose owning project is archived.
332
+ *
333
+ * Cache the archive lookup per normalized project path so one search pass
334
+ * does not re-query the same project row for every session in that folder.
335
+ */
336
+ const normalizedProjectPath = typeof row.project_path === 'string' ? row.project_path.trim() : '';
337
+ if (normalizedProjectPath) {
338
+ if (!projectArchiveStateByPath.has(normalizedProjectPath)) {
339
+ const projectRow = projectsDb.getProjectPath(userId, normalizedProjectPath);
340
+ projectArchiveStateByPath.set(normalizedProjectPath, Boolean(projectRow?.isArchived));
341
+ }
342
+ if (projectArchiveStateByPath.get(normalizedProjectPath) === true) {
343
+ continue;
344
+ }
345
+ }
346
+ normalizedRows.push({
347
+ ...row,
348
+ provider,
349
+ jsonl_path: absoluteJsonlPath,
350
+ });
351
+ }
352
+ return normalizedRows;
353
+ }
354
+ function buildProjectBuckets(userId, searchableSessions) {
355
+ const projectBuckets = new Map();
356
+ const projectMetadataCache = new Map();
357
+ for (const session of searchableSessions) {
358
+ const key = makeProjectKey(session.project_path);
359
+ if (!projectBuckets.has(key)) {
360
+ if (!projectMetadataCache.has(key)) {
361
+ if (key === UNKNOWN_PROJECT_KEY) {
362
+ projectMetadataCache.set(key, {
363
+ projectId: null,
364
+ projectDisplayName: 'Unknown Project',
365
+ });
366
+ }
367
+ else {
368
+ const projectRow = projectsDb.getProjectPath(userId, key);
369
+ const customProjectName = typeof projectRow?.custom_project_name === 'string'
370
+ ? projectRow.custom_project_name.trim()
371
+ : '';
372
+ const displayName = customProjectName || path.basename(key) || key;
373
+ projectMetadataCache.set(key, {
374
+ projectId: projectRow?.project_id ?? null,
375
+ projectDisplayName: displayName,
376
+ });
377
+ }
378
+ }
379
+ const metadata = projectMetadataCache.get(key);
380
+ projectBuckets.set(key, {
381
+ key,
382
+ projectId: metadata.projectId,
383
+ projectName: key,
384
+ projectDisplayName: metadata.projectDisplayName,
385
+ sessions: [],
386
+ });
387
+ }
388
+ const bucket = projectBuckets.get(key);
389
+ bucket.sessions.push(session);
390
+ }
391
+ const buckets = Array.from(projectBuckets.values());
392
+ for (const bucket of buckets) {
393
+ bucket.sessions.sort((left, right) => {
394
+ const leftTs = new Date(left.updated_at || left.created_at || 0).getTime();
395
+ const rightTs = new Date(right.updated_at || right.created_at || 0).getTime();
396
+ return rightTs - leftTs;
397
+ });
398
+ }
399
+ return buckets;
400
+ }
401
+ /**
402
+ * Executes ripgrep with the file list explicitly provided from sessionsDb jsonl paths.
403
+ *
404
+ * This avoids recursive directory walks and uses a fixed known candidate list.
405
+ */
406
+ async function runRipgrepFilesWithMatches(pattern, filePaths, signal) {
407
+ if (!pattern || filePaths.length === 0 || signal?.aborted) {
408
+ return new Set();
409
+ }
410
+ return new Promise((resolve, reject) => {
411
+ const args = [
412
+ '--files-with-matches',
413
+ '--no-messages',
414
+ '--ignore-case',
415
+ '--fixed-strings',
416
+ '--',
417
+ pattern,
418
+ ...filePaths,
419
+ ];
420
+ const rg = spawn(rgPath, args, {
421
+ stdio: ['ignore', 'pipe', 'pipe'],
422
+ windowsHide: true,
423
+ });
424
+ const stdoutChunks = [];
425
+ const stderrChunks = [];
426
+ let aborted = false;
427
+ const abortListener = () => {
428
+ aborted = true;
429
+ rg.kill();
430
+ };
431
+ if (signal) {
432
+ signal.addEventListener('abort', abortListener, { once: true });
433
+ }
434
+ rg.stdout.on('data', (chunk) => {
435
+ stdoutChunks.push(chunk);
436
+ });
437
+ rg.stderr.on('data', (chunk) => {
438
+ stderrChunks.push(chunk);
439
+ });
440
+ rg.on('error', (error) => {
441
+ if (signal) {
442
+ signal.removeEventListener('abort', abortListener);
443
+ }
444
+ if (aborted || signal?.aborted) {
445
+ resolve(new Set());
446
+ return;
447
+ }
448
+ reject(error);
449
+ });
450
+ rg.on('close', (code) => {
451
+ if (signal) {
452
+ signal.removeEventListener('abort', abortListener);
453
+ }
454
+ if (aborted || signal?.aborted) {
455
+ resolve(new Set());
456
+ return;
457
+ }
458
+ if (code !== 0 && code !== 1) {
459
+ const stderr = Buffer.concat(stderrChunks).toString('utf8').trim();
460
+ reject(new Error(`ripgrep failed with code ${String(code)}: ${stderr}`));
461
+ return;
462
+ }
463
+ const stdout = Buffer.concat(stdoutChunks).toString('utf8');
464
+ const matchedPaths = new Set();
465
+ for (const line of stdout.split(/\r?\n/)) {
466
+ const trimmed = line.trim();
467
+ if (!trimmed) {
468
+ continue;
469
+ }
470
+ matchedPaths.add(normalizeComparablePath(trimmed));
471
+ }
472
+ resolve(matchedPaths);
473
+ });
474
+ });
475
+ }
476
+ async function findMatchedFileKeys(searchablePathEntries, rawQuery, words, signal) {
477
+ if (searchablePathEntries.length === 0 || words.length === 0 || signal?.aborted) {
478
+ return new Set();
479
+ }
480
+ const normalizedQuery = rawQuery.trim().replace(/\s+/g, ' ');
481
+ const requireExactPhrase = words.length > 1 && normalizedQuery.length > 0;
482
+ if (requireExactPhrase) {
483
+ let matchedForPhrase = searchablePathEntries.slice();
484
+ // Keep ripgrep as an over-approximation for exact phrase mode by requiring
485
+ // each word to appear somewhere in the file, then defer strict phrase
486
+ // validation to the in-memory matcher.
487
+ for (const word of words) {
488
+ if (signal?.aborted) {
489
+ return new Set();
490
+ }
491
+ const matchedForWord = new Set();
492
+ const fileChunks = chunkArray(matchedForPhrase.map((entry) => entry.absolutePath), RIPGREP_FILE_CHUNK_SIZE);
493
+ let nextChunkIndex = 0;
494
+ const workerCount = Math.min(RIPGREP_CHUNK_CONCURRENCY, fileChunks.length);
495
+ const workers = Array.from({ length: workerCount }, async () => {
496
+ while (nextChunkIndex < fileChunks.length && !signal?.aborted) {
497
+ const currentIndex = nextChunkIndex;
498
+ nextChunkIndex += 1;
499
+ const chunkMatches = await runRipgrepFilesWithMatches(word, fileChunks[currentIndex], signal);
500
+ for (const matchedPath of chunkMatches) {
501
+ matchedForWord.add(matchedPath);
502
+ }
503
+ }
504
+ });
505
+ await Promise.all(workers);
506
+ if (signal?.aborted) {
507
+ return new Set();
508
+ }
509
+ matchedForPhrase = matchedForPhrase.filter((entry) => matchedForWord.has(entry.normalizedPath));
510
+ if (matchedForPhrase.length === 0) {
511
+ break;
512
+ }
513
+ }
514
+ return new Set(matchedForPhrase.map((entry) => entry.normalizedPath));
515
+ }
516
+ let remainingEntries = searchablePathEntries.slice();
517
+ // Run one ripgrep pass per term and intersect by keeping only files that
518
+ // matched every query word.
519
+ for (const word of words) {
520
+ if (signal?.aborted) {
521
+ return new Set();
522
+ }
523
+ const matchedForWord = new Set();
524
+ const fileChunks = chunkArray(remainingEntries.map((entry) => entry.absolutePath), RIPGREP_FILE_CHUNK_SIZE);
525
+ let nextChunkIndex = 0;
526
+ const workerCount = Math.min(RIPGREP_CHUNK_CONCURRENCY, fileChunks.length);
527
+ const workers = Array.from({ length: workerCount }, async () => {
528
+ while (nextChunkIndex < fileChunks.length && !signal?.aborted) {
529
+ const currentIndex = nextChunkIndex;
530
+ nextChunkIndex += 1;
531
+ const chunkMatches = await runRipgrepFilesWithMatches(word, fileChunks[currentIndex], signal);
532
+ for (const matchedPath of chunkMatches) {
533
+ matchedForWord.add(matchedPath);
534
+ }
535
+ }
536
+ });
537
+ await Promise.all(workers);
538
+ if (signal?.aborted) {
539
+ return new Set();
540
+ }
541
+ remainingEntries = remainingEntries.filter((entry) => matchedForWord.has(entry.normalizedPath));
542
+ if (remainingEntries.length === 0) {
543
+ break;
544
+ }
545
+ }
546
+ return new Set(remainingEntries.map((entry) => entry.normalizedPath));
547
+ }
548
+ function addSessionMatch(runtime, matches, match) {
549
+ if (runtime.totalMatches >= runtime.limit || matches.length >= MAX_MATCHES_PER_SESSION) {
550
+ return;
551
+ }
552
+ matches.push(match);
553
+ runtime.totalMatches += 1;
554
+ }
555
+ async function parseClaudeSessionMatches(session, runtime) {
556
+ const fileKey = normalizeComparablePath(session.jsonl_path);
557
+ if (!fileKey) {
558
+ return null;
559
+ }
560
+ if (!runtime.claudeFileResultsCache.has(fileKey)) {
561
+ const sessionsForFile = runtime.claudeSessionsByFileKey.get(fileKey) || [];
562
+ const matchedSessionsForFile = sessionsForFile.filter((candidate) => runtime.matchedSessionKeys.has(getSessionKey(candidate)));
563
+ const targetSessions = matchedSessionsForFile.length > 0
564
+ ? matchedSessionsForFile
565
+ : [session];
566
+ const targetSessionIds = new Set(targetSessions.map((candidate) => candidate.session_id));
567
+ const customNameBySessionId = new Map();
568
+ for (const candidate of targetSessions) {
569
+ customNameBySessionId.set(candidate.session_id, candidate.custom_name ?? null);
570
+ }
571
+ const sessionStateById = new Map();
572
+ const getSessionState = (sessionId) => {
573
+ if (!sessionStateById.has(sessionId)) {
574
+ sessionStateById.set(sessionId, {
575
+ matches: [],
576
+ pendingSummaries: new Map(),
577
+ fallbackUserText: null,
578
+ fallbackAssistantText: null,
579
+ resolvedSummary: null,
580
+ });
581
+ }
582
+ return sessionStateById.get(sessionId);
583
+ };
584
+ let currentSessionId = null;
585
+ try {
586
+ const fileStream = fsSync.createReadStream(session.jsonl_path);
587
+ const rl = readline.createInterface({ input: fileStream, crlfDelay: Infinity });
588
+ for await (const line of rl) {
589
+ if (runtime.totalMatches >= runtime.limit || runtime.isAborted()) {
590
+ break;
591
+ }
592
+ if (!line.trim()) {
593
+ continue;
594
+ }
595
+ let entry;
596
+ try {
597
+ entry = JSON.parse(line);
598
+ }
599
+ catch {
600
+ continue;
601
+ }
602
+ if (entry.sessionId) {
603
+ currentSessionId = String(entry.sessionId);
604
+ }
605
+ const entrySessionId = entry.sessionId
606
+ ? String(entry.sessionId)
607
+ : currentSessionId;
608
+ if (!entrySessionId || !targetSessionIds.has(entrySessionId)) {
609
+ continue;
610
+ }
611
+ const state = getSessionState(entrySessionId);
612
+ if (entry.type === 'summary' && entry.summary) {
613
+ const summaryValue = String(entry.summary);
614
+ if (entry.sessionId) {
615
+ state.resolvedSummary = summaryValue;
616
+ }
617
+ else if (entry.leafUuid) {
618
+ state.pendingSummaries.set(String(entry.leafUuid), summaryValue);
619
+ }
620
+ }
621
+ if (!state.resolvedSummary && entry.parentUuid) {
622
+ const pendingSummary = state.pendingSummaries.get(String(entry.parentUuid));
623
+ if (pendingSummary) {
624
+ state.resolvedSummary = pendingSummary;
625
+ }
626
+ }
627
+ const searchableMessage = extractClaudeSearchableMessage(entry);
628
+ if (!searchableMessage) {
629
+ continue;
630
+ }
631
+ const { text, role } = searchableMessage;
632
+ /**
633
+ * Claude compact summaries are the most faithful session-summary source
634
+ * after a `/compact` because they describe the post-compaction state that
635
+ * the resumed session actually continues from. Prefer them over generic
636
+ * fallback user text when present.
637
+ */
638
+ if (entry.isCompactSummary === true) {
639
+ state.resolvedSummary = text;
640
+ }
641
+ if (role === 'user') {
642
+ state.fallbackUserText = text;
643
+ }
644
+ else {
645
+ state.fallbackAssistantText = text;
646
+ }
647
+ if (!runtime.matchesQuery(text)) {
648
+ continue;
649
+ }
650
+ const { snippet, highlights } = runtime.buildSnippet(text);
651
+ addSessionMatch(runtime, state.matches, {
652
+ role,
653
+ snippet,
654
+ highlights,
655
+ timestamp: entry.timestamp ? String(entry.timestamp) : null,
656
+ provider: 'claude',
657
+ messageUuid: entry.uuid ? String(entry.uuid) : null,
658
+ });
659
+ }
660
+ }
661
+ catch {
662
+ runtime.claudeFileResultsCache.set(fileKey, new Map());
663
+ return null;
664
+ }
665
+ const fileResults = new Map();
666
+ for (const [sessionId, state] of sessionStateById.entries()) {
667
+ if (state.matches.length === 0) {
668
+ continue;
669
+ }
670
+ fileResults.set(sessionId, {
671
+ sessionId,
672
+ provider: 'claude',
673
+ sessionSummary: toSummaryText(customNameBySessionId.get(sessionId) ?? null, state.resolvedSummary || state.fallbackUserText || state.fallbackAssistantText, 'New Session'),
674
+ matches: state.matches,
675
+ });
676
+ }
677
+ runtime.claudeFileResultsCache.set(fileKey, fileResults);
678
+ }
679
+ return runtime.claudeFileResultsCache.get(fileKey)?.get(session.session_id) ?? null;
680
+ }
681
+ function isVisibleCodexUserMessage(payload) {
682
+ if (!payload || payload.type !== 'user_message') {
683
+ return false;
684
+ }
685
+ if (payload.kind && payload.kind !== 'plain') {
686
+ return false;
687
+ }
688
+ return typeof payload.message === 'string' && payload.message.trim().length > 0;
689
+ }
690
+ async function parseCodexSessionMatches(session, runtime) {
691
+ const matches = [];
692
+ let latestUserMessageText = null;
693
+ const seenMessageFingerprints = new Set();
694
+ try {
695
+ const fileStream = fsSync.createReadStream(session.jsonl_path);
696
+ const rl = readline.createInterface({ input: fileStream, crlfDelay: Infinity });
697
+ for await (const line of rl) {
698
+ if (runtime.totalMatches >= runtime.limit || runtime.isAborted()) {
699
+ break;
700
+ }
701
+ if (!line.trim()) {
702
+ continue;
703
+ }
704
+ let entry;
705
+ try {
706
+ entry = JSON.parse(line);
707
+ }
708
+ catch {
709
+ continue;
710
+ }
711
+ let text = null;
712
+ let role = null;
713
+ if (entry.type === 'event_msg' && isVisibleCodexUserMessage(entry.payload)) {
714
+ text = String(entry.payload.message);
715
+ role = 'user';
716
+ }
717
+ else if (entry.type === 'event_msg'
718
+ && entry.payload?.type === 'agent_reasoning'
719
+ && typeof entry.payload?.text === 'string') {
720
+ text = String(entry.payload.text);
721
+ role = 'assistant';
722
+ }
723
+ else if (entry.type === 'response_item' && entry.payload?.type === 'message') {
724
+ const payload = entry.payload;
725
+ if (payload.role === 'user') {
726
+ text = extractCodexText(payload.content);
727
+ role = 'user';
728
+ }
729
+ else if (payload.role === 'assistant') {
730
+ text = extractCodexText(payload.content);
731
+ role = 'assistant';
732
+ }
733
+ }
734
+ else if (entry.type === 'response_item' && entry.payload?.type === 'reasoning') {
735
+ const summaryText = Array.isArray(entry.payload.summary)
736
+ ? entry.payload.summary
737
+ .map((item) => (typeof item?.text === 'string' ? item.text : ''))
738
+ .filter(Boolean)
739
+ .join('\n')
740
+ : '';
741
+ if (summaryText.trim()) {
742
+ text = summaryText;
743
+ role = 'assistant';
744
+ }
745
+ }
746
+ if (!text || !role) {
747
+ continue;
748
+ }
749
+ if (isInternalCodexContent(text)) {
750
+ continue;
751
+ }
752
+ if (role === 'user') {
753
+ latestUserMessageText = text;
754
+ }
755
+ const fingerprint = `${role}:${text.trim().toLowerCase()}`;
756
+ if (seenMessageFingerprints.has(fingerprint)) {
757
+ continue;
758
+ }
759
+ seenMessageFingerprints.add(fingerprint);
760
+ if (!runtime.matchesQuery(text)) {
761
+ continue;
762
+ }
763
+ const { snippet, highlights } = runtime.buildSnippet(text);
764
+ addSessionMatch(runtime, matches, {
765
+ role,
766
+ snippet,
767
+ highlights,
768
+ timestamp: entry.timestamp ? String(entry.timestamp) : null,
769
+ provider: 'codex',
770
+ });
771
+ }
772
+ }
773
+ catch {
774
+ return null;
775
+ }
776
+ if (matches.length === 0) {
777
+ return null;
778
+ }
779
+ return {
780
+ sessionId: session.session_id,
781
+ provider: 'codex',
782
+ sessionSummary: toSummaryText(session.custom_name, latestUserMessageText, 'Codex Session'),
783
+ matches,
784
+ };
785
+ }
786
+ async function parseGeminiSessionMatches(session, runtime) {
787
+ let data;
788
+ try {
789
+ data = await fs.readFile(session.jsonl_path, 'utf8');
790
+ }
791
+ catch {
792
+ return null;
793
+ }
794
+ let parsed;
795
+ try {
796
+ parsed = JSON.parse(data);
797
+ }
798
+ catch {
799
+ return null;
800
+ }
801
+ const sourceMessages = Array.isArray(parsed.messages) ? parsed.messages : [];
802
+ if (sourceMessages.length === 0) {
803
+ return null;
804
+ }
805
+ const matches = [];
806
+ let firstUserText = null;
807
+ for (const msg of sourceMessages) {
808
+ if (runtime.totalMatches >= runtime.limit || runtime.isAborted()) {
809
+ break;
810
+ }
811
+ const role = msg.type === 'user'
812
+ ? 'user'
813
+ : (msg.type === 'gemini' || msg.type === 'assistant')
814
+ ? 'assistant'
815
+ : null;
816
+ if (!role) {
817
+ continue;
818
+ }
819
+ const text = extractGeminiText(msg.content);
820
+ if (!text) {
821
+ continue;
822
+ }
823
+ if (role === 'user' && !firstUserText) {
824
+ firstUserText = text;
825
+ }
826
+ if (!runtime.matchesQuery(text)) {
827
+ continue;
828
+ }
829
+ const { snippet, highlights } = runtime.buildSnippet(text);
830
+ addSessionMatch(runtime, matches, {
831
+ role,
832
+ snippet,
833
+ highlights,
834
+ timestamp: msg.timestamp ? String(msg.timestamp) : null,
835
+ provider: 'gemini',
836
+ });
837
+ }
838
+ if (matches.length === 0) {
839
+ return null;
840
+ }
841
+ return {
842
+ sessionId: session.session_id,
843
+ provider: 'gemini',
844
+ sessionSummary: toSummaryText(session.custom_name, firstUserText, 'Gemini Session'),
845
+ matches,
846
+ };
847
+ }
848
+ async function parseSessionMatches(session, runtime) {
849
+ if (session.provider === 'claude') {
850
+ return parseClaudeSessionMatches(session, runtime);
851
+ }
852
+ if (session.provider === 'codex') {
853
+ return parseCodexSessionMatches(session, runtime);
854
+ }
855
+ return parseGeminiSessionMatches(session, runtime);
856
+ }
857
+ export async function searchConversations(userId, query, limit = 50, onProjectResult = null, signal = null) {
858
+ const safeQuery = typeof query === 'string' ? query.trim() : '';
859
+ const safeLimit = Math.max(1, Math.min(Number.isFinite(limit) ? limit : 50, 200));
860
+ const words = safeQuery.toLowerCase().split(/\s+/).filter((word) => word.length > 0);
861
+ if (words.length === 0) {
862
+ return { results: [], totalMatches: 0, query: safeQuery };
863
+ }
864
+ const isAborted = () => signal?.aborted === true;
865
+ if (isAborted()) {
866
+ return { results: [], totalMatches: 0, query: safeQuery };
867
+ }
868
+ const searchableSessions = normalizeSearchableSessions(userId, sessionsDb.getAllSessions(userId));
869
+ if (searchableSessions.length === 0) {
870
+ return { results: [], totalMatches: 0, query: safeQuery };
871
+ }
872
+ const sessionsByPathKey = new Map();
873
+ const searchablePathEntries = [];
874
+ for (const session of searchableSessions) {
875
+ const normalizedPath = normalizeComparablePath(session.jsonl_path);
876
+ if (!normalizedPath) {
877
+ continue;
878
+ }
879
+ if (!sessionsByPathKey.has(normalizedPath)) {
880
+ sessionsByPathKey.set(normalizedPath, []);
881
+ searchablePathEntries.push({
882
+ normalizedPath,
883
+ absolutePath: session.jsonl_path,
884
+ });
885
+ }
886
+ const pathSessions = sessionsByPathKey.get(normalizedPath);
887
+ pathSessions.push(session);
888
+ }
889
+ const matchedFileKeys = await findMatchedFileKeys(searchablePathEntries, safeQuery, words, signal ?? undefined);
890
+ if (isAborted() || matchedFileKeys.size === 0) {
891
+ return { results: [], totalMatches: 0, query: safeQuery };
892
+ }
893
+ const matchedSessionKeys = new Set();
894
+ for (const fileKey of matchedFileKeys) {
895
+ const sessions = sessionsByPathKey.get(fileKey);
896
+ if (!sessions) {
897
+ continue;
898
+ }
899
+ for (const session of sessions) {
900
+ matchedSessionKeys.add(getSessionKey(session));
901
+ }
902
+ }
903
+ const projectBuckets = buildProjectBuckets(userId, searchableSessions);
904
+ const totalProjects = projectBuckets.length;
905
+ const results = [];
906
+ let scannedProjects = 0;
907
+ const runtime = {
908
+ ...createWordMatcher(safeQuery, words),
909
+ limit: safeLimit,
910
+ totalMatches: 0,
911
+ isAborted,
912
+ matchedSessionKeys,
913
+ claudeSessionsByFileKey: new Map(),
914
+ claudeFileResultsCache: new Map(),
915
+ };
916
+ for (const [fileKey, sessions] of sessionsByPathKey.entries()) {
917
+ const claudeSessions = sessions.filter((session) => session.provider === 'claude');
918
+ if (claudeSessions.length > 0) {
919
+ runtime.claudeSessionsByFileKey.set(fileKey, claudeSessions);
920
+ }
921
+ }
922
+ for (const bucket of projectBuckets) {
923
+ if (runtime.totalMatches >= runtime.limit || runtime.isAborted()) {
924
+ break;
925
+ }
926
+ const projectResult = {
927
+ projectId: bucket.projectId,
928
+ projectName: bucket.projectName,
929
+ projectDisplayName: bucket.projectDisplayName,
930
+ sessions: [],
931
+ };
932
+ for (const session of bucket.sessions) {
933
+ if (runtime.totalMatches >= runtime.limit || runtime.isAborted()) {
934
+ break;
935
+ }
936
+ if (!matchedSessionKeys.has(getSessionKey(session))) {
937
+ continue;
938
+ }
939
+ const sessionResult = await parseSessionMatches(session, runtime);
940
+ if (sessionResult) {
941
+ projectResult.sessions.push(sessionResult);
942
+ }
943
+ }
944
+ scannedProjects += 1;
945
+ if (projectResult.sessions.length > 0) {
946
+ results.push(projectResult);
947
+ onProjectResult?.({
948
+ projectResult,
949
+ totalMatches: runtime.totalMatches,
950
+ scannedProjects,
951
+ totalProjects,
952
+ });
953
+ }
954
+ else if (onProjectResult && scannedProjects % 10 === 0) {
955
+ onProjectResult({
956
+ projectResult: null,
957
+ totalMatches: runtime.totalMatches,
958
+ scannedProjects,
959
+ totalProjects,
960
+ });
961
+ }
962
+ }
963
+ return {
964
+ results,
965
+ totalMatches: runtime.totalMatches,
966
+ query: safeQuery,
967
+ };
968
+ }
969
+ /**
970
+ * Application service for session-conversation search.
971
+ *
972
+ * Provider routes call this service so route handlers stay focused on
973
+ * request parsing/response formatting, while search execution remains
974
+ * centralized in one place.
975
+ */
976
+ export const sessionConversationsSearchService = {
977
+ /**
978
+ * Streams progress updates while the search scans provider session logs.
979
+ */
980
+ async search(input) {
981
+ await searchConversations(input.userId, input.query, input.limit, input.onProgress ?? null, input.signal ?? null);
982
+ },
983
+ };
984
+ //# sourceMappingURL=session-conversations-search.service.js.map