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