claude-code-workflow 6.2.2 → 6.2.4

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 (346) hide show
  1. package/ccw/dist/cli.d.ts +2 -0
  2. package/ccw/dist/cli.d.ts.map +1 -0
  3. package/ccw/dist/cli.js +219 -0
  4. package/ccw/dist/cli.js.map +1 -0
  5. package/ccw/dist/commands/cli.d.ts +32 -0
  6. package/ccw/dist/commands/cli.d.ts.map +1 -0
  7. package/ccw/dist/commands/cli.js +619 -0
  8. package/ccw/dist/commands/cli.js.map +1 -0
  9. package/ccw/dist/commands/core-memory.d.ts +32 -0
  10. package/ccw/dist/commands/core-memory.d.ts.map +1 -0
  11. package/ccw/dist/commands/core-memory.js +640 -0
  12. package/ccw/dist/commands/core-memory.js.map +1 -0
  13. package/ccw/dist/commands/hook.d.ts +16 -0
  14. package/ccw/dist/commands/hook.d.ts.map +1 -0
  15. package/ccw/dist/commands/hook.js +276 -0
  16. package/ccw/dist/commands/hook.js.map +1 -0
  17. package/ccw/dist/commands/install.d.ts +12 -0
  18. package/ccw/dist/commands/install.d.ts.map +1 -0
  19. package/ccw/dist/commands/install.js +443 -0
  20. package/ccw/dist/commands/install.js.map +1 -0
  21. package/ccw/dist/commands/list.d.ts +5 -0
  22. package/ccw/dist/commands/list.d.ts.map +1 -0
  23. package/ccw/dist/commands/list.js +32 -0
  24. package/ccw/dist/commands/list.js.map +1 -0
  25. package/ccw/dist/commands/memory.d.ts +57 -0
  26. package/ccw/dist/commands/memory.d.ts.map +1 -0
  27. package/ccw/dist/commands/memory.js +890 -0
  28. package/ccw/dist/commands/memory.js.map +1 -0
  29. package/ccw/dist/commands/serve.d.ts +12 -0
  30. package/ccw/dist/commands/serve.d.ts.map +1 -0
  31. package/ccw/dist/commands/serve.js +63 -0
  32. package/ccw/dist/commands/serve.js.map +1 -0
  33. package/ccw/dist/commands/session-path-resolver.d.ts +45 -0
  34. package/ccw/dist/commands/session-path-resolver.d.ts.map +1 -0
  35. package/ccw/dist/commands/session-path-resolver.js +302 -0
  36. package/ccw/dist/commands/session-path-resolver.js.map +1 -0
  37. package/ccw/dist/commands/session.d.ts +12 -0
  38. package/ccw/dist/commands/session.d.ts.map +1 -0
  39. package/ccw/dist/commands/session.js +954 -0
  40. package/ccw/dist/commands/session.js.map +1 -0
  41. package/ccw/dist/commands/stop.d.ts +11 -0
  42. package/ccw/dist/commands/stop.d.ts.map +1 -0
  43. package/ccw/dist/commands/stop.js +96 -0
  44. package/ccw/dist/commands/stop.js.map +1 -0
  45. package/ccw/dist/commands/tool.d.ts +29 -0
  46. package/ccw/dist/commands/tool.d.ts.map +1 -0
  47. package/ccw/dist/commands/tool.js +173 -0
  48. package/ccw/dist/commands/tool.js.map +1 -0
  49. package/ccw/dist/commands/uninstall.d.ts +9 -0
  50. package/ccw/dist/commands/uninstall.d.ts.map +1 -0
  51. package/ccw/dist/commands/uninstall.js +239 -0
  52. package/ccw/dist/commands/uninstall.js.map +1 -0
  53. package/ccw/dist/commands/upgrade.d.ts +10 -0
  54. package/ccw/dist/commands/upgrade.d.ts.map +1 -0
  55. package/ccw/dist/commands/upgrade.js +288 -0
  56. package/ccw/dist/commands/upgrade.js.map +1 -0
  57. package/ccw/dist/commands/view.d.ts +14 -0
  58. package/ccw/dist/commands/view.d.ts.map +1 -0
  59. package/ccw/dist/commands/view.js +100 -0
  60. package/ccw/dist/commands/view.js.map +1 -0
  61. package/ccw/dist/config/storage-paths.d.ts +184 -0
  62. package/ccw/dist/config/storage-paths.d.ts.map +1 -0
  63. package/ccw/dist/config/storage-paths.js +536 -0
  64. package/ccw/dist/config/storage-paths.js.map +1 -0
  65. package/ccw/dist/core/cache-manager.d.ts +80 -0
  66. package/ccw/dist/core/cache-manager.d.ts.map +1 -0
  67. package/ccw/dist/core/cache-manager.js +260 -0
  68. package/ccw/dist/core/cache-manager.js.map +1 -0
  69. package/ccw/dist/core/claude-freshness.d.ts +53 -0
  70. package/ccw/dist/core/claude-freshness.d.ts.map +1 -0
  71. package/ccw/dist/core/claude-freshness.js +232 -0
  72. package/ccw/dist/core/claude-freshness.js.map +1 -0
  73. package/ccw/dist/core/core-memory-store.d.ts +320 -0
  74. package/ccw/dist/core/core-memory-store.d.ts.map +1 -0
  75. package/ccw/dist/core/core-memory-store.js +1177 -0
  76. package/ccw/dist/core/core-memory-store.js.map +1 -0
  77. package/ccw/dist/core/dashboard-generator-patch.d.ts +2 -0
  78. package/ccw/dist/core/dashboard-generator-patch.d.ts.map +1 -0
  79. package/ccw/dist/core/dashboard-generator-patch.js +48 -0
  80. package/ccw/dist/core/dashboard-generator-patch.js.map +1 -0
  81. package/ccw/dist/core/dashboard-generator.d.ts +8 -0
  82. package/ccw/dist/core/dashboard-generator.d.ts.map +1 -0
  83. package/ccw/dist/core/dashboard-generator.js +695 -0
  84. package/ccw/dist/core/dashboard-generator.js.map +1 -0
  85. package/ccw/dist/core/data-aggregator.d.ts +145 -0
  86. package/ccw/dist/core/data-aggregator.d.ts.map +1 -0
  87. package/ccw/dist/core/data-aggregator.js +416 -0
  88. package/ccw/dist/core/data-aggregator.js.map +1 -0
  89. package/ccw/dist/core/history-importer.d.ts +102 -0
  90. package/ccw/dist/core/history-importer.d.ts.map +1 -0
  91. package/ccw/dist/core/history-importer.js +493 -0
  92. package/ccw/dist/core/history-importer.js.map +1 -0
  93. package/ccw/dist/core/lite-scanner-complete.d.ts +81 -0
  94. package/ccw/dist/core/lite-scanner-complete.d.ts.map +1 -0
  95. package/ccw/dist/core/lite-scanner-complete.js +368 -0
  96. package/ccw/dist/core/lite-scanner-complete.js.map +1 -0
  97. package/ccw/dist/core/lite-scanner.d.ts +81 -0
  98. package/ccw/dist/core/lite-scanner.d.ts.map +1 -0
  99. package/ccw/dist/core/lite-scanner.js +368 -0
  100. package/ccw/dist/core/lite-scanner.js.map +1 -0
  101. package/ccw/dist/core/manifest.d.ts +88 -0
  102. package/ccw/dist/core/manifest.d.ts.map +1 -0
  103. package/ccw/dist/core/manifest.js +214 -0
  104. package/ccw/dist/core/manifest.js.map +1 -0
  105. package/ccw/dist/core/memory-embedder-bridge.d.ts +83 -0
  106. package/ccw/dist/core/memory-embedder-bridge.d.ts.map +1 -0
  107. package/ccw/dist/core/memory-embedder-bridge.js +181 -0
  108. package/ccw/dist/core/memory-embedder-bridge.js.map +1 -0
  109. package/ccw/dist/core/memory-store.d.ts +249 -0
  110. package/ccw/dist/core/memory-store.d.ts.map +1 -0
  111. package/ccw/dist/core/memory-store.js +781 -0
  112. package/ccw/dist/core/memory-store.js.map +1 -0
  113. package/ccw/dist/core/routes/ccw-routes.d.ts +20 -0
  114. package/ccw/dist/core/routes/ccw-routes.d.ts.map +1 -0
  115. package/ccw/dist/core/routes/ccw-routes.js +70 -0
  116. package/ccw/dist/core/routes/ccw-routes.js.map +1 -0
  117. package/ccw/dist/core/routes/claude-routes.d.ts +19 -0
  118. package/ccw/dist/core/routes/claude-routes.d.ts.map +1 -0
  119. package/ccw/dist/core/routes/claude-routes.js +1017 -0
  120. package/ccw/dist/core/routes/claude-routes.js.map +1 -0
  121. package/ccw/dist/core/routes/cli-routes.d.ts +20 -0
  122. package/ccw/dist/core/routes/cli-routes.d.ts.map +1 -0
  123. package/ccw/dist/core/routes/cli-routes.js +468 -0
  124. package/ccw/dist/core/routes/cli-routes.js.map +1 -0
  125. package/ccw/dist/core/routes/codexlens-routes.d.ts +20 -0
  126. package/ccw/dist/core/routes/codexlens-routes.d.ts.map +1 -0
  127. package/ccw/dist/core/routes/codexlens-routes.js +754 -0
  128. package/ccw/dist/core/routes/codexlens-routes.js.map +1 -0
  129. package/ccw/dist/core/routes/core-memory-routes.d.ts +21 -0
  130. package/ccw/dist/core/routes/core-memory-routes.d.ts.map +1 -0
  131. package/ccw/dist/core/routes/core-memory-routes.js +520 -0
  132. package/ccw/dist/core/routes/core-memory-routes.js.map +1 -0
  133. package/ccw/dist/core/routes/files-routes.d.ts +20 -0
  134. package/ccw/dist/core/routes/files-routes.d.ts.map +1 -0
  135. package/ccw/dist/core/routes/files-routes.js +374 -0
  136. package/ccw/dist/core/routes/files-routes.js.map +1 -0
  137. package/ccw/dist/core/routes/graph-routes.d.ts +20 -0
  138. package/ccw/dist/core/routes/graph-routes.d.ts.map +1 -0
  139. package/ccw/dist/core/routes/graph-routes.js +517 -0
  140. package/ccw/dist/core/routes/graph-routes.js.map +1 -0
  141. package/ccw/dist/core/routes/help-routes.d.ts +20 -0
  142. package/ccw/dist/core/routes/help-routes.d.ts.map +1 -0
  143. package/ccw/dist/core/routes/help-routes.js +250 -0
  144. package/ccw/dist/core/routes/help-routes.js.map +1 -0
  145. package/ccw/dist/core/routes/hooks-routes.d.ts +21 -0
  146. package/ccw/dist/core/routes/hooks-routes.d.ts.map +1 -0
  147. package/ccw/dist/core/routes/hooks-routes.js +346 -0
  148. package/ccw/dist/core/routes/hooks-routes.js.map +1 -0
  149. package/ccw/dist/core/routes/mcp-routes.d.ts +20 -0
  150. package/ccw/dist/core/routes/mcp-routes.d.ts.map +1 -0
  151. package/ccw/dist/core/routes/mcp-routes.js +1129 -0
  152. package/ccw/dist/core/routes/mcp-routes.js.map +1 -0
  153. package/ccw/dist/core/routes/mcp-templates-db.d.ts +54 -0
  154. package/ccw/dist/core/routes/mcp-templates-db.d.ts.map +1 -0
  155. package/ccw/dist/core/routes/mcp-templates-db.js +226 -0
  156. package/ccw/dist/core/routes/mcp-templates-db.js.map +1 -0
  157. package/ccw/dist/core/routes/memory-routes.d.ts +21 -0
  158. package/ccw/dist/core/routes/memory-routes.d.ts.map +1 -0
  159. package/ccw/dist/core/routes/memory-routes.js +1095 -0
  160. package/ccw/dist/core/routes/memory-routes.js.map +1 -0
  161. package/ccw/dist/core/routes/rules-routes.d.ts +20 -0
  162. package/ccw/dist/core/routes/rules-routes.d.ts.map +1 -0
  163. package/ccw/dist/core/routes/rules-routes.js +442 -0
  164. package/ccw/dist/core/routes/rules-routes.js.map +1 -0
  165. package/ccw/dist/core/routes/session-routes.d.ts +20 -0
  166. package/ccw/dist/core/routes/session-routes.d.ts.map +1 -0
  167. package/ccw/dist/core/routes/session-routes.js +423 -0
  168. package/ccw/dist/core/routes/session-routes.js.map +1 -0
  169. package/ccw/dist/core/routes/skills-routes.d.ts +20 -0
  170. package/ccw/dist/core/routes/skills-routes.d.ts.map +1 -0
  171. package/ccw/dist/core/routes/skills-routes.js +533 -0
  172. package/ccw/dist/core/routes/skills-routes.js.map +1 -0
  173. package/ccw/dist/core/routes/status-routes.d.ts +20 -0
  174. package/ccw/dist/core/routes/status-routes.d.ts.map +1 -0
  175. package/ccw/dist/core/routes/status-routes.js +38 -0
  176. package/ccw/dist/core/routes/status-routes.js.map +1 -0
  177. package/ccw/dist/core/routes/system-routes.d.ts +22 -0
  178. package/ccw/dist/core/routes/system-routes.d.ts.map +1 -0
  179. package/ccw/dist/core/routes/system-routes.js +354 -0
  180. package/ccw/dist/core/routes/system-routes.js.map +1 -0
  181. package/ccw/dist/core/server.d.ts +17 -0
  182. package/ccw/dist/core/server.d.ts.map +1 -0
  183. package/ccw/dist/core/server.js +386 -0
  184. package/ccw/dist/core/server.js.map +1 -0
  185. package/ccw/dist/core/session-clustering-service.d.ts +153 -0
  186. package/ccw/dist/core/session-clustering-service.d.ts.map +1 -0
  187. package/ccw/dist/core/session-clustering-service.js +1065 -0
  188. package/ccw/dist/core/session-clustering-service.js.map +1 -0
  189. package/ccw/dist/core/session-scanner.d.ts +32 -0
  190. package/ccw/dist/core/session-scanner.d.ts.map +1 -0
  191. package/ccw/dist/core/session-scanner.js +253 -0
  192. package/ccw/dist/core/session-scanner.js.map +1 -0
  193. package/ccw/dist/core/websocket.d.ts +23 -0
  194. package/ccw/dist/core/websocket.d.ts.map +1 -0
  195. package/ccw/dist/core/websocket.js +168 -0
  196. package/ccw/dist/core/websocket.js.map +1 -0
  197. package/ccw/dist/index.d.ts +10 -0
  198. package/ccw/dist/index.d.ts.map +1 -0
  199. package/ccw/dist/index.js +10 -0
  200. package/ccw/dist/index.js.map +1 -0
  201. package/ccw/dist/mcp-server/index.d.ts +7 -0
  202. package/ccw/dist/mcp-server/index.d.ts.map +1 -0
  203. package/ccw/dist/mcp-server/index.js +157 -0
  204. package/ccw/dist/mcp-server/index.js.map +1 -0
  205. package/ccw/dist/tools/classify-folders.d.ts +26 -0
  206. package/ccw/dist/tools/classify-folders.d.ts.map +1 -0
  207. package/ccw/dist/tools/classify-folders.js +201 -0
  208. package/ccw/dist/tools/classify-folders.js.map +1 -0
  209. package/ccw/dist/tools/cli-config-manager.d.ts +62 -0
  210. package/ccw/dist/tools/cli-config-manager.d.ts.map +1 -0
  211. package/ccw/dist/tools/cli-config-manager.js +221 -0
  212. package/ccw/dist/tools/cli-config-manager.js.map +1 -0
  213. package/ccw/dist/tools/cli-executor.d.ts +373 -0
  214. package/ccw/dist/tools/cli-executor.d.ts.map +1 -0
  215. package/ccw/dist/tools/cli-executor.js +1625 -0
  216. package/ccw/dist/tools/cli-executor.js.map +1 -0
  217. package/ccw/dist/tools/cli-history-store.d.ts +330 -0
  218. package/ccw/dist/tools/cli-history-store.d.ts.map +1 -0
  219. package/ccw/dist/tools/cli-history-store.js +916 -0
  220. package/ccw/dist/tools/cli-history-store.js.map +1 -0
  221. package/ccw/dist/tools/codex-lens.d.ts +118 -0
  222. package/ccw/dist/tools/codex-lens.d.ts.map +1 -0
  223. package/ccw/dist/tools/codex-lens.js +962 -0
  224. package/ccw/dist/tools/codex-lens.js.map +1 -0
  225. package/ccw/dist/tools/convert-tokens-to-css.d.ts +14 -0
  226. package/ccw/dist/tools/convert-tokens-to-css.d.ts.map +1 -0
  227. package/ccw/dist/tools/convert-tokens-to-css.js +244 -0
  228. package/ccw/dist/tools/convert-tokens-to-css.js.map +1 -0
  229. package/ccw/dist/tools/core-memory.d.ts +66 -0
  230. package/ccw/dist/tools/core-memory.d.ts.map +1 -0
  231. package/ccw/dist/tools/core-memory.js +324 -0
  232. package/ccw/dist/tools/core-memory.js.map +1 -0
  233. package/ccw/dist/tools/detect-changed-modules.d.ts +24 -0
  234. package/ccw/dist/tools/detect-changed-modules.d.ts.map +1 -0
  235. package/ccw/dist/tools/detect-changed-modules.js +277 -0
  236. package/ccw/dist/tools/detect-changed-modules.js.map +1 -0
  237. package/ccw/dist/tools/discover-design-files.d.ts +36 -0
  238. package/ccw/dist/tools/discover-design-files.d.ts.map +1 -0
  239. package/ccw/dist/tools/discover-design-files.js +147 -0
  240. package/ccw/dist/tools/discover-design-files.js.map +1 -0
  241. package/ccw/dist/tools/edit-file.d.ts +28 -0
  242. package/ccw/dist/tools/edit-file.d.ts.map +1 -0
  243. package/ccw/dist/tools/edit-file.js +479 -0
  244. package/ccw/dist/tools/edit-file.js.map +1 -0
  245. package/ccw/dist/tools/generate-module-docs.d.ts +22 -0
  246. package/ccw/dist/tools/generate-module-docs.d.ts.map +1 -0
  247. package/ccw/dist/tools/generate-module-docs.js +379 -0
  248. package/ccw/dist/tools/generate-module-docs.js.map +1 -0
  249. package/ccw/dist/tools/get-modules-by-depth.d.ts +15 -0
  250. package/ccw/dist/tools/get-modules-by-depth.d.ts.map +1 -0
  251. package/ccw/dist/tools/get-modules-by-depth.js +296 -0
  252. package/ccw/dist/tools/get-modules-by-depth.js.map +1 -0
  253. package/ccw/dist/tools/index.d.ts +55 -0
  254. package/ccw/dist/tools/index.d.ts.map +1 -0
  255. package/ccw/dist/tools/index.js +304 -0
  256. package/ccw/dist/tools/index.js.map +1 -0
  257. package/ccw/dist/tools/native-session-discovery.d.ts +97 -0
  258. package/ccw/dist/tools/native-session-discovery.d.ts.map +1 -0
  259. package/ccw/dist/tools/native-session-discovery.js +700 -0
  260. package/ccw/dist/tools/native-session-discovery.js.map +1 -0
  261. package/ccw/dist/tools/notifier.d.ts +50 -0
  262. package/ccw/dist/tools/notifier.d.ts.map +1 -0
  263. package/ccw/dist/tools/notifier.js +90 -0
  264. package/ccw/dist/tools/notifier.js.map +1 -0
  265. package/ccw/dist/tools/read-file.d.ts +32 -0
  266. package/ccw/dist/tools/read-file.d.ts.map +1 -0
  267. package/ccw/dist/tools/read-file.js +329 -0
  268. package/ccw/dist/tools/read-file.js.map +1 -0
  269. package/ccw/dist/tools/resume-strategy.d.ts +48 -0
  270. package/ccw/dist/tools/resume-strategy.d.ts.map +1 -0
  271. package/ccw/dist/tools/resume-strategy.js +248 -0
  272. package/ccw/dist/tools/resume-strategy.js.map +1 -0
  273. package/ccw/dist/tools/session-content-parser.d.ts +58 -0
  274. package/ccw/dist/tools/session-content-parser.d.ts.map +1 -0
  275. package/ccw/dist/tools/session-content-parser.js +420 -0
  276. package/ccw/dist/tools/session-content-parser.js.map +1 -0
  277. package/ccw/dist/tools/session-manager.d.ts +9 -0
  278. package/ccw/dist/tools/session-manager.d.ts.map +1 -0
  279. package/ccw/dist/tools/session-manager.js +834 -0
  280. package/ccw/dist/tools/session-manager.js.map +1 -0
  281. package/ccw/dist/tools/smart-context.d.ts +35 -0
  282. package/ccw/dist/tools/smart-context.d.ts.map +1 -0
  283. package/ccw/dist/tools/smart-context.js +182 -0
  284. package/ccw/dist/tools/smart-context.js.map +1 -0
  285. package/ccw/dist/tools/smart-search.d.ts +105 -0
  286. package/ccw/dist/tools/smart-search.d.ts.map +1 -0
  287. package/ccw/dist/tools/smart-search.js +1753 -0
  288. package/ccw/dist/tools/smart-search.js.map +1 -0
  289. package/ccw/dist/tools/storage-manager.d.ts +114 -0
  290. package/ccw/dist/tools/storage-manager.d.ts.map +1 -0
  291. package/ccw/dist/tools/storage-manager.js +392 -0
  292. package/ccw/dist/tools/storage-manager.js.map +1 -0
  293. package/ccw/dist/tools/ui-generate-preview.d.ts +39 -0
  294. package/ccw/dist/tools/ui-generate-preview.d.ts.map +1 -0
  295. package/ccw/dist/tools/ui-generate-preview.js +300 -0
  296. package/ccw/dist/tools/ui-generate-preview.js.map +1 -0
  297. package/ccw/dist/tools/ui-instantiate-prototypes.d.ts +75 -0
  298. package/ccw/dist/tools/ui-instantiate-prototypes.d.ts.map +1 -0
  299. package/ccw/dist/tools/ui-instantiate-prototypes.js +256 -0
  300. package/ccw/dist/tools/ui-instantiate-prototypes.js.map +1 -0
  301. package/ccw/dist/tools/update-module-claude.d.ts +80 -0
  302. package/ccw/dist/tools/update-module-claude.d.ts.map +1 -0
  303. package/ccw/dist/tools/update-module-claude.js +351 -0
  304. package/ccw/dist/tools/update-module-claude.js.map +1 -0
  305. package/ccw/dist/tools/write-file.d.ts +19 -0
  306. package/ccw/dist/tools/write-file.d.ts.map +1 -0
  307. package/ccw/dist/tools/write-file.js +193 -0
  308. package/ccw/dist/tools/write-file.js.map +1 -0
  309. package/ccw/dist/types/config.d.ts +11 -0
  310. package/ccw/dist/types/config.d.ts.map +1 -0
  311. package/ccw/dist/types/config.js +2 -0
  312. package/ccw/dist/types/config.js.map +1 -0
  313. package/ccw/dist/types/index.d.ts +4 -0
  314. package/ccw/dist/types/index.d.ts.map +1 -0
  315. package/ccw/dist/types/index.js +4 -0
  316. package/ccw/dist/types/index.js.map +1 -0
  317. package/ccw/dist/types/session.d.ts +20 -0
  318. package/ccw/dist/types/session.d.ts.map +1 -0
  319. package/ccw/dist/types/session.js +2 -0
  320. package/ccw/dist/types/session.js.map +1 -0
  321. package/ccw/dist/types/tool.d.ts +36 -0
  322. package/ccw/dist/types/tool.d.ts.map +1 -0
  323. package/ccw/dist/types/tool.js +11 -0
  324. package/ccw/dist/types/tool.js.map +1 -0
  325. package/ccw/dist/utils/browser-launcher.d.ts +13 -0
  326. package/ccw/dist/utils/browser-launcher.d.ts.map +1 -0
  327. package/ccw/dist/utils/browser-launcher.js +60 -0
  328. package/ccw/dist/utils/browser-launcher.js.map +1 -0
  329. package/ccw/dist/utils/file-utils.d.ts +25 -0
  330. package/ccw/dist/utils/file-utils.d.ts.map +1 -0
  331. package/ccw/dist/utils/file-utils.js +48 -0
  332. package/ccw/dist/utils/file-utils.js.map +1 -0
  333. package/ccw/dist/utils/path-resolver.d.ts +80 -0
  334. package/ccw/dist/utils/path-resolver.d.ts.map +1 -0
  335. package/ccw/dist/utils/path-resolver.js +260 -0
  336. package/ccw/dist/utils/path-resolver.js.map +1 -0
  337. package/ccw/dist/utils/path-validator.d.ts +49 -0
  338. package/ccw/dist/utils/path-validator.d.ts.map +1 -0
  339. package/ccw/dist/utils/path-validator.js +123 -0
  340. package/ccw/dist/utils/path-validator.js.map +1 -0
  341. package/ccw/dist/utils/ui.d.ts +62 -0
  342. package/ccw/dist/utils/ui.d.ts.map +1 -0
  343. package/ccw/dist/utils/ui.js +129 -0
  344. package/ccw/dist/utils/ui.js.map +1 -0
  345. package/ccw/package.json +1 -1
  346. package/package.json +5 -2
