claude-code-workflow 6.1.4 → 6.2.2

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 (437) hide show
  1. package/.claude/CLAUDE.md +10 -0
  2. package/.claude/agents/action-planning-agent.md +857 -778
  3. package/.claude/agents/cli-execution-agent.md +266 -269
  4. package/.claude/agents/cli-explore-agent.md +2 -2
  5. package/.claude/agents/cli-lite-planning-agent.md +142 -92
  6. package/.claude/agents/cli-planning-agent.md +4 -4
  7. package/.claude/agents/code-developer.md +7 -6
  8. package/.claude/agents/conceptual-planning-agent.md +2 -2
  9. package/.claude/agents/context-search-agent.md +31 -32
  10. package/.claude/agents/doc-generator.md +4 -4
  11. package/.claude/agents/memory-bridge.md +93 -93
  12. package/.claude/agents/test-context-search-agent.md +8 -7
  13. package/.claude/agents/test-fix-agent.md +7 -6
  14. package/.claude/commands/clean.md +516 -0
  15. package/.claude/commands/memory/compact.md +383 -0
  16. package/.claude/commands/memory/docs-full-cli.md +471 -471
  17. package/.claude/commands/memory/docs-related-cli.md +386 -386
  18. package/.claude/commands/memory/docs.md +615 -615
  19. package/.claude/commands/memory/load.md +5 -5
  20. package/.claude/commands/memory/tech-research-rules.md +310 -0
  21. package/.claude/commands/memory/update-full.md +332 -332
  22. package/.claude/commands/memory/workflow-skill-memory.md +4 -4
  23. package/.claude/commands/task/create.md +151 -151
  24. package/.claude/commands/version.md +254 -254
  25. package/.claude/commands/workflow/brainstorm/api-designer.md +587 -585
  26. package/.claude/commands/workflow/brainstorm/artifacts.md +1 -0
  27. package/.claude/commands/workflow/brainstorm/auto-parallel.md +443 -443
  28. package/.claude/commands/workflow/brainstorm/data-architect.md +220 -220
  29. package/.claude/commands/workflow/brainstorm/product-manager.md +200 -200
  30. package/.claude/commands/workflow/brainstorm/product-owner.md +200 -200
  31. package/.claude/commands/workflow/brainstorm/scrum-master.md +200 -200
  32. package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  33. package/.claude/commands/workflow/brainstorm/system-architect.md +389 -387
  34. package/.claude/commands/workflow/brainstorm/ui-designer.md +221 -221
  35. package/.claude/commands/workflow/brainstorm/ux-expert.md +221 -221
  36. package/.claude/commands/workflow/debug.md +321 -0
  37. package/.claude/commands/workflow/execute.md +13 -0
  38. package/.claude/commands/workflow/init.md +165 -164
  39. package/.claude/commands/workflow/lite-execute.md +119 -13
  40. package/.claude/commands/workflow/lite-fix.md +623 -621
  41. package/.claude/commands/workflow/lite-plan.md +610 -592
  42. package/.claude/commands/workflow/plan.md +5 -5
  43. package/.claude/commands/workflow/review-module-cycle.md +2 -0
  44. package/.claude/commands/workflow/review-session-cycle.md +2 -0
  45. package/.claude/commands/workflow/review.md +297 -291
  46. package/.claude/commands/workflow/session/complete.md +153 -500
  47. package/.claude/commands/workflow/session/list.md +95 -95
  48. package/.claude/commands/workflow/session/resume.md +60 -60
  49. package/.claude/commands/workflow/session/start.md +199 -199
  50. package/.claude/commands/workflow/tdd-plan.md +3 -3
  51. package/.claude/commands/workflow/tdd-verify.md +23 -9
  52. package/.claude/commands/workflow/test-cycle-execute.md +2 -0
  53. package/.claude/commands/workflow/test-fix-gen.md +699 -699
  54. package/.claude/commands/workflow/tools/conflict-resolution.md +104 -18
  55. package/.claude/commands/workflow/tools/context-gather.md +436 -434
  56. package/.claude/commands/workflow/tools/task-generate-agent.md +490 -291
  57. package/.claude/commands/workflow/tools/task-generate-tdd.md +18 -10
  58. package/.claude/commands/workflow/tools/test-concept-enhanced.md +2 -1
  59. package/.claude/commands/workflow/tools/test-context-gather.md +1 -0
  60. package/.claude/commands/workflow/tools/test-task-generate.md +1 -0
  61. package/.claude/commands/workflow/ui-design/import-from-code.md +9 -6
  62. package/.claude/skills/command-guide/SKILL.md +5 -5
  63. package/.claude/skills/command-guide/index/all-commands.json +1 -1
  64. package/.claude/skills/command-guide/index/by-category.json +1 -1
  65. package/.claude/skills/command-guide/index/by-use-case.json +1 -1
  66. package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +857 -778
  67. package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +266 -269
  68. package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +2 -2
  69. package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +142 -92
  70. package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +4 -4
  71. package/.claude/skills/command-guide/reference/agents/code-developer.md +7 -6
  72. package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +2 -2
  73. package/.claude/skills/command-guide/reference/agents/context-search-agent.md +31 -32
  74. package/.claude/skills/command-guide/reference/agents/doc-generator.md +4 -4
  75. package/.claude/skills/command-guide/reference/agents/memory-bridge.md +93 -93
  76. package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +8 -7
  77. package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +7 -6
  78. package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +471 -471
  79. package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +386 -386
  80. package/.claude/skills/command-guide/reference/commands/memory/docs.md +17 -16
  81. package/.claude/skills/command-guide/reference/commands/memory/load.md +5 -5
  82. package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +194 -357
  83. package/.claude/skills/command-guide/reference/commands/memory/update-full.md +332 -332
  84. package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +4 -4
  85. package/.claude/skills/command-guide/reference/commands/task/create.md +151 -151
  86. package/.claude/skills/command-guide/reference/commands/version.md +254 -254
  87. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +585 -585
  88. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +443 -443
  89. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +220 -220
  90. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +200 -200
  91. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +200 -200
  92. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +200 -200
  93. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  94. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +387 -387
  95. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +221 -221
  96. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +221 -221
  97. package/.claude/skills/command-guide/reference/commands/workflow/execute.md +25 -20
  98. package/.claude/skills/command-guide/reference/commands/workflow/init.md +164 -164
  99. package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +748 -686
  100. package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +664 -621
  101. package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +645 -592
  102. package/.claude/skills/command-guide/reference/commands/workflow/plan.md +5 -5
  103. package/.claude/skills/command-guide/reference/commands/workflow/review.md +25 -18
  104. package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +547 -500
  105. package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +45 -27
  106. package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +35 -19
  107. package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +90 -33
  108. package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +3 -3
  109. package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +23 -9
  110. package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +699 -699
  111. package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +103 -17
  112. package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +434 -434
  113. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +487 -291
  114. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +17 -10
  115. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +1 -1
  116. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +6 -6
  117. package/.claude/workflows/chinese-response.md +38 -0
  118. package/.claude/workflows/cli-templates/prompts/rules/rule-api.txt +122 -0
  119. package/.claude/workflows/cli-templates/prompts/rules/rule-components.txt +122 -0
  120. package/.claude/workflows/cli-templates/prompts/rules/rule-config.txt +89 -0
  121. package/.claude/workflows/cli-templates/prompts/rules/rule-core.txt +60 -0
  122. package/.claude/workflows/cli-templates/prompts/rules/rule-patterns.txt +70 -0
  123. package/.claude/workflows/cli-templates/prompts/rules/rule-testing.txt +81 -0
  124. package/.claude/workflows/cli-templates/prompts/rules/tech-rules-agent-prompt.txt +89 -0
  125. package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +131 -131
  126. package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +5 -9
  127. package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +5 -9
  128. package/.claude/workflows/cli-templates/protocols/analysis-protocol.md +112 -0
  129. package/.claude/workflows/cli-templates/protocols/write-protocol.md +201 -0
  130. package/.claude/workflows/cli-templates/schemas/conflict-resolution-schema.json +137 -0
  131. package/.claude/workflows/cli-templates/schemas/debug-log-json-schema.json +127 -0
  132. package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +25 -0
  133. package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +25 -0
  134. package/.claude/workflows/cli-tools-usage.md +526 -0
  135. package/{CLAUDE.md → .claude/workflows/coding-philosophy.md} +24 -45
  136. package/.claude/workflows/context-tools.md +84 -0
  137. package/.claude/workflows/file-modification.md +64 -0
  138. package/.claude/workflows/tool-strategy.md +216 -79
  139. package/.claude/workflows/windows-platform.md +16 -0
  140. package/.claude/workflows/workflow-architecture.md +942 -942
  141. package/.codex/AGENTS.md +63 -330
  142. package/.codex/prompts/debug.md +318 -0
  143. package/.codex/prompts/execute.md +273 -0
  144. package/.codex/prompts/lite-execute.md +164 -0
  145. package/.codex/prompts/lite-plan.md +469 -0
  146. package/.codex/prompts.zip +0 -0
  147. package/.gemini/GEMINI.md +25 -164
  148. package/.qwen/QWEN.md +0 -139
  149. package/README.md +29 -9
  150. package/ccw/README.md +30 -6
  151. package/ccw/bin/ccw-mcp.js +7 -0
  152. package/ccw/bin/ccw.js +9 -9
  153. package/ccw/package.json +65 -47
  154. package/ccw/src/.workflow/.cli-history/history.db +0 -0
  155. package/ccw/src/.workflow/.cli-history/history.db-shm +0 -0
  156. package/ccw/src/.workflow/.cli-history/history.db-wal +0 -0
  157. package/ccw/src/cli.ts +244 -0
  158. package/ccw/src/commands/cli.ts +740 -0
  159. package/ccw/src/commands/core-memory.ts +770 -0
  160. package/ccw/src/commands/hook.ts +315 -0
  161. package/ccw/src/commands/install.ts +519 -0
  162. package/ccw/src/commands/{list.js → list.ts} +1 -1
  163. package/ccw/src/commands/memory.ts +1090 -0
  164. package/ccw/src/commands/{serve.js → serve.ts} +14 -5
  165. package/ccw/src/commands/session-path-resolver.ts +372 -0
  166. package/ccw/src/commands/session.ts +1141 -0
  167. package/ccw/src/commands/{stop.js → stop.ts} +16 -6
  168. package/ccw/src/commands/tool.ts +201 -0
  169. package/ccw/src/commands/{uninstall.js → uninstall.ts} +89 -40
  170. package/ccw/src/commands/{upgrade.js → upgrade.ts} +68 -23
  171. package/ccw/src/commands/{view.js → view.ts} +22 -8
  172. package/ccw/src/config/storage-paths.ts +670 -0
  173. package/ccw/src/core/cache-manager.ts +294 -0
  174. package/ccw/src/core/claude-freshness.ts +319 -0
  175. package/ccw/src/core/core-memory-store.ts +1528 -0
  176. package/ccw/src/core/{dashboard-generator-patch.js → dashboard-generator-patch.ts} +18 -0
  177. package/ccw/src/core/{dashboard-generator.js → dashboard-generator.ts} +69 -12
  178. package/ccw/src/core/data-aggregator.ts +584 -0
  179. package/ccw/src/core/history-importer.ts +625 -0
  180. package/ccw/src/core/{lite-scanner.js → lite-scanner-complete.ts} +162 -66
  181. package/ccw/src/core/lite-scanner.ts +469 -0
  182. package/ccw/src/core/{manifest.js → manifest.ts} +104 -34
  183. package/ccw/src/core/memory-embedder-bridge.ts +262 -0
  184. package/ccw/src/core/memory-store.ts +978 -0
  185. package/ccw/src/core/routes/ccw-routes.ts +96 -0
  186. package/ccw/src/core/routes/claude-routes.ts +1183 -0
  187. package/ccw/src/core/routes/cli-routes.ts +561 -0
  188. package/ccw/src/core/routes/codexlens-routes.ts +806 -0
  189. package/ccw/src/core/routes/core-memory-routes.ts +605 -0
  190. package/ccw/src/core/routes/files-routes.ts +428 -0
  191. package/ccw/src/core/routes/graph-routes.md +164 -0
  192. package/ccw/src/core/routes/graph-routes.ts +626 -0
  193. package/ccw/src/core/routes/help-routes.ts +308 -0
  194. package/ccw/src/core/routes/hooks-routes.ts +405 -0
  195. package/ccw/src/core/routes/mcp-routes.ts +1271 -0
  196. package/ccw/src/core/routes/mcp-routes.ts.backup +550 -0
  197. package/ccw/src/core/routes/mcp-templates-db.ts +268 -0
  198. package/ccw/src/core/routes/memory-routes.ts +1206 -0
  199. package/ccw/src/core/routes/rules-routes.ts +526 -0
  200. package/ccw/src/core/routes/session-routes.ts +467 -0
  201. package/ccw/src/core/routes/skills-routes.ts +599 -0
  202. package/ccw/src/core/routes/status-routes.ts +57 -0
  203. package/ccw/src/core/routes/system-routes.ts +427 -0
  204. package/ccw/src/core/server.ts +431 -0
  205. package/ccw/src/core/session-clustering-service.ts +1258 -0
  206. package/ccw/src/core/session-scanner.ts +283 -0
  207. package/ccw/src/core/websocket.ts +190 -0
  208. package/ccw/src/{index.js → index.ts} +1 -0
  209. package/ccw/src/mcp-server/index.ts +186 -0
  210. package/ccw/src/templates/assets/css/github-dark.min.css +10 -0
  211. package/ccw/src/templates/assets/css/github.min.css +10 -0
  212. package/ccw/src/templates/assets/js/cytoscape.min.js +32 -0
  213. package/ccw/src/templates/assets/js/d3.min.js +2 -0
  214. package/ccw/src/templates/assets/js/highlight.min.js +1244 -0
  215. package/ccw/src/templates/assets/js/lucide.min.js +12 -0
  216. package/ccw/src/templates/assets/js/marked.min.js +69 -0
  217. package/ccw/src/templates/assets/js/tailwind.js +83 -0
  218. package/ccw/src/templates/dashboard-css/01-base.css +11 -0
  219. package/ccw/src/templates/dashboard-css/02-session.css +22 -0
  220. package/ccw/src/templates/dashboard-css/04-lite-tasks.css +10 -0
  221. package/ccw/src/templates/dashboard-css/06-cards.css +10 -4
  222. package/ccw/src/templates/dashboard-css/07-managers.css +1178 -7
  223. package/ccw/src/templates/dashboard-css/09-explorer.css +23 -12
  224. package/ccw/src/templates/dashboard-css/10-cli-status.css +337 -0
  225. package/ccw/src/templates/dashboard-css/11-cli-history.css +271 -0
  226. package/ccw/src/templates/dashboard-css/12-cli-legacy.css +796 -0
  227. package/ccw/src/templates/dashboard-css/13-cli-ccw.css +199 -0
  228. package/ccw/src/templates/dashboard-css/14-cli-modals.css +258 -0
  229. package/ccw/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
  230. package/ccw/src/templates/dashboard-css/16-cli-session.css +241 -0
  231. package/ccw/src/templates/dashboard-css/17-cli-conversation.css +283 -0
  232. package/ccw/src/templates/dashboard-css/18-cli-settings.css +160 -0
  233. package/ccw/src/templates/dashboard-css/19-cli-native-session.css +496 -0
  234. package/ccw/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
  235. package/ccw/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
  236. package/ccw/src/templates/dashboard-css/22-cli-semantic.css +240 -0
  237. package/ccw/src/templates/dashboard-css/23-memory.css +2390 -0
  238. package/ccw/src/templates/dashboard-css/24-prompt-history.css +1089 -0
  239. package/ccw/src/templates/dashboard-css/25-skills-rules.css +326 -0
  240. package/ccw/src/templates/dashboard-css/26-claude-manager.css +908 -0
  241. package/ccw/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
  242. package/ccw/src/templates/dashboard-css/28-mcp-manager.css +748 -0
  243. package/ccw/src/templates/dashboard-css/29-help.css +264 -0
  244. package/ccw/src/templates/dashboard-css/30-core-memory.css +1700 -0
  245. package/ccw/src/templates/dashboard-js/api.js +162 -142
  246. package/ccw/src/templates/dashboard-js/components/carousel.js +4 -4
  247. package/ccw/src/templates/dashboard-js/components/cli-history.js +876 -0
  248. package/ccw/src/templates/dashboard-js/components/cli-status.js +978 -0
  249. package/ccw/src/templates/dashboard-js/components/global-notifications.js +508 -219
  250. package/ccw/src/templates/dashboard-js/components/hook-manager.js +1277 -282
  251. package/ccw/src/templates/dashboard-js/components/index-manager.js +302 -0
  252. package/ccw/src/templates/dashboard-js/components/mcp-manager.js +718 -27
  253. package/ccw/src/templates/dashboard-js/components/modals.js +66 -0
  254. package/ccw/src/templates/dashboard-js/components/navigation.js +80 -12
  255. package/ccw/src/templates/dashboard-js/components/notifications.js +758 -194
  256. package/ccw/src/templates/dashboard-js/components/storage-manager.js +478 -0
  257. package/ccw/src/templates/dashboard-js/components/tabs-other.js +157 -6
  258. package/ccw/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
  259. package/ccw/src/templates/dashboard-js/help-i18n.js +272 -0
  260. package/ccw/src/templates/dashboard-js/i18n.js +2807 -0
  261. package/ccw/src/templates/dashboard-js/main.js +15 -0
  262. package/ccw/src/templates/dashboard-js/state.js +243 -42
  263. package/ccw/src/templates/dashboard-js/utils.js +47 -1
  264. package/ccw/src/templates/dashboard-js/views/claude-manager.js +912 -0
  265. package/ccw/src/templates/dashboard-js/views/cli-manager.js +2272 -0
  266. package/ccw/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
  267. package/ccw/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
  268. package/ccw/src/templates/dashboard-js/views/core-memory.js +782 -0
  269. package/ccw/src/templates/dashboard-js/views/explorer.js +888 -852
  270. package/ccw/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
  271. package/ccw/src/templates/dashboard-js/views/help.js +856 -0
  272. package/ccw/src/templates/dashboard-js/views/history.js +337 -0
  273. package/ccw/src/templates/dashboard-js/views/home.js +61 -15
  274. package/ccw/src/templates/dashboard-js/views/hook-manager.js +311 -43
  275. package/ccw/src/templates/dashboard-js/views/lite-tasks.js +204 -28
  276. package/ccw/src/templates/dashboard-js/views/mcp-manager.js +2187 -411
  277. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
  278. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
  279. package/ccw/src/templates/dashboard-js/views/memory.js +1221 -0
  280. package/ccw/src/templates/dashboard-js/views/prompt-history.js +713 -0
  281. package/ccw/src/templates/dashboard-js/views/rules-manager.js +828 -0
  282. package/ccw/src/templates/dashboard-js/views/session-detail.js +54 -53
  283. package/ccw/src/templates/dashboard-js/views/skills-manager.js +819 -0
  284. package/ccw/src/templates/dashboard.html +185 -85
  285. package/ccw/src/templates/hooks-config-example.json +60 -0
  286. package/ccw/src/tools/classify-folders.ts +245 -0
  287. package/ccw/src/tools/cli-config-manager.ts +268 -0
  288. package/ccw/src/tools/cli-executor.ts +2014 -0
  289. package/ccw/src/tools/cli-history-store.ts +1195 -0
  290. package/ccw/src/tools/codex-lens.ts +1141 -0
  291. package/ccw/src/tools/{convert-tokens-to-css.js → convert-tokens-to-css.ts} +73 -23
  292. package/ccw/src/tools/core-memory.ts +444 -0
  293. package/ccw/src/tools/detect-changed-modules.ts +325 -0
  294. package/ccw/src/tools/{discover-design-files.js → discover-design-files.ts} +74 -24
  295. package/ccw/src/tools/edit-file.ts +568 -0
  296. package/ccw/src/tools/{generate-module-docs.js → generate-module-docs.ts} +207 -185
  297. package/ccw/src/tools/{get-modules-by-depth.js → get-modules-by-depth.ts} +120 -79
  298. package/ccw/src/tools/index.ts +370 -0
  299. package/ccw/src/tools/native-session-discovery.ts +795 -0
  300. package/ccw/src/tools/notifier.ts +129 -0
  301. package/ccw/src/tools/read-file.ts +410 -0
  302. package/ccw/src/tools/resume-strategy.ts +345 -0
  303. package/ccw/src/tools/session-content-parser.ts +619 -0
  304. package/ccw/src/tools/session-manager.ts +1026 -0
  305. package/ccw/src/tools/smart-context.ts +228 -0
  306. package/ccw/src/tools/smart-search.ts +2065 -0
  307. package/ccw/src/tools/smart-search.ts.backup +1233 -0
  308. package/ccw/src/tools/storage-manager.ts +455 -0
  309. package/ccw/src/tools/write-file.ts +222 -0
  310. package/ccw/src/types/config.ts +11 -0
  311. package/ccw/src/types/index.ts +3 -0
  312. package/ccw/src/types/session.ts +25 -0
  313. package/ccw/src/types/tool.ts +41 -0
  314. package/ccw/src/utils/{browser-launcher.js → browser-launcher.ts} +10 -8
  315. package/ccw/src/utils/file-utils.ts +48 -0
  316. package/ccw/src/utils/{path-resolver.js → path-resolver.ts} +114 -78
  317. package/ccw/src/utils/path-validator.ts +153 -0
  318. package/ccw/src/utils/{ui.js → ui.ts} +32 -25
  319. package/codex-lens/pyproject.toml +48 -0
  320. package/codex-lens/src/codexlens/.workflow/.cli-history/history.db +0 -0
  321. package/codex-lens/src/codexlens/__init__.py +28 -0
  322. package/codex-lens/src/codexlens/__main__.py +14 -0
  323. package/codex-lens/src/codexlens/__pycache__/__init__.cpython-313.pyc +0 -0
  324. package/codex-lens/src/codexlens/__pycache__/__main__.cpython-313.pyc +0 -0
  325. package/codex-lens/src/codexlens/__pycache__/config.cpython-313.pyc +0 -0
  326. package/codex-lens/src/codexlens/__pycache__/entities.cpython-313.pyc +0 -0
  327. package/codex-lens/src/codexlens/__pycache__/errors.cpython-313.pyc +0 -0
  328. package/codex-lens/src/codexlens/cli/__init__.py +27 -0
  329. package/codex-lens/src/codexlens/cli/__pycache__/__init__.cpython-313.pyc +0 -0
  330. package/codex-lens/src/codexlens/cli/__pycache__/commands.cpython-313.pyc +0 -0
  331. package/codex-lens/src/codexlens/cli/__pycache__/embedding_manager.cpython-313.pyc +0 -0
  332. package/codex-lens/src/codexlens/cli/__pycache__/model_manager.cpython-313.pyc +0 -0
  333. package/codex-lens/src/codexlens/cli/__pycache__/output.cpython-313.pyc +0 -0
  334. package/codex-lens/src/codexlens/cli/commands.py +1931 -0
  335. package/codex-lens/src/codexlens/cli/embedding_manager.py +620 -0
  336. package/codex-lens/src/codexlens/cli/model_manager.py +289 -0
  337. package/codex-lens/src/codexlens/cli/output.py +124 -0
  338. package/codex-lens/src/codexlens/config.py +201 -0
  339. package/codex-lens/src/codexlens/entities.py +121 -0
  340. package/codex-lens/src/codexlens/errors.py +55 -0
  341. package/codex-lens/src/codexlens/indexing/README.md +77 -0
  342. package/codex-lens/src/codexlens/indexing/__init__.py +4 -0
  343. package/codex-lens/src/codexlens/indexing/__pycache__/__init__.cpython-313.pyc +0 -0
  344. package/codex-lens/src/codexlens/indexing/__pycache__/symbol_extractor.cpython-313.pyc +0 -0
  345. package/codex-lens/src/codexlens/indexing/symbol_extractor.py +243 -0
  346. package/codex-lens/src/codexlens/parsers/__init__.py +8 -0
  347. package/codex-lens/src/codexlens/parsers/__pycache__/__init__.cpython-313.pyc +0 -0
  348. package/codex-lens/src/codexlens/parsers/__pycache__/encoding.cpython-313.pyc +0 -0
  349. package/codex-lens/src/codexlens/parsers/__pycache__/factory.cpython-313.pyc +0 -0
  350. package/codex-lens/src/codexlens/parsers/__pycache__/tokenizer.cpython-313.pyc +0 -0
  351. package/codex-lens/src/codexlens/parsers/__pycache__/treesitter_parser.cpython-313.pyc +0 -0
  352. package/codex-lens/src/codexlens/parsers/encoding.py +202 -0
  353. package/codex-lens/src/codexlens/parsers/factory.py +256 -0
  354. package/codex-lens/src/codexlens/parsers/tokenizer.py +98 -0
  355. package/codex-lens/src/codexlens/parsers/treesitter_parser.py +335 -0
  356. package/codex-lens/src/codexlens/search/__init__.py +15 -0
  357. package/codex-lens/src/codexlens/search/__pycache__/__init__.cpython-313.pyc +0 -0
  358. package/codex-lens/src/codexlens/search/__pycache__/chain_search.cpython-313.pyc +0 -0
  359. package/codex-lens/src/codexlens/search/__pycache__/enrichment.cpython-313.pyc +0 -0
  360. package/codex-lens/src/codexlens/search/__pycache__/hybrid_search.cpython-313.pyc +0 -0
  361. package/codex-lens/src/codexlens/search/__pycache__/query_parser.cpython-313.pyc +0 -0
  362. package/codex-lens/src/codexlens/search/__pycache__/ranking.cpython-313.pyc +0 -0
  363. package/codex-lens/src/codexlens/search/chain_search.py +647 -0
  364. package/codex-lens/src/codexlens/search/enrichment.py +150 -0
  365. package/codex-lens/src/codexlens/search/hybrid_search.py +313 -0
  366. package/codex-lens/src/codexlens/search/query_parser.py +242 -0
  367. package/codex-lens/src/codexlens/search/ranking.py +274 -0
  368. package/codex-lens/src/codexlens/semantic/__init__.py +39 -0
  369. package/codex-lens/src/codexlens/semantic/__pycache__/__init__.cpython-313.pyc +0 -0
  370. package/codex-lens/src/codexlens/semantic/__pycache__/ann_index.cpython-313.pyc +0 -0
  371. package/codex-lens/src/codexlens/semantic/__pycache__/chunker.cpython-313.pyc +0 -0
  372. package/codex-lens/src/codexlens/semantic/__pycache__/code_extractor.cpython-313.pyc +0 -0
  373. package/codex-lens/src/codexlens/semantic/__pycache__/embedder.cpython-313.pyc +0 -0
  374. package/codex-lens/src/codexlens/semantic/__pycache__/graph_analyzer.cpython-313.pyc +0 -0
  375. package/codex-lens/src/codexlens/semantic/__pycache__/llm_enhancer.cpython-313.pyc +0 -0
  376. package/codex-lens/src/codexlens/semantic/__pycache__/vector_store.cpython-313.pyc +0 -0
  377. package/codex-lens/src/codexlens/semantic/ann_index.py +414 -0
  378. package/codex-lens/src/codexlens/semantic/chunker.py +448 -0
  379. package/codex-lens/src/codexlens/semantic/code_extractor.py +274 -0
  380. package/codex-lens/src/codexlens/semantic/embedder.py +185 -0
  381. package/codex-lens/src/codexlens/semantic/vector_store.py +955 -0
  382. package/codex-lens/src/codexlens/storage/__init__.py +29 -0
  383. package/codex-lens/src/codexlens/storage/__pycache__/__init__.cpython-313.pyc +0 -0
  384. package/codex-lens/src/codexlens/storage/__pycache__/dir_index.cpython-313.pyc +0 -0
  385. package/codex-lens/src/codexlens/storage/__pycache__/file_cache.cpython-313.pyc +0 -0
  386. package/codex-lens/src/codexlens/storage/__pycache__/index_tree.cpython-313.pyc +0 -0
  387. package/codex-lens/src/codexlens/storage/__pycache__/migration_manager.cpython-313.pyc +0 -0
  388. package/codex-lens/src/codexlens/storage/__pycache__/path_mapper.cpython-313.pyc +0 -0
  389. package/codex-lens/src/codexlens/storage/__pycache__/registry.cpython-313.pyc +0 -0
  390. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_store.cpython-313.pyc +0 -0
  391. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_utils.cpython-313.pyc +0 -0
  392. package/codex-lens/src/codexlens/storage/dir_index.py +1850 -0
  393. package/codex-lens/src/codexlens/storage/file_cache.py +32 -0
  394. package/codex-lens/src/codexlens/storage/index_tree.py +776 -0
  395. package/codex-lens/src/codexlens/storage/migration_manager.py +154 -0
  396. package/codex-lens/src/codexlens/storage/migrations/__init__.py +1 -0
  397. package/codex-lens/src/codexlens/storage/migrations/__pycache__/__init__.cpython-313.pyc +0 -0
  398. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_001_normalize_keywords.cpython-313.pyc +0 -0
  399. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_002_add_token_metadata.cpython-313.pyc +0 -0
  400. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_003_code_relationships.cpython-313.pyc +0 -0
  401. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_004_dual_fts.cpython-313.pyc +0 -0
  402. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_005_cleanup_unused_fields.cpython-313.pyc +0 -0
  403. package/codex-lens/src/codexlens/storage/migrations/migration_001_normalize_keywords.py +123 -0
  404. package/codex-lens/src/codexlens/storage/migrations/migration_002_add_token_metadata.py +48 -0
  405. package/codex-lens/src/codexlens/storage/migrations/migration_004_dual_fts.py +232 -0
  406. package/codex-lens/src/codexlens/storage/migrations/migration_005_cleanup_unused_fields.py +196 -0
  407. package/codex-lens/src/codexlens/storage/path_mapper.py +274 -0
  408. package/codex-lens/src/codexlens/storage/registry.py +670 -0
  409. package/codex-lens/src/codexlens/storage/sqlite_store.py +576 -0
  410. package/codex-lens/src/codexlens/storage/sqlite_utils.py +64 -0
  411. package/package.json +4 -1
  412. package/.claude/commands/memory/tech-research.md +0 -477
  413. package/.claude/scripts/classify-folders.sh +0 -39
  414. package/.claude/scripts/convert_tokens_to_css.sh +0 -229
  415. package/.claude/scripts/detect_changed_modules.sh +0 -161
  416. package/.claude/scripts/discover-design-files.sh +0 -87
  417. package/.claude/scripts/extract-animations.js +0 -243
  418. package/.claude/scripts/extract-computed-styles.js +0 -118
  419. package/.claude/scripts/extract-layout-structure.js +0 -411
  420. package/.claude/scripts/generate_module_docs.sh +0 -717
  421. package/.claude/scripts/get_modules_by_depth.sh +0 -170
  422. package/.claude/scripts/ui-generate-preview.sh +0 -395
  423. package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
  424. package/.claude/scripts/update_module_claude.sh +0 -337
  425. package/.claude/workflows/context-search-strategy.md +0 -77
  426. package/.claude/workflows/intelligent-tools-strategy.md +0 -662
  427. package/ccw/src/cli.js +0 -119
  428. package/ccw/src/commands/install.js +0 -324
  429. package/ccw/src/commands/tool.js +0 -138
  430. package/ccw/src/core/data-aggregator.js +0 -409
  431. package/ccw/src/core/server.js +0 -2063
  432. package/ccw/src/core/session-scanner.js +0 -235
  433. package/ccw/src/tools/classify-folders.js +0 -204
  434. package/ccw/src/tools/detect-changed-modules.js +0 -288
  435. package/ccw/src/tools/edit-file.js +0 -266
  436. package/ccw/src/tools/index.js +0 -176
  437. package/ccw/src/utils/file-utils.js +0 -48
