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,1026 @@
1
+ /**
2
+ * Session Manager Tool - Workflow session lifecycle management
3
+ * Operations: init, list, read, write, update, archive, mkdir, delete, stats
4
+ * Content routing via content_type + path_params
5
+ */
6
+
7
+ import { z } from 'zod';
8
+ import type { ToolSchema, ToolResult } from '../types/tool.js';
9
+ import {
10
+ readFileSync,
11
+ writeFileSync,
12
+ existsSync,
13
+ readdirSync,
14
+ mkdirSync,
15
+ renameSync,
16
+ rmSync,
17
+ statSync,
18
+ } from 'fs';
19
+ import { resolve, join, dirname } from 'path';
20
+
21
+ // Base paths for session storage
22
+ const WORKFLOW_BASE = '.workflow';
23
+ const ACTIVE_BASE = '.workflow/active';
24
+ const ARCHIVE_BASE = '.workflow/archives';
25
+ const LITE_PLAN_BASE = '.workflow/.lite-plan';
26
+ const LITE_FIX_BASE = '.workflow/.lite-fix';
27
+
28
+ // Session ID validation pattern (alphanumeric, hyphen, underscore)
29
+ const SESSION_ID_PATTERN = /^[a-zA-Z0-9_-]+$/;
30
+
31
+ // Zod schemas - using tuple syntax for z.enum
32
+ const ContentTypeEnum = z.enum([
33
+ 'session', 'plan', 'task', 'summary', 'process', 'chat', 'brainstorm',
34
+ 'review-dim', 'review-iter', 'review-fix', 'todo', 'context',
35
+ // Lite-specific content types
36
+ 'lite-plan', 'lite-fix-plan', 'exploration', 'explorations-manifest',
37
+ 'diagnosis', 'diagnoses-manifest', 'clarifications', 'execution-context', 'session-metadata'
38
+ ]);
39
+
40
+ const OperationEnum = z.enum(['init', 'list', 'read', 'write', 'update', 'archive', 'mkdir', 'delete', 'stats']);
41
+
42
+ const LocationEnum = z.enum([
43
+ 'active', 'archived', 'both',
44
+ 'lite-plan', 'lite-fix', 'all'
45
+ ]);
46
+
47
+ const ParamsSchema = z.object({
48
+ operation: OperationEnum,
49
+ session_id: z.string().optional(),
50
+ content_type: ContentTypeEnum.optional(),
51
+ content: z.union([z.string(), z.record(z.string(), z.any())]).optional(),
52
+ path_params: z.record(z.string(), z.string()).optional(),
53
+ metadata: z.record(z.string(), z.any()).optional(),
54
+ location: LocationEnum.optional(),
55
+ include_metadata: z.boolean().optional(),
56
+ dirs: z.array(z.string()).optional(),
57
+ update_status: z.boolean().optional(),
58
+ file_path: z.string().optional(),
59
+ });
60
+
61
+ type Params = z.infer<typeof ParamsSchema>;
62
+ type ContentType = z.infer<typeof ContentTypeEnum>;
63
+ type Operation = z.infer<typeof OperationEnum>;
64
+ type Location = z.infer<typeof LocationEnum>;
65
+
66
+ interface SessionInfo {
67
+ session_id: string;
68
+ location: string;
69
+ metadata?: any;
70
+ }
71
+
72
+ interface SessionLocation {
73
+ path: string;
74
+ location: string;
75
+ }
76
+
77
+ interface TaskStats {
78
+ total: number;
79
+ pending: number;
80
+ in_progress: number;
81
+ completed: number;
82
+ blocked: number;
83
+ cancelled: number;
84
+ }
85
+
86
+ // Cached workflow root (computed once per execution)
87
+ let cachedWorkflowRoot: string | null = null;
88
+
89
+ /**
90
+ * Find project root by traversing up looking for .workflow directory
91
+ * Falls back to cwd if not found
92
+ */
93
+ function findWorkflowRoot(): string {
94
+ if (cachedWorkflowRoot) return cachedWorkflowRoot;
95
+
96
+ let dir = process.cwd();
97
+ const root = dirname(dir) === dir ? dir : null; // filesystem root
98
+
99
+ while (dir && dir !== root) {
100
+ if (existsSync(join(dir, WORKFLOW_BASE))) {
101
+ cachedWorkflowRoot = dir;
102
+ return dir;
103
+ }
104
+ const parent = dirname(dir);
105
+ if (parent === dir) break; // reached filesystem root
106
+ dir = parent;
107
+ }
108
+
109
+ // Fallback to cwd (for init operation)
110
+ cachedWorkflowRoot = process.cwd();
111
+ return cachedWorkflowRoot;
112
+ }
113
+
114
+ /**
115
+ * Validate session ID format
116
+ */
117
+ function validateSessionId(sessionId: string): void {
118
+ if (!sessionId || typeof sessionId !== 'string') {
119
+ throw new Error('session_id must be a non-empty string');
120
+ }
121
+ if (!SESSION_ID_PATTERN.test(sessionId)) {
122
+ throw new Error(
123
+ `Invalid session_id format: "${sessionId}". Only alphanumeric, hyphen, and underscore allowed.`
124
+ );
125
+ }
126
+ if (sessionId.length > 100) {
127
+ throw new Error('session_id must be 100 characters or less');
128
+ }
129
+ }
130
+
131
+ /**
132
+ * Validate path params to prevent path traversal
133
+ */
134
+ function validatePathParams(pathParams: Record<string, unknown>): void {
135
+ for (const [key, value] of Object.entries(pathParams)) {
136
+ if (typeof value !== 'string') continue;
137
+ if (value.includes('..') || value.includes('/') || value.includes('\\')) {
138
+ throw new Error(`Invalid path_params.${key}: path traversal characters not allowed`);
139
+ }
140
+ }
141
+ }
142
+
143
+ /**
144
+ * Content type to file path routing
145
+ * {base} is replaced with session base path
146
+ * Dynamic params: {task_id}, {filename}, {dimension}, {iteration}
147
+ */
148
+ const PATH_ROUTES: Record<ContentType, string> = {
149
+ // Standard WFS content types
150
+ session: '{base}/workflow-session.json',
151
+ plan: '{base}/IMPL_PLAN.md',
152
+ task: '{base}/.task/{task_id}.json',
153
+ summary: '{base}/.summaries/{task_id}-summary.md',
154
+ process: '{base}/.process/{filename}',
155
+ chat: '{base}/.chat/{filename}',
156
+ brainstorm: '{base}/.brainstorming/{filename}',
157
+ 'review-dim': '{base}/.review/dimensions/{dimension}.json',
158
+ 'review-iter': '{base}/.review/iterations/{iteration}.json',
159
+ 'review-fix': '{base}/.review/fixes/{filename}',
160
+ todo: '{base}/TODO_LIST.md',
161
+ context: '{base}/context-package.json',
162
+ // Lite-specific content types
163
+ 'lite-plan': '{base}/plan.json',
164
+ 'lite-fix-plan': '{base}/fix-plan.json',
165
+ 'exploration': '{base}/exploration-{angle}.json',
166
+ 'explorations-manifest': '{base}/explorations-manifest.json',
167
+ 'diagnosis': '{base}/diagnosis-{angle}.json',
168
+ 'diagnoses-manifest': '{base}/diagnoses-manifest.json',
169
+ 'clarifications': '{base}/clarifications.json',
170
+ 'execution-context': '{base}/execution-context.json',
171
+ 'session-metadata': '{base}/session-metadata.json',
172
+ };
173
+
174
+ /**
175
+ * Resolve path with base and parameters
176
+ */
177
+ function resolvePath(
178
+ base: string,
179
+ contentType: ContentType,
180
+ pathParams: Record<string, string> = {}
181
+ ): string {
182
+ const template = PATH_ROUTES[contentType];
183
+ if (!template) {
184
+ throw new Error(
185
+ `Unknown content_type: ${contentType}. Valid types: ${Object.keys(PATH_ROUTES).join(', ')}`
186
+ );
187
+ }
188
+
189
+ let path = template.replace('{base}', base);
190
+
191
+ // Replace dynamic parameters
192
+ for (const [key, value] of Object.entries(pathParams)) {
193
+ path = path.replace(`{${key}}`, value);
194
+ }
195
+
196
+ // Check for unreplaced placeholders
197
+ const unreplaced = path.match(/\{[^}]+\}/g);
198
+ if (unreplaced) {
199
+ throw new Error(
200
+ `Missing path_params: ${unreplaced.join(', ')} for content_type "${contentType}"`
201
+ );
202
+ }
203
+
204
+ return resolve(findWorkflowRoot(), path);
205
+ }
206
+
207
+ /**
208
+ * Get session base path
209
+ */
210
+ function getSessionBase(
211
+ sessionId: string,
212
+ location: 'active' | 'archived' | 'lite-plan' | 'lite-fix' = 'active'
213
+ ): string {
214
+ const locationMap: Record<string, string> = {
215
+ 'active': ACTIVE_BASE,
216
+ 'archived': ARCHIVE_BASE,
217
+ 'lite-plan': LITE_PLAN_BASE,
218
+ 'lite-fix': LITE_FIX_BASE,
219
+ };
220
+ const basePath = locationMap[location] || ACTIVE_BASE;
221
+ return resolve(findWorkflowRoot(), basePath, sessionId);
222
+ }
223
+
224
+ /**
225
+ * Auto-detect session location by searching all known paths
226
+ * Search order: active, archives, lite-plan, lite-fix
227
+ */
228
+ function findSession(sessionId: string): SessionLocation | null {
229
+ const root = findWorkflowRoot();
230
+ const searchPaths = [
231
+ { path: resolve(root, ACTIVE_BASE, sessionId), location: 'active' },
232
+ { path: resolve(root, ARCHIVE_BASE, sessionId), location: 'archived' },
233
+ { path: resolve(root, LITE_PLAN_BASE, sessionId), location: 'lite-plan' },
234
+ { path: resolve(root, LITE_FIX_BASE, sessionId), location: 'lite-fix' },
235
+ ];
236
+
237
+ for (const { path, location } of searchPaths) {
238
+ if (existsSync(path)) {
239
+ return { path, location };
240
+ }
241
+ }
242
+ return null;
243
+ }
244
+
245
+ /**
246
+ * Ensure directory exists
247
+ */
248
+ function ensureDir(dirPath: string): void {
249
+ if (!existsSync(dirPath)) {
250
+ mkdirSync(dirPath, { recursive: true });
251
+ }
252
+ }
253
+
254
+ /**
255
+ * Read JSON file safely
256
+ */
257
+ function readJsonFile(filePath: string): any {
258
+ if (!existsSync(filePath)) {
259
+ throw new Error(`File not found: ${filePath}`);
260
+ }
261
+ try {
262
+ const content = readFileSync(filePath, 'utf8');
263
+ return JSON.parse(content);
264
+ } catch (error) {
265
+ if (error instanceof SyntaxError) {
266
+ throw new Error(`Invalid JSON in ${filePath}: ${error.message}`);
267
+ }
268
+ throw new Error(`Failed to read ${filePath}: ${(error as Error).message}`);
269
+ }
270
+ }
271
+
272
+ /**
273
+ * Write JSON file with formatting
274
+ */
275
+ function writeJsonFile(filePath: string, data: any): void {
276
+ ensureDir(dirname(filePath));
277
+ const content = JSON.stringify(data, null, 2);
278
+ writeFileSync(filePath, content, 'utf8');
279
+ }
280
+
281
+ /**
282
+ * Write text file
283
+ */
284
+ function writeTextFile(filePath: string, content: string): void {
285
+ ensureDir(dirname(filePath));
286
+ writeFileSync(filePath, content, 'utf8');
287
+ }
288
+
289
+ // ============================================================
290
+ // Helper Functions
291
+ // ============================================================
292
+
293
+ /**
294
+ * List sessions in a specific directory
295
+ * @param dirPath - Directory to scan
296
+ * @param location - Location identifier for returned sessions
297
+ * @param prefix - Optional prefix filter (e.g., 'WFS-'), null means no filter
298
+ * @param includeMetadata - Whether to load metadata for each session
299
+ */
300
+ function listSessionsInDir(
301
+ dirPath: string,
302
+ location: string,
303
+ prefix: string | null,
304
+ includeMetadata: boolean
305
+ ): SessionInfo[] {
306
+ if (!existsSync(dirPath)) return [];
307
+
308
+ try {
309
+ const entries = readdirSync(dirPath, { withFileTypes: true });
310
+ return entries
311
+ .filter(e => e.isDirectory() && (prefix === null || e.name.startsWith(prefix)))
312
+ .map(e => {
313
+ const sessionInfo: SessionInfo = { session_id: e.name, location };
314
+ if (includeMetadata) {
315
+ // Try multiple metadata file locations
316
+ const metaPaths = [
317
+ join(dirPath, e.name, 'workflow-session.json'),
318
+ join(dirPath, e.name, 'session-metadata.json'),
319
+ join(dirPath, e.name, 'explorations-manifest.json'),
320
+ join(dirPath, e.name, 'diagnoses-manifest.json'),
321
+ ];
322
+ for (const metaPath of metaPaths) {
323
+ if (existsSync(metaPath)) {
324
+ try {
325
+ sessionInfo.metadata = readJsonFile(metaPath);
326
+ break;
327
+ } catch { /* continue */ }
328
+ }
329
+ }
330
+ }
331
+ return sessionInfo;
332
+ });
333
+ } catch {
334
+ return [];
335
+ }
336
+ }
337
+
338
+ // ============================================================
339
+ // Operation Handlers
340
+ // ============================================================
341
+
342
+ /**
343
+ * Operation: init
344
+ * Create new session with directory structure
345
+ * Supports both WFS sessions and lite sessions (lite-plan, lite-fix)
346
+ */
347
+ function executeInit(params: Params): any {
348
+ const { session_id, metadata, location } = params;
349
+
350
+ if (!session_id) {
351
+ throw new Error('Parameter "session_id" is required for init');
352
+ }
353
+
354
+ // Validate session_id format
355
+ validateSessionId(session_id);
356
+
357
+ // Auto-infer location from metadata.type if location not explicitly provided
358
+ // Priority: explicit location > metadata.type > default 'active'
359
+ const sessionLocation: 'active' | 'archived' | 'lite-plan' | 'lite-fix' =
360
+ (location === 'active' || location === 'archived' || location === 'lite-plan' || location === 'lite-fix')
361
+ ? location
362
+ : (metadata?.type === 'lite-plan' ? 'lite-plan' :
363
+ metadata?.type === 'lite-fix' ? 'lite-fix' :
364
+ 'active');
365
+
366
+ // Check if session already exists (auto-detect all locations)
367
+ const existing = findSession(session_id);
368
+ if (existing) {
369
+ throw new Error(`Session "${session_id}" already exists in ${existing.location}`);
370
+ }
371
+
372
+ const sessionPath = getSessionBase(session_id, sessionLocation);
373
+
374
+ // Create session directory structure based on type
375
+ ensureDir(sessionPath);
376
+
377
+ let directoriesCreated: string[] = [];
378
+ if (sessionLocation === 'lite-plan' || sessionLocation === 'lite-fix') {
379
+ // Lite sessions: minimal structure, files created by workflow
380
+ // No subdirectories needed initially
381
+ directoriesCreated = [];
382
+ } else {
383
+ // WFS sessions: standard structure
384
+ ensureDir(join(sessionPath, '.task'));
385
+ ensureDir(join(sessionPath, '.summaries'));
386
+ ensureDir(join(sessionPath, '.process'));
387
+ directoriesCreated = ['.task', '.summaries', '.process'];
388
+ }
389
+
390
+ // Create session metadata file if provided
391
+ let sessionMetadata = null;
392
+ if (metadata) {
393
+ const sessionFile = sessionLocation.startsWith('lite-')
394
+ ? join(sessionPath, 'session-metadata.json') // Lite sessions
395
+ : join(sessionPath, 'workflow-session.json'); // WFS sessions
396
+
397
+ const sessionData = {
398
+ session_id,
399
+ type: metadata?.type || sessionLocation, // Preserve user-specified type if provided
400
+ status: 'initialized',
401
+ created_at: new Date().toISOString(),
402
+ ...metadata,
403
+ };
404
+ writeJsonFile(sessionFile, sessionData);
405
+ sessionMetadata = sessionData;
406
+ }
407
+
408
+ return {
409
+ operation: 'init',
410
+ session_id,
411
+ location: sessionLocation,
412
+ path: sessionPath,
413
+ directories_created: directoriesCreated,
414
+ metadata: sessionMetadata,
415
+ message: `Session "${session_id}" initialized in ${sessionLocation}`,
416
+ };
417
+ }
418
+
419
+ /**
420
+ * Operation: list
421
+ * List sessions (active, archived, lite-plan, lite-fix, or all)
422
+ */
423
+ function executeList(params: Params): any {
424
+ const { location = 'both', include_metadata = false } = params;
425
+
426
+ const result: {
427
+ operation: string;
428
+ active: SessionInfo[];
429
+ archived: SessionInfo[];
430
+ litePlan: SessionInfo[];
431
+ liteFix: SessionInfo[];
432
+ total: number;
433
+ } = {
434
+ operation: 'list',
435
+ active: [],
436
+ archived: [],
437
+ litePlan: [],
438
+ liteFix: [],
439
+ total: 0,
440
+ };
441
+
442
+ const root = findWorkflowRoot();
443
+
444
+ // Helper to check if location should be included
445
+ const shouldInclude = (loc: string) =>
446
+ location === 'all' || location === 'both' || location === loc;
447
+
448
+ // List active sessions (WFS-* prefix)
449
+ if (shouldInclude('active')) {
450
+ result.active = listSessionsInDir(
451
+ resolve(root, ACTIVE_BASE),
452
+ 'active',
453
+ 'WFS-',
454
+ include_metadata
455
+ );
456
+ }
457
+
458
+ // List archived sessions (WFS-* prefix)
459
+ if (shouldInclude('archived')) {
460
+ result.archived = listSessionsInDir(
461
+ resolve(root, ARCHIVE_BASE),
462
+ 'archived',
463
+ 'WFS-',
464
+ include_metadata
465
+ );
466
+ }
467
+
468
+ // List lite-plan sessions (no prefix filter)
469
+ if (location === 'all' || location === 'lite-plan') {
470
+ result.litePlan = listSessionsInDir(
471
+ resolve(root, LITE_PLAN_BASE),
472
+ 'lite-plan',
473
+ null,
474
+ include_metadata
475
+ );
476
+ }
477
+
478
+ // List lite-fix sessions (no prefix filter)
479
+ if (location === 'all' || location === 'lite-fix') {
480
+ result.liteFix = listSessionsInDir(
481
+ resolve(root, LITE_FIX_BASE),
482
+ 'lite-fix',
483
+ null,
484
+ include_metadata
485
+ );
486
+ }
487
+
488
+ result.total = result.active.length + result.archived.length +
489
+ result.litePlan.length + result.liteFix.length;
490
+
491
+ return result;
492
+ }
493
+
494
+ /**
495
+ * Operation: read
496
+ * Read file content by content_type
497
+ */
498
+ function executeRead(params: Params): any {
499
+ const { session_id, content_type, path_params = {} } = params;
500
+
501
+ if (!session_id) {
502
+ throw new Error('Parameter "session_id" is required for read');
503
+ }
504
+ if (!content_type) {
505
+ throw new Error('Parameter "content_type" is required for read');
506
+ }
507
+
508
+ // Validate inputs
509
+ validateSessionId(session_id);
510
+ validatePathParams(path_params);
511
+
512
+ const session = findSession(session_id);
513
+ if (!session) {
514
+ throw new Error(`Session "${session_id}" not found`);
515
+ }
516
+
517
+ const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
518
+
519
+ if (!existsSync(filePath)) {
520
+ throw new Error(`File not found: ${filePath}`);
521
+ }
522
+
523
+ // Read content
524
+ const rawContent = readFileSync(filePath, 'utf8');
525
+
526
+ // Parse JSON for JSON content types
527
+ const isJson = filePath.endsWith('.json');
528
+ const content = isJson ? JSON.parse(rawContent) : rawContent;
529
+
530
+ return {
531
+ operation: 'read',
532
+ session_id,
533
+ content_type,
534
+ path: filePath,
535
+ location: session.location,
536
+ content,
537
+ is_json: isJson,
538
+ };
539
+ }
540
+
541
+ /**
542
+ * Operation: write
543
+ * Write content to file by content_type
544
+ */
545
+ function executeWrite(params: Params): any {
546
+ const { session_id, content_type, content, path_params = {} } = params;
547
+
548
+ if (!session_id) {
549
+ throw new Error('Parameter "session_id" is required for write');
550
+ }
551
+ if (!content_type) {
552
+ throw new Error('Parameter "content_type" is required for write');
553
+ }
554
+ if (content === undefined) {
555
+ throw new Error('Parameter "content" is required for write');
556
+ }
557
+
558
+ // Validate inputs
559
+ validateSessionId(session_id);
560
+ validatePathParams(path_params);
561
+
562
+ const session = findSession(session_id);
563
+ if (!session) {
564
+ throw new Error(`Session "${session_id}" not found. Use init operation first.`);
565
+ }
566
+
567
+ const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
568
+ const isJson = filePath.endsWith('.json');
569
+
570
+ // Write content
571
+ if (isJson) {
572
+ writeJsonFile(filePath, content);
573
+ } else {
574
+ writeTextFile(filePath, typeof content === 'string' ? content : JSON.stringify(content, null, 2));
575
+ }
576
+
577
+ // Return written content for task/summary types
578
+ const returnContent =
579
+ content_type === 'task' || content_type === 'summary' ? content : undefined;
580
+
581
+ return {
582
+ operation: 'write',
583
+ session_id,
584
+ content_type,
585
+ written_content: returnContent,
586
+ path: filePath,
587
+ location: session.location,
588
+ message: `File written successfully`,
589
+ };
590
+ }
591
+
592
+ /**
593
+ * Operation: update
594
+ * Update existing JSON file with shallow merge
595
+ */
596
+ function executeUpdate(params: Params): any {
597
+ const { session_id, content_type, content, path_params = {} } = params;
598
+
599
+ if (!session_id) {
600
+ throw new Error('Parameter "session_id" is required for update');
601
+ }
602
+ if (!content_type) {
603
+ throw new Error('Parameter "content_type" is required for update');
604
+ }
605
+ if (!content || typeof content !== 'object') {
606
+ throw new Error('Parameter "content" must be an object for update');
607
+ }
608
+
609
+ const session = findSession(session_id);
610
+ if (!session) {
611
+ throw new Error(`Session "${session_id}" not found`);
612
+ }
613
+
614
+ const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
615
+
616
+ if (!filePath.endsWith('.json')) {
617
+ throw new Error('Update operation only supports JSON files');
618
+ }
619
+
620
+ // Read existing content or start with empty object
621
+ let existing: any = {};
622
+ if (existsSync(filePath)) {
623
+ existing = readJsonFile(filePath);
624
+ }
625
+
626
+ // Shallow merge
627
+ const merged = { ...existing, ...(content as object) };
628
+ writeJsonFile(filePath, merged);
629
+
630
+ return {
631
+ operation: 'update',
632
+ session_id,
633
+ content_type,
634
+ path: filePath,
635
+ location: session.location,
636
+ fields_updated: Object.keys(content as object),
637
+ merged_data: merged,
638
+ message: `File updated successfully`,
639
+ };
640
+ }
641
+
642
+ /**
643
+ * Operation: archive
644
+ * Move session from active to archives
645
+ */
646
+ function executeArchive(params: Params): any {
647
+ const { session_id, update_status = true } = params;
648
+
649
+ if (!session_id) {
650
+ throw new Error('Parameter "session_id" is required for archive');
651
+ }
652
+
653
+ // Find session in any location
654
+ const session = findSession(session_id);
655
+ if (!session) {
656
+ throw new Error(`Session "${session_id}" not found`);
657
+ }
658
+
659
+ // Lite sessions do not support archiving
660
+ if (session.location === 'lite-plan' || session.location === 'lite-fix') {
661
+ throw new Error(`Lite sessions (${session.location}) do not support archiving. Use delete operation instead.`);
662
+ }
663
+
664
+ // Determine archive destination based on source location
665
+ let archivePath: string;
666
+
667
+ if (session.location === 'active') {
668
+ archivePath = getSessionBase(session_id, 'archived');
669
+ } else {
670
+ // Already archived
671
+ return {
672
+ operation: 'archive',
673
+ session_id,
674
+ status: 'already_archived',
675
+ path: session.path,
676
+ location: session.location,
677
+ message: `Session "${session_id}" is already archived`,
678
+ };
679
+ }
680
+
681
+ // Update status before archiving
682
+ if (update_status) {
683
+ const metadataFiles = [
684
+ join(session.path, 'workflow-session.json'),
685
+ join(session.path, 'session-metadata.json'),
686
+ join(session.path, 'explorations-manifest.json'),
687
+ ];
688
+ for (const metaFile of metadataFiles) {
689
+ if (existsSync(metaFile)) {
690
+ try {
691
+ const data = readJsonFile(metaFile);
692
+ data.status = 'completed';
693
+ data.archived_at = new Date().toISOString();
694
+ writeJsonFile(metaFile, data);
695
+ break;
696
+ } catch { /* continue */ }
697
+ }
698
+ }
699
+ }
700
+
701
+ // Ensure archive directory exists
702
+ ensureDir(dirname(archivePath));
703
+
704
+ // Move session directory
705
+ renameSync(session.path, archivePath);
706
+
707
+ // Read session metadata after archiving
708
+ let sessionMetadata = null;
709
+ const metadataFiles = [
710
+ join(archivePath, 'workflow-session.json'),
711
+ join(archivePath, 'session-metadata.json'),
712
+ join(archivePath, 'explorations-manifest.json'),
713
+ ];
714
+ for (const metaFile of metadataFiles) {
715
+ if (existsSync(metaFile)) {
716
+ try {
717
+ sessionMetadata = readJsonFile(metaFile);
718
+ break;
719
+ } catch { /* continue */ }
720
+ }
721
+ }
722
+
723
+ return {
724
+ operation: 'archive',
725
+ session_id,
726
+ status: 'archived',
727
+ source: session.path,
728
+ source_location: session.location,
729
+ destination: archivePath,
730
+ metadata: sessionMetadata,
731
+ message: `Session "${session_id}" archived from ${session.location}`,
732
+ };
733
+ }
734
+
735
+ /**
736
+ * Operation: mkdir
737
+ * Create directory structure within session
738
+ */
739
+ function executeMkdir(params: Params): any {
740
+ const { session_id, dirs } = params;
741
+
742
+ if (!session_id) {
743
+ throw new Error('Parameter "session_id" is required for mkdir');
744
+ }
745
+ if (!dirs || !Array.isArray(dirs) || dirs.length === 0) {
746
+ throw new Error('Parameter "dirs" must be a non-empty array');
747
+ }
748
+
749
+ const session = findSession(session_id);
750
+ if (!session) {
751
+ throw new Error(`Session "${session_id}" not found`);
752
+ }
753
+
754
+ const created: string[] = [];
755
+ for (const dir of dirs) {
756
+ const dirPath = join(session.path, dir);
757
+ ensureDir(dirPath);
758
+ created.push(dir);
759
+ }
760
+
761
+ return {
762
+ operation: 'mkdir',
763
+ session_id,
764
+ location: session.location,
765
+ directories_created: created,
766
+ message: `Created ${created.length} directories`,
767
+ };
768
+ }
769
+
770
+ /**
771
+ * Operation: delete
772
+ * Delete a file within session (security: path traversal prevention)
773
+ */
774
+ function executeDelete(params: Params): any {
775
+ const { session_id, file_path } = params;
776
+
777
+ if (!session_id) {
778
+ throw new Error('Parameter "session_id" is required for delete');
779
+ }
780
+ if (!file_path) {
781
+ throw new Error('Parameter "file_path" is required for delete');
782
+ }
783
+
784
+ // Validate session exists
785
+ const session = findSession(session_id);
786
+ if (!session) {
787
+ throw new Error(`Session "${session_id}" not found`);
788
+ }
789
+
790
+ // Security: Prevent path traversal
791
+ if (file_path.includes('..') || file_path.includes('\\')) {
792
+ throw new Error('Invalid file_path: path traversal characters not allowed');
793
+ }
794
+
795
+ // Construct absolute path
796
+ const absolutePath = resolve(session.path, file_path);
797
+
798
+ // Security: Verify path is within session directory
799
+ if (!absolutePath.startsWith(session.path)) {
800
+ throw new Error('Security error: file_path must be within session directory');
801
+ }
802
+
803
+ // Check file exists
804
+ if (!existsSync(absolutePath)) {
805
+ throw new Error(`File not found: ${file_path}`);
806
+ }
807
+
808
+ // Delete the file
809
+ rmSync(absolutePath, { force: true });
810
+
811
+ return {
812
+ operation: 'delete',
813
+ session_id,
814
+ deleted: file_path,
815
+ absolute_path: absolutePath,
816
+ message: `File deleted successfully`,
817
+ };
818
+ }
819
+
820
+ /**
821
+ * Operation: stats
822
+ * Get session statistics (tasks, summaries, plan)
823
+ */
824
+ function executeStats(params: Params): any {
825
+ const { session_id } = params;
826
+
827
+ if (!session_id) {
828
+ throw new Error('Parameter "session_id" is required for stats');
829
+ }
830
+
831
+ // Validate session exists
832
+ const session = findSession(session_id);
833
+ if (!session) {
834
+ throw new Error(`Session "${session_id}" not found`);
835
+ }
836
+
837
+ const taskDir = join(session.path, '.task');
838
+ const summariesDir = join(session.path, '.summaries');
839
+ const planFile = join(session.path, 'IMPL_PLAN.md');
840
+
841
+ // Count tasks by status
842
+ const taskStats: TaskStats = {
843
+ total: 0,
844
+ pending: 0,
845
+ in_progress: 0,
846
+ completed: 0,
847
+ blocked: 0,
848
+ cancelled: 0,
849
+ };
850
+
851
+ if (existsSync(taskDir)) {
852
+ const taskFiles = readdirSync(taskDir).filter((f) => f.endsWith('.json'));
853
+ taskStats.total = taskFiles.length;
854
+
855
+ for (const taskFile of taskFiles) {
856
+ try {
857
+ const taskPath = join(taskDir, taskFile);
858
+ const taskData = readJsonFile(taskPath);
859
+ const status = taskData.status || 'unknown';
860
+ if (status in taskStats) {
861
+ (taskStats as any)[status]++;
862
+ }
863
+ } catch {
864
+ // Skip invalid task files
865
+ }
866
+ }
867
+ }
868
+
869
+ // Count summaries
870
+ let summariesCount = 0;
871
+ if (existsSync(summariesDir)) {
872
+ summariesCount = readdirSync(summariesDir).filter((f) => f.endsWith('.md')).length;
873
+ }
874
+
875
+ // Check for plan
876
+ const hasPlan = existsSync(planFile);
877
+
878
+ return {
879
+ operation: 'stats',
880
+ session_id,
881
+ location: session.location,
882
+ tasks: taskStats,
883
+ summaries: summariesCount,
884
+ has_plan: hasPlan,
885
+ message: `Session statistics retrieved`,
886
+ };
887
+ }
888
+
889
+ // ============================================================
890
+ // Main Execute Function
891
+ // ============================================================
892
+
893
+ /**
894
+ * Route to appropriate operation handler
895
+ */
896
+ async function execute(params: Params): Promise<any> {
897
+ const { operation } = params;
898
+
899
+ if (!operation) {
900
+ throw new Error(
901
+ 'Parameter "operation" is required. Valid operations: init, list, read, write, update, archive, mkdir, delete, stats'
902
+ );
903
+ }
904
+
905
+ switch (operation) {
906
+ case 'init':
907
+ return executeInit(params);
908
+ case 'list':
909
+ return executeList(params);
910
+ case 'read':
911
+ return executeRead(params);
912
+ case 'write':
913
+ return executeWrite(params);
914
+ case 'update':
915
+ return executeUpdate(params);
916
+ case 'archive':
917
+ return executeArchive(params);
918
+ case 'mkdir':
919
+ return executeMkdir(params);
920
+ case 'delete':
921
+ return executeDelete(params);
922
+ case 'stats':
923
+ return executeStats(params);
924
+ default:
925
+ throw new Error(
926
+ `Unknown operation: ${operation}. Valid operations: init, list, read, write, update, archive, mkdir, delete, stats`
927
+ );
928
+ }
929
+ }
930
+
931
+ // ============================================================
932
+ // Tool Definition
933
+ // ============================================================
934
+
935
+ export const schema: ToolSchema = {
936
+ name: 'session_manager',
937
+ description: `Workflow session management.
938
+
939
+ Usage:
940
+ session_manager(operation="init", type="workflow", description="...")
941
+ session_manager(operation="list", location="active|archived|both")
942
+ session_manager(operation="read", sessionId="WFS-xxx", contentType="plan|task|summary")
943
+ session_manager(operation="write", sessionId="WFS-xxx", contentType="plan", content={...})
944
+ session_manager(operation="archive", sessionId="WFS-xxx")
945
+ session_manager(operation="stats", sessionId="WFS-xxx")`,
946
+ inputSchema: {
947
+ type: 'object',
948
+ properties: {
949
+ operation: {
950
+ type: 'string',
951
+ enum: ['init', 'list', 'read', 'write', 'update', 'archive', 'mkdir', 'delete', 'stats'],
952
+ description: 'Operation to perform',
953
+ },
954
+ session_id: {
955
+ type: 'string',
956
+ description: 'Session identifier (e.g., WFS-my-session). Required for all operations except list.',
957
+ },
958
+ content_type: {
959
+ type: 'string',
960
+ enum: [
961
+ 'session',
962
+ 'plan',
963
+ 'task',
964
+ 'summary',
965
+ 'process',
966
+ 'chat',
967
+ 'brainstorm',
968
+ 'review-dim',
969
+ 'review-iter',
970
+ 'review-fix',
971
+ 'todo',
972
+ 'context',
973
+ ],
974
+ description: 'Content type for read/write/update operations',
975
+ },
976
+ content: {
977
+ type: 'object',
978
+ description: 'Content for write/update operations (object for JSON, string for text)',
979
+ },
980
+ path_params: {
981
+ type: 'object',
982
+ description: 'Dynamic path parameters: task_id, filename, dimension, iteration',
983
+ },
984
+ metadata: {
985
+ type: 'object',
986
+ description: 'Session metadata for init operation (project, type, description, etc.)',
987
+ },
988
+ location: {
989
+ type: 'string',
990
+ enum: ['active', 'archived', 'both'],
991
+ description: 'Session location filter for list operation (default: both)',
992
+ },
993
+ include_metadata: {
994
+ type: 'boolean',
995
+ description: 'Include session metadata in list results (default: false)',
996
+ },
997
+ dirs: {
998
+ type: 'array',
999
+ description: 'Directory paths to create for mkdir operation',
1000
+ },
1001
+ update_status: {
1002
+ type: 'boolean',
1003
+ description: 'Update session status to completed when archiving (default: true)',
1004
+ },
1005
+ file_path: {
1006
+ type: 'string',
1007
+ description: 'Relative file path within session for delete operation',
1008
+ },
1009
+ },
1010
+ required: ['operation'],
1011
+ },
1012
+ };
1013
+
1014
+ export async function handler(params: Record<string, unknown>): Promise<ToolResult> {
1015
+ const parsed = ParamsSchema.safeParse(params);
1016
+ if (!parsed.success) {
1017
+ return { success: false, error: `Invalid params: ${parsed.error.message}` };
1018
+ }
1019
+
1020
+ try {
1021
+ const result = await execute(parsed.data);
1022
+ return { success: true, result };
1023
+ } catch (error) {
1024
+ return { success: false, error: (error as Error).message };
1025
+ }
1026
+ }