claude-code-workflow 6.1.3 → 6.2.1

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 (879) hide show
  1. package/README.md +145 -274
  2. package/bin/ccw-mcp.js +7 -0
  3. package/bin/ccw.js +10 -0
  4. package/dist/cli.d.ts +2 -0
  5. package/dist/cli.d.ts.map +1 -0
  6. package/dist/cli.js +219 -0
  7. package/dist/cli.js.map +1 -0
  8. package/dist/commands/cli.d.ts +32 -0
  9. package/dist/commands/cli.d.ts.map +1 -0
  10. package/dist/commands/cli.js +619 -0
  11. package/dist/commands/cli.js.map +1 -0
  12. package/dist/commands/core-memory.d.ts +32 -0
  13. package/dist/commands/core-memory.d.ts.map +1 -0
  14. package/dist/commands/core-memory.js +640 -0
  15. package/dist/commands/core-memory.js.map +1 -0
  16. package/dist/commands/hook.d.ts +16 -0
  17. package/dist/commands/hook.d.ts.map +1 -0
  18. package/dist/commands/hook.js +276 -0
  19. package/dist/commands/hook.js.map +1 -0
  20. package/dist/commands/install.d.ts +12 -0
  21. package/dist/commands/install.d.ts.map +1 -0
  22. package/dist/commands/install.js +443 -0
  23. package/dist/commands/install.js.map +1 -0
  24. package/dist/commands/list.d.ts +5 -0
  25. package/dist/commands/list.d.ts.map +1 -0
  26. package/dist/commands/list.js +32 -0
  27. package/dist/commands/list.js.map +1 -0
  28. package/dist/commands/memory.d.ts +57 -0
  29. package/dist/commands/memory.d.ts.map +1 -0
  30. package/dist/commands/memory.js +890 -0
  31. package/dist/commands/memory.js.map +1 -0
  32. package/dist/commands/serve.d.ts +12 -0
  33. package/dist/commands/serve.d.ts.map +1 -0
  34. package/dist/commands/serve.js +63 -0
  35. package/dist/commands/serve.js.map +1 -0
  36. package/dist/commands/session-path-resolver.d.ts +45 -0
  37. package/dist/commands/session-path-resolver.d.ts.map +1 -0
  38. package/dist/commands/session-path-resolver.js +302 -0
  39. package/dist/commands/session-path-resolver.js.map +1 -0
  40. package/dist/commands/session.d.ts +12 -0
  41. package/dist/commands/session.d.ts.map +1 -0
  42. package/dist/commands/session.js +954 -0
  43. package/dist/commands/session.js.map +1 -0
  44. package/dist/commands/stop.d.ts +11 -0
  45. package/dist/commands/stop.d.ts.map +1 -0
  46. package/dist/commands/stop.js +96 -0
  47. package/dist/commands/stop.js.map +1 -0
  48. package/dist/commands/tool.d.ts +29 -0
  49. package/dist/commands/tool.d.ts.map +1 -0
  50. package/dist/commands/tool.js +173 -0
  51. package/dist/commands/tool.js.map +1 -0
  52. package/dist/commands/uninstall.d.ts +9 -0
  53. package/dist/commands/uninstall.d.ts.map +1 -0
  54. package/dist/commands/uninstall.js +239 -0
  55. package/dist/commands/uninstall.js.map +1 -0
  56. package/dist/commands/upgrade.d.ts +10 -0
  57. package/dist/commands/upgrade.d.ts.map +1 -0
  58. package/dist/commands/upgrade.js +288 -0
  59. package/dist/commands/upgrade.js.map +1 -0
  60. package/dist/commands/view.d.ts +14 -0
  61. package/dist/commands/view.d.ts.map +1 -0
  62. package/dist/commands/view.js +100 -0
  63. package/dist/commands/view.js.map +1 -0
  64. package/dist/config/storage-paths.d.ts +184 -0
  65. package/dist/config/storage-paths.d.ts.map +1 -0
  66. package/dist/config/storage-paths.js +536 -0
  67. package/dist/config/storage-paths.js.map +1 -0
  68. package/dist/core/cache-manager.d.ts +80 -0
  69. package/dist/core/cache-manager.d.ts.map +1 -0
  70. package/dist/core/cache-manager.js +260 -0
  71. package/dist/core/cache-manager.js.map +1 -0
  72. package/dist/core/claude-freshness.d.ts +53 -0
  73. package/dist/core/claude-freshness.d.ts.map +1 -0
  74. package/dist/core/claude-freshness.js +232 -0
  75. package/dist/core/claude-freshness.js.map +1 -0
  76. package/dist/core/core-memory-store.d.ts +320 -0
  77. package/dist/core/core-memory-store.d.ts.map +1 -0
  78. package/dist/core/core-memory-store.js +1177 -0
  79. package/dist/core/core-memory-store.js.map +1 -0
  80. package/dist/core/dashboard-generator-patch.d.ts +2 -0
  81. package/dist/core/dashboard-generator-patch.d.ts.map +1 -0
  82. package/dist/core/dashboard-generator-patch.js +48 -0
  83. package/dist/core/dashboard-generator-patch.js.map +1 -0
  84. package/dist/core/dashboard-generator.d.ts +8 -0
  85. package/dist/core/dashboard-generator.d.ts.map +1 -0
  86. package/dist/core/dashboard-generator.js +695 -0
  87. package/dist/core/dashboard-generator.js.map +1 -0
  88. package/dist/core/data-aggregator.d.ts +145 -0
  89. package/dist/core/data-aggregator.d.ts.map +1 -0
  90. package/dist/core/data-aggregator.js +416 -0
  91. package/dist/core/data-aggregator.js.map +1 -0
  92. package/dist/core/history-importer.d.ts +102 -0
  93. package/dist/core/history-importer.d.ts.map +1 -0
  94. package/dist/core/history-importer.js +493 -0
  95. package/dist/core/history-importer.js.map +1 -0
  96. package/dist/core/lite-scanner-complete.d.ts +81 -0
  97. package/dist/core/lite-scanner-complete.d.ts.map +1 -0
  98. package/dist/core/lite-scanner-complete.js +368 -0
  99. package/dist/core/lite-scanner-complete.js.map +1 -0
  100. package/dist/core/lite-scanner.d.ts +81 -0
  101. package/dist/core/lite-scanner.d.ts.map +1 -0
  102. package/dist/core/lite-scanner.js +368 -0
  103. package/dist/core/lite-scanner.js.map +1 -0
  104. package/dist/core/manifest.d.ts +88 -0
  105. package/dist/core/manifest.d.ts.map +1 -0
  106. package/dist/core/manifest.js +214 -0
  107. package/dist/core/manifest.js.map +1 -0
  108. package/dist/core/memory-embedder-bridge.d.ts +83 -0
  109. package/dist/core/memory-embedder-bridge.d.ts.map +1 -0
  110. package/dist/core/memory-embedder-bridge.js +181 -0
  111. package/dist/core/memory-embedder-bridge.js.map +1 -0
  112. package/dist/core/memory-store.d.ts +249 -0
  113. package/dist/core/memory-store.d.ts.map +1 -0
  114. package/dist/core/memory-store.js +781 -0
  115. package/dist/core/memory-store.js.map +1 -0
  116. package/dist/core/routes/ccw-routes.d.ts +20 -0
  117. package/dist/core/routes/ccw-routes.d.ts.map +1 -0
  118. package/dist/core/routes/ccw-routes.js +70 -0
  119. package/dist/core/routes/ccw-routes.js.map +1 -0
  120. package/dist/core/routes/claude-routes.d.ts +19 -0
  121. package/dist/core/routes/claude-routes.d.ts.map +1 -0
  122. package/dist/core/routes/claude-routes.js +1017 -0
  123. package/dist/core/routes/claude-routes.js.map +1 -0
  124. package/dist/core/routes/cli-routes.d.ts +20 -0
  125. package/dist/core/routes/cli-routes.d.ts.map +1 -0
  126. package/dist/core/routes/cli-routes.js +468 -0
  127. package/dist/core/routes/cli-routes.js.map +1 -0
  128. package/dist/core/routes/codexlens-routes.d.ts +20 -0
  129. package/dist/core/routes/codexlens-routes.d.ts.map +1 -0
  130. package/dist/core/routes/codexlens-routes.js +754 -0
  131. package/dist/core/routes/codexlens-routes.js.map +1 -0
  132. package/dist/core/routes/core-memory-routes.d.ts +21 -0
  133. package/dist/core/routes/core-memory-routes.d.ts.map +1 -0
  134. package/dist/core/routes/core-memory-routes.js +520 -0
  135. package/dist/core/routes/core-memory-routes.js.map +1 -0
  136. package/dist/core/routes/files-routes.d.ts +20 -0
  137. package/dist/core/routes/files-routes.d.ts.map +1 -0
  138. package/dist/core/routes/files-routes.js +374 -0
  139. package/dist/core/routes/files-routes.js.map +1 -0
  140. package/dist/core/routes/graph-routes.d.ts +20 -0
  141. package/dist/core/routes/graph-routes.d.ts.map +1 -0
  142. package/dist/core/routes/graph-routes.js +517 -0
  143. package/dist/core/routes/graph-routes.js.map +1 -0
  144. package/dist/core/routes/help-routes.d.ts +20 -0
  145. package/dist/core/routes/help-routes.d.ts.map +1 -0
  146. package/dist/core/routes/help-routes.js +250 -0
  147. package/dist/core/routes/help-routes.js.map +1 -0
  148. package/dist/core/routes/hooks-routes.d.ts +21 -0
  149. package/dist/core/routes/hooks-routes.d.ts.map +1 -0
  150. package/dist/core/routes/hooks-routes.js +346 -0
  151. package/dist/core/routes/hooks-routes.js.map +1 -0
  152. package/dist/core/routes/mcp-routes.d.ts +20 -0
  153. package/dist/core/routes/mcp-routes.d.ts.map +1 -0
  154. package/dist/core/routes/mcp-routes.js +1129 -0
  155. package/dist/core/routes/mcp-routes.js.map +1 -0
  156. package/dist/core/routes/mcp-templates-db.d.ts +54 -0
  157. package/dist/core/routes/mcp-templates-db.d.ts.map +1 -0
  158. package/dist/core/routes/mcp-templates-db.js +226 -0
  159. package/dist/core/routes/mcp-templates-db.js.map +1 -0
  160. package/dist/core/routes/memory-routes.d.ts +21 -0
  161. package/dist/core/routes/memory-routes.d.ts.map +1 -0
  162. package/dist/core/routes/memory-routes.js +1095 -0
  163. package/dist/core/routes/memory-routes.js.map +1 -0
  164. package/dist/core/routes/rules-routes.d.ts +20 -0
  165. package/dist/core/routes/rules-routes.d.ts.map +1 -0
  166. package/dist/core/routes/rules-routes.js +442 -0
  167. package/dist/core/routes/rules-routes.js.map +1 -0
  168. package/dist/core/routes/session-routes.d.ts +20 -0
  169. package/dist/core/routes/session-routes.d.ts.map +1 -0
  170. package/dist/core/routes/session-routes.js +423 -0
  171. package/dist/core/routes/session-routes.js.map +1 -0
  172. package/dist/core/routes/skills-routes.d.ts +20 -0
  173. package/dist/core/routes/skills-routes.d.ts.map +1 -0
  174. package/dist/core/routes/skills-routes.js +533 -0
  175. package/dist/core/routes/skills-routes.js.map +1 -0
  176. package/dist/core/routes/status-routes.d.ts +20 -0
  177. package/dist/core/routes/status-routes.d.ts.map +1 -0
  178. package/dist/core/routes/status-routes.js +38 -0
  179. package/dist/core/routes/status-routes.js.map +1 -0
  180. package/dist/core/routes/system-routes.d.ts +22 -0
  181. package/dist/core/routes/system-routes.d.ts.map +1 -0
  182. package/dist/core/routes/system-routes.js +354 -0
  183. package/dist/core/routes/system-routes.js.map +1 -0
  184. package/dist/core/server.d.ts +17 -0
  185. package/dist/core/server.d.ts.map +1 -0
  186. package/dist/core/server.js +386 -0
  187. package/dist/core/server.js.map +1 -0
  188. package/dist/core/session-clustering-service.d.ts +153 -0
  189. package/dist/core/session-clustering-service.d.ts.map +1 -0
  190. package/dist/core/session-clustering-service.js +1065 -0
  191. package/dist/core/session-clustering-service.js.map +1 -0
  192. package/dist/core/session-scanner.d.ts +32 -0
  193. package/dist/core/session-scanner.d.ts.map +1 -0
  194. package/dist/core/session-scanner.js +253 -0
  195. package/dist/core/session-scanner.js.map +1 -0
  196. package/dist/core/websocket.d.ts +23 -0
  197. package/dist/core/websocket.d.ts.map +1 -0
  198. package/dist/core/websocket.js +168 -0
  199. package/dist/core/websocket.js.map +1 -0
  200. package/dist/index.d.ts +10 -0
  201. package/dist/index.d.ts.map +1 -0
  202. package/dist/index.js +10 -0
  203. package/dist/index.js.map +1 -0
  204. package/dist/mcp-server/index.d.ts +7 -0
  205. package/dist/mcp-server/index.d.ts.map +1 -0
  206. package/dist/mcp-server/index.js +157 -0
  207. package/dist/mcp-server/index.js.map +1 -0
  208. package/dist/tools/classify-folders.d.ts +26 -0
  209. package/dist/tools/classify-folders.d.ts.map +1 -0
  210. package/dist/tools/classify-folders.js +201 -0
  211. package/dist/tools/classify-folders.js.map +1 -0
  212. package/dist/tools/cli-config-manager.d.ts +62 -0
  213. package/dist/tools/cli-config-manager.d.ts.map +1 -0
  214. package/dist/tools/cli-config-manager.js +221 -0
  215. package/dist/tools/cli-config-manager.js.map +1 -0
  216. package/dist/tools/cli-executor.d.ts +373 -0
  217. package/dist/tools/cli-executor.d.ts.map +1 -0
  218. package/dist/tools/cli-executor.js +1625 -0
  219. package/dist/tools/cli-executor.js.map +1 -0
  220. package/dist/tools/cli-history-store.d.ts +330 -0
  221. package/dist/tools/cli-history-store.d.ts.map +1 -0
  222. package/dist/tools/cli-history-store.js +916 -0
  223. package/dist/tools/cli-history-store.js.map +1 -0
  224. package/dist/tools/codex-lens.d.ts +118 -0
  225. package/dist/tools/codex-lens.d.ts.map +1 -0
  226. package/dist/tools/codex-lens.js +962 -0
  227. package/dist/tools/codex-lens.js.map +1 -0
  228. package/dist/tools/convert-tokens-to-css.d.ts +14 -0
  229. package/dist/tools/convert-tokens-to-css.d.ts.map +1 -0
  230. package/dist/tools/convert-tokens-to-css.js +244 -0
  231. package/dist/tools/convert-tokens-to-css.js.map +1 -0
  232. package/dist/tools/core-memory.d.ts +66 -0
  233. package/dist/tools/core-memory.d.ts.map +1 -0
  234. package/dist/tools/core-memory.js +324 -0
  235. package/dist/tools/core-memory.js.map +1 -0
  236. package/dist/tools/detect-changed-modules.d.ts +24 -0
  237. package/dist/tools/detect-changed-modules.d.ts.map +1 -0
  238. package/dist/tools/detect-changed-modules.js +277 -0
  239. package/dist/tools/detect-changed-modules.js.map +1 -0
  240. package/dist/tools/discover-design-files.d.ts +36 -0
  241. package/dist/tools/discover-design-files.d.ts.map +1 -0
  242. package/dist/tools/discover-design-files.js +147 -0
  243. package/dist/tools/discover-design-files.js.map +1 -0
  244. package/dist/tools/edit-file.d.ts +28 -0
  245. package/dist/tools/edit-file.d.ts.map +1 -0
  246. package/dist/tools/edit-file.js +479 -0
  247. package/dist/tools/edit-file.js.map +1 -0
  248. package/dist/tools/generate-module-docs.d.ts +22 -0
  249. package/dist/tools/generate-module-docs.d.ts.map +1 -0
  250. package/dist/tools/generate-module-docs.js +379 -0
  251. package/dist/tools/generate-module-docs.js.map +1 -0
  252. package/dist/tools/get-modules-by-depth.d.ts +15 -0
  253. package/dist/tools/get-modules-by-depth.d.ts.map +1 -0
  254. package/dist/tools/get-modules-by-depth.js +296 -0
  255. package/dist/tools/get-modules-by-depth.js.map +1 -0
  256. package/dist/tools/index.d.ts +55 -0
  257. package/dist/tools/index.d.ts.map +1 -0
  258. package/dist/tools/index.js +304 -0
  259. package/dist/tools/index.js.map +1 -0
  260. package/dist/tools/native-session-discovery.d.ts +97 -0
  261. package/dist/tools/native-session-discovery.d.ts.map +1 -0
  262. package/dist/tools/native-session-discovery.js +700 -0
  263. package/dist/tools/native-session-discovery.js.map +1 -0
  264. package/dist/tools/notifier.d.ts +50 -0
  265. package/dist/tools/notifier.d.ts.map +1 -0
  266. package/dist/tools/notifier.js +90 -0
  267. package/dist/tools/notifier.js.map +1 -0
  268. package/dist/tools/read-file.d.ts +32 -0
  269. package/dist/tools/read-file.d.ts.map +1 -0
  270. package/dist/tools/read-file.js +329 -0
  271. package/dist/tools/read-file.js.map +1 -0
  272. package/dist/tools/resume-strategy.d.ts +48 -0
  273. package/dist/tools/resume-strategy.d.ts.map +1 -0
  274. package/dist/tools/resume-strategy.js +248 -0
  275. package/dist/tools/resume-strategy.js.map +1 -0
  276. package/dist/tools/session-content-parser.d.ts +58 -0
  277. package/dist/tools/session-content-parser.d.ts.map +1 -0
  278. package/dist/tools/session-content-parser.js +420 -0
  279. package/dist/tools/session-content-parser.js.map +1 -0
  280. package/dist/tools/session-manager.d.ts +9 -0
  281. package/dist/tools/session-manager.d.ts.map +1 -0
  282. package/dist/tools/session-manager.js +834 -0
  283. package/dist/tools/session-manager.js.map +1 -0
  284. package/dist/tools/smart-context.d.ts +35 -0
  285. package/dist/tools/smart-context.d.ts.map +1 -0
  286. package/dist/tools/smart-context.js +182 -0
  287. package/dist/tools/smart-context.js.map +1 -0
  288. package/dist/tools/smart-search.d.ts +105 -0
  289. package/dist/tools/smart-search.d.ts.map +1 -0
  290. package/dist/tools/smart-search.js +1753 -0
  291. package/dist/tools/smart-search.js.map +1 -0
  292. package/dist/tools/storage-manager.d.ts +114 -0
  293. package/dist/tools/storage-manager.d.ts.map +1 -0
  294. package/dist/tools/storage-manager.js +392 -0
  295. package/dist/tools/storage-manager.js.map +1 -0
  296. package/dist/tools/ui-generate-preview.d.ts +39 -0
  297. package/dist/tools/ui-generate-preview.d.ts.map +1 -0
  298. package/dist/tools/ui-generate-preview.js +300 -0
  299. package/dist/tools/ui-generate-preview.js.map +1 -0
  300. package/dist/tools/ui-instantiate-prototypes.d.ts +75 -0
  301. package/dist/tools/ui-instantiate-prototypes.d.ts.map +1 -0
  302. package/dist/tools/ui-instantiate-prototypes.js +256 -0
  303. package/dist/tools/ui-instantiate-prototypes.js.map +1 -0
  304. package/dist/tools/update-module-claude.d.ts +80 -0
  305. package/dist/tools/update-module-claude.d.ts.map +1 -0
  306. package/dist/tools/update-module-claude.js +351 -0
  307. package/dist/tools/update-module-claude.js.map +1 -0
  308. package/dist/tools/write-file.d.ts +19 -0
  309. package/dist/tools/write-file.d.ts.map +1 -0
  310. package/dist/tools/write-file.js +193 -0
  311. package/dist/tools/write-file.js.map +1 -0
  312. package/dist/types/config.d.ts +11 -0
  313. package/dist/types/config.d.ts.map +1 -0
  314. package/dist/types/config.js +2 -0
  315. package/dist/types/config.js.map +1 -0
  316. package/dist/types/index.d.ts +4 -0
  317. package/dist/types/index.d.ts.map +1 -0
  318. package/dist/types/index.js +4 -0
  319. package/dist/types/index.js.map +1 -0
  320. package/dist/types/session.d.ts +20 -0
  321. package/dist/types/session.d.ts.map +1 -0
  322. package/dist/types/session.js +2 -0
  323. package/dist/types/session.js.map +1 -0
  324. package/dist/types/tool.d.ts +36 -0
  325. package/dist/types/tool.d.ts.map +1 -0
  326. package/dist/types/tool.js +11 -0
  327. package/dist/types/tool.js.map +1 -0
  328. package/dist/utils/browser-launcher.d.ts +13 -0
  329. package/dist/utils/browser-launcher.d.ts.map +1 -0
  330. package/dist/utils/browser-launcher.js +60 -0
  331. package/dist/utils/browser-launcher.js.map +1 -0
  332. package/dist/utils/file-utils.d.ts +25 -0
  333. package/dist/utils/file-utils.d.ts.map +1 -0
  334. package/dist/utils/file-utils.js +48 -0
  335. package/dist/utils/file-utils.js.map +1 -0
  336. package/dist/utils/path-resolver.d.ts +80 -0
  337. package/dist/utils/path-resolver.d.ts.map +1 -0
  338. package/dist/utils/path-resolver.js +260 -0
  339. package/dist/utils/path-resolver.js.map +1 -0
  340. package/dist/utils/path-validator.d.ts +49 -0
  341. package/dist/utils/path-validator.d.ts.map +1 -0
  342. package/dist/utils/path-validator.js +123 -0
  343. package/dist/utils/path-validator.js.map +1 -0
  344. package/dist/utils/ui.d.ts +62 -0
  345. package/dist/utils/ui.d.ts.map +1 -0
  346. package/dist/utils/ui.js +129 -0
  347. package/dist/utils/ui.js.map +1 -0
  348. package/package.json +65 -67
  349. package/src/.workflow/.cli-history/history.db +0 -0
  350. package/src/.workflow/.cli-history/history.db-shm +0 -0
  351. package/src/.workflow/.cli-history/history.db-wal +0 -0
  352. package/src/cli.ts +244 -0
  353. package/src/commands/cli.ts +740 -0
  354. package/src/commands/core-memory.ts +770 -0
  355. package/src/commands/hook.ts +315 -0
  356. package/src/commands/install.ts +519 -0
  357. package/src/commands/list.ts +37 -0
  358. package/src/commands/memory.ts +1090 -0
  359. package/src/commands/serve.ts +76 -0
  360. package/src/commands/session-path-resolver.ts +372 -0
  361. package/src/commands/session.ts +1141 -0
  362. package/src/commands/stop.ts +111 -0
  363. package/src/commands/tool.ts +201 -0
  364. package/src/commands/uninstall.ts +287 -0
  365. package/src/commands/upgrade.ts +352 -0
  366. package/src/commands/view.ts +119 -0
  367. package/src/config/storage-paths.ts +670 -0
  368. package/src/core/cache-manager.ts +294 -0
  369. package/src/core/claude-freshness.ts +319 -0
  370. package/src/core/core-memory-store.ts +1528 -0
  371. package/src/core/dashboard-generator-patch.ts +47 -0
  372. package/src/core/dashboard-generator.ts +739 -0
  373. package/src/core/data-aggregator.ts +584 -0
  374. package/src/core/history-importer.ts +625 -0
  375. package/src/core/lite-scanner-complete.ts +469 -0
  376. package/src/core/lite-scanner.ts +469 -0
  377. package/src/core/manifest.ts +271 -0
  378. package/src/core/memory-embedder-bridge.ts +262 -0
  379. package/src/core/memory-store.ts +978 -0
  380. package/src/core/routes/ccw-routes.ts +96 -0
  381. package/src/core/routes/claude-routes.ts +1183 -0
  382. package/src/core/routes/cli-routes.ts +561 -0
  383. package/src/core/routes/codexlens-routes.ts +806 -0
  384. package/src/core/routes/core-memory-routes.ts +605 -0
  385. package/src/core/routes/files-routes.ts +428 -0
  386. package/src/core/routes/graph-routes.md +164 -0
  387. package/src/core/routes/graph-routes.ts +626 -0
  388. package/src/core/routes/help-routes.ts +308 -0
  389. package/src/core/routes/hooks-routes.ts +405 -0
  390. package/src/core/routes/mcp-routes.ts +1271 -0
  391. package/src/core/routes/mcp-routes.ts.backup +550 -0
  392. package/src/core/routes/mcp-templates-db.ts +268 -0
  393. package/src/core/routes/memory-routes.ts +1206 -0
  394. package/src/core/routes/rules-routes.ts +526 -0
  395. package/src/core/routes/session-routes.ts +467 -0
  396. package/src/core/routes/skills-routes.ts +599 -0
  397. package/src/core/routes/status-routes.ts +57 -0
  398. package/src/core/routes/system-routes.ts +427 -0
  399. package/src/core/server.ts +431 -0
  400. package/src/core/session-clustering-service.ts +1258 -0
  401. package/src/core/session-scanner.ts +283 -0
  402. package/src/core/websocket.ts +190 -0
  403. package/src/index.ts +10 -0
  404. package/src/mcp-server/index.ts +186 -0
  405. package/src/templates/assets/css/github-dark.min.css +10 -0
  406. package/src/templates/assets/css/github.min.css +10 -0
  407. package/src/templates/assets/js/cytoscape.min.js +32 -0
  408. package/src/templates/assets/js/d3.min.js +2 -0
  409. package/src/templates/assets/js/highlight.min.js +1244 -0
  410. package/src/templates/assets/js/lucide.min.js +12 -0
  411. package/src/templates/assets/js/marked.min.js +69 -0
  412. package/src/templates/assets/js/tailwind.js +83 -0
  413. package/src/templates/dashboard-css/01-base.css +302 -0
  414. package/src/templates/dashboard-css/02-session.css +748 -0
  415. package/src/templates/dashboard-css/04-lite-tasks.css +1181 -0
  416. package/src/templates/dashboard-css/06-cards.css +1576 -0
  417. package/src/templates/dashboard-css/07-managers.css +2107 -0
  418. package/src/templates/dashboard-css/09-explorer.css +1408 -0
  419. package/src/templates/dashboard-css/10-cli-status.css +337 -0
  420. package/src/templates/dashboard-css/11-cli-history.css +271 -0
  421. package/src/templates/dashboard-css/12-cli-legacy.css +796 -0
  422. package/src/templates/dashboard-css/13-cli-ccw.css +199 -0
  423. package/src/templates/dashboard-css/14-cli-modals.css +258 -0
  424. package/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
  425. package/src/templates/dashboard-css/16-cli-session.css +241 -0
  426. package/src/templates/dashboard-css/17-cli-conversation.css +283 -0
  427. package/src/templates/dashboard-css/18-cli-settings.css +160 -0
  428. package/src/templates/dashboard-css/19-cli-native-session.css +496 -0
  429. package/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
  430. package/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
  431. package/src/templates/dashboard-css/22-cli-semantic.css +240 -0
  432. package/src/templates/dashboard-css/23-memory.css +2390 -0
  433. package/src/templates/dashboard-css/24-prompt-history.css +1089 -0
  434. package/src/templates/dashboard-css/25-skills-rules.css +326 -0
  435. package/src/templates/dashboard-css/26-claude-manager.css +908 -0
  436. package/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
  437. package/src/templates/dashboard-css/28-mcp-manager.css +748 -0
  438. package/src/templates/dashboard-css/29-help.css +264 -0
  439. package/src/templates/dashboard-css/30-core-memory.css +1700 -0
  440. package/src/templates/dashboard-js/api.js +220 -0
  441. package/src/templates/dashboard-js/components/carousel.js +398 -0
  442. package/src/templates/dashboard-js/components/cli-history.js +876 -0
  443. package/src/templates/dashboard-js/components/cli-status.js +978 -0
  444. package/src/templates/dashboard-js/components/global-notifications.js +508 -0
  445. package/src/templates/dashboard-js/components/hook-manager.js +1278 -0
  446. package/src/templates/dashboard-js/components/index-manager.js +302 -0
  447. package/src/templates/dashboard-js/components/mcp-manager.js +1219 -0
  448. package/src/templates/dashboard-js/components/modals.js +326 -0
  449. package/src/templates/dashboard-js/components/navigation.js +313 -0
  450. package/src/templates/dashboard-js/components/notifications.js +758 -0
  451. package/src/templates/dashboard-js/components/storage-manager.js +478 -0
  452. package/src/templates/dashboard-js/components/tabs-other.js +424 -0
  453. package/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
  454. package/src/templates/dashboard-js/help-i18n.js +272 -0
  455. package/src/templates/dashboard-js/i18n.js +2807 -0
  456. package/src/templates/dashboard-js/main.js +87 -0
  457. package/src/templates/dashboard-js/state.js +243 -0
  458. package/src/templates/dashboard-js/utils.js +199 -0
  459. package/src/templates/dashboard-js/views/claude-manager.js +912 -0
  460. package/src/templates/dashboard-js/views/cli-manager.js +2272 -0
  461. package/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
  462. package/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
  463. package/src/templates/dashboard-js/views/core-memory.js +782 -0
  464. package/src/templates/dashboard-js/views/explorer.js +888 -0
  465. package/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
  466. package/src/templates/dashboard-js/views/help.js +856 -0
  467. package/src/templates/dashboard-js/views/history.js +337 -0
  468. package/src/templates/dashboard-js/views/home.js +243 -0
  469. package/src/templates/dashboard-js/views/hook-manager.js +660 -0
  470. package/src/templates/dashboard-js/views/lite-tasks.js +861 -0
  471. package/src/templates/dashboard-js/views/mcp-manager.js +2187 -0
  472. package/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
  473. package/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
  474. package/src/templates/dashboard-js/views/memory.js +1221 -0
  475. package/src/templates/dashboard-js/views/prompt-history.js +713 -0
  476. package/src/templates/dashboard-js/views/rules-manager.js +828 -0
  477. package/src/templates/dashboard-js/views/session-detail.js +781 -0
  478. package/src/templates/dashboard-js/views/skills-manager.js +819 -0
  479. package/src/templates/dashboard.html +831 -0
  480. package/src/templates/hooks-config-example.json +60 -0
  481. package/src/tools/classify-folders.ts +245 -0
  482. package/src/tools/cli-config-manager.ts +268 -0
  483. package/src/tools/cli-executor.ts +2014 -0
  484. package/src/tools/cli-history-store.ts +1195 -0
  485. package/src/tools/codex-lens.ts +1141 -0
  486. package/src/tools/convert-tokens-to-css.ts +300 -0
  487. package/src/tools/core-memory.ts +444 -0
  488. package/src/tools/detect-changed-modules.ts +325 -0
  489. package/src/tools/discover-design-files.ts +184 -0
  490. package/src/tools/edit-file.ts +568 -0
  491. package/src/tools/generate-module-docs.ts +438 -0
  492. package/src/tools/get-modules-by-depth.ts +349 -0
  493. package/src/tools/index.ts +370 -0
  494. package/src/tools/native-session-discovery.ts +795 -0
  495. package/src/tools/notifier.ts +129 -0
  496. package/src/tools/read-file.ts +410 -0
  497. package/src/tools/resume-strategy.ts +345 -0
  498. package/src/tools/session-content-parser.ts +619 -0
  499. package/src/tools/session-manager.ts +1026 -0
  500. package/src/tools/smart-context.ts +228 -0
  501. package/src/tools/smart-search.ts +2065 -0
  502. package/src/tools/smart-search.ts.backup +1233 -0
  503. package/src/tools/storage-manager.ts +455 -0
  504. package/src/tools/write-file.ts +222 -0
  505. package/src/types/config.ts +11 -0
  506. package/src/types/index.ts +3 -0
  507. package/src/types/session.ts +25 -0
  508. package/src/types/tool.ts +41 -0
  509. package/src/utils/browser-launcher.ts +62 -0
  510. package/src/utils/file-utils.ts +48 -0
  511. package/src/utils/path-resolver.ts +315 -0
  512. package/src/utils/path-validator.ts +153 -0
  513. package/src/utils/ui.ts +155 -0
  514. package/.claude/agents/action-planning-agent.md +0 -778
  515. package/.claude/agents/cli-execution-agent.md +0 -270
  516. package/.claude/agents/cli-explore-agent.md +0 -182
  517. package/.claude/agents/cli-lite-planning-agent.md +0 -396
  518. package/.claude/agents/cli-planning-agent.md +0 -558
  519. package/.claude/agents/code-developer.md +0 -310
  520. package/.claude/agents/conceptual-planning-agent.md +0 -308
  521. package/.claude/agents/context-search-agent.md +0 -582
  522. package/.claude/agents/doc-generator.md +0 -330
  523. package/.claude/agents/memory-bridge.md +0 -94
  524. package/.claude/agents/test-context-search-agent.md +0 -399
  525. package/.claude/agents/test-fix-agent.md +0 -343
  526. package/.claude/agents/ui-design-agent.md +0 -593
  527. package/.claude/agents/universal-executor.md +0 -131
  528. package/.claude/commands/cli/cli-init.md +0 -440
  529. package/.claude/commands/enhance-prompt.md +0 -93
  530. package/.claude/commands/memory/code-map-memory.md +0 -687
  531. package/.claude/commands/memory/docs-full-cli.md +0 -471
  532. package/.claude/commands/memory/docs-related-cli.md +0 -386
  533. package/.claude/commands/memory/docs.md +0 -615
  534. package/.claude/commands/memory/load-skill-memory.md +0 -182
  535. package/.claude/commands/memory/load.md +0 -240
  536. package/.claude/commands/memory/skill-memory.md +0 -525
  537. package/.claude/commands/memory/style-skill-memory.md +0 -396
  538. package/.claude/commands/memory/tech-research.md +0 -477
  539. package/.claude/commands/memory/update-full.md +0 -332
  540. package/.claude/commands/memory/update-related.md +0 -332
  541. package/.claude/commands/memory/workflow-skill-memory.md +0 -517
  542. package/.claude/commands/task/breakdown.md +0 -204
  543. package/.claude/commands/task/create.md +0 -152
  544. package/.claude/commands/task/execute.md +0 -270
  545. package/.claude/commands/task/replan.md +0 -437
  546. package/.claude/commands/version.md +0 -254
  547. package/.claude/commands/workflow/action-plan-verify.md +0 -447
  548. package/.claude/commands/workflow/brainstorm/api-designer.md +0 -585
  549. package/.claude/commands/workflow/brainstorm/artifacts.md +0 -452
  550. package/.claude/commands/workflow/brainstorm/auto-parallel.md +0 -443
  551. package/.claude/commands/workflow/brainstorm/data-architect.md +0 -220
  552. package/.claude/commands/workflow/brainstorm/product-manager.md +0 -200
  553. package/.claude/commands/workflow/brainstorm/product-owner.md +0 -200
  554. package/.claude/commands/workflow/brainstorm/scrum-master.md +0 -200
  555. package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +0 -200
  556. package/.claude/commands/workflow/brainstorm/synthesis.md +0 -398
  557. package/.claude/commands/workflow/brainstorm/system-architect.md +0 -387
  558. package/.claude/commands/workflow/brainstorm/ui-designer.md +0 -221
  559. package/.claude/commands/workflow/brainstorm/ux-expert.md +0 -221
  560. package/.claude/commands/workflow/execute.md +0 -460
  561. package/.claude/commands/workflow/init.md +0 -164
  562. package/.claude/commands/workflow/lite-execute.md +0 -686
  563. package/.claude/commands/workflow/lite-fix.md +0 -621
  564. package/.claude/commands/workflow/lite-plan.md +0 -592
  565. package/.claude/commands/workflow/plan.md +0 -551
  566. package/.claude/commands/workflow/replan.md +0 -515
  567. package/.claude/commands/workflow/review-fix.md +0 -606
  568. package/.claude/commands/workflow/review-module-cycle.md +0 -765
  569. package/.claude/commands/workflow/review-session-cycle.md +0 -776
  570. package/.claude/commands/workflow/review.md +0 -291
  571. package/.claude/commands/workflow/session/complete.md +0 -500
  572. package/.claude/commands/workflow/session/list.md +0 -96
  573. package/.claude/commands/workflow/session/resume.md +0 -61
  574. package/.claude/commands/workflow/session/start.md +0 -200
  575. package/.claude/commands/workflow/tdd-plan.md +0 -460
  576. package/.claude/commands/workflow/tdd-verify.md +0 -386
  577. package/.claude/commands/workflow/test-cycle-execute.md +0 -498
  578. package/.claude/commands/workflow/test-fix-gen.md +0 -699
  579. package/.claude/commands/workflow/test-gen.md +0 -529
  580. package/.claude/commands/workflow/tools/conflict-resolution.md +0 -680
  581. package/.claude/commands/workflow/tools/context-gather.md +0 -434
  582. package/.claude/commands/workflow/tools/task-generate-agent.md +0 -291
  583. package/.claude/commands/workflow/tools/task-generate-tdd.md +0 -518
  584. package/.claude/commands/workflow/tools/tdd-coverage-analysis.md +0 -309
  585. package/.claude/commands/workflow/tools/test-concept-enhanced.md +0 -163
  586. package/.claude/commands/workflow/tools/test-context-gather.md +0 -235
  587. package/.claude/commands/workflow/tools/test-task-generate.md +0 -256
  588. package/.claude/commands/workflow/ui-design/animation-extract.md +0 -1150
  589. package/.claude/commands/workflow/ui-design/codify-style.md +0 -652
  590. package/.claude/commands/workflow/ui-design/design-sync.md +0 -454
  591. package/.claude/commands/workflow/ui-design/explore-auto.md +0 -678
  592. package/.claude/commands/workflow/ui-design/generate.md +0 -504
  593. package/.claude/commands/workflow/ui-design/imitate-auto.md +0 -745
  594. package/.claude/commands/workflow/ui-design/import-from-code.md +0 -537
  595. package/.claude/commands/workflow/ui-design/layout-extract.md +0 -788
  596. package/.claude/commands/workflow/ui-design/reference-page-generator.md +0 -356
  597. package/.claude/commands/workflow/ui-design/style-extract.md +0 -773
  598. package/.claude/scripts/classify-folders.sh +0 -39
  599. package/.claude/scripts/convert_tokens_to_css.sh +0 -229
  600. package/.claude/scripts/detect_changed_modules.sh +0 -161
  601. package/.claude/scripts/discover-design-files.sh +0 -87
  602. package/.claude/scripts/extract-animations.js +0 -243
  603. package/.claude/scripts/extract-computed-styles.js +0 -118
  604. package/.claude/scripts/extract-layout-structure.js +0 -411
  605. package/.claude/scripts/generate_module_docs.sh +0 -717
  606. package/.claude/scripts/get_modules_by_depth.sh +0 -170
  607. package/.claude/scripts/ui-generate-preview.sh +0 -395
  608. package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
  609. package/.claude/scripts/update_module_claude.sh +0 -337
  610. package/.claude/skills/command-guide/SKILL.md +0 -388
  611. package/.claude/skills/command-guide/UPDATE-GUIDELINE.md +0 -592
  612. package/.claude/skills/command-guide/guides/cli-tools-guide.md +0 -410
  613. package/.claude/skills/command-guide/guides/examples.md +0 -537
  614. package/.claude/skills/command-guide/guides/getting-started.md +0 -242
  615. package/.claude/skills/command-guide/guides/implementation-details.md +0 -1010
  616. package/.claude/skills/command-guide/guides/index-structure.md +0 -326
  617. package/.claude/skills/command-guide/guides/troubleshooting.md +0 -92
  618. package/.claude/skills/command-guide/guides/ui-design-workflow-guide.md +0 -316
  619. package/.claude/skills/command-guide/guides/workflow-patterns.md +0 -662
  620. package/.claude/skills/command-guide/index/all-commands.json +0 -772
  621. package/.claude/skills/command-guide/index/by-category.json +0 -800
  622. package/.claude/skills/command-guide/index/by-use-case.json +0 -786
  623. package/.claude/skills/command-guide/index/command-relationships.json +0 -307
  624. package/.claude/skills/command-guide/index/essential-commands.json +0 -112
  625. package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +0 -778
  626. package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +0 -270
  627. package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +0 -182
  628. package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +0 -396
  629. package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +0 -558
  630. package/.claude/skills/command-guide/reference/agents/code-developer.md +0 -310
  631. package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +0 -308
  632. package/.claude/skills/command-guide/reference/agents/context-search-agent.md +0 -582
  633. package/.claude/skills/command-guide/reference/agents/doc-generator.md +0 -330
  634. package/.claude/skills/command-guide/reference/agents/memory-bridge.md +0 -94
  635. package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +0 -399
  636. package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +0 -343
  637. package/.claude/skills/command-guide/reference/agents/ui-design-agent.md +0 -593
  638. package/.claude/skills/command-guide/reference/agents/universal-executor.md +0 -131
  639. package/.claude/skills/command-guide/reference/commands/cli/cli-init.md +0 -440
  640. package/.claude/skills/command-guide/reference/commands/enhance-prompt.md +0 -93
  641. package/.claude/skills/command-guide/reference/commands/memory/code-map-memory.md +0 -687
  642. package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +0 -471
  643. package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +0 -386
  644. package/.claude/skills/command-guide/reference/commands/memory/docs.md +0 -615
  645. package/.claude/skills/command-guide/reference/commands/memory/load-skill-memory.md +0 -182
  646. package/.claude/skills/command-guide/reference/commands/memory/load.md +0 -240
  647. package/.claude/skills/command-guide/reference/commands/memory/skill-memory.md +0 -525
  648. package/.claude/skills/command-guide/reference/commands/memory/style-skill-memory.md +0 -396
  649. package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +0 -477
  650. package/.claude/skills/command-guide/reference/commands/memory/update-full.md +0 -332
  651. package/.claude/skills/command-guide/reference/commands/memory/update-related.md +0 -332
  652. package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +0 -517
  653. package/.claude/skills/command-guide/reference/commands/task/breakdown.md +0 -204
  654. package/.claude/skills/command-guide/reference/commands/task/create.md +0 -152
  655. package/.claude/skills/command-guide/reference/commands/task/execute.md +0 -270
  656. package/.claude/skills/command-guide/reference/commands/task/replan.md +0 -437
  657. package/.claude/skills/command-guide/reference/commands/version.md +0 -254
  658. package/.claude/skills/command-guide/reference/commands/workflow/action-plan-verify.md +0 -447
  659. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +0 -585
  660. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/artifacts.md +0 -452
  661. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +0 -443
  662. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +0 -220
  663. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +0 -200
  664. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +0 -200
  665. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +0 -200
  666. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +0 -200
  667. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/synthesis.md +0 -398
  668. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +0 -387
  669. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +0 -221
  670. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +0 -221
  671. package/.claude/skills/command-guide/reference/commands/workflow/execute.md +0 -460
  672. package/.claude/skills/command-guide/reference/commands/workflow/init.md +0 -164
  673. package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +0 -686
  674. package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +0 -621
  675. package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +0 -592
  676. package/.claude/skills/command-guide/reference/commands/workflow/plan.md +0 -551
  677. package/.claude/skills/command-guide/reference/commands/workflow/replan.md +0 -515
  678. package/.claude/skills/command-guide/reference/commands/workflow/review-fix.md +0 -606
  679. package/.claude/skills/command-guide/reference/commands/workflow/review-module-cycle.md +0 -765
  680. package/.claude/skills/command-guide/reference/commands/workflow/review-session-cycle.md +0 -776
  681. package/.claude/skills/command-guide/reference/commands/workflow/review.md +0 -291
  682. package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +0 -500
  683. package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +0 -96
  684. package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +0 -61
  685. package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +0 -200
  686. package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +0 -460
  687. package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +0 -386
  688. package/.claude/skills/command-guide/reference/commands/workflow/test-cycle-execute.md +0 -498
  689. package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +0 -699
  690. package/.claude/skills/command-guide/reference/commands/workflow/test-gen.md +0 -529
  691. package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +0 -680
  692. package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +0 -434
  693. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +0 -291
  694. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +0 -518
  695. package/.claude/skills/command-guide/reference/commands/workflow/tools/tdd-coverage-analysis.md +0 -309
  696. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +0 -163
  697. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-context-gather.md +0 -235
  698. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-task-generate.md +0 -256
  699. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/animation-extract.md +0 -1150
  700. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/codify-style.md +0 -652
  701. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/design-sync.md +0 -454
  702. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/explore-auto.md +0 -678
  703. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/generate.md +0 -504
  704. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/imitate-auto.md +0 -745
  705. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +0 -537
  706. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/layout-extract.md +0 -788
  707. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/reference-page-generator.md +0 -356
  708. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/style-extract.md +0 -773
  709. package/.claude/skills/command-guide/scripts/analyze_commands.py +0 -502
  710. package/.claude/skills/command-guide/scripts/update-index.sh +0 -130
  711. package/.claude/skills/command-guide/templates/issue-bug.md +0 -104
  712. package/.claude/skills/command-guide/templates/issue-diagnosis.md +0 -275
  713. package/.claude/skills/command-guide/templates/issue-feature.md +0 -97
  714. package/.claude/skills/command-guide/templates/issue-question.md +0 -141
  715. package/.claude/skills/prompt-enhancer/SKILL.md +0 -124
  716. package/.claude/workflows/_template-compare-matrix.html +0 -692
  717. package/.claude/workflows/cli-templates/fix-plan-template.json +0 -75
  718. package/.claude/workflows/cli-templates/fix-progress-template.json +0 -48
  719. package/.claude/workflows/cli-templates/memory/style-skill-memory/skill-md-template.md +0 -299
  720. package/.claude/workflows/cli-templates/planning-roles/data-architect.md +0 -120
  721. package/.claude/workflows/cli-templates/planning-roles/product-manager.md +0 -119
  722. package/.claude/workflows/cli-templates/planning-roles/product-owner.md +0 -261
  723. package/.claude/workflows/cli-templates/planning-roles/scrum-master.md +0 -186
  724. package/.claude/workflows/cli-templates/planning-roles/subject-matter-expert.md +0 -281
  725. package/.claude/workflows/cli-templates/planning-roles/synthesis-role.md +0 -414
  726. package/.claude/workflows/cli-templates/planning-roles/system-architect.md +0 -106
  727. package/.claude/workflows/cli-templates/planning-roles/test-strategist.md +0 -124
  728. package/.claude/workflows/cli-templates/planning-roles/ui-designer.md +0 -379
  729. package/.claude/workflows/cli-templates/planning-roles/ux-expert.md +0 -240
  730. package/.claude/workflows/cli-templates/prompts/analysis/01-diagnose-bug-root-cause.txt +0 -127
  731. package/.claude/workflows/cli-templates/prompts/analysis/01-trace-code-execution.txt +0 -115
  732. package/.claude/workflows/cli-templates/prompts/analysis/02-analyze-code-patterns.txt +0 -37
  733. package/.claude/workflows/cli-templates/prompts/analysis/02-analyze-technical-document.txt +0 -33
  734. package/.claude/workflows/cli-templates/prompts/analysis/02-review-architecture.txt +0 -29
  735. package/.claude/workflows/cli-templates/prompts/analysis/02-review-code-quality.txt +0 -28
  736. package/.claude/workflows/cli-templates/prompts/analysis/03-analyze-performance.txt +0 -29
  737. package/.claude/workflows/cli-templates/prompts/analysis/03-assess-security-risks.txt +0 -29
  738. package/.claude/workflows/cli-templates/prompts/analysis/03-review-quality-standards.txt +0 -29
  739. package/.claude/workflows/cli-templates/prompts/development/02-generate-tests.txt +0 -70
  740. package/.claude/workflows/cli-templates/prompts/development/02-implement-component-ui.txt +0 -55
  741. package/.claude/workflows/cli-templates/prompts/development/02-implement-feature.txt +0 -58
  742. package/.claude/workflows/cli-templates/prompts/development/02-refactor-codebase.txt +0 -55
  743. package/.claude/workflows/cli-templates/prompts/development/03-debug-runtime-issues.txt +0 -55
  744. package/.claude/workflows/cli-templates/prompts/documentation/api.txt +0 -15
  745. package/.claude/workflows/cli-templates/prompts/documentation/folder-navigation.txt +0 -27
  746. package/.claude/workflows/cli-templates/prompts/documentation/module-readme.txt +0 -49
  747. package/.claude/workflows/cli-templates/prompts/documentation/project-architecture.txt +0 -41
  748. package/.claude/workflows/cli-templates/prompts/documentation/project-examples.txt +0 -35
  749. package/.claude/workflows/cli-templates/prompts/documentation/project-readme.txt +0 -35
  750. package/.claude/workflows/cli-templates/prompts/memory/02-document-module-structure.txt +0 -165
  751. package/.claude/workflows/cli-templates/prompts/planning/01-plan-architecture-design.txt +0 -109
  752. package/.claude/workflows/cli-templates/prompts/planning/02-breakdown-task-steps.txt +0 -30
  753. package/.claude/workflows/cli-templates/prompts/planning/02-design-component-spec.txt +0 -28
  754. package/.claude/workflows/cli-templates/prompts/planning/03-evaluate-concept-feasibility.txt +0 -127
  755. package/.claude/workflows/cli-templates/prompts/planning/03-plan-migration-strategy.txt +0 -30
  756. package/.claude/workflows/cli-templates/prompts/tech/tech-module-format.txt +0 -359
  757. package/.claude/workflows/cli-templates/prompts/tech/tech-skill-index.txt +0 -185
  758. package/.claude/workflows/cli-templates/prompts/test/test-concept-analysis.txt +0 -179
  759. package/.claude/workflows/cli-templates/prompts/universal/00-universal-creative-style.txt +0 -95
  760. package/.claude/workflows/cli-templates/prompts/universal/00-universal-rigorous-style.txt +0 -92
  761. package/.claude/workflows/cli-templates/prompts/verification/codex-technical.txt +0 -28
  762. package/.claude/workflows/cli-templates/prompts/verification/cross-validation.txt +0 -28
  763. package/.claude/workflows/cli-templates/prompts/verification/gemini-strategic.txt +0 -27
  764. package/.claude/workflows/cli-templates/prompts/workflow/analysis-results-structure.txt +0 -224
  765. package/.claude/workflows/cli-templates/prompts/workflow/codex-feasibility-validation.txt +0 -176
  766. package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +0 -131
  767. package/.claude/workflows/cli-templates/prompts/workflow/impl-plan-template.txt +0 -286
  768. package/.claude/workflows/cli-templates/prompts/workflow/skill-aggregation.txt +0 -172
  769. package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +0 -98
  770. package/.claude/workflows/cli-templates/prompts/workflow/skill-index.txt +0 -224
  771. package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +0 -98
  772. package/.claude/workflows/cli-templates/prompts/workflow/skill-sessions-timeline.txt +0 -53
  773. package/.claude/workflows/cli-templates/prompts/workflow/task-json-agent-mode.txt +0 -123
  774. package/.claude/workflows/cli-templates/prompts/workflow/task-json-cli-mode.txt +0 -182
  775. package/.claude/workflows/cli-templates/schemas/diagnosis-json-schema.json +0 -234
  776. package/.claude/workflows/cli-templates/schemas/explore-json-schema.json +0 -124
  777. package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +0 -273
  778. package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +0 -219
  779. package/.claude/workflows/cli-templates/schemas/project-json-schema.json +0 -221
  780. package/.claude/workflows/cli-templates/schemas/review-deep-dive-results-schema.json +0 -82
  781. package/.claude/workflows/cli-templates/schemas/review-dimension-results-schema.json +0 -51
  782. package/.claude/workflows/cli-templates/tech-stacks/go-dev.md +0 -91
  783. package/.claude/workflows/cli-templates/tech-stacks/java-dev.md +0 -107
  784. package/.claude/workflows/cli-templates/tech-stacks/javascript-dev.md +0 -58
  785. package/.claude/workflows/cli-templates/tech-stacks/python-dev.md +0 -79
  786. package/.claude/workflows/cli-templates/tech-stacks/react-dev.md +0 -103
  787. package/.claude/workflows/cli-templates/tech-stacks/typescript-dev.md +0 -83
  788. package/.claude/workflows/cli-templates/ui-design/systems/animation-tokens.json +0 -247
  789. package/.claude/workflows/cli-templates/ui-design/systems/design-tokens.json +0 -342
  790. package/.claude/workflows/cli-templates/ui-design/systems/layout-templates.json +0 -145
  791. package/.claude/workflows/context-search-strategy.md +0 -77
  792. package/.claude/workflows/intelligent-tools-strategy.md +0 -662
  793. package/.claude/workflows/review-directory-specification.md +0 -336
  794. package/.claude/workflows/task-core.md +0 -214
  795. package/.claude/workflows/tool-strategy.md +0 -79
  796. package/.claude/workflows/workflow-architecture.md +0 -942
  797. package/.codex/AGENTS.md +0 -330
  798. package/.gemini/GEMINI.md +0 -164
  799. package/.qwen/QWEN.md +0 -164
  800. package/CLAUDE.md +0 -91
  801. package/LICENSE +0 -21
  802. package/ccw/README.md +0 -121
  803. package/ccw/bin/ccw.js +0 -10
  804. package/ccw/package.json +0 -47
  805. package/ccw/src/cli.js +0 -119
  806. package/ccw/src/commands/install.js +0 -324
  807. package/ccw/src/commands/list.js +0 -37
  808. package/ccw/src/commands/serve.js +0 -67
  809. package/ccw/src/commands/stop.js +0 -101
  810. package/ccw/src/commands/tool.js +0 -138
  811. package/ccw/src/commands/uninstall.js +0 -238
  812. package/ccw/src/commands/upgrade.js +0 -307
  813. package/ccw/src/commands/view.js +0 -105
  814. package/ccw/src/core/dashboard-generator-patch.js +0 -29
  815. package/ccw/src/core/dashboard-generator.js +0 -682
  816. package/ccw/src/core/data-aggregator.js +0 -409
  817. package/ccw/src/core/lite-scanner.js +0 -314
  818. package/ccw/src/core/manifest.js +0 -201
  819. package/ccw/src/core/server.js +0 -2063
  820. package/ccw/src/core/session-scanner.js +0 -235
  821. package/ccw/src/index.js +0 -9
  822. package/ccw/src/templates/dashboard-css/01-base.css +0 -291
  823. package/ccw/src/templates/dashboard-css/02-session.css +0 -726
  824. package/ccw/src/templates/dashboard-css/04-lite-tasks.css +0 -843
  825. package/ccw/src/templates/dashboard-css/06-cards.css +0 -1570
  826. package/ccw/src/templates/dashboard-css/07-managers.css +0 -936
  827. package/ccw/src/templates/dashboard-css/09-explorer.css +0 -1397
  828. package/ccw/src/templates/dashboard-js/api.js +0 -200
  829. package/ccw/src/templates/dashboard-js/components/carousel.js +0 -398
  830. package/ccw/src/templates/dashboard-js/components/global-notifications.js +0 -219
  831. package/ccw/src/templates/dashboard-js/components/hook-manager.js +0 -283
  832. package/ccw/src/templates/dashboard-js/components/mcp-manager.js +0 -528
  833. package/ccw/src/templates/dashboard-js/components/modals.js +0 -260
  834. package/ccw/src/templates/dashboard-js/components/navigation.js +0 -245
  835. package/ccw/src/templates/dashboard-js/components/notifications.js +0 -194
  836. package/ccw/src/templates/dashboard-js/components/tabs-other.js +0 -273
  837. package/ccw/src/templates/dashboard-js/main.js +0 -72
  838. package/ccw/src/templates/dashboard-js/state.js +0 -42
  839. package/ccw/src/templates/dashboard-js/utils.js +0 -153
  840. package/ccw/src/templates/dashboard-js/views/explorer.js +0 -852
  841. package/ccw/src/templates/dashboard-js/views/home.js +0 -197
  842. package/ccw/src/templates/dashboard-js/views/hook-manager.js +0 -392
  843. package/ccw/src/templates/dashboard-js/views/lite-tasks.js +0 -395
  844. package/ccw/src/templates/dashboard-js/views/mcp-manager.js +0 -411
  845. package/ccw/src/templates/dashboard-js/views/session-detail.js +0 -780
  846. package/ccw/src/templates/dashboard.html +0 -731
  847. package/ccw/src/tools/classify-folders.js +0 -204
  848. package/ccw/src/tools/convert-tokens-to-css.js +0 -250
  849. package/ccw/src/tools/detect-changed-modules.js +0 -288
  850. package/ccw/src/tools/discover-design-files.js +0 -134
  851. package/ccw/src/tools/edit-file.js +0 -266
  852. package/ccw/src/tools/generate-module-docs.js +0 -416
  853. package/ccw/src/tools/get-modules-by-depth.js +0 -308
  854. package/ccw/src/tools/index.js +0 -176
  855. package/ccw/src/utils/browser-launcher.js +0 -60
  856. package/ccw/src/utils/file-utils.js +0 -48
  857. package/ccw/src/utils/path-resolver.js +0 -279
  858. package/ccw/src/utils/ui.js +0 -148
  859. /package/{ccw/src → src}/templates/dashboard-css/03-tasks.css +0 -0
  860. /package/{ccw/src → src}/templates/dashboard-css/05-context.css +0 -0
  861. /package/{ccw/src → src}/templates/dashboard-css/08-review.css +0 -0
  862. /package/{ccw/src → src}/templates/dashboard-js/components/_conflict_tab.js +0 -0
  863. /package/{ccw/src → src}/templates/dashboard-js/components/_exp_helpers.js +0 -0
  864. /package/{ccw/src → src}/templates/dashboard-js/components/_review_tab.js +0 -0
  865. /package/{ccw/src → src}/templates/dashboard-js/components/flowchart.js +0 -0
  866. /package/{ccw/src → src}/templates/dashboard-js/components/sidebar.js +0 -0
  867. /package/{ccw/src → src}/templates/dashboard-js/components/tabs-context.js +0 -0
  868. /package/{ccw/src → src}/templates/dashboard-js/components/task-drawer-core.js +0 -0
  869. /package/{ccw/src → src}/templates/dashboard-js/components/task-drawer-renderers.js +0 -0
  870. /package/{ccw/src → src}/templates/dashboard-js/components/theme.js +0 -0
  871. /package/{ccw/src → src}/templates/dashboard-js/components/version-check.js +0 -0
  872. /package/{ccw/src → src}/templates/dashboard-js/views/fix-session.js +0 -0
  873. /package/{ccw/src → src}/templates/dashboard-js/views/project-overview.js +0 -0
  874. /package/{ccw/src → src}/templates/dashboard-js/views/review-session.js +0 -0
  875. /package/{ccw/src → src}/templates/review-cycle-dashboard.html +0 -0
  876. /package/{ccw/src → src}/templates/workflow-dashboard.html +0 -0
  877. /package/{ccw/src → src}/tools/ui-generate-preview.js +0 -0
  878. /package/{ccw/src → src}/tools/ui-instantiate-prototypes.js +0 -0
  879. /package/{ccw/src → src}/tools/update-module-claude.js +0 -0