@@ -0,0 +1,1233 @@
1
+ /**
2
+ * Smart Search Tool - Unified intelligent search with CodexLens integration
3
+ *
4
+ * Features:
5
+ * - Intent classification with automatic mode selection
6
+ * - CodexLens integration (init, hybrid, vector, semantic)
7
+ * - Ripgrep fallback for exact mode
8
+ * - Index status checking and warnings
9
+ * - Multi-backend search routing with RRF ranking
10
+ *
11
+ * Actions:
12
+ * - init: Initialize CodexLens index
13
+ * - search: Intelligent search with auto mode selection
14
+ * - status: Check index status
15
+ */
16
+
17
+ import { z } from 'zod';
18
+ import type { ToolSchema, ToolResult } from '../types/tool.js';
19
+ import { spawn, execSync } from 'child_process';
20
+ import {
21
+ ensureReady as ensureCodexLensReady,
22
+ executeCodexLens,
23
+ } from './codex-lens.js';
24
+ import type { ProgressInfo } from './codex-lens.js';
25
+
26
+ // Define Zod schema for validation
27
+ const ParamsSchema = z.object({
28
+ action: z.enum(['init', 'search', 'search_files', 'status']).default('search'),
29
+ query: z.string().optional(),
30
+ mode: z.enum(['auto', 'hybrid', 'exact', 'ripgrep', 'priority']).default('auto'),
31
+ output_mode: z.enum(['full', 'files_only', 'count']).default('full'),
32
+ path: z.string().optional(),
33
+ paths: z.array(z.string()).default([]),
34
+ contextLines: z.number().default(0),
35
+ maxResults: z.number().default(10),
36
+ includeHidden: z.boolean().default(false),
37
+ languages: z.array(z.string()).optional(),
38
+ limit: z.number().default(10),
39
+ enrich: z.boolean().default(false),
40
+ });
41
+
42
+ type Params = z.infer<typeof ParamsSchema>;
43
+
44
+ // Search mode constants
45
+ const SEARCH_MODES = ['auto', 'hybrid', 'exact', 'ripgrep', 'priority'] as const;
46
+
47
+ // Classification confidence threshold
48
+ const CONFIDENCE_THRESHOLD = 0.7;
49
+
50
+ interface Classification {
51
+ mode: string;
52
+ confidence: number;
53
+ reasoning: string;
54
+ }
55
+
56
+ interface ExactMatch {
57
+ file: string;
58
+ line: number;
59
+ column: number;
60
+ content: string;
61
+ }
62
+
63
+ interface RelationshipInfo {
64
+ type: string; // 'calls', 'imports', 'called_by', 'imported_by'
65
+ direction: 'outgoing' | 'incoming';
66
+ target?: string; // Target symbol name (for outgoing)
67
+ source?: string; // Source symbol name (for incoming)
68
+ file: string; // File path
69
+ line?: number; // Line number
70
+ }
71
+
72
+ interface SemanticMatch {
73
+ file: string;
74
+ score: number;
75
+ content: string;
76
+ symbol: string | null;
77
+ relationships?: RelationshipInfo[];
78
+ }
79
+
80
+ interface GraphMatch {
81
+ file: string;
82
+ symbols: unknown;
83
+ relationships: unknown[];
84
+ }
85
+
86
+ interface SearchMetadata {
87
+ mode?: string;
88
+ backend?: string;
89
+ count?: number;
90
+ query?: string;
91
+ classified_as?: string;
92
+ confidence?: number;
93
+ reasoning?: string;
94
+ embeddings_coverage_percent?: number;
95
+ warning?: string;
96
+ note?: string;
97
+ index_status?: 'indexed' | 'not_indexed' | 'partial';
98
+ fallback_history?: string[];
99
+ // Init action specific
100
+ action?: string;
101
+ path?: string;
102
+ progress?: {
103
+ stage: string;
104
+ message: string;
105
+ percent: number;
106
+ filesProcessed?: number;
107
+ totalFiles?: number;
108
+ };
109
+ progressHistory?: ProgressInfo[];
110
+ }
111
+
112
+ interface SearchResult {
113
+ success: boolean;
114
+ results?: ExactMatch[] | SemanticMatch[] | GraphMatch[] | unknown;
115
+ output?: string;
116
+ metadata?: SearchMetadata;
117
+ error?: string;
118
+ status?: unknown;
119
+ message?: string;
120
+ }
121
+
122
+ interface IndexStatus {
123
+ indexed: boolean;
124
+ has_embeddings: boolean;
125
+ file_count?: number;
126
+ embeddings_coverage_percent?: number;
127
+ warning?: string;
128
+ }
129
+
130
+ /**
131
+ * Strip ANSI color codes from string (for JSON parsing)
132
+ */
133
+ function stripAnsi(str: string): string {
134
+ return str.replace(/\x1b\[[0-9;]*m/g, '');
135
+ }
136
+
137
+ /**
138
+ * Check if CodexLens index exists for current directory
139
+ * @param path - Directory path to check
140
+ * @returns Index status
141
+ */
142
+ async function checkIndexStatus(path: string = '.'): Promise<IndexStatus> {
143
+ try {
144
+ const result = await executeCodexLens(['status', '--json'], { cwd: path });
145
+
146
+ if (!result.success) {
147
+ return {
148
+ indexed: false,
149
+ has_embeddings: false,
150
+ warning: 'No CodexLens index found. Run smart_search(action="init") to create index for better search results.',
151
+ };
152
+ }
153
+
154
+ // Parse status output
155
+ try {
156
+ // Strip ANSI color codes from JSON output
157
+ const cleanOutput = stripAnsi(result.output || '{}');
158
+ const parsed = JSON.parse(cleanOutput);
159
+ // Handle both direct and nested response formats (status returns {success, result: {...}})
160
+ const status = parsed.result || parsed;
161
+ const indexed = status.projects_count > 0 || status.total_files > 0;
162
+
163
+ // Get embeddings coverage from comprehensive status
164
+ const embeddingsData = status.embeddings || {};
165
+ const embeddingsCoverage = embeddingsData.coverage_percent || 0;
166
+ const has_embeddings = embeddingsCoverage >= 50; // Threshold: 50%
167
+
168
+ let warning: string | undefined;
169
+ if (!indexed) {
170
+ warning = 'No CodexLens index found. Run smart_search(action="init") to create index for better search results.';
171
+ } else if (embeddingsCoverage === 0) {
172
+ warning = 'Index exists but no embeddings generated. Run: codexlens embeddings-generate --recursive';
173
+ } else if (embeddingsCoverage < 50) {
174
+ warning = `Embeddings coverage is ${embeddingsCoverage.toFixed(1)}% (below 50%). Hybrid search will use exact mode. Run: codexlens embeddings-generate --recursive`;
175
+ }
176
+
177
+ return {
178
+ indexed,
179
+ has_embeddings,
180
+ file_count: status.total_files,
181
+ embeddings_coverage_percent: embeddingsCoverage,
182
+ warning,
183
+ };
184
+ } catch {
185
+ return {
186
+ indexed: false,
187
+ has_embeddings: false,
188
+ warning: 'Failed to parse index status',
189
+ };
190
+ }
191
+ } catch {
192
+ return {
193
+ indexed: false,
194
+ has_embeddings: false,
195
+ warning: 'CodexLens not available',
196
+ };
197
+ }
198
+ }
199
+
200
+ /**
201
+ * Detection heuristics for intent classification
202
+ */
203
+
204
+ /**
205
+ * Detect literal string query (simple alphanumeric or quoted strings)
206
+ */
207
+ function detectLiteral(query: string): boolean {
208
+ return /^[a-zA-Z0-9_-]+$/.test(query) || /^["'].*["']$/.test(query);
209
+ }
210
+
211
+ /**
212
+ * Detect regex pattern (contains regex metacharacters)
213
+ */
214
+ function detectRegex(query: string): boolean {
215
+ return /[.*+?^${}()|[\]\\]/.test(query);
216
+ }
217
+
218
+ /**
219
+ * Detect natural language query (sentence structure, questions, multi-word phrases)
220
+ */
221
+ function detectNaturalLanguage(query: string): boolean {
222
+ return query.split(/\s+/).length >= 3 || /\?$/.test(query);
223
+ }
224
+
225
+ /**
226
+ * Detect file path query (path separators, file extensions)
227
+ */
228
+ function detectFilePath(query: string): boolean {
229
+ return /[/\\]/.test(query) || /\.[a-z]{2,4}$/i.test(query);
230
+ }
231
+
232
+ /**
233
+ * Detect relationship query (import, export, dependency keywords)
234
+ */
235
+ function detectRelationship(query: string): boolean {
236
+ return /(import|export|uses?|depends?|calls?|extends?)\s/i.test(query);
237
+ }
238
+
239
+ /**
240
+ * Classify query intent and recommend search mode
241
+ * Simple mapping: hybrid (NL + index + embeddings) | exact (index or insufficient embeddings) | ripgrep (no index)
242
+ * @param query - Search query string
243
+ * @param hasIndex - Whether CodexLens index exists
244
+ * @param hasSufficientEmbeddings - Whether embeddings coverage >= 50%
245
+ * @returns Classification result
246
+ */
247
+ function classifyIntent(query: string, hasIndex: boolean = false, hasSufficientEmbeddings: boolean = false): Classification {
248
+ // Detect query patterns
249
+ const isNaturalLanguage = detectNaturalLanguage(query);
250
+
251
+ // Simple decision tree
252
+ let mode: string;
253
+ let confidence: number;
254
+
255
+ if (!hasIndex) {
256
+ // No index: use ripgrep
257
+ mode = 'ripgrep';
258
+ confidence = 1.0;
259
+ } else if (isNaturalLanguage && hasSufficientEmbeddings) {
260
+ // Natural language + sufficient embeddings: use hybrid
261
+ mode = 'hybrid';
262
+ confidence = 0.9;
263
+ } else {
264
+ // Simple query OR insufficient embeddings: use exact
265
+ mode = 'exact';
266
+ confidence = 0.8;
267
+ }
268
+
269
+ // Build reasoning string
270
+ const detectedPatterns: string[] = [];
271
+ if (detectLiteral(query)) detectedPatterns.push('literal');
272
+ if (detectRegex(query)) detectedPatterns.push('regex');
273
+ if (detectNaturalLanguage(query)) detectedPatterns.push('natural language');
274
+ if (detectFilePath(query)) detectedPatterns.push('file path');
275
+ if (detectRelationship(query)) detectedPatterns.push('relationship');
276
+
277
+ const reasoning = `Query classified as ${mode} (confidence: ${confidence.toFixed(2)}, detected: ${detectedPatterns.join(', ')}, index: ${hasIndex ? 'available' : 'not available'}, embeddings: ${hasSufficientEmbeddings ? 'sufficient' : 'insufficient'})`;
278
+
279
+ return { mode, confidence, reasoning };
280
+ }
281
+
282
+ /**
283
+ * Check if a tool is available in PATH
284
+ * @param toolName - Tool executable name
285
+ * @returns True if available
286
+ */
287
+ function checkToolAvailability(toolName: string): boolean {
288
+ try {
289
+ const isWindows = process.platform === 'win32';
290
+ const command = isWindows ? 'where' : 'which';
291
+ execSync(`${command} ${toolName}`, { stdio: 'ignore' });
292
+ return true;
293
+ } catch {
294
+ return false;
295
+ }
296
+ }
297
+
298
+ /**
299
+ * Build ripgrep command arguments
300
+ * @param params - Search parameters
301
+ * @returns Command and arguments
302
+ */
303
+ function buildRipgrepCommand(params: {
304
+ query: string;
305
+ paths: string[];
306
+ contextLines: number;
307
+ maxResults: number;
308
+ includeHidden: boolean;
309
+ }): { command: string; args: string[] } {
310
+ const { query, paths = ['.'], contextLines = 0, maxResults = 10, includeHidden = false } = params;
311
+
312
+ const args = [
313
+ '-n', // Show line numbers
314
+ '--color=never', // Disable color output
315
+ '--json', // Output in JSON format
316
+ ];
317
+
318
+ // Add context lines if specified
319
+ if (contextLines > 0) {
320
+ args.push('-C', contextLines.toString());
321
+ }
322
+
323
+ // Add max results limit
324
+ if (maxResults > 0) {
325
+ args.push('--max-count', maxResults.toString());
326
+ }
327
+
328
+ // Include hidden files if specified
329
+ if (includeHidden) {
330
+ args.push('--hidden');
331
+ }
332
+
333
+ // Use literal/fixed string matching for exact mode
334
+ args.push('-F', query);
335
+
336
+ // Add search paths
337
+ args.push(...paths);
338
+
339
+ return { command: 'rg', args };
340
+ }
341
+
342
+ /**
343
+ * Action: init - Initialize CodexLens index (FTS only, no embeddings)
344
+ * For semantic/vector search, use ccw view dashboard or codexlens CLI directly
345
+ */
346
+ async function executeInitAction(params: Params): Promise<SearchResult> {
347
+ const { path = '.', languages } = params;
348
+
349
+ // Check CodexLens availability
350
+ const readyStatus = await ensureCodexLensReady();
351
+ if (!readyStatus.ready) {
352
+ return {
353
+ success: false,
354
+ error: `CodexLens not available: ${readyStatus.error}. CodexLens will be auto-installed on first use.`,
355
+ };
356
+ }
357
+
358
+ // Build args with --no-embeddings for FTS-only index (faster)
359
+ const args = ['init', path, '--no-embeddings'];
360
+ if (languages && languages.length > 0) {
361
+ args.push('--languages', languages.join(','));
362
+ }
363
+
364
+ // Track progress updates
365
+ const progressUpdates: ProgressInfo[] = [];
366
+ let lastProgress: ProgressInfo | null = null;
367
+
368
+ const result = await executeCodexLens(args, {
369
+ cwd: path,
370
+ timeout: 1800000, // 30 minutes for large codebases
371
+ onProgress: (progress: ProgressInfo) => {
372
+ progressUpdates.push(progress);
373
+ lastProgress = progress;
374
+ },
375
+ });
376
+
377
+ // Build metadata with progress info
378
+ const metadata: SearchMetadata = {
379
+ action: 'init',
380
+ path,
381
+ };
382
+
383
+ if (lastProgress !== null) {
384
+ const p = lastProgress as ProgressInfo;
385
+ metadata.progress = {
386
+ stage: p.stage,
387
+ message: p.message,
388
+ percent: p.percent,
389
+ filesProcessed: p.filesProcessed,
390
+ totalFiles: p.totalFiles,
391
+ };
392
+ }
393
+
394
+ if (progressUpdates.length > 0) {
395
+ metadata.progressHistory = progressUpdates.slice(-5); // Keep last 5 progress updates
396
+ }
397
+
398
+ const successMessage = result.success
399
+ ? `FTS index created for ${path}. Note: For semantic/vector search, create vector index via "ccw view" dashboard or run "codexlens init ${path}" (without --no-embeddings).`
400
+ : undefined;
401
+
402
+ return {
403
+ success: result.success,
404
+ error: result.error,
405
+ message: successMessage,
406
+ metadata,
407
+ };
408
+ }
409
+
410
+ /**
411
+ * Action: status - Check CodexLens index status
412
+ */
413
+ async function executeStatusAction(params: Params): Promise<SearchResult> {
414
+ const { path = '.' } = params;
415
+
416
+ const indexStatus = await checkIndexStatus(path);
417
+
418
+ return {
419
+ success: true,
420
+ status: indexStatus,
421
+ message: indexStatus.warning || `Index status: ${indexStatus.indexed ? 'indexed' : 'not indexed'}, embeddings: ${indexStatus.has_embeddings ? 'available' : 'not available'}`,
422
+ };
423
+ }
424
+
425
+ /**
426
+ * Mode: auto - Intent classification and mode selection
427
+ * Routes to: hybrid (NL + index) | exact (index) | ripgrep (no index)
428
+ */
429
+ async function executeAutoMode(params: Params): Promise<SearchResult> {
430
+ const { query, path = '.' } = params;
431
+
432
+ if (!query) {
433
+ return {
434
+ success: false,
435
+ error: 'Query is required for search action',
436
+ };
437
+ }
438
+
439
+ // Check index status
440
+ const indexStatus = await checkIndexStatus(path);
441
+
442
+ // Classify intent with index and embeddings awareness
443
+ const classification = classifyIntent(
444
+ query,
445
+ indexStatus.indexed,
446
+ indexStatus.has_embeddings // This now considers 50% threshold
447
+ );
448
+
449
+ // Route to appropriate mode based on classification
450
+ let result: SearchResult;
451
+
452
+ switch (classification.mode) {
453
+ case 'hybrid':
454
+ result = await executeHybridMode(params);
455
+ break;
456
+
457
+ case 'exact':
458
+ result = await executeCodexLensExactMode(params);
459
+ break;
460
+
461
+ case 'ripgrep':
462
+ result = await executeRipgrepMode(params);
463
+ break;
464
+
465
+ default:
466
+ // Fallback to ripgrep
467
+ result = await executeRipgrepMode(params);
468
+ break;
469
+ }
470
+
471
+ // Add classification metadata
472
+ if (result.metadata) {
473
+ result.metadata.classified_as = classification.mode;
474
+ result.metadata.confidence = classification.confidence;
475
+ result.metadata.reasoning = classification.reasoning;
476
+ result.metadata.embeddings_coverage_percent = indexStatus.embeddings_coverage_percent;
477
+ result.metadata.index_status = indexStatus.indexed
478
+ ? (indexStatus.has_embeddings ? 'indexed' : 'partial')
479
+ : 'not_indexed';
480
+
481
+ // Add warning if needed
482
+ if (indexStatus.warning) {
483
+ result.metadata.warning = indexStatus.warning;
484
+ }
485
+ }
486
+
487
+ return result;
488
+ }
489
+
490
+ /**
491
+ * Mode: ripgrep - Fast literal string matching using ripgrep
492
+ * No index required, fallback to CodexLens if ripgrep unavailable
493
+ */
494
+ async function executeRipgrepMode(params: Params): Promise<SearchResult> {
495
+ const { query, paths = [], contextLines = 0, maxResults = 10, includeHidden = false, path = '.' } = params;
496
+
497
+ if (!query) {
498
+ return {
499
+ success: false,
500
+ error: 'Query is required for search',
501
+ };
502
+ }
503
+
504
+ // Check if ripgrep is available
505
+ const hasRipgrep = checkToolAvailability('rg');
506
+
507
+ // If ripgrep not available, fall back to CodexLens exact mode
508
+ if (!hasRipgrep) {
509
+ const readyStatus = await ensureCodexLensReady();
510
+ if (!readyStatus.ready) {
511
+ return {
512
+ success: false,
513
+ error: 'Neither ripgrep nor CodexLens available. Install ripgrep (rg) or CodexLens for search functionality.',
514
+ };
515
+ }
516
+
517
+ // Use CodexLens exact mode as fallback
518
+ const args = ['search', query, '--limit', maxResults.toString(), '--mode', 'exact', '--json'];
519
+ const result = await executeCodexLens(args, { cwd: path });
520
+
521
+ if (!result.success) {
522
+ return {
523
+ success: false,
524
+ error: result.error,
525
+ metadata: {
526
+ mode: 'ripgrep',
527
+ backend: 'codexlens-fallback',
528
+ count: 0,
529
+ query,
530
+ },
531
+ };
532
+ }
533
+
534
+ // Parse results
535
+ let results: SemanticMatch[] = [];
536
+ try {
537
+ const parsed = JSON.parse(stripAnsi(result.output || '{}'));
538
+ const data = parsed.result?.results || parsed.results || parsed;
539
+ results = (Array.isArray(data) ? data : []).map((item: any) => ({
540
+ file: item.path || item.file,
541
+ score: item.score || 0,
542
+ content: item.excerpt || item.content || '',
543
+ symbol: item.symbol || null,
544
+ }));
545
+ } catch {
546
+ // Keep empty results
547
+ }
548
+
549
+ return {
550
+ success: true,
551
+ results,
552
+ metadata: {
553
+ mode: 'ripgrep',
554
+ backend: 'codexlens-fallback',
555
+ count: results.length,
556
+ query,
557
+ note: 'Using CodexLens exact mode (ripgrep not available)',
558
+ },
559
+ };
560
+ }
561
+
562
+ // Use ripgrep
563
+ const { command, args } = buildRipgrepCommand({
564
+ query,
565
+ paths: paths.length > 0 ? paths : [path],
566
+ contextLines,
567
+ maxResults,
568
+ includeHidden,
569
+ });
570
+
571
+ return new Promise((resolve) => {
572
+ const child = spawn(command, args, {
573
+ cwd: path || process.cwd(),
574
+ stdio: ['ignore', 'pipe', 'pipe'],
575
+ });
576
+
577
+ let stdout = '';
578
+ let stderr = '';
579
+
580
+ child.stdout.on('data', (data) => {
581
+ stdout += data.toString();
582
+ });
583
+
584
+ child.stderr.on('data', (data) => {
585
+ stderr += data.toString();
586
+ });
587
+
588
+ child.on('close', (code) => {
589
+ const results: ExactMatch[] = [];
590
+
591
+ if (code === 0 || (code === 1 && stdout.trim())) {
592
+ const lines = stdout.split('\n').filter((line) => line.trim());
593
+
594
+ for (const line of lines) {
595
+ try {
596
+ const item = JSON.parse(line);
597
+
598
+ if (item.type === 'match') {
599
+ const match: ExactMatch = {
600
+ file: item.data.path.text,
601
+ line: item.data.line_number,
602
+ column:
603
+ item.data.submatches && item.data.submatches[0]
604
+ ? item.data.submatches[0].start + 1
605
+ : 1,
606
+ content: item.data.lines.text.trim(),
607
+ };
608
+ results.push(match);
609
+ }
610
+ } catch {
611
+ continue;
612
+ }
613
+ }
614
+
615
+ resolve({
616
+ success: true,
617
+ results,
618
+ metadata: {
619
+ mode: 'ripgrep',
620
+ backend: 'ripgrep',
621
+ count: results.length,
622
+ query,
623
+ },
624
+ });
625
+ } else {
626
+ resolve({
627
+ success: false,
628
+ error: `ripgrep execution failed with code ${code}: ${stderr}`,
629
+ results: [],
630
+ });
631
+ }
632
+ });
633
+
634
+ child.on('error', (error) => {
635
+ resolve({
636
+ success: false,
637
+ error: `Failed to spawn ripgrep: ${error.message}`,
638
+ results: [],
639
+ });
640
+ });
641
+ });
642
+ }
643
+
644
+ /**
645
+ * Mode: exact - CodexLens exact/FTS search
646
+ * Requires index
647
+ */
648
+ async function executeCodexLensExactMode(params: Params): Promise<SearchResult> {
649
+ const { query, path = '.', maxResults = 10, enrich = false } = params;
650
+
651
+ if (!query) {
652
+ return {
653
+ success: false,
654
+ error: 'Query is required for search',
655
+ };
656
+ }
657
+
658
+ // Check CodexLens availability
659
+ const readyStatus = await ensureCodexLensReady();
660
+ if (!readyStatus.ready) {
661
+ return {
662
+ success: false,
663
+ error: `CodexLens not available: ${readyStatus.error}`,
664
+ };
665
+ }
666
+
667
+ // Check index status
668
+ const indexStatus = await checkIndexStatus(path);
669
+
670
+ const args = ['search', query, '--limit', maxResults.toString(), '--mode', 'exact', '--json'];
671
+ if (enrich) {
672
+ args.push('--enrich');
673
+ }
674
+ const result = await executeCodexLens(args, { cwd: path });
675
+
676
+ if (!result.success) {
677
+ return {
678
+ success: false,
679
+ error: result.error,
680
+ metadata: {
681
+ mode: 'exact',
682
+ backend: 'codexlens',
683
+ count: 0,
684
+ query,
685
+ warning: indexStatus.warning,
686
+ },
687
+ };
688
+ }
689
+
690
+ // Parse results
691
+ let results: SemanticMatch[] = [];
692
+ try {
693
+ const parsed = JSON.parse(stripAnsi(result.output || '{}'));
694
+ const data = parsed.result?.results || parsed.results || parsed;
695
+ results = (Array.isArray(data) ? data : []).map((item: any) => ({
696
+ file: item.path || item.file,
697
+ score: item.score || 0,
698
+ content: item.excerpt || item.content || '',
699
+ symbol: item.symbol || null,
700
+ }));
701
+ } catch {
702
+ // Keep empty results
703
+ }
704
+
705
+ return {
706
+ success: true,
707
+ results,
708
+ metadata: {
709
+ mode: 'exact',
710
+ backend: 'codexlens',
711
+ count: results.length,
712
+ query,
713
+ warning: indexStatus.warning,
714
+ },
715
+ };
716
+ }
717
+
718
+ /**
719
+ * Mode: hybrid - Best quality search with RRF fusion
720
+ * Uses CodexLens hybrid mode (exact + fuzzy + vector)
721
+ * Requires index with embeddings
722
+ */
723
+ async function executeHybridMode(params: Params): Promise<SearchResult> {
724
+ const { query, path = '.', maxResults = 10, enrich = false } = params;
725
+
726
+ if (!query) {
727
+ return {
728
+ success: false,
729
+ error: 'Query is required for search',
730
+ };
731
+ }
732
+
733
+ // Check CodexLens availability
734
+ const readyStatus = await ensureCodexLensReady();
735
+ if (!readyStatus.ready) {
736
+ return {
737
+ success: false,
738
+ error: `CodexLens not available: ${readyStatus.error}`,
739
+ };
740
+ }
741
+
742
+ // Check index status
743
+ const indexStatus = await checkIndexStatus(path);
744
+
745
+ const args = ['search', query, '--limit', maxResults.toString(), '--mode', 'hybrid', '--json'];
746
+ if (enrich) {
747
+ args.push('--enrich');
748
+ }
749
+ const result = await executeCodexLens(args, { cwd: path });
750
+
751
+ if (!result.success) {
752
+ return {
753
+ success: false,
754
+ error: result.error,
755
+ metadata: {
756
+ mode: 'hybrid',
757
+ backend: 'codexlens',
758
+ count: 0,
759
+ query,
760
+ warning: indexStatus.warning,
761
+ },
762
+ };
763
+ }
764
+
765
+ // Parse results
766
+ let results: SemanticMatch[] = [];
767
+ try {
768
+ const parsed = JSON.parse(stripAnsi(result.output || '{}'));
769
+ const data = parsed.result?.results || parsed.results || parsed;
770
+ results = (Array.isArray(data) ? data : []).map((item: any) => ({
771
+ file: item.path || item.file,
772
+ score: item.score || 0,
773
+ content: item.excerpt || item.content || '',
774
+ symbol: item.symbol || null,
775
+ }));
776
+ } catch {
777
+ return {
778
+ success: true,
779
+ results: [],
780
+ output: result.output,
781
+ metadata: {
782
+ mode: 'hybrid',
783
+ backend: 'codexlens',
784
+ count: 0,
785
+ query,
786
+ warning: indexStatus.warning || 'Failed to parse JSON output',
787
+ },
788
+ };
789
+ }
790
+
791
+ return {
792
+ success: true,
793
+ results,
794
+ metadata: {
795
+ mode: 'hybrid',
796
+ backend: 'codexlens',
797
+ count: results.length,
798
+ query,
799
+ note: 'Hybrid mode uses RRF fusion (exact + fuzzy + vector) for best results',
800
+ warning: indexStatus.warning,
801
+ },
802
+ };
803
+ }
804
+
805
+ /**
806
+ * TypeScript implementation of Reciprocal Rank Fusion
807
+ * Reference: codex-lens/src/codexlens/search/ranking.py
808
+ * Formula: score(d) = Σ weight_source / (k + rank_source(d))
809
+ */
810
+ function applyRRFFusion(
811
+ resultsMap: Map<string, any[]>,
812
+ weights: Record<string, number>,
813
+ limit: number,
814
+ k: number = 60,
815
+ ): any[] {
816
+ const pathScores = new Map<string, { score: number; result: any; sources: string[] }>();
817
+
818
+ resultsMap.forEach((results, source) => {
819
+ const weight = weights[source] || 0;
820
+ if (weight === 0 || !results) return;
821
+
822
+ results.forEach((result, rank) => {
823
+ const path = result.file || result.path;
824
+ if (!path) return;
825
+
826
+ const rrfContribution = weight / (k + rank + 1);
827
+
828
+ if (!pathScores.has(path)) {
829
+ pathScores.set(path, { score: 0, result, sources: [] });
830
+ }
831
+ const entry = pathScores.get(path)!;
832
+ entry.score += rrfContribution;
833
+ if (!entry.sources.includes(source)) {
834
+ entry.sources.push(source);
835
+ }
836
+ });
837
+ });
838
+
839
+ // Sort by fusion score descending
840
+ return Array.from(pathScores.values())
841
+ .sort((a, b) => b.score - a.score)
842
+ .slice(0, limit)
843
+ .map(item => ({
844
+ ...item.result,
845
+ fusion_score: item.score,
846
+ matched_backends: item.sources,
847
+ }));
848
+ }
849
+
850
+ /**
851
+ * Promise wrapper with timeout support
852
+ * @param promise - The promise to wrap
853
+ * @param ms - Timeout in milliseconds
854
+ * @param modeName - Name of the mode for error message
855
+ * @returns A new promise that rejects on timeout
856
+ */
857
+ function withTimeout<T>(promise: Promise<T>, ms: number, modeName: string): Promise<T> {
858
+ return new Promise((resolve, reject) => {
859
+ const timer = setTimeout(() => {
860
+ reject(new Error(`'${modeName}' search timed out after ${ms}ms`));
861
+ }, ms);
862
+
863
+ promise
864
+ .then(resolve)
865
+ .catch(reject)
866
+ .finally(() => clearTimeout(timer));
867
+ });
868
+ }
869
+
870
+ /**
871
+ * Mode: priority - Fallback search strategy: hybrid -> exact -> ripgrep
872
+ * Returns results from the first backend that succeeds and provides results.
873
+ * More efficient than parallel mode - stops as soon as valid results are found.
874
+ */
875
+ async function executePriorityFallbackMode(params: Params): Promise<SearchResult> {
876
+ const { query, path = '.' } = params;
877
+ const fallbackHistory: string[] = [];
878
+
879
+ if (!query) {
880
+ return { success: false, error: 'Query is required for search' };
881
+ }
882
+
883
+ // Check index status first
884
+ const indexStatus = await checkIndexStatus(path);
885
+
886
+ // 1. Try Hybrid search (highest priority) - 90s timeout for large indexes
887
+ if (indexStatus.indexed && indexStatus.has_embeddings) {
888
+ try {
889
+ const hybridResult = await withTimeout(executeHybridMode(params), 90000, 'hybrid');
890
+ if (hybridResult.success && hybridResult.results && (hybridResult.results as any[]).length > 0) {
891
+ fallbackHistory.push('hybrid: success');
892
+ return {
893
+ ...hybridResult,
894
+ metadata: {
895
+ ...hybridResult.metadata,
896
+ mode: 'priority',
897
+ note: 'Result from hybrid search (semantic + vector).',
898
+ fallback_history: fallbackHistory,
899
+ },
900
+ };
901
+ }
902
+ fallbackHistory.push('hybrid: no results');
903
+ } catch (error) {
904
+ fallbackHistory.push(`hybrid: ${(error as Error).message}`);
905
+ }
906
+ } else {
907
+ fallbackHistory.push(`hybrid: skipped (${!indexStatus.indexed ? 'no index' : 'no embeddings'})`);
908
+ }
909
+
910
+ // 2. Fallback to Exact search - 10s timeout
911
+ if (indexStatus.indexed) {
912
+ try {
913
+ const exactResult = await withTimeout(executeCodexLensExactMode(params), 10000, 'exact');
914
+ if (exactResult.success && exactResult.results && (exactResult.results as any[]).length > 0) {
915
+ fallbackHistory.push('exact: success');
916
+ return {
917
+ ...exactResult,
918
+ metadata: {
919
+ ...exactResult.metadata,
920
+ mode: 'priority',
921
+ note: 'Result from exact/FTS search (fallback from hybrid).',
922
+ fallback_history: fallbackHistory,
923
+ },
924
+ };
925
+ }
926
+ fallbackHistory.push('exact: no results');
927
+ } catch (error) {
928
+ fallbackHistory.push(`exact: ${(error as Error).message}`);
929
+ }
930
+ } else {
931
+ fallbackHistory.push('exact: skipped (no index)');
932
+ }
933
+
934
+ // 3. Final fallback to Ripgrep - 5s timeout
935
+ try {
936
+ const ripgrepResult = await withTimeout(executeRipgrepMode(params), 5000, 'ripgrep');
937
+ fallbackHistory.push(ripgrepResult.success ? 'ripgrep: success' : 'ripgrep: failed');
938
+ return {
939
+ ...ripgrepResult,
940
+ metadata: {
941
+ ...ripgrepResult.metadata,
942
+ mode: 'priority',
943
+ note: 'Result from ripgrep search (final fallback).',
944
+ fallback_history: fallbackHistory,
945
+ },
946
+ };
947
+ } catch (error) {
948
+ fallbackHistory.push(`ripgrep: ${(error as Error).message}`);
949
+ }
950
+
951
+ // All modes failed
952
+ return {
953
+ success: false,
954
+ error: 'All search backends in priority mode failed or returned no results.',
955
+ metadata: {
956
+ mode: 'priority',
957
+ query,
958
+ fallback_history: fallbackHistory,
959
+ } as any,
960
+ };
961
+ }
962
+
963
+ // Tool schema for MCP
964
+ export const schema: ToolSchema = {
965
+ name: 'smart_search',
966
+ description: `Intelligent code search with five modes. Use "auto" mode (default) for intelligent routing.
967
+
968
+ **Usage:**
969
+ smart_search(query="authentication logic") # auto mode - routes to best backend
970
+ smart_search(query="MyClass", mode="exact") # exact mode - precise FTS matching
971
+ smart_search(query="auth", mode="ripgrep") # ripgrep mode - fast literal search (no index)
972
+ smart_search(query="how to auth", mode="hybrid") # hybrid mode - semantic search (requires index)
973
+
974
+ **Index Management:**
975
+ smart_search(action="init") # Create FTS index for current directory
976
+ smart_search(action="status") # Check index and embedding status
977
+
978
+ **Graph Enrichment:**
979
+ smart_search(query="func", enrich=true) # Enrich results with code relationships (calls, imports, called_by, imported_by)
980
+
981
+ **Modes:** auto (intelligent routing), hybrid (semantic, needs index), exact (FTS), ripgrep (fast, no index), priority (fallback: hybrid→exact→ripgrep)`,
982
+ inputSchema: {
983
+ type: 'object',
984
+ properties: {
985
+ action: {
986
+ type: 'string',
987
+ enum: ['init', 'search', 'search_files', 'status'],
988
+ description: 'Action to perform: init (create FTS index, no embeddings), search (default), search_files (paths only), status (check index)',
989
+ default: 'search',
990
+ },
991
+ query: {
992
+ type: 'string',
993
+ description: 'Search query (required for search/search_files actions)',
994
+ },
995
+ mode: {
996
+ type: 'string',
997
+ enum: SEARCH_MODES,
998
+ description: 'Search mode: auto (default), hybrid (best quality), exact (CodexLens FTS), ripgrep (fast, no index), priority (fallback: hybrid->exact->ripgrep)',
999
+ default: 'auto',
1000
+ },
1001
+ output_mode: {
1002
+ type: 'string',
1003
+ enum: ['full', 'files_only', 'count'],
1004
+ description: 'Output format: full (default), files_only (paths only), count (per-file counts)',
1005
+ default: 'full',
1006
+ },
1007
+ path: {
1008
+ type: 'string',
1009
+ description: 'Directory path for init/search actions (default: current directory)',
1010
+ },
1011
+ paths: {
1012
+ type: 'array',
1013
+ description: 'Multiple paths to search within (for search action)',
1014
+ items: {
1015
+ type: 'string',
1016
+ },
1017
+ default: [],
1018
+ },
1019
+ contextLines: {
1020
+ type: 'number',
1021
+ description: 'Number of context lines around matches (exact mode only)',
1022
+ default: 0,
1023
+ },
1024
+ maxResults: {
1025
+ type: 'number',
1026
+ description: 'Maximum number of results (default: 10)',
1027
+ default: 10,
1028
+ },
1029
+ limit: {
1030
+ type: 'number',
1031
+ description: 'Alias for maxResults',
1032
+ default: 10,
1033
+ },
1034
+ includeHidden: {
1035
+ type: 'boolean',
1036
+ description: 'Include hidden files/directories',
1037
+ default: false,
1038
+ },
1039
+ languages: {
1040
+ type: 'array',
1041
+ items: { type: 'string' },
1042
+ description: 'Languages to index (for init action). Example: ["javascript", "typescript"]',
1043
+ },
1044
+ enrich: {
1045
+ type: 'boolean',
1046
+ description: 'Enrich search results with code graph relationships (calls, imports, called_by, imported_by).',
1047
+ default: false,
1048
+ },
1049
+ },
1050
+ required: [],
1051
+ },
1052
+ };
1053
+
1054
+ /**
1055
+ * Transform results based on output_mode
1056
+ */
1057
+ function transformOutput(
1058
+ results: ExactMatch[] | SemanticMatch[] | GraphMatch[] | unknown[],
1059
+ outputMode: 'full' | 'files_only' | 'count'
1060
+ ): unknown {
1061
+ if (!Array.isArray(results)) {
1062
+ return results;
1063
+ }
1064
+
1065
+ switch (outputMode) {
1066
+ case 'files_only': {
1067
+ // Extract unique file paths
1068
+ const files = [...new Set(results.map((r: any) => r.file))].filter(Boolean);
1069
+ return { files, count: files.length };
1070
+ }
1071
+ case 'count': {
1072
+ // Count matches per file
1073
+ const counts: Record<string, number> = {};
1074
+ for (const r of results) {
1075
+ const file = (r as any).file;
1076
+ if (file) {
1077
+ counts[file] = (counts[file] || 0) + 1;
1078
+ }
1079
+ }
1080
+ return {
1081
+ files: Object.entries(counts).map(([file, count]) => ({ file, count })),
1082
+ total: results.length,
1083
+ };
1084
+ }
1085
+ case 'full':
1086
+ default:
1087
+ return results;
1088
+ }
1089
+ }
1090
+
1091
+ // Handler function
1092
+ export async function handler(params: Record<string, unknown>): Promise<ToolResult<SearchResult>> {
1093
+ const parsed = ParamsSchema.safeParse(params);
1094
+ if (!parsed.success) {
1095
+ return { success: false, error: `Invalid params: ${parsed.error.message}` };
1096
+ }
1097
+
1098
+ const { action, mode, output_mode } = parsed.data;
1099
+
1100
+ // Sync limit and maxResults - use the larger of the two if both provided
1101
+ // This ensures user-provided values take precedence over defaults
1102
+ const effectiveLimit = Math.max(parsed.data.limit || 10, parsed.data.maxResults || 10);
1103
+ parsed.data.maxResults = effectiveLimit;
1104
+ parsed.data.limit = effectiveLimit;
1105
+
1106
+ try {
1107
+ let result: SearchResult;
1108
+
1109
+ // Handle actions
1110
+ switch (action) {
1111
+ case 'init':
1112
+ result = await executeInitAction(parsed.data);
1113
+ break;
1114
+
1115
+ case 'status':
1116
+ result = await executeStatusAction(parsed.data);
1117
+ break;
1118
+
1119
+ case 'search_files':
1120
+ // For search_files, use search mode but force files_only output
1121
+ parsed.data.output_mode = 'files_only';
1122
+ // Fall through to search
1123
+
1124
+ case 'search':
1125
+ default:
1126
+ // Handle search modes: auto | hybrid | exact | ripgrep | priority
1127
+ switch (mode) {
1128
+ case 'auto':
1129
+ result = await executeAutoMode(parsed.data);
1130
+ break;
1131
+ case 'hybrid':
1132
+ result = await executeHybridMode(parsed.data);
1133
+ break;
1134
+ case 'exact':
1135
+ result = await executeCodexLensExactMode(parsed.data);
1136
+ break;
1137
+ case 'ripgrep':
1138
+ result = await executeRipgrepMode(parsed.data);
1139
+ break;
1140
+ case 'priority':
1141
+ result = await executePriorityFallbackMode(parsed.data);
1142
+ break;
1143
+ default:
1144
+ throw new Error(`Unsupported mode: ${mode}. Use: auto, hybrid, exact, ripgrep, or priority`);
1145
+ }
1146
+ break;
1147
+ }
1148
+
1149
+ // Transform output based on output_mode (for search actions only)
1150
+ if (action === 'search' || action === 'search_files') {
1151
+ if (result.success && result.results && output_mode !== 'full') {
1152
+ result.results = transformOutput(result.results as any[], output_mode);
1153
+ }
1154
+ }
1155
+
1156
+ return result.success ? { success: true, result } : { success: false, error: result.error };
1157
+ } catch (error) {
1158
+ return { success: false, error: (error as Error).message };
1159
+ }
1160
+ }
1161
+
1162
+ /**
1163
+ * Execute init action with external progress callback
1164
+ * Used by MCP server for streaming progress
1165
+ */
1166
+ export async function executeInitWithProgress(
1167
+ params: Record<string, unknown>,
1168
+ onProgress?: (progress: ProgressInfo) => void
1169
+ ): Promise<SearchResult> {
1170
+ const path = (params.path as string) || '.';
1171
+ const languages = params.languages as string[] | undefined;
1172
+
1173
+ // Check CodexLens availability
1174
+ const readyStatus = await ensureCodexLensReady();
1175
+ if (!readyStatus.ready) {
1176
+ return {
1177
+ success: false,
1178
+ error: `CodexLens not available: ${readyStatus.error}. CodexLens will be auto-installed on first use.`,
1179
+ };
1180
+ }
1181
+
1182
+ const args = ['init', path];
1183
+ if (languages && languages.length > 0) {
1184
+ args.push('--languages', languages.join(','));
1185
+ }
1186
+
1187
+ // Track progress updates
1188
+ const progressUpdates: ProgressInfo[] = [];
1189
+ let lastProgress: ProgressInfo | null = null;
1190
+
1191
+ const result = await executeCodexLens(args, {
1192
+ cwd: path,
1193
+ timeout: 1800000, // 30 minutes for large codebases
1194
+ onProgress: (progress: ProgressInfo) => {
1195
+ progressUpdates.push(progress);
1196
+ lastProgress = progress;
1197
+ // Call external progress callback if provided
1198
+ if (onProgress) {
1199
+ onProgress(progress);
1200
+ }
1201
+ },
1202
+ });
1203
+
1204
+ // Build metadata with progress info
1205
+ const metadata: SearchMetadata = {
1206
+ action: 'init',
1207
+ path,
1208
+ };
1209
+
1210
+ if (lastProgress !== null) {
1211
+ const p = lastProgress as ProgressInfo;
1212
+ metadata.progress = {
1213
+ stage: p.stage,
1214
+ message: p.message,
1215
+ percent: p.percent,
1216
+ filesProcessed: p.filesProcessed,
1217
+ totalFiles: p.totalFiles,
1218
+ };
1219
+ }
1220
+
1221
+ if (progressUpdates.length > 0) {
1222
+ metadata.progressHistory = progressUpdates.slice(-5);
1223
+ }
1224
+
1225
+ return {
1226
+ success: result.success,
1227
+ error: result.error,
1228
+ message: result.success
1229
+ ? `CodexLens index created successfully for ${path}`
1230
+ : undefined,
1231
+ metadata,
1232
+ };
1233
+ }