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,1141 @@
1
+ /**
2
+ * CodexLens Tool - Bridge between CCW and CodexLens Python package
3
+ * Provides code indexing and semantic search via spawned Python process
4
+ *
5
+ * Features:
6
+ * - Automatic venv bootstrap at ~/.codexlens/venv
7
+ * - JSON protocol communication
8
+ * - Symbol extraction and semantic search
9
+ * - FTS5 full-text search
10
+ */
11
+
12
+ import { z } from 'zod';
13
+ import type { ToolSchema, ToolResult } from '../types/tool.js';
14
+ import { spawn, execSync, exec } from 'child_process';
15
+ import { existsSync, mkdirSync } from 'fs';
16
+ import { join, dirname } from 'path';
17
+ import { homedir } from 'os';
18
+ import { fileURLToPath } from 'url';
19
+
20
+ // Get directory of this module
21
+ const __filename = fileURLToPath(import.meta.url);
22
+ const __dirname = dirname(__filename);
23
+
24
+ // CodexLens configuration
25
+ const CODEXLENS_DATA_DIR = join(homedir(), '.codexlens');
26
+ const CODEXLENS_VENV = join(CODEXLENS_DATA_DIR, 'venv');
27
+ const VENV_PYTHON =
28
+ process.platform === 'win32'
29
+ ? join(CODEXLENS_VENV, 'Scripts', 'python.exe')
30
+ : join(CODEXLENS_VENV, 'bin', 'python');
31
+
32
+ // Bootstrap status cache
33
+ let bootstrapChecked = false;
34
+ let bootstrapReady = false;
35
+
36
+ // Track running indexing process for cancellation
37
+ let currentIndexingProcess: ReturnType<typeof spawn> | null = null;
38
+ let currentIndexingAborted = false;
39
+
40
+ // Define Zod schema for validation
41
+ const ParamsSchema = z.object({
42
+ action: z.enum([
43
+ 'init',
44
+ 'search',
45
+ 'search_files',
46
+ 'status',
47
+ 'symbol',
48
+ 'check',
49
+ 'update',
50
+ 'bootstrap',
51
+ ]),
52
+ path: z.string().optional(),
53
+ query: z.string().optional(),
54
+ mode: z.enum(['auto', 'text', 'semantic', 'exact', 'fuzzy', 'hybrid', 'vector', 'pure-vector']).default('auto'),
55
+ format: z.enum(['json', 'text', 'pretty']).default('json'),
56
+ languages: z.array(z.string()).optional(),
57
+ limit: z.number().default(20),
58
+ enrich: z.boolean().default(false),
59
+ // Additional fields for internal functions
60
+ file: z.string().optional(),
61
+ key: z.string().optional(),
62
+ value: z.string().optional(),
63
+ newPath: z.string().optional(),
64
+ all: z.boolean().optional(),
65
+ });
66
+
67
+ type Params = z.infer<typeof ParamsSchema>;
68
+
69
+ interface ReadyStatus {
70
+ ready: boolean;
71
+ error?: string;
72
+ version?: string;
73
+ }
74
+
75
+ interface SemanticStatus {
76
+ available: boolean;
77
+ backend?: string;
78
+ error?: string;
79
+ }
80
+
81
+ interface BootstrapResult {
82
+ success: boolean;
83
+ error?: string;
84
+ message?: string;
85
+ }
86
+
87
+ interface ExecuteResult {
88
+ success: boolean;
89
+ output?: string;
90
+ error?: string;
91
+ message?: string;
92
+ results?: unknown;
93
+ files?: unknown;
94
+ symbols?: unknown;
95
+ status?: unknown;
96
+ config?: unknown;
97
+ cleanResult?: unknown;
98
+ ready?: boolean;
99
+ version?: string;
100
+ }
101
+
102
+ interface ExecuteOptions {
103
+ timeout?: number;
104
+ cwd?: string;
105
+ onProgress?: (progress: ProgressInfo) => void;
106
+ }
107
+
108
+ interface ProgressInfo {
109
+ stage: string;
110
+ message: string;
111
+ percent: number;
112
+ filesProcessed?: number;
113
+ totalFiles?: number;
114
+ }
115
+
116
+ /**
117
+ * Detect available Python 3 executable
118
+ * @returns Python executable command
119
+ */
120
+ function getSystemPython(): string {
121
+ const commands = process.platform === 'win32' ? ['python', 'py', 'python3'] : ['python3', 'python'];
122
+
123
+ for (const cmd of commands) {
124
+ try {
125
+ const version = execSync(`${cmd} --version 2>&1`, { encoding: 'utf8' });
126
+ if (version.includes('Python 3')) {
127
+ return cmd;
128
+ }
129
+ } catch {
130
+ // Try next command
131
+ }
132
+ }
133
+ throw new Error('Python 3 not found. Please install Python 3 and ensure it is in PATH.');
134
+ }
135
+
136
+ /**
137
+ * Check if CodexLens venv exists and has required packages
138
+ * @returns Ready status
139
+ */
140
+ async function checkVenvStatus(): Promise<ReadyStatus> {
141
+ // Check venv exists
142
+ if (!existsSync(CODEXLENS_VENV)) {
143
+ return { ready: false, error: 'Venv not found' };
144
+ }
145
+
146
+ // Check python executable exists
147
+ if (!existsSync(VENV_PYTHON)) {
148
+ return { ready: false, error: 'Python executable not found in venv' };
149
+ }
150
+
151
+ // Check codexlens is importable
152
+ return new Promise((resolve) => {
153
+ const child = spawn(VENV_PYTHON, ['-c', 'import codexlens; print(codexlens.__version__)'], {
154
+ stdio: ['ignore', 'pipe', 'pipe'],
155
+ timeout: 10000,
156
+ });
157
+
158
+ let stdout = '';
159
+ let stderr = '';
160
+
161
+ child.stdout.on('data', (data) => {
162
+ stdout += data.toString();
163
+ });
164
+ child.stderr.on('data', (data) => {
165
+ stderr += data.toString();
166
+ });
167
+
168
+ child.on('close', (code) => {
169
+ if (code === 0) {
170
+ resolve({ ready: true, version: stdout.trim() });
171
+ } else {
172
+ resolve({ ready: false, error: `CodexLens not installed: ${stderr}` });
173
+ }
174
+ });
175
+
176
+ child.on('error', (err) => {
177
+ resolve({ ready: false, error: `Failed to check venv: ${err.message}` });
178
+ });
179
+ });
180
+ }
181
+
182
+ /**
183
+ * Check if semantic search dependencies are installed
184
+ * @returns Semantic status
185
+ */
186
+ async function checkSemanticStatus(): Promise<SemanticStatus> {
187
+ // First check if CodexLens is installed
188
+ const venvStatus = await checkVenvStatus();
189
+ if (!venvStatus.ready) {
190
+ return { available: false, error: 'CodexLens not installed' };
191
+ }
192
+
193
+ // Check semantic module availability
194
+ return new Promise((resolve) => {
195
+ const checkCode = `
196
+ import sys
197
+ try:
198
+ from codexlens.semantic import SEMANTIC_AVAILABLE, SEMANTIC_BACKEND
199
+ if SEMANTIC_AVAILABLE:
200
+ print(f"available:{SEMANTIC_BACKEND}")
201
+ else:
202
+ print("unavailable")
203
+ except Exception as e:
204
+ print(f"error:{e}")
205
+ `;
206
+ const child = spawn(VENV_PYTHON, ['-c', checkCode], {
207
+ stdio: ['ignore', 'pipe', 'pipe'],
208
+ timeout: 15000,
209
+ });
210
+
211
+ let stdout = '';
212
+ let stderr = '';
213
+
214
+ child.stdout.on('data', (data) => {
215
+ stdout += data.toString();
216
+ });
217
+ child.stderr.on('data', (data) => {
218
+ stderr += data.toString();
219
+ });
220
+
221
+ child.on('close', (code) => {
222
+ const output = stdout.trim();
223
+ if (output.startsWith('available:')) {
224
+ const backend = output.split(':')[1];
225
+ resolve({ available: true, backend });
226
+ } else if (output === 'unavailable') {
227
+ resolve({ available: false, error: 'Semantic dependencies not installed' });
228
+ } else {
229
+ resolve({ available: false, error: output || stderr || 'Unknown error' });
230
+ }
231
+ });
232
+
233
+ child.on('error', (err) => {
234
+ resolve({ available: false, error: `Check failed: ${err.message}` });
235
+ });
236
+ });
237
+ }
238
+
239
+ /**
240
+ * Install semantic search dependencies (fastembed, ONNX-based, ~200MB)
241
+ * @returns Bootstrap result
242
+ */
243
+ async function installSemantic(): Promise<BootstrapResult> {
244
+ // First ensure CodexLens is installed
245
+ const venvStatus = await checkVenvStatus();
246
+ if (!venvStatus.ready) {
247
+ return { success: false, error: 'CodexLens not installed. Install CodexLens first.' };
248
+ }
249
+
250
+ const pipPath =
251
+ process.platform === 'win32'
252
+ ? join(CODEXLENS_VENV, 'Scripts', 'pip.exe')
253
+ : join(CODEXLENS_VENV, 'bin', 'pip');
254
+
255
+ return new Promise((resolve) => {
256
+ console.log('[CodexLens] Installing semantic search dependencies (fastembed)...');
257
+ console.log('[CodexLens] Using ONNX-based fastembed backend (~200MB)');
258
+
259
+ const child = spawn(pipPath, ['install', 'numpy>=1.24', 'fastembed>=0.2'], {
260
+ stdio: ['ignore', 'pipe', 'pipe'],
261
+ timeout: 600000, // 10 minutes for potential model download
262
+ });
263
+
264
+ let stdout = '';
265
+ let stderr = '';
266
+
267
+ child.stdout.on('data', (data) => {
268
+ stdout += data.toString();
269
+ // Log progress
270
+ const line = data.toString().trim();
271
+ if (line.includes('Downloading') || line.includes('Installing') || line.includes('Collecting')) {
272
+ console.log(`[CodexLens] ${line}`);
273
+ }
274
+ });
275
+
276
+ child.stderr.on('data', (data) => {
277
+ stderr += data.toString();
278
+ });
279
+
280
+ child.on('close', (code) => {
281
+ if (code === 0) {
282
+ console.log('[CodexLens] Semantic dependencies installed successfully');
283
+ resolve({ success: true });
284
+ } else {
285
+ resolve({ success: false, error: `Installation failed: ${stderr || stdout}` });
286
+ }
287
+ });
288
+
289
+ child.on('error', (err) => {
290
+ resolve({ success: false, error: `Failed to run pip: ${err.message}` });
291
+ });
292
+ });
293
+ }
294
+
295
+ /**
296
+ * Bootstrap CodexLens venv with required packages
297
+ * @returns Bootstrap result
298
+ */
299
+ async function bootstrapVenv(): Promise<BootstrapResult> {
300
+ // Ensure data directory exists
301
+ if (!existsSync(CODEXLENS_DATA_DIR)) {
302
+ mkdirSync(CODEXLENS_DATA_DIR, { recursive: true });
303
+ }
304
+
305
+ // Create venv if not exists
306
+ if (!existsSync(CODEXLENS_VENV)) {
307
+ try {
308
+ console.log('[CodexLens] Creating virtual environment...');
309
+ const pythonCmd = getSystemPython();
310
+ execSync(`${pythonCmd} -m venv "${CODEXLENS_VENV}"`, { stdio: 'inherit' });
311
+ } catch (err) {
312
+ return { success: false, error: `Failed to create venv: ${(err as Error).message}` };
313
+ }
314
+ }
315
+
316
+ // Install codexlens with semantic extras
317
+ try {
318
+ console.log('[CodexLens] Installing codexlens package...');
319
+ const pipPath =
320
+ process.platform === 'win32'
321
+ ? join(CODEXLENS_VENV, 'Scripts', 'pip.exe')
322
+ : join(CODEXLENS_VENV, 'bin', 'pip');
323
+
324
+ // Try multiple local paths, then fall back to PyPI
325
+ const possiblePaths = [
326
+ join(process.cwd(), 'codex-lens'),
327
+ join(__dirname, '..', '..', '..', 'codex-lens'), // ccw/src/tools -> project root
328
+ join(homedir(), 'codex-lens'),
329
+ ];
330
+
331
+ let installed = false;
332
+ for (const localPath of possiblePaths) {
333
+ if (existsSync(join(localPath, 'pyproject.toml'))) {
334
+ console.log(`[CodexLens] Installing from local path: ${localPath}`);
335
+ execSync(`"${pipPath}" install -e "${localPath}"`, { stdio: 'inherit' });
336
+ installed = true;
337
+ break;
338
+ }
339
+ }
340
+
341
+ if (!installed) {
342
+ console.log('[CodexLens] Installing from PyPI...');
343
+ execSync(`"${pipPath}" install codexlens`, { stdio: 'inherit' });
344
+ }
345
+
346
+ return { success: true };
347
+ } catch (err) {
348
+ return { success: false, error: `Failed to install codexlens: ${(err as Error).message}` };
349
+ }
350
+ }
351
+
352
+ /**
353
+ * Ensure CodexLens is ready to use
354
+ * @returns Ready status
355
+ */
356
+ async function ensureReady(): Promise<ReadyStatus> {
357
+ // Use cached result if already checked
358
+ if (bootstrapChecked && bootstrapReady) {
359
+ return { ready: true };
360
+ }
361
+
362
+ // Check current status
363
+ const status = await checkVenvStatus();
364
+ if (status.ready) {
365
+ bootstrapChecked = true;
366
+ bootstrapReady = true;
367
+ return { ready: true, version: status.version };
368
+ }
369
+
370
+ // Attempt bootstrap
371
+ const bootstrap = await bootstrapVenv();
372
+ if (!bootstrap.success) {
373
+ return { ready: false, error: bootstrap.error };
374
+ }
375
+
376
+ // Verify after bootstrap
377
+ const recheck = await checkVenvStatus();
378
+ bootstrapChecked = true;
379
+ bootstrapReady = recheck.ready;
380
+
381
+ return recheck;
382
+ }
383
+
384
+ /**
385
+ * Parse progress info from CodexLens output
386
+ * @param line - Output line to parse
387
+ * @returns Progress info or null
388
+ */
389
+ function parseProgressLine(line: string): ProgressInfo | null {
390
+ // Parse file processing progress: "Processing file X/Y: path"
391
+ const fileMatch = line.match(/Processing file (\d+)\/(\d+):\s*(.+)/i);
392
+ if (fileMatch) {
393
+ const current = parseInt(fileMatch[1], 10);
394
+ const total = parseInt(fileMatch[2], 10);
395
+ return {
396
+ stage: 'indexing',
397
+ message: `Processing ${fileMatch[3]}`,
398
+ percent: Math.round((current / total) * 80) + 10, // 10-90%
399
+ filesProcessed: current,
400
+ totalFiles: total,
401
+ };
402
+ }
403
+
404
+ // Parse stage messages
405
+ if (line.includes('Discovering files')) {
406
+ return { stage: 'discover', message: 'Discovering files...', percent: 5 };
407
+ }
408
+ if (line.includes('Building index')) {
409
+ return { stage: 'build', message: 'Building index...', percent: 10 };
410
+ }
411
+ if (line.includes('Extracting symbols')) {
412
+ return { stage: 'symbols', message: 'Extracting symbols...', percent: 50 };
413
+ }
414
+ if (line.includes('Generating embeddings') || line.includes('Creating embeddings')) {
415
+ return { stage: 'embeddings', message: 'Generating embeddings...', percent: 70 };
416
+ }
417
+ // Note: "Finalizing index" and "Building ANN" are handled separately below
418
+ // Only match generic "Complete" here (not "Finalizing" which has specific handlers)
419
+
420
+ // Parse indexed count: "Indexed X files" - FTS complete, but embeddings may follow
421
+ const indexedMatch = line.match(/Indexed (\d+) files/i);
422
+ if (indexedMatch) {
423
+ return {
424
+ stage: 'fts_complete', // Not 'complete' - embeddings generation may still be pending
425
+ message: `Indexed ${indexedMatch[1]} files, generating embeddings...`,
426
+ percent: 60, // FTS done, embeddings starting
427
+ filesProcessed: parseInt(indexedMatch[1], 10),
428
+ };
429
+ }
430
+
431
+ // Parse embedding batch progress: "Batch X: N files, M chunks"
432
+ const batchMatch = line.match(/Batch (\d+):\s*(\d+) files,\s*(\d+) chunks/i);
433
+ if (batchMatch) {
434
+ return {
435
+ stage: 'embeddings',
436
+ message: `Embedding batch ${batchMatch[1]}: ${batchMatch[3]} chunks`,
437
+ percent: 70, // Stay at 70% during embedding batches
438
+ };
439
+ }
440
+
441
+ // Parse embedding progress with file count
442
+ const embedProgressMatch = line.match(/Processing (\d+) files/i);
443
+ if (embedProgressMatch && line.toLowerCase().includes('embed')) {
444
+ return {
445
+ stage: 'embeddings',
446
+ message: `Processing ${embedProgressMatch[1]} files for embeddings`,
447
+ percent: 75,
448
+ };
449
+ }
450
+
451
+ // Parse finalizing ANN index
452
+ if (line.includes('Finalizing index') || line.includes('Building ANN')) {
453
+ return { stage: 'finalizing', message: 'Finalizing vector index...', percent: 90 };
454
+ }
455
+
456
+ // Parse embeddings complete message
457
+ const embedCompleteMatch = line.match(/Embeddings complete:\s*(\d+)\s*chunks/i);
458
+ if (embedCompleteMatch) {
459
+ return {
460
+ stage: 'embeddings_complete',
461
+ message: `Embeddings complete: ${embedCompleteMatch[1]} chunks`,
462
+ percent: 95,
463
+ };
464
+ }
465
+
466
+ // Parse generic completion (but not "Embeddings complete" which is handled above)
467
+ if (line.includes('Complete') && !line.toLowerCase().includes('embeddings complete')) {
468
+ return { stage: 'complete', message: 'Complete', percent: 98 };
469
+ }
470
+
471
+ return null;
472
+ }
473
+
474
+ /**
475
+ * Execute CodexLens CLI command with real-time progress updates
476
+ * @param args - CLI arguments
477
+ * @param options - Execution options
478
+ * @returns Execution result
479
+ */
480
+ async function executeCodexLens(args: string[], options: ExecuteOptions = {}): Promise<ExecuteResult> {
481
+ const { timeout = 300000, cwd = process.cwd(), onProgress } = options; // Default 5 min
482
+
483
+ // Ensure ready
484
+ const readyStatus = await ensureReady();
485
+ if (!readyStatus.ready) {
486
+ return { success: false, error: readyStatus.error };
487
+ }
488
+
489
+ return new Promise((resolve) => {
490
+ // Build command string - quote paths for shell execution
491
+ const quotedPython = `"${VENV_PYTHON}"`;
492
+ const cmdArgs = args.map(arg => {
493
+ // Quote arguments that contain spaces or special characters
494
+ if (arg.includes(' ') || arg.includes('\\')) {
495
+ return `"${arg}"`;
496
+ }
497
+ return arg;
498
+ });
499
+
500
+ // Build full command - on Windows, prepend cd to handle different drives
501
+ let fullCmd: string;
502
+ if (process.platform === 'win32' && cwd) {
503
+ // Use cd /d to change drive and directory, then run command
504
+ fullCmd = `cd /d "${cwd}" && ${quotedPython} -m codexlens ${cmdArgs.join(' ')}`;
505
+ } else {
506
+ fullCmd = `${quotedPython} -m codexlens ${cmdArgs.join(' ')}`;
507
+ }
508
+
509
+ // Use spawn with shell for real-time progress updates
510
+ // spawn streams output in real-time, unlike exec which buffers until completion
511
+ const child = spawn(fullCmd, [], {
512
+ cwd: process.platform === 'win32' ? undefined : cwd,
513
+ shell: process.platform === 'win32' ? process.env.ComSpec || true : true,
514
+ timeout,
515
+ });
516
+
517
+ // Track indexing process for cancellation (only for init commands)
518
+ const isIndexingCommand = args.includes('init');
519
+ if (isIndexingCommand) {
520
+ currentIndexingProcess = child;
521
+ currentIndexingAborted = false;
522
+ }
523
+
524
+ let stdout = '';
525
+ let stderr = '';
526
+ let stdoutLineBuffer = '';
527
+ let stderrLineBuffer = '';
528
+ let timeoutHandle: NodeJS.Timeout | null = null;
529
+ let resolved = false;
530
+
531
+ // Helper to safely resolve only once
532
+ const safeResolve = (result: ExecuteResult) => {
533
+ if (resolved) return;
534
+ resolved = true;
535
+ if (timeoutHandle) {
536
+ clearTimeout(timeoutHandle);
537
+ timeoutHandle = null;
538
+ }
539
+ // Clear indexing process tracking
540
+ if (isIndexingCommand) {
541
+ currentIndexingProcess = null;
542
+ }
543
+ resolve(result);
544
+ };
545
+
546
+ // Set up timeout handler
547
+ if (timeout > 0) {
548
+ timeoutHandle = setTimeout(() => {
549
+ if (!resolved) {
550
+ child.kill('SIGTERM');
551
+ // Give it a moment to die gracefully, then force kill
552
+ setTimeout(() => {
553
+ if (!resolved) {
554
+ child.kill('SIGKILL');
555
+ }
556
+ }, 5000);
557
+ safeResolve({ success: false, error: 'Command timed out' });
558
+ }
559
+ }, timeout);
560
+ }
561
+
562
+ // Process stdout line by line for real-time progress
563
+ child.stdout?.on('data', (data: Buffer) => {
564
+ const chunk = data.toString();
565
+ stdoutLineBuffer += chunk;
566
+ stdout += chunk;
567
+
568
+ // Process complete lines
569
+ const lines = stdoutLineBuffer.split('\n');
570
+ stdoutLineBuffer = lines.pop() || ''; // Keep incomplete line in buffer
571
+
572
+ for (const line of lines) {
573
+ const trimmedLine = line.trim();
574
+ if (trimmedLine && onProgress) {
575
+ const progress = parseProgressLine(trimmedLine);
576
+ if (progress) {
577
+ onProgress(progress);
578
+ }
579
+ }
580
+ }
581
+ });
582
+
583
+ // Collect stderr
584
+ child.stderr?.on('data', (data: Buffer) => {
585
+ const chunk = data.toString();
586
+ stderrLineBuffer += chunk;
587
+ stderr += chunk;
588
+
589
+ // Also check stderr for progress (some tools output progress to stderr)
590
+ const lines = stderrLineBuffer.split('\n');
591
+ stderrLineBuffer = lines.pop() || '';
592
+
593
+ for (const line of lines) {
594
+ const trimmedLine = line.trim();
595
+ if (trimmedLine && onProgress) {
596
+ const progress = parseProgressLine(trimmedLine);
597
+ if (progress) {
598
+ onProgress(progress);
599
+ }
600
+ }
601
+ }
602
+ });
603
+
604
+ // Handle process errors (spawn failure)
605
+ child.on('error', (err) => {
606
+ safeResolve({ success: false, error: `Failed to start process: ${err.message}` });
607
+ });
608
+
609
+ // Handle process completion
610
+ child.on('close', (code) => {
611
+ // Process any remaining buffered content
612
+ if (stdoutLineBuffer.trim() && onProgress) {
613
+ const progress = parseProgressLine(stdoutLineBuffer.trim());
614
+ if (progress) {
615
+ onProgress(progress);
616
+ }
617
+ }
618
+
619
+ if (code === 0) {
620
+ safeResolve({ success: true, output: stdout.trim() });
621
+ } else {
622
+ safeResolve({ success: false, error: stderr.trim() || `Process exited with code ${code}` });
623
+ }
624
+ });
625
+ });
626
+ }
627
+
628
+ /**
629
+ * Initialize CodexLens index for a directory
630
+ * @param params - Parameters
631
+ * @returns Execution result
632
+ */
633
+ async function initIndex(params: Params): Promise<ExecuteResult> {
634
+ const { path = '.', languages } = params;
635
+
636
+ const args = ['init', path];
637
+ if (languages && languages.length > 0) {
638
+ args.push('--languages', languages.join(','));
639
+ }
640
+
641
+ return executeCodexLens(args, { cwd: path });
642
+ }
643
+
644
+ /**
645
+ * Search code using CodexLens
646
+ * @param params - Search parameters
647
+ * @returns Execution result
648
+ */
649
+ async function searchCode(params: Params): Promise<ExecuteResult> {
650
+ const { query, path = '.', limit = 20, mode = 'auto', enrich = false } = params;
651
+
652
+ if (!query) {
653
+ return { success: false, error: 'Query is required for search action' };
654
+ }
655
+
656
+ // Map MCP mode names to CLI mode names
657
+ const modeMap: Record<string, string> = {
658
+ 'text': 'exact',
659
+ 'semantic': 'pure-vector',
660
+ 'auto': 'auto',
661
+ 'exact': 'exact',
662
+ 'fuzzy': 'fuzzy',
663
+ 'hybrid': 'hybrid',
664
+ 'vector': 'vector',
665
+ 'pure-vector': 'pure-vector',
666
+ };
667
+
668
+ const cliMode = modeMap[mode] || 'auto';
669
+ const args = ['search', query, '--limit', limit.toString(), '--mode', cliMode, '--json'];
670
+
671
+ if (enrich) {
672
+ args.push('--enrich');
673
+ }
674
+
675
+ const result = await executeCodexLens(args, { cwd: path });
676
+
677
+ if (result.success && result.output) {
678
+ try {
679
+ result.results = JSON.parse(result.output);
680
+ delete result.output;
681
+ } catch {
682
+ // Keep raw output if JSON parse fails
683
+ }
684
+ }
685
+
686
+ return result;
687
+ }
688
+
689
+ /**
690
+ * Search code and return only file paths
691
+ * @param params - Search parameters
692
+ * @returns Execution result
693
+ */
694
+ async function searchFiles(params: Params): Promise<ExecuteResult> {
695
+ const { query, path = '.', limit = 20, mode = 'auto', enrich = false } = params;
696
+
697
+ if (!query) {
698
+ return { success: false, error: 'Query is required for search_files action' };
699
+ }
700
+
701
+ // Map MCP mode names to CLI mode names
702
+ const modeMap: Record<string, string> = {
703
+ 'text': 'exact',
704
+ 'semantic': 'pure-vector',
705
+ 'auto': 'auto',
706
+ 'exact': 'exact',
707
+ 'fuzzy': 'fuzzy',
708
+ 'hybrid': 'hybrid',
709
+ 'vector': 'vector',
710
+ 'pure-vector': 'pure-vector',
711
+ };
712
+
713
+ const cliMode = modeMap[mode] || 'auto';
714
+ const args = ['search', query, '--files-only', '--limit', limit.toString(), '--mode', cliMode, '--json'];
715
+
716
+ if (enrich) {
717
+ args.push('--enrich');
718
+ }
719
+
720
+ const result = await executeCodexLens(args, { cwd: path });
721
+
722
+ if (result.success && result.output) {
723
+ try {
724
+ result.files = JSON.parse(result.output);
725
+ delete result.output;
726
+ } catch {
727
+ // Keep raw output if JSON parse fails
728
+ }
729
+ }
730
+
731
+ return result;
732
+ }
733
+
734
+ /**
735
+ * Extract symbols from a file
736
+ * @param params - Parameters
737
+ * @returns Execution result
738
+ */
739
+ async function extractSymbols(params: Params): Promise<ExecuteResult> {
740
+ const { file } = params;
741
+
742
+ if (!file) {
743
+ return { success: false, error: 'File is required for symbol action' };
744
+ }
745
+
746
+ const args = ['symbol', file, '--json'];
747
+
748
+ const result = await executeCodexLens(args);
749
+
750
+ if (result.success && result.output) {
751
+ try {
752
+ result.symbols = JSON.parse(result.output);
753
+ delete result.output;
754
+ } catch {
755
+ // Keep raw output if JSON parse fails
756
+ }
757
+ }
758
+
759
+ return result;
760
+ }
761
+
762
+ /**
763
+ * Get index status
764
+ * @param params - Parameters
765
+ * @returns Execution result
766
+ */
767
+ async function getStatus(params: Params): Promise<ExecuteResult> {
768
+ const { path = '.' } = params;
769
+
770
+ const args = ['status', '--json'];
771
+
772
+ const result = await executeCodexLens(args, { cwd: path });
773
+
774
+ if (result.success && result.output) {
775
+ try {
776
+ result.status = JSON.parse(result.output);
777
+ delete result.output;
778
+ } catch {
779
+ // Keep raw output if JSON parse fails
780
+ }
781
+ }
782
+
783
+ return result;
784
+ }
785
+
786
+ /**
787
+ * Show configuration
788
+ * @param params - Parameters
789
+ * @returns Execution result
790
+ */
791
+ async function configShow(): Promise<ExecuteResult> {
792
+ const args = ['config', 'show', '--json'];
793
+ const result = await executeCodexLens(args);
794
+
795
+ if (result.success && result.output) {
796
+ try {
797
+ result.config = JSON.parse(result.output);
798
+ delete result.output;
799
+ } catch {
800
+ // Keep raw output if JSON parse fails
801
+ }
802
+ }
803
+
804
+ return result;
805
+ }
806
+
807
+ /**
808
+ * Set configuration value
809
+ * @param params - Parameters
810
+ * @returns Execution result
811
+ */
812
+ async function configSet(params: Params): Promise<ExecuteResult> {
813
+ const { key, value } = params;
814
+
815
+ if (!key) {
816
+ return { success: false, error: 'key is required for config_set action' };
817
+ }
818
+ if (!value) {
819
+ return { success: false, error: 'value is required for config_set action' };
820
+ }
821
+
822
+ const args = ['config', 'set', key, value, '--json'];
823
+ const result = await executeCodexLens(args);
824
+
825
+ if (result.success && result.output) {
826
+ try {
827
+ result.config = JSON.parse(result.output);
828
+ delete result.output;
829
+ } catch {
830
+ // Keep raw output if JSON parse fails
831
+ }
832
+ }
833
+
834
+ return result;
835
+ }
836
+
837
+ /**
838
+ * Migrate indexes to new location
839
+ * @param params - Parameters
840
+ * @returns Execution result
841
+ */
842
+ async function configMigrate(params: Params): Promise<ExecuteResult> {
843
+ const { newPath } = params;
844
+
845
+ if (!newPath) {
846
+ return { success: false, error: 'newPath is required for config_migrate action' };
847
+ }
848
+
849
+ const args = ['config', 'migrate', newPath, '--json'];
850
+ const result = await executeCodexLens(args, { timeout: 300000 }); // 5 min for migration
851
+
852
+ if (result.success && result.output) {
853
+ try {
854
+ result.config = JSON.parse(result.output);
855
+ delete result.output;
856
+ } catch {
857
+ // Keep raw output if JSON parse fails
858
+ }
859
+ }
860
+
861
+ return result;
862
+ }
863
+
864
+ /**
865
+ * Clean indexes
866
+ * @param params - Parameters
867
+ * @returns Execution result
868
+ */
869
+ async function cleanIndexes(params: Params): Promise<ExecuteResult> {
870
+ const { path, all } = params;
871
+
872
+ const args = ['clean'];
873
+
874
+ if (all) {
875
+ args.push('--all');
876
+ } else if (path) {
877
+ args.push(path);
878
+ }
879
+
880
+ args.push('--json');
881
+ const result = await executeCodexLens(args);
882
+
883
+ if (result.success && result.output) {
884
+ try {
885
+ result.cleanResult = JSON.parse(result.output);
886
+ delete result.output;
887
+ } catch {
888
+ // Keep raw output if JSON parse fails
889
+ }
890
+ }
891
+
892
+ return result;
893
+ }
894
+
895
+ // Tool schema for MCP
896
+ export const schema: ToolSchema = {
897
+ name: 'codex_lens',
898
+ description: `CodexLens - Code indexing and semantic search.
899
+
900
+ Usage:
901
+ codex_lens(action="init", path=".") # Index directory (auto-generates embeddings if available)
902
+ codex_lens(action="search", query="func") # Search code (auto: hybrid if embeddings exist, else exact)
903
+ codex_lens(action="search", query="func", mode="hybrid") # Force hybrid search
904
+ codex_lens(action="search_files", query="x") # Search, return paths only
905
+
906
+ Graph Enrichment:
907
+ codex_lens(action="search", query="func", enrich=true) # Enrich results with code relationships
908
+
909
+ Search Modes:
910
+ - auto: Auto-detect (hybrid if embeddings exist, exact otherwise) [default]
911
+ - exact/text: Exact FTS for code identifiers
912
+ - hybrid: Exact + Fuzzy + Vector fusion (best results, requires embeddings)
913
+ - fuzzy: Typo-tolerant search
914
+ - vector: Semantic + keyword
915
+ - pure-vector/semantic: Pure semantic search
916
+
917
+ Note: For advanced operations (config, status, clean), use CLI directly: codexlens --help`,
918
+ inputSchema: {
919
+ type: 'object',
920
+ properties: {
921
+ action: {
922
+ type: 'string',
923
+ enum: [
924
+ 'init',
925
+ 'search',
926
+ 'search_files',
927
+ 'status',
928
+ 'symbol',
929
+ 'check',
930
+ 'update',
931
+ 'bootstrap',
932
+ ],
933
+ description: 'Action to perform: init/update (index directory), search (search code), search_files (search files only), status (index status), symbol (extract symbols), check (check if ready), bootstrap (setup venv)',
934
+ },
935
+ path: {
936
+ type: 'string',
937
+ description: 'Target directory path (for init, search, search_files). Defaults to current directory.',
938
+ },
939
+ query: {
940
+ type: 'string',
941
+ description: 'Search query (required for search and search_files actions)',
942
+ },
943
+ mode: {
944
+ type: 'string',
945
+ enum: ['auto', 'text', 'semantic', 'exact', 'fuzzy', 'hybrid', 'vector', 'pure-vector'],
946
+ description: 'Search mode: auto (default, hybrid if embeddings exist), text/exact (FTS), hybrid (best), fuzzy, vector, semantic/pure-vector',
947
+ default: 'auto',
948
+ },
949
+ format: {
950
+ type: 'string',
951
+ enum: ['json', 'text', 'pretty'],
952
+ description: 'Output format: json (default), text, pretty',
953
+ default: 'json',
954
+ },
955
+ languages: {
956
+ type: 'array',
957
+ items: { type: 'string' },
958
+ description: 'Languages to index (for init action). Example: ["javascript", "typescript", "python"]',
959
+ },
960
+ limit: {
961
+ type: 'number',
962
+ description: 'Maximum number of search results (for search and search_files actions)',
963
+ default: 20,
964
+ },
965
+ enrich: {
966
+ type: 'boolean',
967
+ description: 'Enrich search results with code graph relationships (calls, imports)',
968
+ default: false,
969
+ },
970
+ },
971
+ required: ['action'],
972
+ },
973
+ };
974
+
975
+ // Handler function
976
+ export async function handler(params: Record<string, unknown>): Promise<ToolResult<ExecuteResult>> {
977
+ const parsed = ParamsSchema.safeParse(params);
978
+ if (!parsed.success) {
979
+ return { success: false, error: `Invalid params: ${parsed.error.message}` };
980
+ }
981
+
982
+ const { action } = parsed.data;
983
+
984
+ try {
985
+ let result: ExecuteResult;
986
+
987
+ switch (action) {
988
+ case 'init':
989
+ result = await initIndex(parsed.data);
990
+ break;
991
+
992
+ case 'search':
993
+ result = await searchCode(parsed.data);
994
+ break;
995
+
996
+ case 'search_files':
997
+ result = await searchFiles(parsed.data);
998
+ break;
999
+
1000
+ case 'status':
1001
+ result = await getStatus(parsed.data);
1002
+ break;
1003
+
1004
+ case 'symbol':
1005
+ result = await extractSymbols(parsed.data);
1006
+ break;
1007
+
1008
+ case 'check':
1009
+ const checkStatus = await ensureReady();
1010
+ result = {
1011
+ success: checkStatus.ready,
1012
+ ready: checkStatus.ready,
1013
+ version: checkStatus.version,
1014
+ error: checkStatus.error,
1015
+ };
1016
+ break;
1017
+
1018
+ case 'update':
1019
+ // Update is an alias for init (incremental update)
1020
+ result = await initIndex(parsed.data);
1021
+ break;
1022
+
1023
+ case 'bootstrap':
1024
+ const bootstrapResult = await bootstrapVenv();
1025
+ result = {
1026
+ success: bootstrapResult.success,
1027
+ message: bootstrapResult.message,
1028
+ error: bootstrapResult.error,
1029
+ };
1030
+ break;
1031
+
1032
+ default:
1033
+ throw new Error(
1034
+ `Unknown action: ${action}. Valid actions: init, search, search_files, status, symbol, check, update, bootstrap`
1035
+ );
1036
+ }
1037
+
1038
+ return result.success ? { success: true, result } : { success: false, error: result.error };
1039
+ } catch (error) {
1040
+ return { success: false, error: (error as Error).message };
1041
+ }
1042
+ }
1043
+
1044
+ /**
1045
+ * Uninstall CodexLens by removing the venv directory
1046
+ * @returns Uninstall result
1047
+ */
1048
+ async function uninstallCodexLens(): Promise<BootstrapResult> {
1049
+ try {
1050
+ // Check if venv exists
1051
+ if (!existsSync(CODEXLENS_VENV)) {
1052
+ return { success: false, error: 'CodexLens not installed (venv not found)' };
1053
+ }
1054
+
1055
+ console.log('[CodexLens] Uninstalling CodexLens...');
1056
+ console.log(`[CodexLens] Removing directory: ${CODEXLENS_DATA_DIR}`);
1057
+
1058
+ // Remove the entire .codexlens directory
1059
+ const fs = await import('fs');
1060
+ fs.rmSync(CODEXLENS_DATA_DIR, { recursive: true, force: true });
1061
+
1062
+ // Reset bootstrap cache
1063
+ bootstrapChecked = false;
1064
+ bootstrapReady = false;
1065
+
1066
+ console.log('[CodexLens] CodexLens uninstalled successfully');
1067
+ return { success: true, message: 'CodexLens uninstalled successfully' };
1068
+ } catch (err) {
1069
+ return { success: false, error: `Failed to uninstall CodexLens: ${(err as Error).message}` };
1070
+ }
1071
+ }
1072
+
1073
+ /**
1074
+ * Cancel the currently running indexing process
1075
+ * @returns Result indicating if cancellation was successful
1076
+ */
1077
+ function cancelIndexing(): { success: boolean; message?: string; error?: string } {
1078
+ if (!currentIndexingProcess) {
1079
+ return { success: false, error: 'No indexing process is currently running' };
1080
+ }
1081
+
1082
+ if (currentIndexingAborted) {
1083
+ return { success: false, error: 'Indexing process is already being cancelled' };
1084
+ }
1085
+
1086
+ try {
1087
+ currentIndexingAborted = true;
1088
+
1089
+ // Send SIGTERM first for graceful shutdown
1090
+ if (process.platform === 'win32') {
1091
+ // On Windows, use taskkill to kill the process tree
1092
+ const { execSync } = require('child_process');
1093
+ try {
1094
+ execSync(`taskkill /pid ${currentIndexingProcess.pid} /T /F`, { stdio: 'ignore' });
1095
+ } catch {
1096
+ // Process may have already exited
1097
+ }
1098
+ } else {
1099
+ // On Unix, send SIGTERM
1100
+ currentIndexingProcess.kill('SIGTERM');
1101
+
1102
+ // Force kill after 3 seconds if still running
1103
+ setTimeout(() => {
1104
+ if (currentIndexingProcess) {
1105
+ currentIndexingProcess.kill('SIGKILL');
1106
+ }
1107
+ }, 3000);
1108
+ }
1109
+
1110
+ console.log('[CodexLens] Indexing process cancelled');
1111
+ return { success: true, message: 'Indexing cancelled successfully' };
1112
+ } catch (err) {
1113
+ return { success: false, error: `Failed to cancel indexing: ${(err as Error).message}` };
1114
+ }
1115
+ }
1116
+
1117
+ /**
1118
+ * Check if an indexing process is currently running
1119
+ * @returns True if indexing is in progress
1120
+ */
1121
+ function isIndexingInProgress(): boolean {
1122
+ return currentIndexingProcess !== null && !currentIndexingAborted;
1123
+ }
1124
+
1125
+ // Export types
1126
+ export type { ProgressInfo, ExecuteOptions };
1127
+
1128
+ // Export for direct usage
1129
+ export { ensureReady, executeCodexLens, checkVenvStatus, bootstrapVenv, checkSemanticStatus, installSemantic, uninstallCodexLens, cancelIndexing, isIndexingInProgress };
1130
+
1131
+ // Backward-compatible export for tests
1132
+ export const codexLensTool = {
1133
+ name: schema.name,
1134
+ description: schema.description,
1135
+ parameters: schema.inputSchema,
1136
+ execute: async (params: Record<string, unknown>) => {
1137
+ const result = await handler(params);
1138
+ // Return the result directly - tests expect {success: boolean, ...} format
1139
+ return result.success ? result.result : { success: false, error: result.error };
1140
+ }
1141
+ };