@@ -0,0 +1,954 @@
1
+ /**
2
+ * Session Command - Workflow session lifecycle management
3
+ * Adapter for session_manager tool providing direct CLI access
4
+ */
5
+ import chalk from 'chalk';
6
+ import http from 'http';
7
+ import { executeTool } from '../tools/index.js';
8
+ import { resolveFilePath, PathResolutionError } from './session-path-resolver.js';
9
+ // Handle EPIPE errors gracefully (occurs when piping to head/jq that closes early)
10
+ process.stdout.on('error', (err) => {
11
+ if (err.code === 'EPIPE') {
12
+ process.exit(0);
13
+ }
14
+ throw err;
15
+ });
16
+ /**
17
+ * Notify dashboard of granular events (fire and forget)
18
+ * @param {Object} data - Event data
19
+ */
20
+ function notifyDashboard(data) {
21
+ const DASHBOARD_PORT = process.env.CCW_PORT || 3456;
22
+ const payload = JSON.stringify({
23
+ ...data,
24
+ timestamp: new Date().toISOString()
25
+ });
26
+ const req = http.request({
27
+ hostname: 'localhost',
28
+ port: DASHBOARD_PORT,
29
+ path: '/api/hook',
30
+ method: 'POST',
31
+ headers: {
32
+ 'Content-Type': 'application/json',
33
+ 'Content-Length': Buffer.byteLength(payload)
34
+ }
35
+ });
36
+ // Fire and forget - log errors only in debug mode
37
+ req.on('error', (err) => {
38
+ if (process.env.DEBUG)
39
+ console.error('[Dashboard] Notification failed:', err.message);
40
+ });
41
+ req.write(payload);
42
+ req.end();
43
+ }
44
+ /**
45
+ * List sessions
46
+ * @param {Object} options - CLI options
47
+ */
48
+ async function listAction(options) {
49
+ const params = {
50
+ operation: 'list',
51
+ location: options.location || 'both',
52
+ include_metadata: options.metadata !== false
53
+ };
54
+ const result = await executeTool('session_manager', params);
55
+ if (!result.success) {
56
+ console.error(chalk.red(`Error: ${result.error}`));
57
+ process.exit(1);
58
+ }
59
+ const { active = [], archived = [], total } = result.result;
60
+ console.log(chalk.bold.cyan('\nWorkflow Sessions\n'));
61
+ if (active.length > 0) {
62
+ console.log(chalk.bold.white('Active Sessions:'));
63
+ for (const session of active) {
64
+ const meta = session.metadata || {};
65
+ console.log(chalk.green(` [ACTIVE] ${session.session_id}`));
66
+ if (meta.description)
67
+ console.log(chalk.gray(` ${meta.description}`));
68
+ if (meta.status)
69
+ console.log(chalk.gray(` Status: ${meta.status}`));
70
+ }
71
+ console.log();
72
+ }
73
+ if (archived.length > 0) {
74
+ console.log(chalk.bold.white('Archived Sessions:'));
75
+ for (const session of archived) {
76
+ const meta = session.metadata || {};
77
+ console.log(chalk.blue(` [ARCHIVED] ${session.session_id}`));
78
+ if (meta.description)
79
+ console.log(chalk.gray(` ${meta.description}`));
80
+ }
81
+ console.log();
82
+ }
83
+ if (total === 0) {
84
+ console.log(chalk.yellow('No sessions found'));
85
+ }
86
+ else {
87
+ console.log(chalk.gray(`Total: ${total} session(s)`));
88
+ }
89
+ }
90
+ /**
91
+ * Initialize a new session
92
+ * @param {string} sessionId - Session ID
93
+ * @param {Object} options - CLI options
94
+ */
95
+ async function initAction(sessionId, options) {
96
+ if (!sessionId) {
97
+ console.error(chalk.red('Session ID is required'));
98
+ console.error(chalk.gray('Usage: ccw session init <session_id> [--location <location>] [--type <type>] [--content <json>]'));
99
+ process.exit(1);
100
+ }
101
+ // Auto-infer location from type if not explicitly provided
102
+ // When type is 'lite-plan' or 'lite-fix', default location should match the type
103
+ const sessionLocation = options.location ||
104
+ (options.type === 'lite-plan' ? 'lite-plan' :
105
+ options.type === 'lite-fix' ? 'lite-fix' :
106
+ 'active');
107
+ // Infer type from location if not explicitly provided
108
+ const sessionType = options.type || (sessionLocation === 'active' ? 'workflow' : sessionLocation);
109
+ // Parse custom metadata from --content if provided
110
+ let customMetadata = {};
111
+ if (options.content) {
112
+ try {
113
+ customMetadata = JSON.parse(options.content);
114
+ }
115
+ catch (e) {
116
+ const error = e;
117
+ console.error(chalk.red('Invalid JSON in --content parameter'));
118
+ console.error(chalk.gray(`Parse error: ${error.message}`));
119
+ process.exit(1);
120
+ }
121
+ }
122
+ // Filter custom metadata: only allow safe fields, block system-critical fields
123
+ const blockedFields = ['session_id', 'type', 'status', 'created_at', 'updated_at', 'archived_at'];
124
+ const filteredCustomMetadata = {};
125
+ for (const key in customMetadata) {
126
+ if (!blockedFields.includes(key)) {
127
+ filteredCustomMetadata[key] = customMetadata[key];
128
+ }
129
+ else {
130
+ console.warn(chalk.yellow(`⚠ WARNING: Field '${key}' in --content is reserved and will be ignored`));
131
+ }
132
+ }
133
+ // Merge metadata: defaults < custom (filtered) < required fields
134
+ const metadata = Object.assign({
135
+ session_id: sessionId,
136
+ type: sessionType,
137
+ status: 'planning',
138
+ created_at: new Date().toISOString()
139
+ }, filteredCustomMetadata, // User custom fields (filtered)
140
+ {
141
+ session_id: sessionId, // Force override - always use CLI param
142
+ type: sessionType // Force override - always use --type or default
143
+ });
144
+ const params = {
145
+ operation: 'init',
146
+ session_id: sessionId,
147
+ metadata: metadata,
148
+ location: sessionLocation // Always pass location to session_manager
149
+ };
150
+ const result = await executeTool('session_manager', params);
151
+ if (!result.success) {
152
+ console.error(chalk.red(`Error: ${result.error}`));
153
+ process.exit(1);
154
+ }
155
+ // Emit SESSION_CREATED event
156
+ notifyDashboard({
157
+ type: 'SESSION_CREATED',
158
+ sessionId: sessionId,
159
+ payload: result.result
160
+ });
161
+ // Lite sessions (lite-plan, lite-fix) use session-metadata.json, others use workflow-session.json
162
+ const metadataFile = sessionLocation.startsWith('lite-') ? 'session-metadata.json' : 'workflow-session.json';
163
+ console.log(chalk.green(`✓ Session "${sessionId}" initialized`));
164
+ console.log(chalk.gray(` Location: ${result.result.path}`));
165
+ console.log(chalk.gray(` Metadata: ${metadataFile} created`));
166
+ }
167
+ /**
168
+ * Get session information (location and path)
169
+ * Helper function for path resolution
170
+ */
171
+ async function getSessionInfo(sessionId) {
172
+ // Use session_manager to find the session
173
+ const findParams = {
174
+ operation: 'list',
175
+ location: 'all',
176
+ include_metadata: false
177
+ };
178
+ const result = await executeTool('session_manager', findParams);
179
+ if (!result.success) {
180
+ throw new Error(`Failed to list sessions: ${result.error}`);
181
+ }
182
+ const resultData = result.result;
183
+ const allSessions = [
184
+ ...(resultData.active || []).map((s) => ({ ...s, location: 'active' })),
185
+ ...(resultData.archived || []).map((s) => ({ ...s, location: 'archived' })),
186
+ ...(resultData.litePlan || []).map((s) => ({ ...s, location: 'lite-plan' })),
187
+ ...(resultData.liteFix || []).map((s) => ({ ...s, location: 'lite-fix' })),
188
+ ];
189
+ const session = allSessions.find((s) => s.session_id === sessionId || s.id === sessionId);
190
+ if (!session) {
191
+ throw new Error(`Session "${sessionId}" not found in active, archived, lite-plan, or lite-fix locations`);
192
+ }
193
+ // Return actual session path from the session object
194
+ return {
195
+ path: session.path || '',
196
+ location: session.location
197
+ };
198
+ }
199
+ /**
200
+ * Read session content (NEW - with path resolution)
201
+ * @param {string} sessionId - Session ID
202
+ * @param {string} filename - Filename or relative path
203
+ * @param {Object} options - CLI options
204
+ */
205
+ async function readAction(sessionId, filename, options) {
206
+ if (!sessionId) {
207
+ console.error(chalk.red('Session ID is required'));
208
+ console.error(chalk.gray('Usage: ccw session <session-id> read <filename|path>'));
209
+ process.exit(1);
210
+ }
211
+ // Backward compatibility: if --type is provided, use legacy implementation
212
+ if (options.type) {
213
+ console.warn(chalk.yellow('⚠ WARNING: --type parameter is deprecated'));
214
+ console.warn(chalk.gray(' Old: ccw session read WFS-001 --type task --task-id IMPL-001'));
215
+ console.warn(chalk.gray(' New: ccw session WFS-001 read IMPL-001.json'));
216
+ console.log();
217
+ return readActionLegacy(sessionId, options);
218
+ }
219
+ if (!filename) {
220
+ console.error(chalk.red('Filename is required'));
221
+ console.error(chalk.gray('Usage: ccw session <session-id> read <filename|path>'));
222
+ console.error(chalk.gray(''));
223
+ console.error(chalk.gray('Examples:'));
224
+ console.error(chalk.gray(' ccw session WFS-001 read IMPL-001.json'));
225
+ console.error(chalk.gray(' ccw session WFS-001 read IMPL_PLAN.md'));
226
+ console.error(chalk.gray(' ccw session WFS-001 read .task/IMPL-001.json'));
227
+ process.exit(1);
228
+ }
229
+ try {
230
+ // Get session context
231
+ const session = await getSessionInfo(sessionId);
232
+ const context = {
233
+ sessionPath: session.path,
234
+ sessionLocation: session.location
235
+ };
236
+ // Resolve filename to content_type
237
+ const resolved = resolveFilePath(filename, context);
238
+ // Call session_manager tool
239
+ const params = {
240
+ operation: 'read',
241
+ session_id: sessionId,
242
+ content_type: resolved.contentType,
243
+ };
244
+ if (resolved.pathParams) {
245
+ params.path_params = resolved.pathParams;
246
+ }
247
+ const result = await executeTool('session_manager', params);
248
+ if (!result.success) {
249
+ console.error(chalk.red(`Error: ${result.error}`));
250
+ process.exit(1);
251
+ }
252
+ // Output raw content for piping
253
+ if (options.raw) {
254
+ console.log(typeof result.result.content === 'string'
255
+ ? result.result.content
256
+ : JSON.stringify(result.result.content, null, 2));
257
+ }
258
+ else {
259
+ console.log(JSON.stringify(result, null, 2));
260
+ }
261
+ }
262
+ catch (error) {
263
+ if (error instanceof PathResolutionError) {
264
+ console.error(chalk.red(`Error: ${error.message}`));
265
+ if (error.suggestions.length > 0) {
266
+ console.log(chalk.yellow('\nSuggestions:'));
267
+ error.suggestions.forEach(s => console.log(chalk.gray(` ${s}`)));
268
+ }
269
+ process.exit(1);
270
+ }
271
+ throw error;
272
+ }
273
+ }
274
+ /**
275
+ * Read session content (LEGACY - with --type parameter)
276
+ * @param {string} sessionId - Session ID
277
+ * @param {Object} options - CLI options
278
+ */
279
+ async function readActionLegacy(sessionId, options) {
280
+ if (!sessionId) {
281
+ console.error(chalk.red('Session ID is required'));
282
+ console.error(chalk.gray('Usage: ccw session read <session_id> --type <content_type>'));
283
+ process.exit(1);
284
+ }
285
+ const params = {
286
+ operation: 'read',
287
+ session_id: sessionId,
288
+ content_type: options.type || 'session'
289
+ };
290
+ // Add path_params if provided
291
+ if (options.taskId)
292
+ params.path_params = { ...(params.path_params || {}), task_id: options.taskId };
293
+ if (options.filename)
294
+ params.path_params = { ...(params.path_params || {}), filename: options.filename };
295
+ if (options.dimension)
296
+ params.path_params = { ...(params.path_params || {}), dimension: options.dimension };
297
+ if (options.iteration)
298
+ params.path_params = { ...(params.path_params || {}), iteration: options.iteration };
299
+ const result = await executeTool('session_manager', params);
300
+ if (!result.success) {
301
+ console.error(chalk.red(`Error: ${result.error}`));
302
+ process.exit(1);
303
+ }
304
+ // Output raw content for piping
305
+ if (options.raw) {
306
+ console.log(typeof result.result.content === 'string'
307
+ ? result.result.content
308
+ : JSON.stringify(result.result.content, null, 2));
309
+ }
310
+ else {
311
+ console.log(JSON.stringify(result, null, 2));
312
+ }
313
+ }
314
+ /**
315
+ * Write session content (NEW - with path resolution)
316
+ * @param {string} sessionId - Session ID
317
+ * @param {string} filename - Filename or relative path
318
+ * @param {string} contentString - Content to write
319
+ * @param {Object} options - CLI options
320
+ */
321
+ async function writeAction(sessionId, filename, contentString, options) {
322
+ if (!sessionId) {
323
+ console.error(chalk.red('Session ID is required'));
324
+ console.error(chalk.gray('Usage: ccw session <session-id> write <filename|path> <content>'));
325
+ process.exit(1);
326
+ }
327
+ // Backward compatibility: if --type is provided, use legacy implementation
328
+ if (options.type) {
329
+ console.warn(chalk.yellow('⚠ WARNING: --type parameter is deprecated'));
330
+ console.warn(chalk.gray(' Old: ccw session write WFS-001 --type plan --content "# Plan"'));
331
+ console.warn(chalk.gray(' New: ccw session WFS-001 write IMPL_PLAN.md "# Plan"'));
332
+ console.log();
333
+ return writeActionLegacy(sessionId, options);
334
+ }
335
+ if (!filename || !contentString) {
336
+ console.error(chalk.red('Filename and content are required'));
337
+ console.error(chalk.gray('Usage: ccw session <session-id> write <filename|path> <content>'));
338
+ console.error(chalk.gray(''));
339
+ console.error(chalk.gray('Examples:'));
340
+ console.error(chalk.gray(' ccw session WFS-001 write IMPL_PLAN.md "# Implementation Plan"'));
341
+ console.error(chalk.gray(' ccw session WFS-001 write IMPL-001.json \'{"id":"IMPL-001","status":"pending"}\''));
342
+ console.error(chalk.gray(' ccw session WFS-001 write .task/IMPL-001.json \'{"status":"completed"}\''));
343
+ process.exit(1);
344
+ }
345
+ try {
346
+ // Get session context
347
+ const session = await getSessionInfo(sessionId);
348
+ const context = {
349
+ sessionPath: session.path,
350
+ sessionLocation: session.location
351
+ };
352
+ // Resolve filename to content_type
353
+ const resolved = resolveFilePath(filename, context);
354
+ // Parse content (try JSON first, fallback to string)
355
+ let content;
356
+ try {
357
+ content = JSON.parse(contentString);
358
+ }
359
+ catch {
360
+ content = contentString;
361
+ }
362
+ // Call session_manager tool
363
+ const params = {
364
+ operation: 'write',
365
+ session_id: sessionId,
366
+ content_type: resolved.contentType,
367
+ content,
368
+ };
369
+ if (resolved.pathParams) {
370
+ params.path_params = resolved.pathParams;
371
+ }
372
+ const result = await executeTool('session_manager', params);
373
+ if (!result.success) {
374
+ console.error(chalk.red(`Error: ${result.error}`));
375
+ process.exit(1);
376
+ }
377
+ // Emit granular event based on content_type
378
+ const contentType = resolved.contentType;
379
+ let eventType = 'CONTENT_WRITTEN';
380
+ let entityId = null;
381
+ switch (contentType) {
382
+ case 'task':
383
+ eventType = 'TASK_CREATED';
384
+ entityId = resolved.pathParams?.task_id || content.task_id;
385
+ break;
386
+ case 'summary':
387
+ eventType = 'SUMMARY_WRITTEN';
388
+ entityId = resolved.pathParams?.task_id;
389
+ break;
390
+ case 'plan':
391
+ eventType = 'PLAN_UPDATED';
392
+ break;
393
+ case 'review-dim':
394
+ eventType = 'REVIEW_UPDATED';
395
+ entityId = resolved.pathParams?.dimension;
396
+ break;
397
+ case 'review-iter':
398
+ eventType = 'REVIEW_UPDATED';
399
+ entityId = resolved.pathParams?.iteration;
400
+ break;
401
+ case 'review-fix':
402
+ eventType = 'REVIEW_UPDATED';
403
+ entityId = resolved.pathParams?.filename;
404
+ break;
405
+ case 'session':
406
+ eventType = 'SESSION_UPDATED';
407
+ break;
408
+ }
409
+ notifyDashboard({
410
+ type: eventType,
411
+ sessionId: sessionId,
412
+ entityId: entityId,
413
+ contentType: contentType,
414
+ payload: result.result.written_content || content
415
+ });
416
+ console.log(chalk.green(`✓ Content written to ${resolved.resolvedPath}`));
417
+ }
418
+ catch (error) {
419
+ if (error instanceof PathResolutionError) {
420
+ console.error(chalk.red(`Error: ${error.message}`));
421
+ if (error.suggestions.length > 0) {
422
+ console.log(chalk.yellow('\nSuggestions:'));
423
+ error.suggestions.forEach(s => console.log(chalk.gray(` ${s}`)));
424
+ }
425
+ process.exit(1);
426
+ }
427
+ throw error;
428
+ }
429
+ }
430
+ /**
431
+ * Write session content (LEGACY - with --type parameter)
432
+ * @param {string} sessionId - Session ID
433
+ * @param {Object} options - CLI options
434
+ */
435
+ async function writeActionLegacy(sessionId, options) {
436
+ if (!sessionId) {
437
+ console.error(chalk.red('Session ID is required'));
438
+ console.error(chalk.gray('Usage: ccw session write <session_id> --type <content_type> --content <json>'));
439
+ process.exit(1);
440
+ }
441
+ if (!options.content) {
442
+ console.error(chalk.red('Content is required (--content)'));
443
+ process.exit(1);
444
+ }
445
+ let content;
446
+ try {
447
+ content = JSON.parse(options.content);
448
+ }
449
+ catch {
450
+ // If not JSON, treat as string content
451
+ content = options.content;
452
+ }
453
+ const params = {
454
+ operation: 'write',
455
+ session_id: sessionId,
456
+ content_type: options.type || 'session',
457
+ content
458
+ };
459
+ // Add path_params if provided
460
+ if (options.taskId)
461
+ params.path_params = { ...(params.path_params || {}), task_id: options.taskId };
462
+ if (options.filename)
463
+ params.path_params = { ...(params.path_params || {}), filename: options.filename };
464
+ const result = await executeTool('session_manager', params);
465
+ if (!result.success) {
466
+ console.error(chalk.red(`Error: ${result.error}`));
467
+ process.exit(1);
468
+ }
469
+ // Emit granular event based on content_type
470
+ const contentType = params.content_type;
471
+ let eventType = 'CONTENT_WRITTEN';
472
+ let entityId = null;
473
+ switch (contentType) {
474
+ case 'task':
475
+ eventType = 'TASK_CREATED';
476
+ entityId = options.taskId || content.task_id;
477
+ break;
478
+ case 'summary':
479
+ eventType = 'SUMMARY_WRITTEN';
480
+ entityId = options.taskId;
481
+ break;
482
+ case 'plan':
483
+ eventType = 'PLAN_UPDATED';
484
+ break;
485
+ case 'review-dim':
486
+ eventType = 'REVIEW_UPDATED';
487
+ entityId = options.dimension;
488
+ break;
489
+ case 'review-iter':
490
+ eventType = 'REVIEW_UPDATED';
491
+ entityId = options.iteration;
492
+ break;
493
+ case 'review-fix':
494
+ eventType = 'REVIEW_UPDATED';
495
+ entityId = options.filename;
496
+ break;
497
+ case 'session':
498
+ eventType = 'SESSION_UPDATED';
499
+ break;
500
+ }
501
+ notifyDashboard({
502
+ type: eventType,
503
+ sessionId: sessionId,
504
+ entityId: entityId,
505
+ contentType: contentType,
506
+ payload: result.result.written_content || content
507
+ });
508
+ console.log(chalk.green(`✓ Content written to ${result.result.path}`));
509
+ }
510
+ /**
511
+ * Update session content (merge)
512
+ * @param {string} sessionId - Session ID
513
+ * @param {Object} options - CLI options
514
+ */
515
+ async function updateAction(sessionId, options) {
516
+ if (!sessionId) {
517
+ console.error(chalk.red('Session ID is required'));
518
+ console.error(chalk.gray('Usage: ccw session update <session_id> --content <json>'));
519
+ process.exit(1);
520
+ }
521
+ if (!options.content) {
522
+ console.error(chalk.red('Content is required (--content)'));
523
+ process.exit(1);
524
+ }
525
+ let content;
526
+ try {
527
+ content = JSON.parse(options.content);
528
+ }
529
+ catch (e) {
530
+ const error = e;
531
+ console.error(chalk.red('Content must be valid JSON for update operation'));
532
+ console.error(chalk.gray(`Parse error: ${error.message}`));
533
+ process.exit(1);
534
+ }
535
+ const params = {
536
+ operation: 'update',
537
+ session_id: sessionId,
538
+ content_type: options.type || 'session',
539
+ content
540
+ };
541
+ // Add path_params if task update
542
+ if (options.taskId)
543
+ params.path_params = { task_id: options.taskId };
544
+ const result = await executeTool('session_manager', params);
545
+ if (!result.success) {
546
+ console.error(chalk.red(`Error: ${result.error}`));
547
+ process.exit(1);
548
+ }
549
+ // Emit granular event based on content_type
550
+ const eventType = params.content_type === 'task' ? 'TASK_UPDATED' : 'SESSION_UPDATED';
551
+ notifyDashboard({
552
+ type: eventType,
553
+ sessionId: sessionId,
554
+ entityId: options.taskId || null,
555
+ payload: result.result.merged_data || content
556
+ });
557
+ console.log(chalk.green(`✓ Session "${sessionId}" updated`));
558
+ }
559
+ /**
560
+ * Archive a session
561
+ * @param {string} sessionId - Session ID
562
+ * @param {Object} options - CLI options
563
+ */
564
+ async function archiveAction(sessionId, options) {
565
+ if (!sessionId) {
566
+ console.error(chalk.red('Session ID is required'));
567
+ console.error(chalk.gray('Usage: ccw session archive <session_id>'));
568
+ process.exit(1);
569
+ }
570
+ const params = {
571
+ operation: 'archive',
572
+ session_id: sessionId,
573
+ update_status: options.updateStatus !== false
574
+ };
575
+ const result = await executeTool('session_manager', params);
576
+ if (!result.success) {
577
+ console.error(chalk.red(`Error: ${result.error}`));
578
+ process.exit(1);
579
+ }
580
+ // Emit SESSION_ARCHIVED event
581
+ notifyDashboard({
582
+ type: 'SESSION_ARCHIVED',
583
+ sessionId: sessionId,
584
+ payload: result.result
585
+ });
586
+ console.log(chalk.green(`✓ Session "${sessionId}" archived`));
587
+ console.log(chalk.gray(` Location: ${result.result.destination}`));
588
+ }
589
+ /**
590
+ * Update session status (shortcut)
591
+ * @param {string} sessionId - Session ID
592
+ * @param {string} newStatus - New status value
593
+ */
594
+ async function statusAction(sessionId, newStatus) {
595
+ if (!sessionId) {
596
+ console.error(chalk.red('Session ID is required'));
597
+ console.error(chalk.gray('Usage: ccw session status <session_id> <status>'));
598
+ process.exit(1);
599
+ }
600
+ if (!newStatus) {
601
+ console.error(chalk.red('Status is required'));
602
+ console.error(chalk.gray('Valid statuses: planning, active, implementing, reviewing, completed, paused'));
603
+ process.exit(1);
604
+ }
605
+ const validStatuses = ['planning', 'active', 'implementing', 'reviewing', 'completed', 'paused'];
606
+ if (!validStatuses.includes(newStatus)) {
607
+ console.error(chalk.red(`Invalid status: ${newStatus}`));
608
+ console.error(chalk.gray(`Valid statuses: ${validStatuses.join(', ')}`));
609
+ process.exit(1);
610
+ }
611
+ const params = {
612
+ operation: 'update',
613
+ session_id: sessionId,
614
+ content_type: 'session',
615
+ content: { status: newStatus, updated_at: new Date().toISOString() }
616
+ };
617
+ const result = await executeTool('session_manager', params);
618
+ if (!result.success) {
619
+ console.error(chalk.red(`Error: ${result.error}`));
620
+ process.exit(1);
621
+ }
622
+ // Emit SESSION_UPDATED event
623
+ notifyDashboard({
624
+ type: 'SESSION_UPDATED',
625
+ sessionId: sessionId,
626
+ payload: { status: newStatus }
627
+ });
628
+ console.log(chalk.green(`✓ Session "${sessionId}" status → ${newStatus}`));
629
+ }
630
+ /**
631
+ * Update task status (shortcut)
632
+ * @param {string} sessionId - Session ID
633
+ * @param {string} taskId - Task ID
634
+ * @param {string} newStatus - New status value
635
+ */
636
+ async function taskAction(sessionId, taskId, newStatus) {
637
+ if (!sessionId) {
638
+ console.error(chalk.red('Session ID is required'));
639
+ console.error(chalk.gray('Usage: ccw session task <session_id> <task_id> <status>'));
640
+ process.exit(1);
641
+ }
642
+ if (!taskId) {
643
+ console.error(chalk.red('Task ID is required'));
644
+ console.error(chalk.gray('Usage: ccw session task <session_id> <task_id> <status>'));
645
+ process.exit(1);
646
+ }
647
+ if (!newStatus) {
648
+ console.error(chalk.red('Status is required'));
649
+ console.error(chalk.gray('Valid statuses: pending, in_progress, completed, blocked, cancelled'));
650
+ process.exit(1);
651
+ }
652
+ const validStatuses = ['pending', 'in_progress', 'completed', 'blocked', 'cancelled'];
653
+ if (!validStatuses.includes(newStatus)) {
654
+ console.error(chalk.red(`Invalid status: ${newStatus}`));
655
+ console.error(chalk.gray(`Valid statuses: ${validStatuses.join(', ')}`));
656
+ process.exit(1);
657
+ }
658
+ // First, read the current task to get existing status
659
+ const readParams = {
660
+ operation: 'read',
661
+ session_id: sessionId,
662
+ content_type: 'task',
663
+ path_params: { task_id: taskId }
664
+ };
665
+ const readResult = await executeTool('session_manager', readParams);
666
+ let currentTask = {};
667
+ let oldStatus = 'unknown';
668
+ if (readResult.success) {
669
+ currentTask = readResult.result.content || {};
670
+ oldStatus = currentTask.status || 'unknown';
671
+ }
672
+ // Build status history entry
673
+ const historyEntry = {
674
+ from: oldStatus,
675
+ to: newStatus,
676
+ changed_at: new Date().toISOString()
677
+ };
678
+ // Update task with new status and appended history
679
+ const params = {
680
+ operation: 'update',
681
+ session_id: sessionId,
682
+ content_type: 'task',
683
+ path_params: { task_id: taskId },
684
+ content: {
685
+ status: newStatus,
686
+ updated_at: new Date().toISOString(),
687
+ status_history: [...(currentTask.status_history || []), historyEntry]
688
+ }
689
+ };
690
+ const result = await executeTool('session_manager', params);
691
+ if (!result.success) {
692
+ console.error(chalk.red(`Error: ${result.error}`));
693
+ process.exit(1);
694
+ }
695
+ // Emit TASK_UPDATED event
696
+ notifyDashboard({
697
+ type: 'TASK_UPDATED',
698
+ sessionId: sessionId,
699
+ entityId: taskId,
700
+ payload: { status: newStatus }
701
+ });
702
+ console.log(chalk.green(`✓ Task "${taskId}" status → ${newStatus}`));
703
+ }
704
+ /**
705
+ * Create directory within session
706
+ * @param {string} sessionId - Session ID
707
+ * @param {Object} options - CLI options
708
+ */
709
+ async function mkdirAction(sessionId, options) {
710
+ if (!sessionId) {
711
+ console.error(chalk.red('Session ID is required'));
712
+ console.error(chalk.gray('Usage: ccw session mkdir <session_id> --subdir <subdir>'));
713
+ process.exit(1);
714
+ }
715
+ if (!options.subdir) {
716
+ console.error(chalk.red('Subdirectory is required (--subdir)'));
717
+ process.exit(1);
718
+ }
719
+ const params = {
720
+ operation: 'mkdir',
721
+ session_id: sessionId,
722
+ dirs: [options.subdir] // Convert single subdir to array
723
+ };
724
+ const result = await executeTool('session_manager', params);
725
+ if (!result.success) {
726
+ console.error(chalk.red(`Error: ${result.error}`));
727
+ process.exit(1);
728
+ }
729
+ // Emit DIRECTORY_CREATED event
730
+ notifyDashboard({
731
+ type: 'DIRECTORY_CREATED',
732
+ sessionId: sessionId,
733
+ payload: { directories: result.result.directories_created }
734
+ });
735
+ console.log(chalk.green(`✓ Directory created: ${result.result.directories_created.join(', ')}`));
736
+ }
737
+ /**
738
+ * Delete file within session
739
+ * @param {string} sessionId - Session ID
740
+ * @param {string} filePath - Relative file path
741
+ */
742
+ async function deleteAction(sessionId, filePath) {
743
+ if (!sessionId) {
744
+ console.error(chalk.red('Session ID is required'));
745
+ console.error(chalk.gray('Usage: ccw session delete <session_id> <file_path>'));
746
+ process.exit(1);
747
+ }
748
+ if (!filePath) {
749
+ console.error(chalk.red('File path is required'));
750
+ console.error(chalk.gray('Usage: ccw session delete <session_id> <file_path>'));
751
+ process.exit(1);
752
+ }
753
+ const params = {
754
+ operation: 'delete',
755
+ session_id: sessionId,
756
+ file_path: filePath
757
+ };
758
+ const result = await executeTool('session_manager', params);
759
+ if (!result.success) {
760
+ console.error(chalk.red(`Error: ${result.error}`));
761
+ process.exit(1);
762
+ }
763
+ // Emit FILE_DELETED event
764
+ notifyDashboard({
765
+ type: 'FILE_DELETED',
766
+ sessionId: sessionId,
767
+ payload: { file_path: filePath }
768
+ });
769
+ console.log(chalk.green(`✓ File deleted: ${result.result.deleted}`));
770
+ }
771
+ /**
772
+ * Get session statistics
773
+ * @param {string} sessionId - Session ID
774
+ */
775
+ async function statsAction(sessionId, options = {}) {
776
+ if (!sessionId) {
777
+ console.error(chalk.red('Session ID is required'));
778
+ console.error(chalk.gray('Usage: ccw session stats <session_id>'));
779
+ process.exit(1);
780
+ }
781
+ const params = {
782
+ operation: 'stats',
783
+ session_id: sessionId
784
+ };
785
+ const result = await executeTool('session_manager', params);
786
+ if (!result.success) {
787
+ console.error(chalk.red(`Error: ${result.error}`));
788
+ process.exit(1);
789
+ }
790
+ const { tasks, summaries, has_plan, location } = result.result;
791
+ console.log(chalk.bold.cyan(`\nSession Statistics: ${sessionId}`));
792
+ console.log(chalk.gray(`Location: ${location}\n`));
793
+ console.log(chalk.bold.white('Tasks:'));
794
+ console.log(chalk.gray(` Total: ${tasks.total}`));
795
+ console.log(chalk.green(` Completed: ${tasks.completed}`));
796
+ console.log(chalk.yellow(` In Progress: ${tasks.in_progress}`));
797
+ console.log(chalk.blue(` Pending: ${tasks.pending}`));
798
+ console.log(chalk.red(` Blocked: ${tasks.blocked}`));
799
+ console.log(chalk.gray(` Cancelled: ${tasks.cancelled}\n`));
800
+ console.log(chalk.bold.white('Documentation:'));
801
+ console.log(chalk.gray(` Summaries: ${summaries}`));
802
+ console.log(chalk.gray(` Plan: ${has_plan ? 'Yes' : 'No'}`));
803
+ }
804
+ async function execAction(jsonParams) {
805
+ if (!jsonParams) {
806
+ console.error(chalk.red('JSON parameters required'));
807
+ console.error(chalk.gray('Usage: ccw session exec \'{"operation":"list","location":"active"}\''));
808
+ process.exit(1);
809
+ }
810
+ let params;
811
+ try {
812
+ params = JSON.parse(jsonParams);
813
+ }
814
+ catch (e) {
815
+ const error = e;
816
+ console.error(chalk.red('Invalid JSON'));
817
+ console.error(chalk.gray(`Parse error: ${error.message}`));
818
+ process.exit(1);
819
+ }
820
+ const result = await executeTool('session_manager', params);
821
+ // Emit notification for write operations
822
+ if (result.success && params.operation) {
823
+ const writeOps = ['init', 'write', 'update', 'archive', 'mkdir', 'delete'];
824
+ if (writeOps.includes(params.operation)) {
825
+ const eventMap = {
826
+ init: 'SESSION_CREATED',
827
+ write: 'CONTENT_WRITTEN',
828
+ update: 'SESSION_UPDATED',
829
+ archive: 'SESSION_ARCHIVED',
830
+ mkdir: 'DIRECTORY_CREATED',
831
+ delete: 'FILE_DELETED'
832
+ };
833
+ notifyDashboard({
834
+ type: eventMap[params.operation] || 'SESSION_UPDATED',
835
+ sessionId: params.session_id,
836
+ operation: params.operation,
837
+ payload: result.result
838
+ });
839
+ }
840
+ }
841
+ console.log(JSON.stringify(result, null, 2));
842
+ }
843
+ /**
844
+ * Session command entry point
845
+ * @param {string} subcommand - Subcommand
846
+ * @param {string[]} args - Arguments
847
+ * @param {Object} options - CLI options
848
+ */
849
+ export async function sessionCommand(subcommand, args, options) {
850
+ let argsArray = Array.isArray(args) ? args : (args ? [args] : []);
851
+ // Detect new format: ccw session WFS-xxx <operation> <args>
852
+ // If subcommand looks like a session ID, rearrange parameters
853
+ // Exception: 'init' should always use traditional format (ccw session init WFS-xxx)
854
+ const isSessionId = subcommand && (subcommand.startsWith('WFS-') ||
855
+ subcommand === 'manifest' ||
856
+ subcommand === 'project' ||
857
+ /^[A-Z][A-Z0-9]*-[A-Z0-9]+/.test(subcommand) // Generic session ID pattern (uppercase prefix + dash + alphanumeric)
858
+ );
859
+ if (isSessionId && argsArray.length > 0) {
860
+ const operation = argsArray[0];
861
+ // Reject new format for init operation (semantic error)
862
+ if (operation === 'init') {
863
+ console.error(chalk.red('Error: Invalid format for init operation'));
864
+ console.error(chalk.gray('Correct: ccw session init <session-id>'));
865
+ console.error(chalk.gray(`Wrong: ccw session <session-id> init`));
866
+ console.error(chalk.yellow('\nReason: Session must be initialized before it can be referenced'));
867
+ process.exit(1);
868
+ }
869
+ // New format detected: session-id comes first
870
+ const sessionId = subcommand;
871
+ const operationArgs = argsArray.slice(1);
872
+ // Rearrange: operation becomes subcommand, session-id goes into args
873
+ subcommand = operation;
874
+ argsArray = [sessionId, ...operationArgs];
875
+ }
876
+ switch (subcommand) {
877
+ case 'list':
878
+ await listAction(options);
879
+ break;
880
+ case 'init':
881
+ await initAction(argsArray[0], options);
882
+ break;
883
+ case 'read':
884
+ // args[0] = session-id, args[1] = filename (optional for backward compat)
885
+ await readAction(argsArray[0], argsArray[1], options);
886
+ break;
887
+ case 'write':
888
+ // args[0] = session-id, args[1] = filename, args[2] = content
889
+ await writeAction(argsArray[0], argsArray[1], argsArray[2], options);
890
+ break;
891
+ case 'update':
892
+ await updateAction(argsArray[0], options);
893
+ break;
894
+ case 'archive':
895
+ await archiveAction(argsArray[0], options);
896
+ break;
897
+ case 'status':
898
+ await statusAction(argsArray[0], argsArray[1]);
899
+ break;
900
+ case 'task':
901
+ await taskAction(argsArray[0], argsArray[1], argsArray[2]);
902
+ break;
903
+ case 'mkdir':
904
+ await mkdirAction(argsArray[0], options);
905
+ break;
906
+ case 'delete':
907
+ await deleteAction(argsArray[0], argsArray[1]);
908
+ break;
909
+ case 'stats':
910
+ await statsAction(argsArray[0], options);
911
+ break;
912
+ case 'exec':
913
+ await execAction(argsArray[0]);
914
+ break;
915
+ default:
916
+ console.log(chalk.bold.cyan('\nCCW Session Management\n'));
917
+ console.log('Subcommands:');
918
+ console.log(chalk.gray(' list List all sessions'));
919
+ console.log(chalk.gray(' <session-id> init [metadata] Initialize new session'));
920
+ console.log(chalk.gray(' <session-id> read <filename|path> Read session content'));
921
+ console.log(chalk.gray(' <session-id> write <filename> <content> Write session content'));
922
+ console.log(chalk.gray(' <session-id> stats Get session statistics'));
923
+ console.log(chalk.gray(' <session-id> archive Archive session'));
924
+ console.log(chalk.gray(' <session-id> status <status> Update session status'));
925
+ console.log(chalk.gray(' <session-id> task <task-id> <status> Update task status'));
926
+ console.log(chalk.gray(' <session-id> delete <file-path> Delete file within session'));
927
+ console.log(chalk.gray(' <session-id> update Update session (merge)'));
928
+ console.log(chalk.gray(' <session-id> mkdir Create subdirectory'));
929
+ console.log(chalk.gray(' exec <json> Execute raw operation'));
930
+ console.log();
931
+ console.log('Filename/Path Examples:');
932
+ console.log(chalk.gray(' IMPL-001.json Task file (auto: .task/)'));
933
+ console.log(chalk.gray(' .task/IMPL-001.json Task file (explicit path)'));
934
+ console.log(chalk.gray(' IMPL_PLAN.md Implementation plan'));
935
+ console.log(chalk.gray(' TODO_LIST.md TODO list'));
936
+ console.log(chalk.gray(' workflow-session.json Session metadata'));
937
+ console.log(chalk.gray(' .review/dimensions/security.json Review dimension'));
938
+ console.log();
939
+ console.log('Status Values:');
940
+ console.log(chalk.gray(' Session: planning, active, implementing, reviewing, completed, paused'));
941
+ console.log(chalk.gray(' Task: pending, in_progress, completed, blocked, cancelled'));
942
+ console.log();
943
+ console.log('Examples:');
944
+ console.log(chalk.gray(' ccw session list'));
945
+ console.log(chalk.gray(' ccw session WFS-001 init'));
946
+ console.log(chalk.gray(' ccw session WFS-001 read IMPL_PLAN.md'));
947
+ console.log(chalk.gray(' ccw session WFS-001 read IMPL-001.json'));
948
+ console.log(chalk.gray(' ccw session WFS-001 write IMPL_PLAN.md "# Plan"'));
949
+ console.log(chalk.gray(' ccw session WFS-001 write IMPL-001.json \'{"status":"pending"}\''));
950
+ console.log(chalk.gray(' ccw session WFS-001 stats'));
951
+ console.log(chalk.gray(' ccw session WFS-001 archive'));
952
+ }
953
+ }
954
+ //# sourceMappingURL=session.js.map