@@ -0,0 +1,2014 @@
1
+ /**
2
+ * CLI Executor Tool - Unified execution for external CLI tools
3
+ * Supports Gemini, Qwen, and Codex with streaming output
4
+ */
5
+
6
+ import { z } from 'zod';
7
+ import type { ToolSchema, ToolResult } from '../types/tool.js';
8
+ import { spawn, ChildProcess } from 'child_process';
9
+ import { existsSync, mkdirSync, readFileSync, writeFileSync, unlinkSync, readdirSync, statSync } from 'fs';
10
+ import { join, relative } from 'path';
11
+
12
+ // Native resume support
13
+ import {
14
+ trackNewSession,
15
+ getNativeResumeArgs,
16
+ supportsNativeResume,
17
+ calculateProjectHash
18
+ } from './native-session-discovery.js';
19
+ import {
20
+ determineResumeStrategy,
21
+ buildContextPrefix,
22
+ getResumeModeDescription,
23
+ type ResumeDecision
24
+ } from './resume-strategy.js';
25
+ import {
26
+ isToolEnabled as isToolEnabledFromConfig,
27
+ enableTool as enableToolFromConfig,
28
+ disableTool as disableToolFromConfig,
29
+ getPrimaryModel
30
+ } from './cli-config-manager.js';
31
+ import { StoragePaths, ensureStorageDir } from '../config/storage-paths.js';
32
+
33
+ // Lazy-loaded SQLite store module
34
+ let sqliteStoreModule: typeof import('./cli-history-store.js') | null = null;
35
+
36
+ /**
37
+ * Get or initialize SQLite store (async)
38
+ */
39
+ async function getSqliteStore(baseDir: string) {
40
+ if (!sqliteStoreModule) {
41
+ sqliteStoreModule = await import('./cli-history-store.js');
42
+ }
43
+ return sqliteStoreModule.getHistoryStore(baseDir);
44
+ }
45
+
46
+ /**
47
+ * Get SQLite store (sync - uses cached module)
48
+ */
49
+ function getSqliteStoreSync(baseDir: string) {
50
+ if (!sqliteStoreModule) {
51
+ throw new Error('SQLite store not initialized. Call an async function first.');
52
+ }
53
+ return sqliteStoreModule.getHistoryStore(baseDir);
54
+ }
55
+
56
+ // Define Zod schema for validation
57
+ const ParamsSchema = z.object({
58
+ tool: z.enum(['gemini', 'qwen', 'codex']),
59
+ prompt: z.string().min(1, 'Prompt is required'),
60
+ mode: z.enum(['analysis', 'write', 'auto']).default('analysis'),
61
+ format: z.enum(['plain', 'yaml', 'json']).default('plain'), // Multi-turn prompt concatenation format
62
+ model: z.string().optional(),
63
+ cd: z.string().optional(),
64
+ includeDirs: z.string().optional(),
65
+ timeout: z.number().default(300000),
66
+ resume: z.union([z.boolean(), z.string()]).optional(), // true = last, string = single ID or comma-separated IDs
67
+ id: z.string().optional(), // Custom execution ID (e.g., IMPL-001-step1)
68
+ noNative: z.boolean().optional(), // Force prompt concatenation instead of native resume
69
+ category: z.enum(['user', 'internal', 'insight']).default('user'), // Execution category for tracking
70
+ parentExecutionId: z.string().optional(), // Parent execution ID for fork/retry scenarios
71
+ });
72
+
73
+ // Execution category types
74
+ export type ExecutionCategory = 'user' | 'internal' | 'insight';
75
+
76
+ type Params = z.infer<typeof ParamsSchema>;
77
+
78
+ // Prompt concatenation format types
79
+ type PromptFormat = 'plain' | 'yaml' | 'json';
80
+
81
+ interface ToolAvailability {
82
+ available: boolean;
83
+ path: string | null;
84
+ }
85
+
86
+ // Tool availability cache with TTL
87
+ interface CachedToolAvailability {
88
+ result: ToolAvailability;
89
+ timestamp: number;
90
+ }
91
+
92
+ // Cache storage: Map<toolName, CachedToolAvailability>
93
+ const toolAvailabilityCache = new Map<string, CachedToolAvailability>();
94
+ const CACHE_TTL_MS = 5 * 60 * 1000; // 5 minutes
95
+
96
+ /**
97
+ * Check if cache entry is still valid
98
+ */
99
+ function isCacheValid(cached: CachedToolAvailability): boolean {
100
+ return Date.now() - cached.timestamp < CACHE_TTL_MS;
101
+ }
102
+
103
+ /**
104
+ * Clear expired cache entries
105
+ */
106
+ function clearExpiredCache(): void {
107
+ const now = Date.now();
108
+ const entriesToDelete: string[] = [];
109
+
110
+ toolAvailabilityCache.forEach((cached, tool) => {
111
+ if (now - cached.timestamp >= CACHE_TTL_MS) {
112
+ entriesToDelete.push(tool);
113
+ }
114
+ });
115
+
116
+ entriesToDelete.forEach(tool => toolAvailabilityCache.delete(tool));
117
+ }
118
+
119
+ /**
120
+ * Clear all cache entries (useful for testing or forced refresh)
121
+ */
122
+ export function clearToolCache(): void {
123
+ toolAvailabilityCache.clear();
124
+ }
125
+
126
+ // Single turn in a conversation
127
+ interface ConversationTurn {
128
+ turn: number;
129
+ timestamp: string;
130
+ prompt: string;
131
+ duration_ms: number;
132
+ status: 'success' | 'error' | 'timeout';
133
+ exit_code: number | null;
134
+ output: {
135
+ stdout: string;
136
+ stderr: string;
137
+ truncated: boolean;
138
+ };
139
+ }
140
+
141
+ // Multi-turn conversation record
142
+ interface ConversationRecord {
143
+ id: string;
144
+ created_at: string;
145
+ updated_at: string;
146
+ tool: string;
147
+ model: string;
148
+ mode: string;
149
+ category: ExecutionCategory; // user | internal | insight
150
+ total_duration_ms: number;
151
+ turn_count: number;
152
+ latest_status: 'success' | 'error' | 'timeout';
153
+ turns: ConversationTurn[];
154
+ parent_execution_id?: string; // For fork/retry scenarios
155
+ }
156
+
157
+ // Legacy single execution record (for backward compatibility)
158
+ interface ExecutionRecord {
159
+ id: string;
160
+ timestamp: string;
161
+ tool: string;
162
+ model: string;
163
+ mode: string;
164
+ prompt: string;
165
+ status: 'success' | 'error' | 'timeout';
166
+ exit_code: number | null;
167
+ duration_ms: number;
168
+ output: {
169
+ stdout: string;
170
+ stderr: string;
171
+ truncated: boolean;
172
+ };
173
+ }
174
+
175
+ interface HistoryIndex {
176
+ version: number;
177
+ total_executions: number;
178
+ executions: {
179
+ id: string;
180
+ timestamp: string; // created_at for conversations
181
+ updated_at?: string; // last update time
182
+ tool: string;
183
+ status: string;
184
+ duration_ms: number;
185
+ turn_count?: number; // number of turns in conversation
186
+ prompt_preview: string;
187
+ }[];
188
+ }
189
+
190
+ interface ExecutionOutput {
191
+ success: boolean;
192
+ execution: ExecutionRecord;
193
+ conversation: ConversationRecord; // Full conversation record
194
+ stdout: string;
195
+ stderr: string;
196
+ }
197
+
198
+ /**
199
+ * Check if a CLI tool is available (with caching)
200
+ */
201
+ async function checkToolAvailability(tool: string): Promise<ToolAvailability> {
202
+ // Check cache first
203
+ const cached = toolAvailabilityCache.get(tool);
204
+ if (cached && isCacheValid(cached)) {
205
+ return cached.result;
206
+ }
207
+
208
+ // Clear expired entries periodically
209
+ clearExpiredCache();
210
+
211
+ // Perform actual check
212
+ return new Promise((resolve) => {
213
+ const isWindows = process.platform === 'win32';
214
+ const command = isWindows ? 'where' : 'which';
215
+
216
+ // Direct spawn - where/which are system commands that don't need shell wrapper
217
+ const child = spawn(command, [tool], {
218
+ shell: false,
219
+ stdio: ['ignore', 'pipe', 'pipe']
220
+ });
221
+
222
+ let stdout = '';
223
+ child.stdout!.on('data', (data) => { stdout += data.toString(); });
224
+
225
+ child.on('close', (code) => {
226
+ const result: ToolAvailability = code === 0 && stdout.trim()
227
+ ? { available: true, path: stdout.trim().split('\n')[0] }
228
+ : { available: false, path: null };
229
+
230
+ // Only cache positive results to avoid caching transient failures
231
+ if (result.available) {
232
+ toolAvailabilityCache.set(tool, {
233
+ result,
234
+ timestamp: Date.now()
235
+ });
236
+ }
237
+
238
+ resolve(result);
239
+ });
240
+
241
+ child.on('error', () => {
242
+ // Don't cache errors - they may be transient
243
+ resolve({ available: false, path: null });
244
+ });
245
+
246
+ // Timeout after 5 seconds
247
+ setTimeout(() => {
248
+ child.kill();
249
+ // Don't cache timeouts - they may be transient
250
+ resolve({ available: false, path: null });
251
+ }, 5000);
252
+ });
253
+ }
254
+
255
+ // Native resume configuration
256
+ interface NativeResumeConfig {
257
+ enabled: boolean;
258
+ sessionId?: string; // Native UUID
259
+ isLatest?: boolean; // Use latest/--last flag
260
+ }
261
+
262
+ /**
263
+ * Build command arguments based on tool and options
264
+ */
265
+ function buildCommand(params: {
266
+ tool: string;
267
+ prompt: string;
268
+ mode: string;
269
+ model?: string;
270
+ dir?: string;
271
+ include?: string;
272
+ nativeResume?: NativeResumeConfig;
273
+ }): { command: string; args: string[]; useStdin: boolean } {
274
+ const { tool, prompt, mode = 'analysis', model, dir, include, nativeResume } = params;
275
+
276
+ let command = tool;
277
+ let args: string[] = [];
278
+ // Default to stdin for all tools to avoid escaping issues on Windows
279
+ let useStdin = true;
280
+
281
+ switch (tool) {
282
+ case 'gemini':
283
+ // Native resume: gemini -r <uuid> or -r latest
284
+ if (nativeResume?.enabled) {
285
+ if (nativeResume.isLatest) {
286
+ args.push('-r', 'latest');
287
+ } else if (nativeResume.sessionId) {
288
+ args.push('-r', nativeResume.sessionId);
289
+ }
290
+ }
291
+ if (model) {
292
+ args.push('-m', model);
293
+ }
294
+ if (mode === 'write') {
295
+ args.push('--approval-mode', 'yolo');
296
+ }
297
+ if (include) {
298
+ args.push('--include-directories', include);
299
+ }
300
+ break;
301
+
302
+ case 'qwen':
303
+ // Native resume: qwen --continue (latest) or --resume <uuid>
304
+ if (nativeResume?.enabled) {
305
+ if (nativeResume.isLatest) {
306
+ args.push('--continue');
307
+ } else if (nativeResume.sessionId) {
308
+ args.push('--resume', nativeResume.sessionId);
309
+ }
310
+ }
311
+ if (model) {
312
+ args.push('-m', model);
313
+ }
314
+ if (mode === 'write') {
315
+ args.push('--approval-mode', 'yolo');
316
+ }
317
+ if (include) {
318
+ args.push('--include-directories', include);
319
+ }
320
+ break;
321
+
322
+ case 'codex':
323
+ // Codex supports stdin when using `-` as prompt argument
324
+ // Using stdin avoids Windows command line escaping issues with multi-line/special char prompts
325
+ useStdin = true;
326
+ // Native resume: codex resume <uuid> [prompt] or --last
327
+ if (nativeResume?.enabled) {
328
+ args.push('resume');
329
+ if (nativeResume.isLatest) {
330
+ args.push('--last');
331
+ } else if (nativeResume.sessionId) {
332
+ args.push(nativeResume.sessionId);
333
+ }
334
+ // Codex resume still supports additional flags
335
+ if (dir) {
336
+ args.push('-C', dir);
337
+ }
338
+ // Permission configuration based on mode:
339
+ // - analysis: --full-auto (read-only sandbox, no prompts) - safer for read operations
340
+ // - write/auto: --dangerously-bypass-approvals-and-sandbox (full access for modifications)
341
+ if (mode === 'write' || mode === 'auto') {
342
+ args.push('--dangerously-bypass-approvals-and-sandbox');
343
+ } else {
344
+ args.push('--full-auto');
345
+ }
346
+ if (model) {
347
+ args.push('-m', model);
348
+ }
349
+ if (include) {
350
+ const dirs = include.split(',').map(d => d.trim()).filter(d => d);
351
+ for (const addDir of dirs) {
352
+ args.push('--add-dir', addDir);
353
+ }
354
+ }
355
+ // Use `-` to indicate reading prompt from stdin
356
+ args.push('-');
357
+ } else {
358
+ // Standard exec mode
359
+ args.push('exec');
360
+ if (dir) {
361
+ args.push('-C', dir);
362
+ }
363
+ // Permission configuration based on mode:
364
+ // - analysis: --full-auto (read-only sandbox, no prompts) - safer for read operations
365
+ // - write/auto: --dangerously-bypass-approvals-and-sandbox (full access for modifications)
366
+ if (mode === 'write' || mode === 'auto') {
367
+ args.push('--dangerously-bypass-approvals-and-sandbox');
368
+ } else {
369
+ args.push('--full-auto');
370
+ }
371
+ if (model) {
372
+ args.push('-m', model);
373
+ }
374
+ if (include) {
375
+ const dirs = include.split(',').map(d => d.trim()).filter(d => d);
376
+ for (const addDir of dirs) {
377
+ args.push('--add-dir', addDir);
378
+ }
379
+ }
380
+ // Use `-` to indicate reading prompt from stdin (avoids Windows escaping issues)
381
+ args.push('-');
382
+ }
383
+ break;
384
+
385
+ case 'claude':
386
+ // Claude Code: claude -p "prompt" for non-interactive mode
387
+ args.push('-p'); // Print mode (non-interactive)
388
+ // Native resume: claude --resume <session-id> or --continue
389
+ if (nativeResume?.enabled) {
390
+ if (nativeResume.isLatest) {
391
+ args.push('--continue');
392
+ } else if (nativeResume.sessionId) {
393
+ args.push('--resume', nativeResume.sessionId);
394
+ }
395
+ }
396
+ if (model) {
397
+ args.push('--model', model);
398
+ }
399
+ // Permission modes: write/auto → bypassPermissions, analysis → default
400
+ if (mode === 'write' || mode === 'auto') {
401
+ args.push('--permission-mode', 'bypassPermissions');
402
+ } else {
403
+ args.push('--permission-mode', 'default');
404
+ }
405
+ // Output format for better parsing
406
+ args.push('--output-format', 'text');
407
+ // Add directories
408
+ if (include) {
409
+ const dirs = include.split(',').map(d => d.trim()).filter(d => d);
410
+ for (const addDir of dirs) {
411
+ args.push('--add-dir', addDir);
412
+ }
413
+ }
414
+ break;
415
+
416
+ default:
417
+ throw new Error(`Unknown CLI tool: ${tool}`);
418
+ }
419
+
420
+ return { command, args, useStdin };
421
+ }
422
+
423
+ /**
424
+ * Ensure history directory exists (uses centralized storage)
425
+ */
426
+ function ensureHistoryDir(baseDir: string): string {
427
+ const paths = StoragePaths.project(baseDir);
428
+ ensureStorageDir(paths.cliHistory);
429
+ return paths.cliHistory;
430
+ }
431
+
432
+ /**
433
+ * Save conversation to SQLite
434
+ * @param baseDir - Project base directory (NOT historyDir)
435
+ */
436
+ async function saveConversationAsync(baseDir: string, conversation: ConversationRecord): Promise<void> {
437
+ const store = await getSqliteStore(baseDir);
438
+ store.saveConversation(conversation);
439
+ }
440
+
441
+ /**
442
+ * Sync wrapper for saveConversation (uses cached SQLite module)
443
+ * @param baseDir - Project base directory (NOT historyDir)
444
+ */
445
+ function saveConversation(baseDir: string, conversation: ConversationRecord): void {
446
+ try {
447
+ const store = getSqliteStoreSync(baseDir);
448
+ store.saveConversation(conversation);
449
+ } catch {
450
+ // If sync not available, queue for async save
451
+ saveConversationAsync(baseDir, conversation).catch(err => {
452
+ console.error('[CLI Executor] Failed to save conversation:', err.message);
453
+ });
454
+ }
455
+ }
456
+
457
+ /**
458
+ * Load existing conversation by ID from SQLite
459
+ * @param baseDir - Project base directory (NOT historyDir)
460
+ */
461
+ async function loadConversationAsync(baseDir: string, conversationId: string): Promise<ConversationRecord | null> {
462
+ const store = await getSqliteStore(baseDir);
463
+ return store.getConversation(conversationId);
464
+ }
465
+
466
+ /**
467
+ * Sync wrapper for loadConversation (uses cached SQLite module)
468
+ * @param baseDir - Project base directory (NOT historyDir)
469
+ */
470
+ function loadConversation(baseDir: string, conversationId: string): ConversationRecord | null {
471
+ try {
472
+ const store = getSqliteStoreSync(baseDir);
473
+ return store.getConversation(conversationId);
474
+ } catch {
475
+ // SQLite not initialized yet, return null
476
+ return null;
477
+ }
478
+ }
479
+
480
+ /**
481
+ * Convert legacy ExecutionRecord to ConversationRecord
482
+ */
483
+ function convertToConversation(record: ExecutionRecord): ConversationRecord {
484
+ return {
485
+ id: record.id,
486
+ created_at: record.timestamp,
487
+ updated_at: record.timestamp,
488
+ tool: record.tool,
489
+ model: record.model,
490
+ mode: record.mode,
491
+ category: 'user', // Legacy records default to user category
492
+ total_duration_ms: record.duration_ms,
493
+ turn_count: 1,
494
+ latest_status: record.status,
495
+ turns: [{
496
+ turn: 1,
497
+ timestamp: record.timestamp,
498
+ prompt: record.prompt,
499
+ duration_ms: record.duration_ms,
500
+ status: record.status,
501
+ exit_code: record.exit_code,
502
+ output: record.output
503
+ }]
504
+ };
505
+ }
506
+
507
+ /**
508
+ * Merge multiple conversations into a unified context
509
+ * Returns merged turns sorted by timestamp with source tracking
510
+ */
511
+ interface MergedTurn extends ConversationTurn {
512
+ source_id: string; // Original conversation ID
513
+ }
514
+
515
+ interface MergeResult {
516
+ mergedTurns: MergedTurn[];
517
+ sourceConversations: ConversationRecord[];
518
+ totalDuration: number;
519
+ }
520
+
521
+ function mergeConversations(conversations: ConversationRecord[]): MergeResult {
522
+ const mergedTurns: MergedTurn[] = [];
523
+
524
+ // Collect all turns with source tracking
525
+ for (const conv of conversations) {
526
+ for (const turn of conv.turns) {
527
+ mergedTurns.push({
528
+ ...turn,
529
+ source_id: conv.id
530
+ });
531
+ }
532
+ }
533
+
534
+ // Sort by timestamp
535
+ mergedTurns.sort((a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime());
536
+
537
+ // Re-number turns
538
+ mergedTurns.forEach((turn, idx) => {
539
+ turn.turn = idx + 1;
540
+ });
541
+
542
+ // Calculate total duration
543
+ const totalDuration = mergedTurns.reduce((sum, t) => sum + t.duration_ms, 0);
544
+
545
+ return {
546
+ mergedTurns,
547
+ sourceConversations: conversations,
548
+ totalDuration
549
+ };
550
+ }
551
+
552
+ /**
553
+ * Build prompt from merged conversations
554
+ */
555
+ function buildMergedPrompt(
556
+ mergeResult: MergeResult,
557
+ newPrompt: string,
558
+ format: PromptFormat = 'plain'
559
+ ): string {
560
+ const concatenator = createPromptConcatenator({ format });
561
+
562
+ // Set metadata for merged conversations
563
+ concatenator.setMetadata(
564
+ 'merged_sources',
565
+ mergeResult.sourceConversations.map(c => c.id).join(', ')
566
+ );
567
+
568
+ // Add all merged turns with source tracking
569
+ for (const turn of mergeResult.mergedTurns) {
570
+ concatenator.addFromConversationTurn(turn, turn.source_id);
571
+ }
572
+
573
+ return concatenator.build(newPrompt);
574
+ }
575
+
576
+ /**
577
+ * Execute CLI tool with streaming output
578
+ */
579
+ async function executeCliTool(
580
+ params: Record<string, unknown>,
581
+ onOutput?: ((data: { type: string; data: string }) => void) | null
582
+ ): Promise<ExecutionOutput> {
583
+ const parsed = ParamsSchema.safeParse(params);
584
+ if (!parsed.success) {
585
+ throw new Error(`Invalid params: ${parsed.error.message}`);
586
+ }
587
+
588
+ const { tool, prompt, mode, format, model, cd, includeDirs, timeout, resume, id: customId, noNative, category, parentExecutionId } = parsed.data;
589
+
590
+ // Determine working directory early (needed for conversation lookup)
591
+ const workingDir = cd || process.cwd();
592
+ ensureHistoryDir(workingDir); // Ensure history directory exists
593
+
594
+ // Get SQLite store for native session lookup
595
+ const store = await getSqliteStore(workingDir);
596
+
597
+ // Determine conversation ID and load existing conversation
598
+ // Logic:
599
+ // - If --resume <id1,id2,...> (multiple IDs): merge conversations
600
+ // - With --id: create new merged conversation
601
+ // - Without --id: append to ALL source conversations
602
+ // - If --resume <id> AND --id <newId>: fork - read context from resume ID, create new conversation with newId
603
+ // - If --id provided (no resume): use that ID (create new or append)
604
+ // - If --resume <id> without --id: use resume ID (append to existing)
605
+ // - No params: create new with auto-generated ID
606
+ let conversationId: string;
607
+ let existingConversation: ConversationRecord | null = null;
608
+ let contextConversation: ConversationRecord | null = null; // For fork scenario
609
+ let mergeResult: MergeResult | null = null; // For merge scenario
610
+ let sourceConversations: ConversationRecord[] = []; // All source conversations for merge
611
+
612
+ // Parse resume IDs (can be comma-separated for merge)
613
+ const resumeIds: string[] = resume
614
+ ? (typeof resume === 'string' ? resume.split(',').map(id => id.trim()).filter(Boolean) : [])
615
+ : [];
616
+ const isMerge = resumeIds.length > 1;
617
+ const resumeId = resumeIds.length === 1 ? resumeIds[0] : null;
618
+
619
+ if (isMerge) {
620
+ // Merge scenario: multiple resume IDs
621
+ sourceConversations = resumeIds
622
+ .map(id => loadConversation(workingDir, id))
623
+ .filter((c): c is ConversationRecord => c !== null);
624
+
625
+ if (sourceConversations.length === 0) {
626
+ throw new Error('No valid conversations found for merge');
627
+ }
628
+
629
+ mergeResult = mergeConversations(sourceConversations);
630
+
631
+ if (customId) {
632
+ // Create new merged conversation with custom ID
633
+ conversationId = customId;
634
+ existingConversation = loadConversation(workingDir, customId);
635
+ } else {
636
+ // Will append to ALL source conversations (handled in save logic)
637
+ // Use first source conversation ID as primary
638
+ conversationId = sourceConversations[0].id;
639
+ existingConversation = sourceConversations[0];
640
+ }
641
+ } else if (customId && resumeId) {
642
+ // Fork: read context from resume ID, but create new conversation with custom ID
643
+ conversationId = customId;
644
+ contextConversation = loadConversation(workingDir, resumeId);
645
+ existingConversation = loadConversation(workingDir, customId);
646
+ } else if (customId) {
647
+ // Use custom ID - may be new or existing
648
+ conversationId = customId;
649
+ existingConversation = loadConversation(workingDir, customId);
650
+ } else if (resumeId) {
651
+ // Resume single ID without new ID - append to existing conversation
652
+ conversationId = resumeId;
653
+ existingConversation = loadConversation(workingDir, resumeId);
654
+ } else if (resume) {
655
+ // resume=true: get last conversation for this tool
656
+ const history = getExecutionHistory(workingDir, { limit: 1, tool });
657
+ if (history.executions.length > 0) {
658
+ conversationId = history.executions[0].id;
659
+ existingConversation = loadConversation(workingDir, conversationId);
660
+ } else {
661
+ // No previous conversation, create new
662
+ conversationId = `${Date.now()}-${tool}`;
663
+ }
664
+ } else {
665
+ // New conversation with auto-generated ID
666
+ conversationId = `${Date.now()}-${tool}`;
667
+ }
668
+
669
+ // Determine resume strategy (native vs prompt-concat vs hybrid)
670
+ let resumeDecision: ResumeDecision | null = null;
671
+ let nativeResumeConfig: NativeResumeConfig | undefined;
672
+
673
+ // resume=true (latest) - use native latest if supported
674
+ if (resume === true && !noNative && supportsNativeResume(tool)) {
675
+ resumeDecision = {
676
+ strategy: 'native',
677
+ isLatest: true,
678
+ primaryConversationId: conversationId
679
+ };
680
+ }
681
+ // Use strategy engine for complex scenarios
682
+ else if (resumeIds.length > 0 && !noNative) {
683
+ resumeDecision = determineResumeStrategy({
684
+ tool,
685
+ resumeIds,
686
+ customId,
687
+ forcePromptConcat: noNative,
688
+ getNativeSessionId: (ccwId) => store.getNativeSessionId(ccwId),
689
+ getConversation: (ccwId) => loadConversation(workingDir, ccwId),
690
+ getConversationTool: (ccwId) => {
691
+ const conv = loadConversation(workingDir, ccwId);
692
+ return conv?.tool || null;
693
+ }
694
+ });
695
+ }
696
+
697
+ // Configure native resume if strategy decided to use it
698
+ if (resumeDecision && (resumeDecision.strategy === 'native' || resumeDecision.strategy === 'hybrid')) {
699
+ nativeResumeConfig = {
700
+ enabled: true,
701
+ sessionId: resumeDecision.nativeSessionId,
702
+ isLatest: resumeDecision.isLatest
703
+ };
704
+ }
705
+
706
+ // Build final prompt with conversation context
707
+ // For native: minimal prompt (native tool handles context)
708
+ // For hybrid: context prefix from other conversations + new prompt
709
+ // For prompt-concat: full multi-turn prompt
710
+ let finalPrompt = prompt;
711
+
712
+ if (resumeDecision?.strategy === 'native') {
713
+ // Native mode: just use the new prompt, tool handles context
714
+ finalPrompt = prompt;
715
+ } else if (resumeDecision?.strategy === 'hybrid' && resumeDecision.contextTurns?.length) {
716
+ // Hybrid mode: add context prefix from other conversations
717
+ const contextPrefix = buildContextPrefix(resumeDecision.contextTurns, format);
718
+ finalPrompt = contextPrefix + prompt;
719
+ } else if (mergeResult && mergeResult.mergedTurns.length > 0) {
720
+ // Full merge: use merged prompt
721
+ finalPrompt = buildMergedPrompt(mergeResult, prompt, format);
722
+ } else {
723
+ // Standard prompt-concat
724
+ const conversationForContext = contextConversation || existingConversation;
725
+ if (conversationForContext && conversationForContext.turns.length > 0) {
726
+ finalPrompt = buildMultiTurnPrompt(conversationForContext, prompt, format);
727
+ }
728
+ }
729
+
730
+ // Check tool availability
731
+ const toolStatus = await checkToolAvailability(tool);
732
+ if (!toolStatus.available) {
733
+ throw new Error(`CLI tool not available: ${tool}. Please ensure it is installed and in PATH.`);
734
+ }
735
+
736
+ // Log resume mode for debugging
737
+ if (resumeDecision) {
738
+ const modeDesc = getResumeModeDescription(resumeDecision);
739
+ if (onOutput) {
740
+ onOutput({ type: 'stderr', data: `[Resume mode: ${modeDesc}]\n` });
741
+ }
742
+ }
743
+
744
+ // Use configured primary model if no explicit model provided
745
+ const effectiveModel = model || getPrimaryModel(workingDir, tool);
746
+
747
+ // Build command
748
+ const { command, args, useStdin } = buildCommand({
749
+ tool,
750
+ prompt: finalPrompt,
751
+ mode,
752
+ model: effectiveModel,
753
+ dir: cd,
754
+ include: includeDirs,
755
+ nativeResume: nativeResumeConfig
756
+ });
757
+
758
+ const startTime = Date.now();
759
+
760
+ return new Promise((resolve, reject) => {
761
+ // Windows requires shell: true for npm global commands (.cmd files)
762
+ // Unix-like systems can use shell: false for direct execution
763
+ const isWindows = process.platform === 'win32';
764
+ const child = spawn(command, args, {
765
+ cwd: workingDir,
766
+ shell: isWindows, // Enable shell on Windows for .cmd files
767
+ stdio: [useStdin ? 'pipe' : 'ignore', 'pipe', 'pipe']
768
+ });
769
+
770
+ // Write prompt to stdin if using stdin mode (for gemini/qwen)
771
+ if (useStdin && child.stdin) {
772
+ child.stdin.write(finalPrompt);
773
+ child.stdin.end();
774
+ }
775
+
776
+ let stdout = '';
777
+ let stderr = '';
778
+ let timedOut = false;
779
+
780
+ // Handle stdout
781
+ child.stdout!.on('data', (data) => {
782
+ const text = data.toString();
783
+ stdout += text;
784
+ if (onOutput) {
785
+ onOutput({ type: 'stdout', data: text });
786
+ }
787
+ });
788
+
789
+ // Handle stderr
790
+ child.stderr!.on('data', (data) => {
791
+ const text = data.toString();
792
+ stderr += text;
793
+ if (onOutput) {
794
+ onOutput({ type: 'stderr', data: text });
795
+ }
796
+ });
797
+
798
+ // Handle completion
799
+ child.on('close', async (code) => {
800
+ const endTime = Date.now();
801
+ const duration = endTime - startTime;
802
+
803
+ // Determine status
804
+ let status: 'success' | 'error' | 'timeout' = 'success';
805
+ if (timedOut) {
806
+ status = 'timeout';
807
+ } else if (code !== 0) {
808
+ // Check if HTTP 429 but results exist (Gemini quirk)
809
+ if (stderr.includes('429') && stdout.trim()) {
810
+ status = 'success';
811
+ } else {
812
+ status = 'error';
813
+ }
814
+ }
815
+
816
+ // Create new turn
817
+ const newTurnOutput = {
818
+ stdout: stdout.substring(0, 10240), // Truncate to 10KB
819
+ stderr: stderr.substring(0, 2048), // Truncate to 2KB
820
+ truncated: stdout.length > 10240 || stderr.length > 2048
821
+ };
822
+
823
+ // Determine base turn number for merge scenarios
824
+ const baseTurnNumber = isMerge && mergeResult
825
+ ? mergeResult.mergedTurns.length + 1
826
+ : (existingConversation ? existingConversation.turns.length + 1 : 1);
827
+
828
+ const newTurn: ConversationTurn = {
829
+ turn: baseTurnNumber,
830
+ timestamp: new Date(startTime).toISOString(),
831
+ prompt,
832
+ duration_ms: duration,
833
+ status,
834
+ exit_code: code,
835
+ output: newTurnOutput
836
+ };
837
+
838
+ // Create or update conversation record
839
+ let conversation: ConversationRecord;
840
+
841
+ if (isMerge && mergeResult && !customId) {
842
+ // Merge without --id: append to ALL source conversations
843
+ // Save new turn to each source conversation
844
+ const savedConversations: ConversationRecord[] = [];
845
+ for (const srcConv of sourceConversations) {
846
+ const turnForSrc: ConversationTurn = {
847
+ ...newTurn,
848
+ turn: srcConv.turns.length + 1 // Use each conversation's turn count
849
+ };
850
+ const updatedConv: ConversationRecord = {
851
+ ...srcConv,
852
+ updated_at: new Date().toISOString(),
853
+ total_duration_ms: srcConv.total_duration_ms + duration,
854
+ turn_count: srcConv.turns.length + 1,
855
+ latest_status: status,
856
+ turns: [...srcConv.turns, turnForSrc]
857
+ };
858
+ savedConversations.push(updatedConv);
859
+ }
860
+ // Use first conversation as primary
861
+ conversation = savedConversations[0];
862
+ // Save all source conversations
863
+ try {
864
+ for (const conv of savedConversations) {
865
+ saveConversation(workingDir, conv);
866
+ }
867
+ } catch (err) {
868
+ console.error('[CLI Executor] Failed to save merged histories:', (err as Error).message);
869
+ }
870
+ } else if (isMerge && mergeResult && customId) {
871
+ // Merge with --id: create new conversation with merged turns + new turn
872
+ // Convert merged turns to regular turns (without source_id)
873
+ const mergedTurns: ConversationTurn[] = mergeResult.mergedTurns.map((mt, idx) => ({
874
+ turn: idx + 1,
875
+ timestamp: mt.timestamp,
876
+ prompt: mt.prompt,
877
+ duration_ms: mt.duration_ms,
878
+ status: mt.status,
879
+ exit_code: mt.exit_code,
880
+ output: mt.output
881
+ }));
882
+
883
+ conversation = existingConversation
884
+ ? {
885
+ ...existingConversation,
886
+ updated_at: new Date().toISOString(),
887
+ total_duration_ms: existingConversation.total_duration_ms + duration,
888
+ turn_count: existingConversation.turns.length + 1,
889
+ latest_status: status,
890
+ turns: [...existingConversation.turns, newTurn]
891
+ }
892
+ : {
893
+ id: conversationId,
894
+ created_at: new Date(startTime).toISOString(),
895
+ updated_at: new Date().toISOString(),
896
+ tool,
897
+ model: model || 'default',
898
+ mode,
899
+ category,
900
+ total_duration_ms: mergeResult.totalDuration + duration,
901
+ turn_count: mergedTurns.length + 1,
902
+ latest_status: status,
903
+ turns: [...mergedTurns, newTurn]
904
+ };
905
+ // Save merged conversation
906
+ try {
907
+ saveConversation(workingDir, conversation);
908
+ } catch (err) {
909
+ console.error('[CLI Executor] Failed to save merged conversation:', (err as Error).message);
910
+ }
911
+ } else {
912
+ // Normal scenario: single conversation
913
+ conversation = existingConversation
914
+ ? {
915
+ ...existingConversation,
916
+ updated_at: new Date().toISOString(),
917
+ total_duration_ms: existingConversation.total_duration_ms + duration,
918
+ turn_count: existingConversation.turns.length + 1,
919
+ latest_status: status,
920
+ turns: [...existingConversation.turns, newTurn]
921
+ }
922
+ : {
923
+ id: conversationId,
924
+ created_at: new Date(startTime).toISOString(),
925
+ updated_at: new Date().toISOString(),
926
+ tool,
927
+ model: model || 'default',
928
+ mode,
929
+ category,
930
+ total_duration_ms: duration,
931
+ turn_count: 1,
932
+ latest_status: status,
933
+ turns: [newTurn],
934
+ parent_execution_id: parentExecutionId
935
+ };
936
+ // Try to save conversation to history
937
+ try {
938
+ saveConversation(workingDir, conversation);
939
+ } catch (err) {
940
+ // Non-fatal: continue even if history save fails
941
+ console.error('[CLI Executor] Failed to save history:', (err as Error).message);
942
+ }
943
+ }
944
+
945
+ // Track native session after execution (awaited to prevent process hang)
946
+ // Pass prompt for precise matching in parallel execution scenarios
947
+ try {
948
+ const nativeSession = await trackNewSession(tool, new Date(startTime), workingDir, prompt);
949
+ if (nativeSession) {
950
+ // Save native session mapping
951
+ try {
952
+ store.saveNativeSessionMapping({
953
+ ccw_id: conversationId,
954
+ tool,
955
+ native_session_id: nativeSession.sessionId,
956
+ native_session_path: nativeSession.filePath,
957
+ project_hash: nativeSession.projectHash,
958
+ created_at: new Date().toISOString()
959
+ });
960
+ } catch (err) {
961
+ console.error('[CLI Executor] Failed to save native session mapping:', (err as Error).message);
962
+ }
963
+ }
964
+ } catch (err) {
965
+ console.error('[CLI Executor] Failed to track native session:', (err as Error).message);
966
+ }
967
+
968
+ // Create legacy execution record for backward compatibility
969
+ const execution: ExecutionRecord = {
970
+ id: conversationId,
971
+ timestamp: new Date(startTime).toISOString(),
972
+ tool,
973
+ model: model || 'default',
974
+ mode,
975
+ prompt,
976
+ status,
977
+ exit_code: code,
978
+ duration_ms: duration,
979
+ output: newTurnOutput
980
+ };
981
+
982
+ resolve({
983
+ success: status === 'success',
984
+ execution,
985
+ conversation,
986
+ stdout,
987
+ stderr
988
+ });
989
+ });
990
+
991
+ // Handle errors
992
+ child.on('error', (error) => {
993
+ reject(new Error(`Failed to spawn ${tool}: ${error.message}`));
994
+ });
995
+
996
+ // Timeout handling
997
+ const timeoutId = setTimeout(() => {
998
+ timedOut = true;
999
+ child.kill('SIGTERM');
1000
+ setTimeout(() => {
1001
+ if (!child.killed) {
1002
+ child.kill('SIGKILL');
1003
+ }
1004
+ }, 5000);
1005
+ }, timeout);
1006
+
1007
+ child.on('close', () => {
1008
+ clearTimeout(timeoutId);
1009
+ });
1010
+ });
1011
+ }
1012
+
1013
+ // Tool schema for MCP
1014
+ export const schema: ToolSchema = {
1015
+ name: 'cli_executor',
1016
+ description: `Execute external CLI tools (gemini/qwen/codex) with unified interface.
1017
+ Modes:
1018
+ - analysis: Read-only operations (default)
1019
+ - write: File modifications allowed
1020
+ - auto: Full autonomous operations (codex only)`,
1021
+ inputSchema: {
1022
+ type: 'object',
1023
+ properties: {
1024
+ tool: {
1025
+ type: 'string',
1026
+ enum: ['gemini', 'qwen', 'codex'],
1027
+ description: 'CLI tool to execute'
1028
+ },
1029
+ prompt: {
1030
+ type: 'string',
1031
+ description: 'Prompt to send to the CLI tool'
1032
+ },
1033
+ mode: {
1034
+ type: 'string',
1035
+ enum: ['analysis', 'write', 'auto'],
1036
+ description: 'Execution mode (default: analysis)',
1037
+ default: 'analysis'
1038
+ },
1039
+ model: {
1040
+ type: 'string',
1041
+ description: 'Model override (tool-specific)'
1042
+ },
1043
+ cd: {
1044
+ type: 'string',
1045
+ description: 'Working directory for execution (-C for codex)'
1046
+ },
1047
+ includeDirs: {
1048
+ type: 'string',
1049
+ description: 'Additional directories (comma-separated). Maps to --include-directories for gemini/qwen, --add-dir for codex'
1050
+ },
1051
+ timeout: {
1052
+ type: 'number',
1053
+ description: 'Timeout in milliseconds (default: 300000 = 5 minutes)',
1054
+ default: 300000
1055
+ }
1056
+ },
1057
+ required: ['tool', 'prompt']
1058
+ }
1059
+ };
1060
+
1061
+ // Handler function
1062
+ export async function handler(params: Record<string, unknown>): Promise<ToolResult<ExecutionOutput>> {
1063
+ try {
1064
+ const result = await executeCliTool(params);
1065
+ return {
1066
+ success: result.success,
1067
+ result
1068
+ };
1069
+ } catch (error) {
1070
+ return {
1071
+ success: false,
1072
+ error: `CLI execution failed: ${(error as Error).message}`
1073
+ };
1074
+ }
1075
+ }
1076
+
1077
+ /**
1078
+ * Find all project directories with CLI history in centralized storage
1079
+ * Returns list of project base directories (NOT history directories)
1080
+ */
1081
+ function findProjectsWithHistory(): string[] {
1082
+ const projectDirs: string[] = [];
1083
+ const projectsRoot = join(StoragePaths.global.root(), 'projects');
1084
+
1085
+ if (!existsSync(projectsRoot)) {
1086
+ return projectDirs;
1087
+ }
1088
+
1089
+ try {
1090
+ const entries = readdirSync(projectsRoot, { withFileTypes: true });
1091
+ for (const entry of entries) {
1092
+ if (entry.isDirectory()) {
1093
+ const paths = StoragePaths.projectById(entry.name);
1094
+ if (existsSync(paths.historyDb)) {
1095
+ // Return project ID as identifier (actual project path is hashed)
1096
+ projectDirs.push(entry.name);
1097
+ }
1098
+ }
1099
+ }
1100
+ } catch {
1101
+ // Ignore permission errors
1102
+ }
1103
+
1104
+ return projectDirs;
1105
+ }
1106
+
1107
+ /**
1108
+ * Get execution history from SQLite (centralized storage)
1109
+ */
1110
+ export async function getExecutionHistoryAsync(baseDir: string, options: {
1111
+ limit?: number;
1112
+ tool?: string | null;
1113
+ status?: string | null;
1114
+ category?: ExecutionCategory | null;
1115
+ search?: string | null;
1116
+ recursive?: boolean;
1117
+ } = {}): Promise<{
1118
+ total: number;
1119
+ count: number;
1120
+ executions: (HistoryIndex['executions'][0] & { sourceDir?: string })[];
1121
+ }> {
1122
+ const { limit = 50, tool = null, status = null, category = null, search = null, recursive = false } = options;
1123
+
1124
+ // Recursive mode: aggregate data from parent and all child projects
1125
+ if (recursive) {
1126
+ const { scanChildProjectsAsync } = await import('../config/storage-paths.js');
1127
+ const childProjects = await scanChildProjectsAsync(baseDir);
1128
+
1129
+ let allExecutions: (HistoryIndex['executions'][0] & { sourceDir?: string })[] = [];
1130
+ let totalCount = 0;
1131
+
1132
+ // Query parent project - apply limit at source to reduce memory footprint
1133
+ try {
1134
+ const parentStore = await getSqliteStore(baseDir);
1135
+ const parentResult = parentStore.getHistory({ limit, tool, status, category, search });
1136
+ totalCount += parentResult.total;
1137
+
1138
+ for (const exec of parentResult.executions) {
1139
+ allExecutions.push({ ...exec, sourceDir: baseDir });
1140
+ }
1141
+ } catch (error) {
1142
+ if (process.env.DEBUG) {
1143
+ console.error(`[CLI History] Failed to query parent project ${baseDir}:`, error);
1144
+ }
1145
+ }
1146
+
1147
+ // Query all child projects - apply limit to each child
1148
+ for (const child of childProjects) {
1149
+ try {
1150
+ const childStore = await getSqliteStore(child.projectPath);
1151
+ const childResult = childStore.getHistory({ limit, tool, status, category, search });
1152
+ totalCount += childResult.total;
1153
+
1154
+ for (const exec of childResult.executions) {
1155
+ allExecutions.push({
1156
+ ...exec,
1157
+ sourceDir: child.relativePath // Show relative path for clarity
1158
+ });
1159
+ }
1160
+ } catch (error) {
1161
+ if (process.env.DEBUG) {
1162
+ console.error(`[CLI History] Failed to query child project ${child.projectPath}:`, error);
1163
+ }
1164
+ }
1165
+ }
1166
+
1167
+ // Sort by timestamp (newest first) and apply limit
1168
+ allExecutions.sort((a, b) => Number(b.timestamp) - Number(a.timestamp));
1169
+ const limitedExecutions = allExecutions.slice(0, limit);
1170
+
1171
+ return {
1172
+ total: totalCount,
1173
+ count: limitedExecutions.length,
1174
+ executions: limitedExecutions
1175
+ };
1176
+ }
1177
+
1178
+ // Non-recursive mode: only query current project
1179
+ const store = await getSqliteStore(baseDir);
1180
+ return store.getHistory({ limit, tool, status, category, search });
1181
+ }
1182
+
1183
+ /**
1184
+ * Get execution history (sync version - uses cached SQLite module)
1185
+ */
1186
+ export function getExecutionHistory(baseDir: string, options: {
1187
+ limit?: number;
1188
+ tool?: string | null;
1189
+ status?: string | null;
1190
+ recursive?: boolean;
1191
+ } = {}): {
1192
+ total: number;
1193
+ count: number;
1194
+ executions: (HistoryIndex['executions'][0] & { sourceDir?: string })[];
1195
+ } {
1196
+ const { limit = 50, tool = null, status = null, recursive = false } = options;
1197
+
1198
+ try {
1199
+ if (recursive) {
1200
+ const { scanChildProjects } = require('../config/storage-paths.js');
1201
+ const childProjects = scanChildProjects(baseDir);
1202
+
1203
+ let allExecutions: (HistoryIndex['executions'][0] & { sourceDir?: string })[] = [];
1204
+ let totalCount = 0;
1205
+
1206
+ // Query parent project - apply limit at source
1207
+ try {
1208
+ const parentStore = getSqliteStoreSync(baseDir);
1209
+ const parentResult = parentStore.getHistory({ limit, tool, status });
1210
+ totalCount += parentResult.total;
1211
+
1212
+ for (const exec of parentResult.executions) {
1213
+ allExecutions.push({ ...exec, sourceDir: baseDir });
1214
+ }
1215
+ } catch (error) {
1216
+ if (process.env.DEBUG) {
1217
+ console.error(`[CLI History Sync] Failed to query parent project ${baseDir}:`, error);
1218
+ }
1219
+ }
1220
+
1221
+ // Query all child projects - apply limit to each child
1222
+ for (const child of childProjects) {
1223
+ try {
1224
+ const childStore = getSqliteStoreSync(child.projectPath);
1225
+ const childResult = childStore.getHistory({ limit, tool, status });
1226
+ totalCount += childResult.total;
1227
+
1228
+ for (const exec of childResult.executions) {
1229
+ allExecutions.push({
1230
+ ...exec,
1231
+ sourceDir: child.relativePath
1232
+ });
1233
+ }
1234
+ } catch (error) {
1235
+ if (process.env.DEBUG) {
1236
+ console.error(`[CLI History Sync] Failed to query child project ${child.projectPath}:`, error);
1237
+ }
1238
+ }
1239
+ }
1240
+
1241
+ // Sort by timestamp (newest first) and apply limit
1242
+ allExecutions.sort((a, b) => Number(b.timestamp) - Number(a.timestamp));
1243
+
1244
+ return {
1245
+ total: totalCount,
1246
+ count: Math.min(allExecutions.length, limit),
1247
+ executions: allExecutions.slice(0, limit)
1248
+ };
1249
+ }
1250
+
1251
+ const store = getSqliteStoreSync(baseDir);
1252
+ return store.getHistory({ limit, tool, status });
1253
+ } catch {
1254
+ // SQLite not initialized, return empty
1255
+ return { total: 0, count: 0, executions: [] };
1256
+ }
1257
+ }
1258
+
1259
+ /**
1260
+ * Get conversation detail by ID (returns ConversationRecord)
1261
+ */
1262
+ export function getConversationDetail(baseDir: string, conversationId: string): ConversationRecord | null {
1263
+ // Pass baseDir directly - loadConversation will resolve the correct storage path
1264
+ return loadConversation(baseDir, conversationId);
1265
+ }
1266
+
1267
+ /**
1268
+ * Get conversation detail with native session info
1269
+ */
1270
+ export function getConversationDetailWithNativeInfo(baseDir: string, conversationId: string) {
1271
+ try {
1272
+ const store = getSqliteStoreSync(baseDir);
1273
+ return store.getConversationWithNativeInfo(conversationId);
1274
+ } catch {
1275
+ // SQLite not initialized, return null
1276
+ return null;
1277
+ }
1278
+ }
1279
+
1280
+ /**
1281
+ * Get execution detail by ID (legacy, returns ExecutionRecord for backward compatibility)
1282
+ */
1283
+ export function getExecutionDetail(baseDir: string, executionId: string): ExecutionRecord | null {
1284
+ const conversation = getConversationDetail(baseDir, executionId);
1285
+ if (!conversation) return null;
1286
+
1287
+ // Convert to legacy ExecutionRecord format (using latest turn)
1288
+ const latestTurn = conversation.turns[conversation.turns.length - 1];
1289
+ return {
1290
+ id: conversation.id,
1291
+ timestamp: conversation.created_at,
1292
+ tool: conversation.tool,
1293
+ model: conversation.model,
1294
+ mode: conversation.mode,
1295
+ prompt: latestTurn.prompt,
1296
+ status: conversation.latest_status,
1297
+ exit_code: latestTurn.exit_code,
1298
+ duration_ms: conversation.total_duration_ms,
1299
+ output: latestTurn.output
1300
+ };
1301
+ }
1302
+
1303
+ /**
1304
+ * Delete execution by ID (async version)
1305
+ */
1306
+ export async function deleteExecutionAsync(baseDir: string, executionId: string): Promise<{ success: boolean; error?: string }> {
1307
+ const store = await getSqliteStore(baseDir);
1308
+ return store.deleteConversation(executionId);
1309
+ }
1310
+
1311
+ /**
1312
+ * Delete execution by ID (sync version - uses cached SQLite module)
1313
+ */
1314
+ export function deleteExecution(baseDir: string, executionId: string): { success: boolean; error?: string } {
1315
+ try {
1316
+ const store = getSqliteStoreSync(baseDir);
1317
+ return store.deleteConversation(executionId);
1318
+ } catch {
1319
+ return { success: false, error: 'SQLite store not initialized' };
1320
+ }
1321
+ }
1322
+
1323
+ /**
1324
+ * Batch delete executions (async)
1325
+ */
1326
+ export async function batchDeleteExecutionsAsync(baseDir: string, ids: string[]): Promise<{
1327
+ success: boolean;
1328
+ deleted: number;
1329
+ total: number;
1330
+ errors?: string[];
1331
+ }> {
1332
+ const store = await getSqliteStore(baseDir);
1333
+ const result = store.batchDelete(ids);
1334
+ return { ...result, total: ids.length };
1335
+ }
1336
+
1337
+ /**
1338
+ * Get status of all CLI tools
1339
+ */
1340
+ export async function getCliToolsStatus(): Promise<Record<string, ToolAvailability>> {
1341
+ const tools = ['gemini', 'qwen', 'codex', 'claude'];
1342
+ const results: Record<string, ToolAvailability> = {};
1343
+
1344
+ await Promise.all(tools.map(async (tool) => {
1345
+ results[tool] = await checkToolAvailability(tool);
1346
+ }));
1347
+
1348
+ return results;
1349
+ }
1350
+
1351
+ // CLI tool package mapping
1352
+ const CLI_TOOL_PACKAGES: Record<string, string> = {
1353
+ gemini: '@google/gemini-cli',
1354
+ qwen: '@qwen-code/qwen-code',
1355
+ codex: '@openai/codex',
1356
+ claude: '@anthropic-ai/claude-code'
1357
+ };
1358
+
1359
+ // Disabled tools storage (in-memory fallback, main storage is in cli-config.json)
1360
+ const disabledTools = new Set<string>();
1361
+
1362
+ // Default working directory for config operations
1363
+ let configBaseDir = process.cwd();
1364
+
1365
+ /**
1366
+ * Set the base directory for config operations
1367
+ */
1368
+ export function setConfigBaseDir(dir: string): void {
1369
+ configBaseDir = dir;
1370
+ }
1371
+
1372
+ /**
1373
+ * Install a CLI tool via npm
1374
+ */
1375
+ export async function installCliTool(tool: string): Promise<{ success: boolean; error?: string }> {
1376
+ const packageName = CLI_TOOL_PACKAGES[tool];
1377
+ if (!packageName) {
1378
+ return { success: false, error: `Unknown tool: ${tool}` };
1379
+ }
1380
+
1381
+ return new Promise((resolve) => {
1382
+ const child = spawn('npm', ['install', '-g', packageName], {
1383
+ shell: true,
1384
+ stdio: ['ignore', 'pipe', 'pipe']
1385
+ });
1386
+
1387
+ let stderr = '';
1388
+ child.stderr?.on('data', (data) => { stderr += data.toString(); });
1389
+
1390
+ child.on('close', (code) => {
1391
+ // Clear cache to force re-check
1392
+ toolAvailabilityCache.delete(tool);
1393
+
1394
+ if (code === 0) {
1395
+ resolve({ success: true });
1396
+ } else {
1397
+ resolve({ success: false, error: stderr || `npm install failed with code ${code}` });
1398
+ }
1399
+ });
1400
+
1401
+ child.on('error', (err) => {
1402
+ resolve({ success: false, error: err.message });
1403
+ });
1404
+
1405
+ // Timeout after 2 minutes
1406
+ setTimeout(() => {
1407
+ child.kill();
1408
+ resolve({ success: false, error: 'Installation timed out' });
1409
+ }, 120000);
1410
+ });
1411
+ }
1412
+
1413
+ /**
1414
+ * Uninstall a CLI tool via npm
1415
+ */
1416
+ export async function uninstallCliTool(tool: string): Promise<{ success: boolean; error?: string }> {
1417
+ const packageName = CLI_TOOL_PACKAGES[tool];
1418
+ if (!packageName) {
1419
+ return { success: false, error: `Unknown tool: ${tool}` };
1420
+ }
1421
+
1422
+ return new Promise((resolve) => {
1423
+ const child = spawn('npm', ['uninstall', '-g', packageName], {
1424
+ shell: true,
1425
+ stdio: ['ignore', 'pipe', 'pipe']
1426
+ });
1427
+
1428
+ let stderr = '';
1429
+ child.stderr?.on('data', (data) => { stderr += data.toString(); });
1430
+
1431
+ child.on('close', (code) => {
1432
+ // Clear cache to force re-check
1433
+ toolAvailabilityCache.delete(tool);
1434
+
1435
+ if (code === 0) {
1436
+ resolve({ success: true });
1437
+ } else {
1438
+ resolve({ success: false, error: stderr || `npm uninstall failed with code ${code}` });
1439
+ }
1440
+ });
1441
+
1442
+ child.on('error', (err) => {
1443
+ resolve({ success: false, error: err.message });
1444
+ });
1445
+
1446
+ // Timeout after 1 minute
1447
+ setTimeout(() => {
1448
+ child.kill();
1449
+ resolve({ success: false, error: 'Uninstallation timed out' });
1450
+ }, 60000);
1451
+ });
1452
+ }
1453
+
1454
+ /**
1455
+ * Enable a CLI tool (updates config file)
1456
+ */
1457
+ export function enableCliTool(tool: string): { success: boolean } {
1458
+ try {
1459
+ enableToolFromConfig(configBaseDir, tool);
1460
+ disabledTools.delete(tool); // Also update in-memory fallback
1461
+ return { success: true };
1462
+ } catch (err) {
1463
+ console.error('[cli-executor] Error enabling tool:', err);
1464
+ disabledTools.delete(tool); // Fallback to in-memory
1465
+ return { success: true };
1466
+ }
1467
+ }
1468
+
1469
+ /**
1470
+ * Disable a CLI tool (updates config file)
1471
+ */
1472
+ export function disableCliTool(tool: string): { success: boolean } {
1473
+ try {
1474
+ disableToolFromConfig(configBaseDir, tool);
1475
+ disabledTools.add(tool); // Also update in-memory fallback
1476
+ return { success: true };
1477
+ } catch (err) {
1478
+ console.error('[cli-executor] Error disabling tool:', err);
1479
+ disabledTools.add(tool); // Fallback to in-memory
1480
+ return { success: true };
1481
+ }
1482
+ }
1483
+
1484
+ /**
1485
+ * Check if a tool is enabled (reads from config file)
1486
+ */
1487
+ export function isToolEnabled(tool: string): boolean {
1488
+ try {
1489
+ return isToolEnabledFromConfig(configBaseDir, tool);
1490
+ } catch {
1491
+ // Fallback to in-memory check
1492
+ return !disabledTools.has(tool);
1493
+ }
1494
+ }
1495
+
1496
+ /**
1497
+ * Get full status of all CLI tools including enabled state
1498
+ */
1499
+ export async function getCliToolsFullStatus(): Promise<Record<string, {
1500
+ available: boolean;
1501
+ enabled: boolean;
1502
+ path: string | null;
1503
+ packageName: string;
1504
+ }>> {
1505
+ const tools = Object.keys(CLI_TOOL_PACKAGES);
1506
+ const results: Record<string, {
1507
+ available: boolean;
1508
+ enabled: boolean;
1509
+ path: string | null;
1510
+ packageName: string;
1511
+ }> = {};
1512
+
1513
+ await Promise.all(tools.map(async (tool) => {
1514
+ const availability = await checkToolAvailability(tool);
1515
+ results[tool] = {
1516
+ available: availability.available,
1517
+ enabled: isToolEnabled(tool),
1518
+ path: availability.path,
1519
+ packageName: CLI_TOOL_PACKAGES[tool]
1520
+ };
1521
+ }));
1522
+
1523
+ return results;
1524
+ }
1525
+
1526
+ // ========== Prompt Concatenation System ==========
1527
+
1528
+ /**
1529
+ * Turn data structure for concatenation
1530
+ */
1531
+ interface TurnData {
1532
+ turn: number;
1533
+ timestamp?: string;
1534
+ role: 'user' | 'assistant';
1535
+ content: string;
1536
+ status?: string;
1537
+ duration_ms?: number;
1538
+ source_id?: string; // For merged conversations
1539
+ }
1540
+
1541
+ /**
1542
+ * Prompt concatenation options
1543
+ */
1544
+ interface ConcatOptions {
1545
+ format: PromptFormat;
1546
+ includeMetadata?: boolean;
1547
+ includeTurnMarkers?: boolean;
1548
+ maxOutputLength?: number; // Truncate output for context efficiency
1549
+ }
1550
+
1551
+ /**
1552
+ * PromptConcatenator - Dedicated class for building multi-turn prompts
1553
+ * Supports multiple output formats: plain text, YAML, JSON
1554
+ */
1555
+ class PromptConcatenator {
1556
+ private turns: TurnData[] = [];
1557
+ private options: ConcatOptions;
1558
+ private metadata: Record<string, unknown> = {};
1559
+
1560
+ constructor(options: Partial<ConcatOptions> = {}) {
1561
+ this.options = {
1562
+ format: options.format || 'plain',
1563
+ includeMetadata: options.includeMetadata ?? true,
1564
+ includeTurnMarkers: options.includeTurnMarkers ?? true,
1565
+ maxOutputLength: options.maxOutputLength || 8192
1566
+ };
1567
+ }
1568
+
1569
+ /**
1570
+ * Set metadata for the conversation
1571
+ */
1572
+ setMetadata(key: string, value: unknown): this {
1573
+ this.metadata[key] = value;
1574
+ return this;
1575
+ }
1576
+
1577
+ /**
1578
+ * Add a user turn
1579
+ */
1580
+ addUserTurn(content: string, options: Partial<Omit<TurnData, 'role' | 'content'>> = {}): this {
1581
+ this.turns.push({
1582
+ turn: this.turns.length + 1,
1583
+ role: 'user',
1584
+ content,
1585
+ ...options
1586
+ });
1587
+ return this;
1588
+ }
1589
+
1590
+ /**
1591
+ * Add an assistant turn
1592
+ */
1593
+ addAssistantTurn(content: string, options: Partial<Omit<TurnData, 'role' | 'content'>> = {}): this {
1594
+ // Truncate output if needed
1595
+ const truncatedContent = content.length > this.options.maxOutputLength!
1596
+ ? content.substring(0, this.options.maxOutputLength!) + '\n... [truncated]'
1597
+ : content;
1598
+
1599
+ this.turns.push({
1600
+ turn: this.turns.length + 1,
1601
+ role: 'assistant',
1602
+ content: truncatedContent,
1603
+ ...options
1604
+ });
1605
+ return this;
1606
+ }
1607
+
1608
+ /**
1609
+ * Add a conversation turn from ConversationTurn
1610
+ */
1611
+ addFromConversationTurn(turn: ConversationTurn, sourceId?: string): this {
1612
+ this.addUserTurn(turn.prompt, {
1613
+ turn: turn.turn * 2 - 1,
1614
+ timestamp: turn.timestamp,
1615
+ source_id: sourceId
1616
+ });
1617
+ this.addAssistantTurn(turn.output.stdout || '[No output]', {
1618
+ turn: turn.turn * 2,
1619
+ timestamp: turn.timestamp,
1620
+ status: turn.status,
1621
+ duration_ms: turn.duration_ms,
1622
+ source_id: sourceId
1623
+ });
1624
+ return this;
1625
+ }
1626
+
1627
+ /**
1628
+ * Load turns from an existing conversation
1629
+ */
1630
+ loadConversation(conversation: ConversationRecord): this {
1631
+ for (const turn of conversation.turns) {
1632
+ this.addFromConversationTurn(turn);
1633
+ }
1634
+ return this;
1635
+ }
1636
+
1637
+ /**
1638
+ * Build the final prompt in plain text format
1639
+ */
1640
+ private buildPlainText(newPrompt: string): string {
1641
+ const parts: string[] = [];
1642
+
1643
+ // Metadata section
1644
+ if (this.options.includeMetadata && Object.keys(this.metadata).length > 0) {
1645
+ parts.push('=== CONTEXT ===');
1646
+ for (const [key, value] of Object.entries(this.metadata)) {
1647
+ parts.push(`${key}: ${String(value)}`);
1648
+ }
1649
+ parts.push('');
1650
+ }
1651
+
1652
+ // Conversation history
1653
+ if (this.turns.length > 0) {
1654
+ parts.push('=== CONVERSATION HISTORY ===');
1655
+ parts.push('');
1656
+
1657
+ let currentTurn = 0;
1658
+ for (let i = 0; i < this.turns.length; i += 2) {
1659
+ currentTurn++;
1660
+ const userTurn = this.turns[i];
1661
+ const assistantTurn = this.turns[i + 1];
1662
+
1663
+ if (this.options.includeTurnMarkers) {
1664
+ const sourceMarker = userTurn.source_id ? ` [${userTurn.source_id}]` : '';
1665
+ parts.push(`--- Turn ${currentTurn}${sourceMarker} ---`);
1666
+ }
1667
+
1668
+ parts.push('USER:');
1669
+ parts.push(userTurn.content);
1670
+ parts.push('');
1671
+
1672
+ if (assistantTurn) {
1673
+ parts.push('ASSISTANT:');
1674
+ parts.push(assistantTurn.content);
1675
+ parts.push('');
1676
+ }
1677
+ }
1678
+ }
1679
+
1680
+ // New request
1681
+ parts.push('=== NEW REQUEST ===');
1682
+ parts.push('');
1683
+ parts.push(newPrompt);
1684
+
1685
+ return parts.join('\n');
1686
+ }
1687
+
1688
+ /**
1689
+ * Build the final prompt in YAML format
1690
+ */
1691
+ private buildYaml(newPrompt: string): string {
1692
+ const yamlLines: string[] = [];
1693
+
1694
+ // Metadata
1695
+ if (this.options.includeMetadata && Object.keys(this.metadata).length > 0) {
1696
+ yamlLines.push('context:');
1697
+ for (const [key, value] of Object.entries(this.metadata)) {
1698
+ yamlLines.push(` ${key}: ${this.yamlValue(value)}`);
1699
+ }
1700
+ yamlLines.push('');
1701
+ }
1702
+
1703
+ // Conversation history
1704
+ if (this.turns.length > 0) {
1705
+ yamlLines.push('conversation:');
1706
+
1707
+ let currentTurn = 0;
1708
+ for (let i = 0; i < this.turns.length; i += 2) {
1709
+ currentTurn++;
1710
+ const userTurn = this.turns[i];
1711
+ const assistantTurn = this.turns[i + 1];
1712
+
1713
+ yamlLines.push(` - turn: ${currentTurn}`);
1714
+ if (userTurn.source_id) {
1715
+ yamlLines.push(` source: ${userTurn.source_id}`);
1716
+ }
1717
+ if (userTurn.timestamp) {
1718
+ yamlLines.push(` timestamp: ${userTurn.timestamp}`);
1719
+ }
1720
+
1721
+ // User message
1722
+ yamlLines.push(' user: |');
1723
+ const userLines = userTurn.content.split('\n');
1724
+ for (const line of userLines) {
1725
+ yamlLines.push(` ${line}`);
1726
+ }
1727
+
1728
+ // Assistant message
1729
+ if (assistantTurn) {
1730
+ if (assistantTurn.status) {
1731
+ yamlLines.push(` status: ${assistantTurn.status}`);
1732
+ }
1733
+ if (assistantTurn.duration_ms) {
1734
+ yamlLines.push(` duration_ms: ${assistantTurn.duration_ms}`);
1735
+ }
1736
+ yamlLines.push(' assistant: |');
1737
+ const assistantLines = assistantTurn.content.split('\n');
1738
+ for (const line of assistantLines) {
1739
+ yamlLines.push(` ${line}`);
1740
+ }
1741
+ }
1742
+ yamlLines.push('');
1743
+ }
1744
+ }
1745
+
1746
+ // New request
1747
+ yamlLines.push('new_request: |');
1748
+ const requestLines = newPrompt.split('\n');
1749
+ for (const line of requestLines) {
1750
+ yamlLines.push(` ${line}`);
1751
+ }
1752
+
1753
+ return yamlLines.join('\n');
1754
+ }
1755
+
1756
+ /**
1757
+ * Build the final prompt in JSON format
1758
+ */
1759
+ private buildJson(newPrompt: string): string {
1760
+ const data: Record<string, unknown> = {};
1761
+
1762
+ // Metadata
1763
+ if (this.options.includeMetadata && Object.keys(this.metadata).length > 0) {
1764
+ data.context = this.metadata;
1765
+ }
1766
+
1767
+ // Conversation history
1768
+ if (this.turns.length > 0) {
1769
+ const conversation: Array<{
1770
+ turn: number;
1771
+ source?: string;
1772
+ timestamp?: string;
1773
+ user: string;
1774
+ assistant?: string;
1775
+ status?: string;
1776
+ duration_ms?: number;
1777
+ }> = [];
1778
+
1779
+ for (let i = 0; i < this.turns.length; i += 2) {
1780
+ const userTurn = this.turns[i];
1781
+ const assistantTurn = this.turns[i + 1];
1782
+
1783
+ const turnData: typeof conversation[0] = {
1784
+ turn: Math.ceil((i + 1) / 2),
1785
+ user: userTurn.content
1786
+ };
1787
+
1788
+ if (userTurn.source_id) turnData.source = userTurn.source_id;
1789
+ if (userTurn.timestamp) turnData.timestamp = userTurn.timestamp;
1790
+ if (assistantTurn) {
1791
+ turnData.assistant = assistantTurn.content;
1792
+ if (assistantTurn.status) turnData.status = assistantTurn.status;
1793
+ if (assistantTurn.duration_ms) turnData.duration_ms = assistantTurn.duration_ms;
1794
+ }
1795
+
1796
+ conversation.push(turnData);
1797
+ }
1798
+
1799
+ data.conversation = conversation;
1800
+ }
1801
+
1802
+ data.new_request = newPrompt;
1803
+
1804
+ return JSON.stringify(data, null, 2);
1805
+ }
1806
+
1807
+ /**
1808
+ * Helper to format YAML values
1809
+ */
1810
+ private yamlValue(value: unknown): string {
1811
+ if (typeof value === 'string') {
1812
+ // Quote strings that might be interpreted as other types
1813
+ if (/[:\[\]{}#&*!|>'"@`]/.test(value) || value === '') {
1814
+ return `"${value.replace(/"/g, '\\"')}"`;
1815
+ }
1816
+ return value;
1817
+ }
1818
+ if (typeof value === 'number' || typeof value === 'boolean') {
1819
+ return String(value);
1820
+ }
1821
+ if (value === null || value === undefined) {
1822
+ return 'null';
1823
+ }
1824
+ return JSON.stringify(value);
1825
+ }
1826
+
1827
+ /**
1828
+ * Build the final prompt string
1829
+ */
1830
+ build(newPrompt: string): string {
1831
+ switch (this.options.format) {
1832
+ case 'yaml':
1833
+ return this.buildYaml(newPrompt);
1834
+ case 'json':
1835
+ return this.buildJson(newPrompt);
1836
+ case 'plain':
1837
+ default:
1838
+ return this.buildPlainText(newPrompt);
1839
+ }
1840
+ }
1841
+
1842
+ /**
1843
+ * Reset the concatenator for reuse
1844
+ */
1845
+ reset(): this {
1846
+ this.turns = [];
1847
+ this.metadata = {};
1848
+ return this;
1849
+ }
1850
+ }
1851
+
1852
+ /**
1853
+ * Create a prompt concatenator with specified options
1854
+ */
1855
+ function createPromptConcatenator(options?: Partial<ConcatOptions>): PromptConcatenator {
1856
+ return new PromptConcatenator(options);
1857
+ }
1858
+
1859
+ /**
1860
+ * Quick helper to build a multi-turn prompt in any format
1861
+ */
1862
+ function buildPrompt(
1863
+ conversation: ConversationRecord,
1864
+ newPrompt: string,
1865
+ format: PromptFormat = 'plain'
1866
+ ): string {
1867
+ return createPromptConcatenator({ format })
1868
+ .loadConversation(conversation)
1869
+ .build(newPrompt);
1870
+ }
1871
+
1872
+ /**
1873
+ * Build multi-turn prompt with full conversation history
1874
+ * Uses the PromptConcatenator with plain text format by default
1875
+ */
1876
+ function buildMultiTurnPrompt(
1877
+ conversation: ConversationRecord,
1878
+ newPrompt: string,
1879
+ format: PromptFormat = 'plain'
1880
+ ): string {
1881
+ return buildPrompt(conversation, newPrompt, format);
1882
+ }
1883
+
1884
+ /**
1885
+ * Build continuation prompt with previous conversation context (legacy)
1886
+ */
1887
+ function buildContinuationPrompt(previous: ExecutionRecord, additionalPrompt?: string): string {
1888
+ const parts: string[] = [];
1889
+
1890
+ // Add previous conversation context
1891
+ parts.push('=== PREVIOUS CONVERSATION ===');
1892
+ parts.push('');
1893
+ parts.push('USER PROMPT:');
1894
+ parts.push(previous.prompt);
1895
+ parts.push('');
1896
+ parts.push('ASSISTANT RESPONSE:');
1897
+ parts.push(previous.output.stdout || '[No output recorded]');
1898
+ parts.push('');
1899
+ parts.push('=== CONTINUATION ===');
1900
+ parts.push('');
1901
+
1902
+ if (additionalPrompt) {
1903
+ parts.push(additionalPrompt);
1904
+ } else {
1905
+ parts.push('Continue from where we left off. What should we do next?');
1906
+ }
1907
+
1908
+ return parts.join('\n');
1909
+ }
1910
+
1911
+ /**
1912
+ * Get previous execution for resume
1913
+ * @param baseDir - Working directory
1914
+ * @param tool - Tool to filter by
1915
+ * @param resume - true for last, or execution ID string
1916
+ */
1917
+ function getPreviousExecution(baseDir: string, tool: string, resume: boolean | string): ExecutionRecord | null {
1918
+ if (typeof resume === 'string') {
1919
+ // Resume specific execution by ID
1920
+ return getExecutionDetail(baseDir, resume);
1921
+ } else if (resume === true) {
1922
+ // Resume last execution for this tool
1923
+ const history = getExecutionHistory(baseDir, { limit: 1, tool });
1924
+ if (history.executions.length === 0) {
1925
+ return null;
1926
+ }
1927
+ return getExecutionDetail(baseDir, history.executions[0].id);
1928
+ }
1929
+ return null;
1930
+ }
1931
+
1932
+ /**
1933
+ * Get latest execution for a specific tool
1934
+ */
1935
+ export function getLatestExecution(baseDir: string, tool?: string): ExecutionRecord | null {
1936
+ const history = getExecutionHistory(baseDir, { limit: 1, tool: tool || null });
1937
+ if (history.executions.length === 0) {
1938
+ return null;
1939
+ }
1940
+ return getExecutionDetail(baseDir, history.executions[0].id);
1941
+ }
1942
+
1943
+ // ========== Native Session Content Functions ==========
1944
+
1945
+ /**
1946
+ * Get native session content by CCW ID
1947
+ * Parses the native session file and returns full conversation data
1948
+ */
1949
+ export async function getNativeSessionContent(baseDir: string, ccwId: string) {
1950
+ const store = await getSqliteStore(baseDir);
1951
+ return store.getNativeSessionContent(ccwId);
1952
+ }
1953
+
1954
+ /**
1955
+ * Get formatted native conversation text
1956
+ */
1957
+ export async function getFormattedNativeConversation(baseDir: string, ccwId: string, options?: {
1958
+ includeThoughts?: boolean;
1959
+ includeToolCalls?: boolean;
1960
+ includeTokens?: boolean;
1961
+ maxContentLength?: number;
1962
+ }) {
1963
+ const store = await getSqliteStore(baseDir);
1964
+ return store.getFormattedNativeConversation(ccwId, options);
1965
+ }
1966
+
1967
+ /**
1968
+ * Get conversation pairs from native session
1969
+ */
1970
+ export async function getNativeConversationPairs(baseDir: string, ccwId: string) {
1971
+ const store = await getSqliteStore(baseDir);
1972
+ return store.getNativeConversationPairs(ccwId);
1973
+ }
1974
+
1975
+ /**
1976
+ * Get enriched conversation (CCW + native session merged)
1977
+ */
1978
+ export async function getEnrichedConversation(baseDir: string, ccwId: string) {
1979
+ const store = await getSqliteStore(baseDir);
1980
+ return store.getEnrichedConversation(ccwId);
1981
+ }
1982
+
1983
+ /**
1984
+ * Get history with native session info
1985
+ */
1986
+ export async function getHistoryWithNativeInfo(baseDir: string, options?: {
1987
+ limit?: number;
1988
+ offset?: number;
1989
+ tool?: string | null;
1990
+ status?: string | null;
1991
+ category?: ExecutionCategory | null;
1992
+ search?: string | null;
1993
+ }) {
1994
+ const store = await getSqliteStore(baseDir);
1995
+ return store.getHistoryWithNativeInfo(options || {});
1996
+ }
1997
+
1998
+ // Export types
1999
+ export type { ConversationRecord, ConversationTurn, ExecutionRecord, PromptFormat, ConcatOptions };
2000
+
2001
+ // Export utility functions and tool definition for backward compatibility
2002
+ export { executeCliTool, checkToolAvailability };
2003
+
2004
+ // Export prompt concatenation utilities
2005
+ export { PromptConcatenator, createPromptConcatenator, buildPrompt, buildMultiTurnPrompt };
2006
+
2007
+ // Note: Async storage functions (getExecutionHistoryAsync, deleteExecutionAsync,
2008
+ // batchDeleteExecutionsAsync) are exported at declaration site - SQLite storage only
2009
+
2010
+ // Export tool definition (for legacy imports) - This allows direct calls to execute with onOutput
2011
+ export const cliExecutorTool = {
2012
+ schema,
2013
+ execute: executeCliTool // Use executeCliTool directly which supports onOutput callback
2014
+ };