claude-code-workflow 6.1.4 → 6.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (437) hide show
  1. package/.claude/CLAUDE.md +10 -0
  2. package/.claude/agents/action-planning-agent.md +857 -778
  3. package/.claude/agents/cli-execution-agent.md +266 -269
  4. package/.claude/agents/cli-explore-agent.md +2 -2
  5. package/.claude/agents/cli-lite-planning-agent.md +142 -92
  6. package/.claude/agents/cli-planning-agent.md +4 -4
  7. package/.claude/agents/code-developer.md +7 -6
  8. package/.claude/agents/conceptual-planning-agent.md +2 -2
  9. package/.claude/agents/context-search-agent.md +31 -32
  10. package/.claude/agents/doc-generator.md +4 -4
  11. package/.claude/agents/memory-bridge.md +93 -93
  12. package/.claude/agents/test-context-search-agent.md +8 -7
  13. package/.claude/agents/test-fix-agent.md +7 -6
  14. package/.claude/commands/clean.md +516 -0
  15. package/.claude/commands/memory/compact.md +383 -0
  16. package/.claude/commands/memory/docs-full-cli.md +471 -471
  17. package/.claude/commands/memory/docs-related-cli.md +386 -386
  18. package/.claude/commands/memory/docs.md +615 -615
  19. package/.claude/commands/memory/load.md +5 -5
  20. package/.claude/commands/memory/tech-research-rules.md +310 -0
  21. package/.claude/commands/memory/update-full.md +332 -332
  22. package/.claude/commands/memory/workflow-skill-memory.md +4 -4
  23. package/.claude/commands/task/create.md +151 -151
  24. package/.claude/commands/version.md +254 -254
  25. package/.claude/commands/workflow/brainstorm/api-designer.md +587 -585
  26. package/.claude/commands/workflow/brainstorm/artifacts.md +1 -0
  27. package/.claude/commands/workflow/brainstorm/auto-parallel.md +443 -443
  28. package/.claude/commands/workflow/brainstorm/data-architect.md +220 -220
  29. package/.claude/commands/workflow/brainstorm/product-manager.md +200 -200
  30. package/.claude/commands/workflow/brainstorm/product-owner.md +200 -200
  31. package/.claude/commands/workflow/brainstorm/scrum-master.md +200 -200
  32. package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  33. package/.claude/commands/workflow/brainstorm/system-architect.md +389 -387
  34. package/.claude/commands/workflow/brainstorm/ui-designer.md +221 -221
  35. package/.claude/commands/workflow/brainstorm/ux-expert.md +221 -221
  36. package/.claude/commands/workflow/debug.md +321 -0
  37. package/.claude/commands/workflow/execute.md +13 -0
  38. package/.claude/commands/workflow/init.md +165 -164
  39. package/.claude/commands/workflow/lite-execute.md +119 -13
  40. package/.claude/commands/workflow/lite-fix.md +623 -621
  41. package/.claude/commands/workflow/lite-plan.md +610 -592
  42. package/.claude/commands/workflow/plan.md +5 -5
  43. package/.claude/commands/workflow/review-module-cycle.md +2 -0
  44. package/.claude/commands/workflow/review-session-cycle.md +2 -0
  45. package/.claude/commands/workflow/review.md +297 -291
  46. package/.claude/commands/workflow/session/complete.md +153 -500
  47. package/.claude/commands/workflow/session/list.md +95 -95
  48. package/.claude/commands/workflow/session/resume.md +60 -60
  49. package/.claude/commands/workflow/session/start.md +199 -199
  50. package/.claude/commands/workflow/tdd-plan.md +3 -3
  51. package/.claude/commands/workflow/tdd-verify.md +23 -9
  52. package/.claude/commands/workflow/test-cycle-execute.md +2 -0
  53. package/.claude/commands/workflow/test-fix-gen.md +699 -699
  54. package/.claude/commands/workflow/tools/conflict-resolution.md +104 -18
  55. package/.claude/commands/workflow/tools/context-gather.md +436 -434
  56. package/.claude/commands/workflow/tools/task-generate-agent.md +490 -291
  57. package/.claude/commands/workflow/tools/task-generate-tdd.md +18 -10
  58. package/.claude/commands/workflow/tools/test-concept-enhanced.md +2 -1
  59. package/.claude/commands/workflow/tools/test-context-gather.md +1 -0
  60. package/.claude/commands/workflow/tools/test-task-generate.md +1 -0
  61. package/.claude/commands/workflow/ui-design/import-from-code.md +9 -6
  62. package/.claude/skills/command-guide/SKILL.md +5 -5
  63. package/.claude/skills/command-guide/index/all-commands.json +1 -1
  64. package/.claude/skills/command-guide/index/by-category.json +1 -1
  65. package/.claude/skills/command-guide/index/by-use-case.json +1 -1
  66. package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +857 -778
  67. package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +266 -269
  68. package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +2 -2
  69. package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +142 -92
  70. package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +4 -4
  71. package/.claude/skills/command-guide/reference/agents/code-developer.md +7 -6
  72. package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +2 -2
  73. package/.claude/skills/command-guide/reference/agents/context-search-agent.md +31 -32
  74. package/.claude/skills/command-guide/reference/agents/doc-generator.md +4 -4
  75. package/.claude/skills/command-guide/reference/agents/memory-bridge.md +93 -93
  76. package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +8 -7
  77. package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +7 -6
  78. package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +471 -471
  79. package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +386 -386
  80. package/.claude/skills/command-guide/reference/commands/memory/docs.md +17 -16
  81. package/.claude/skills/command-guide/reference/commands/memory/load.md +5 -5
  82. package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +194 -357
  83. package/.claude/skills/command-guide/reference/commands/memory/update-full.md +332 -332
  84. package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +4 -4
  85. package/.claude/skills/command-guide/reference/commands/task/create.md +151 -151
  86. package/.claude/skills/command-guide/reference/commands/version.md +254 -254
  87. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +585 -585
  88. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +443 -443
  89. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +220 -220
  90. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +200 -200
  91. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +200 -200
  92. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +200 -200
  93. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  94. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +387 -387
  95. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +221 -221
  96. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +221 -221
  97. package/.claude/skills/command-guide/reference/commands/workflow/execute.md +25 -20
  98. package/.claude/skills/command-guide/reference/commands/workflow/init.md +164 -164
  99. package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +748 -686
  100. package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +664 -621
  101. package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +645 -592
  102. package/.claude/skills/command-guide/reference/commands/workflow/plan.md +5 -5
  103. package/.claude/skills/command-guide/reference/commands/workflow/review.md +25 -18
  104. package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +547 -500
  105. package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +45 -27
  106. package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +35 -19
  107. package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +90 -33
  108. package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +3 -3
  109. package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +23 -9
  110. package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +699 -699
  111. package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +103 -17
  112. package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +434 -434
  113. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +487 -291
  114. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +17 -10
  115. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +1 -1
  116. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +6 -6
  117. package/.claude/workflows/chinese-response.md +38 -0
  118. package/.claude/workflows/cli-templates/prompts/rules/rule-api.txt +122 -0
  119. package/.claude/workflows/cli-templates/prompts/rules/rule-components.txt +122 -0
  120. package/.claude/workflows/cli-templates/prompts/rules/rule-config.txt +89 -0
  121. package/.claude/workflows/cli-templates/prompts/rules/rule-core.txt +60 -0
  122. package/.claude/workflows/cli-templates/prompts/rules/rule-patterns.txt +70 -0
  123. package/.claude/workflows/cli-templates/prompts/rules/rule-testing.txt +81 -0
  124. package/.claude/workflows/cli-templates/prompts/rules/tech-rules-agent-prompt.txt +89 -0
  125. package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +131 -131
  126. package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +5 -9
  127. package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +5 -9
  128. package/.claude/workflows/cli-templates/protocols/analysis-protocol.md +112 -0
  129. package/.claude/workflows/cli-templates/protocols/write-protocol.md +201 -0
  130. package/.claude/workflows/cli-templates/schemas/conflict-resolution-schema.json +137 -0
  131. package/.claude/workflows/cli-templates/schemas/debug-log-json-schema.json +127 -0
  132. package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +25 -0
  133. package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +25 -0
  134. package/.claude/workflows/cli-tools-usage.md +526 -0
  135. package/{CLAUDE.md → .claude/workflows/coding-philosophy.md} +24 -45
  136. package/.claude/workflows/context-tools.md +84 -0
  137. package/.claude/workflows/file-modification.md +64 -0
  138. package/.claude/workflows/tool-strategy.md +216 -79
  139. package/.claude/workflows/windows-platform.md +16 -0
  140. package/.claude/workflows/workflow-architecture.md +942 -942
  141. package/.codex/AGENTS.md +63 -330
  142. package/.codex/prompts/debug.md +318 -0
  143. package/.codex/prompts/execute.md +273 -0
  144. package/.codex/prompts/lite-execute.md +164 -0
  145. package/.codex/prompts/lite-plan.md +469 -0
  146. package/.codex/prompts.zip +0 -0
  147. package/.gemini/GEMINI.md +25 -164
  148. package/.qwen/QWEN.md +0 -139
  149. package/README.md +29 -9
  150. package/ccw/README.md +30 -6
  151. package/ccw/bin/ccw-mcp.js +7 -0
  152. package/ccw/bin/ccw.js +9 -9
  153. package/ccw/package.json +65 -47
  154. package/ccw/src/.workflow/.cli-history/history.db +0 -0
  155. package/ccw/src/.workflow/.cli-history/history.db-shm +0 -0
  156. package/ccw/src/.workflow/.cli-history/history.db-wal +0 -0
  157. package/ccw/src/cli.ts +244 -0
  158. package/ccw/src/commands/cli.ts +740 -0
  159. package/ccw/src/commands/core-memory.ts +770 -0
  160. package/ccw/src/commands/hook.ts +315 -0
  161. package/ccw/src/commands/install.ts +519 -0
  162. package/ccw/src/commands/{list.js → list.ts} +1 -1
  163. package/ccw/src/commands/memory.ts +1090 -0
  164. package/ccw/src/commands/{serve.js → serve.ts} +14 -5
  165. package/ccw/src/commands/session-path-resolver.ts +372 -0
  166. package/ccw/src/commands/session.ts +1141 -0
  167. package/ccw/src/commands/{stop.js → stop.ts} +16 -6
  168. package/ccw/src/commands/tool.ts +201 -0
  169. package/ccw/src/commands/{uninstall.js → uninstall.ts} +89 -40
  170. package/ccw/src/commands/{upgrade.js → upgrade.ts} +68 -23
  171. package/ccw/src/commands/{view.js → view.ts} +22 -8
  172. package/ccw/src/config/storage-paths.ts +670 -0
  173. package/ccw/src/core/cache-manager.ts +294 -0
  174. package/ccw/src/core/claude-freshness.ts +319 -0
  175. package/ccw/src/core/core-memory-store.ts +1528 -0
  176. package/ccw/src/core/{dashboard-generator-patch.js → dashboard-generator-patch.ts} +18 -0
  177. package/ccw/src/core/{dashboard-generator.js → dashboard-generator.ts} +69 -12
  178. package/ccw/src/core/data-aggregator.ts +584 -0
  179. package/ccw/src/core/history-importer.ts +625 -0
  180. package/ccw/src/core/{lite-scanner.js → lite-scanner-complete.ts} +162 -66
  181. package/ccw/src/core/lite-scanner.ts +469 -0
  182. package/ccw/src/core/{manifest.js → manifest.ts} +104 -34
  183. package/ccw/src/core/memory-embedder-bridge.ts +262 -0
  184. package/ccw/src/core/memory-store.ts +978 -0
  185. package/ccw/src/core/routes/ccw-routes.ts +96 -0
  186. package/ccw/src/core/routes/claude-routes.ts +1183 -0
  187. package/ccw/src/core/routes/cli-routes.ts +561 -0
  188. package/ccw/src/core/routes/codexlens-routes.ts +806 -0
  189. package/ccw/src/core/routes/core-memory-routes.ts +605 -0
  190. package/ccw/src/core/routes/files-routes.ts +428 -0
  191. package/ccw/src/core/routes/graph-routes.md +164 -0
  192. package/ccw/src/core/routes/graph-routes.ts +626 -0
  193. package/ccw/src/core/routes/help-routes.ts +308 -0
  194. package/ccw/src/core/routes/hooks-routes.ts +405 -0
  195. package/ccw/src/core/routes/mcp-routes.ts +1271 -0
  196. package/ccw/src/core/routes/mcp-routes.ts.backup +550 -0
  197. package/ccw/src/core/routes/mcp-templates-db.ts +268 -0
  198. package/ccw/src/core/routes/memory-routes.ts +1206 -0
  199. package/ccw/src/core/routes/rules-routes.ts +526 -0
  200. package/ccw/src/core/routes/session-routes.ts +467 -0
  201. package/ccw/src/core/routes/skills-routes.ts +599 -0
  202. package/ccw/src/core/routes/status-routes.ts +57 -0
  203. package/ccw/src/core/routes/system-routes.ts +427 -0
  204. package/ccw/src/core/server.ts +431 -0
  205. package/ccw/src/core/session-clustering-service.ts +1258 -0
  206. package/ccw/src/core/session-scanner.ts +283 -0
  207. package/ccw/src/core/websocket.ts +190 -0
  208. package/ccw/src/{index.js → index.ts} +1 -0
  209. package/ccw/src/mcp-server/index.ts +186 -0
  210. package/ccw/src/templates/assets/css/github-dark.min.css +10 -0
  211. package/ccw/src/templates/assets/css/github.min.css +10 -0
  212. package/ccw/src/templates/assets/js/cytoscape.min.js +32 -0
  213. package/ccw/src/templates/assets/js/d3.min.js +2 -0
  214. package/ccw/src/templates/assets/js/highlight.min.js +1244 -0
  215. package/ccw/src/templates/assets/js/lucide.min.js +12 -0
  216. package/ccw/src/templates/assets/js/marked.min.js +69 -0
  217. package/ccw/src/templates/assets/js/tailwind.js +83 -0
  218. package/ccw/src/templates/dashboard-css/01-base.css +11 -0
  219. package/ccw/src/templates/dashboard-css/02-session.css +22 -0
  220. package/ccw/src/templates/dashboard-css/04-lite-tasks.css +10 -0
  221. package/ccw/src/templates/dashboard-css/06-cards.css +10 -4
  222. package/ccw/src/templates/dashboard-css/07-managers.css +1178 -7
  223. package/ccw/src/templates/dashboard-css/09-explorer.css +23 -12
  224. package/ccw/src/templates/dashboard-css/10-cli-status.css +337 -0
  225. package/ccw/src/templates/dashboard-css/11-cli-history.css +271 -0
  226. package/ccw/src/templates/dashboard-css/12-cli-legacy.css +796 -0
  227. package/ccw/src/templates/dashboard-css/13-cli-ccw.css +199 -0
  228. package/ccw/src/templates/dashboard-css/14-cli-modals.css +258 -0
  229. package/ccw/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
  230. package/ccw/src/templates/dashboard-css/16-cli-session.css +241 -0
  231. package/ccw/src/templates/dashboard-css/17-cli-conversation.css +283 -0
  232. package/ccw/src/templates/dashboard-css/18-cli-settings.css +160 -0
  233. package/ccw/src/templates/dashboard-css/19-cli-native-session.css +496 -0
  234. package/ccw/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
  235. package/ccw/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
  236. package/ccw/src/templates/dashboard-css/22-cli-semantic.css +240 -0
  237. package/ccw/src/templates/dashboard-css/23-memory.css +2390 -0
  238. package/ccw/src/templates/dashboard-css/24-prompt-history.css +1089 -0
  239. package/ccw/src/templates/dashboard-css/25-skills-rules.css +326 -0
  240. package/ccw/src/templates/dashboard-css/26-claude-manager.css +908 -0
  241. package/ccw/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
  242. package/ccw/src/templates/dashboard-css/28-mcp-manager.css +748 -0
  243. package/ccw/src/templates/dashboard-css/29-help.css +264 -0
  244. package/ccw/src/templates/dashboard-css/30-core-memory.css +1700 -0
  245. package/ccw/src/templates/dashboard-js/api.js +162 -142
  246. package/ccw/src/templates/dashboard-js/components/carousel.js +4 -4
  247. package/ccw/src/templates/dashboard-js/components/cli-history.js +876 -0
  248. package/ccw/src/templates/dashboard-js/components/cli-status.js +978 -0
  249. package/ccw/src/templates/dashboard-js/components/global-notifications.js +508 -219
  250. package/ccw/src/templates/dashboard-js/components/hook-manager.js +1277 -282
  251. package/ccw/src/templates/dashboard-js/components/index-manager.js +302 -0
  252. package/ccw/src/templates/dashboard-js/components/mcp-manager.js +718 -27
  253. package/ccw/src/templates/dashboard-js/components/modals.js +66 -0
  254. package/ccw/src/templates/dashboard-js/components/navigation.js +80 -12
  255. package/ccw/src/templates/dashboard-js/components/notifications.js +758 -194
  256. package/ccw/src/templates/dashboard-js/components/storage-manager.js +478 -0
  257. package/ccw/src/templates/dashboard-js/components/tabs-other.js +157 -6
  258. package/ccw/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
  259. package/ccw/src/templates/dashboard-js/help-i18n.js +272 -0
  260. package/ccw/src/templates/dashboard-js/i18n.js +2807 -0
  261. package/ccw/src/templates/dashboard-js/main.js +15 -0
  262. package/ccw/src/templates/dashboard-js/state.js +243 -42
  263. package/ccw/src/templates/dashboard-js/utils.js +47 -1
  264. package/ccw/src/templates/dashboard-js/views/claude-manager.js +912 -0
  265. package/ccw/src/templates/dashboard-js/views/cli-manager.js +2272 -0
  266. package/ccw/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
  267. package/ccw/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
  268. package/ccw/src/templates/dashboard-js/views/core-memory.js +782 -0
  269. package/ccw/src/templates/dashboard-js/views/explorer.js +888 -852
  270. package/ccw/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
  271. package/ccw/src/templates/dashboard-js/views/help.js +856 -0
  272. package/ccw/src/templates/dashboard-js/views/history.js +337 -0
  273. package/ccw/src/templates/dashboard-js/views/home.js +61 -15
  274. package/ccw/src/templates/dashboard-js/views/hook-manager.js +311 -43
  275. package/ccw/src/templates/dashboard-js/views/lite-tasks.js +204 -28
  276. package/ccw/src/templates/dashboard-js/views/mcp-manager.js +2187 -411
  277. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
  278. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
  279. package/ccw/src/templates/dashboard-js/views/memory.js +1221 -0
  280. package/ccw/src/templates/dashboard-js/views/prompt-history.js +713 -0
  281. package/ccw/src/templates/dashboard-js/views/rules-manager.js +828 -0
  282. package/ccw/src/templates/dashboard-js/views/session-detail.js +54 -53
  283. package/ccw/src/templates/dashboard-js/views/skills-manager.js +819 -0
  284. package/ccw/src/templates/dashboard.html +185 -85
  285. package/ccw/src/templates/hooks-config-example.json +60 -0
  286. package/ccw/src/tools/classify-folders.ts +245 -0
  287. package/ccw/src/tools/cli-config-manager.ts +268 -0
  288. package/ccw/src/tools/cli-executor.ts +2014 -0
  289. package/ccw/src/tools/cli-history-store.ts +1195 -0
  290. package/ccw/src/tools/codex-lens.ts +1141 -0
  291. package/ccw/src/tools/{convert-tokens-to-css.js → convert-tokens-to-css.ts} +73 -23
  292. package/ccw/src/tools/core-memory.ts +444 -0
  293. package/ccw/src/tools/detect-changed-modules.ts +325 -0
  294. package/ccw/src/tools/{discover-design-files.js → discover-design-files.ts} +74 -24
  295. package/ccw/src/tools/edit-file.ts +568 -0
  296. package/ccw/src/tools/{generate-module-docs.js → generate-module-docs.ts} +207 -185
  297. package/ccw/src/tools/{get-modules-by-depth.js → get-modules-by-depth.ts} +120 -79
  298. package/ccw/src/tools/index.ts +370 -0
  299. package/ccw/src/tools/native-session-discovery.ts +795 -0
  300. package/ccw/src/tools/notifier.ts +129 -0
  301. package/ccw/src/tools/read-file.ts +410 -0
  302. package/ccw/src/tools/resume-strategy.ts +345 -0
  303. package/ccw/src/tools/session-content-parser.ts +619 -0
  304. package/ccw/src/tools/session-manager.ts +1026 -0
  305. package/ccw/src/tools/smart-context.ts +228 -0
  306. package/ccw/src/tools/smart-search.ts +2065 -0
  307. package/ccw/src/tools/smart-search.ts.backup +1233 -0
  308. package/ccw/src/tools/storage-manager.ts +455 -0
  309. package/ccw/src/tools/write-file.ts +222 -0
  310. package/ccw/src/types/config.ts +11 -0
  311. package/ccw/src/types/index.ts +3 -0
  312. package/ccw/src/types/session.ts +25 -0
  313. package/ccw/src/types/tool.ts +41 -0
  314. package/ccw/src/utils/{browser-launcher.js → browser-launcher.ts} +10 -8
  315. package/ccw/src/utils/file-utils.ts +48 -0
  316. package/ccw/src/utils/{path-resolver.js → path-resolver.ts} +114 -78
  317. package/ccw/src/utils/path-validator.ts +153 -0
  318. package/ccw/src/utils/{ui.js → ui.ts} +32 -25
  319. package/codex-lens/pyproject.toml +48 -0
  320. package/codex-lens/src/codexlens/.workflow/.cli-history/history.db +0 -0
  321. package/codex-lens/src/codexlens/__init__.py +28 -0
  322. package/codex-lens/src/codexlens/__main__.py +14 -0
  323. package/codex-lens/src/codexlens/__pycache__/__init__.cpython-313.pyc +0 -0
  324. package/codex-lens/src/codexlens/__pycache__/__main__.cpython-313.pyc +0 -0
  325. package/codex-lens/src/codexlens/__pycache__/config.cpython-313.pyc +0 -0
  326. package/codex-lens/src/codexlens/__pycache__/entities.cpython-313.pyc +0 -0
  327. package/codex-lens/src/codexlens/__pycache__/errors.cpython-313.pyc +0 -0
  328. package/codex-lens/src/codexlens/cli/__init__.py +27 -0
  329. package/codex-lens/src/codexlens/cli/__pycache__/__init__.cpython-313.pyc +0 -0
  330. package/codex-lens/src/codexlens/cli/__pycache__/commands.cpython-313.pyc +0 -0
  331. package/codex-lens/src/codexlens/cli/__pycache__/embedding_manager.cpython-313.pyc +0 -0
  332. package/codex-lens/src/codexlens/cli/__pycache__/model_manager.cpython-313.pyc +0 -0
  333. package/codex-lens/src/codexlens/cli/__pycache__/output.cpython-313.pyc +0 -0
  334. package/codex-lens/src/codexlens/cli/commands.py +1931 -0
  335. package/codex-lens/src/codexlens/cli/embedding_manager.py +620 -0
  336. package/codex-lens/src/codexlens/cli/model_manager.py +289 -0
  337. package/codex-lens/src/codexlens/cli/output.py +124 -0
  338. package/codex-lens/src/codexlens/config.py +201 -0
  339. package/codex-lens/src/codexlens/entities.py +121 -0
  340. package/codex-lens/src/codexlens/errors.py +55 -0
  341. package/codex-lens/src/codexlens/indexing/README.md +77 -0
  342. package/codex-lens/src/codexlens/indexing/__init__.py +4 -0
  343. package/codex-lens/src/codexlens/indexing/__pycache__/__init__.cpython-313.pyc +0 -0
  344. package/codex-lens/src/codexlens/indexing/__pycache__/symbol_extractor.cpython-313.pyc +0 -0
  345. package/codex-lens/src/codexlens/indexing/symbol_extractor.py +243 -0
  346. package/codex-lens/src/codexlens/parsers/__init__.py +8 -0
  347. package/codex-lens/src/codexlens/parsers/__pycache__/__init__.cpython-313.pyc +0 -0
  348. package/codex-lens/src/codexlens/parsers/__pycache__/encoding.cpython-313.pyc +0 -0
  349. package/codex-lens/src/codexlens/parsers/__pycache__/factory.cpython-313.pyc +0 -0
  350. package/codex-lens/src/codexlens/parsers/__pycache__/tokenizer.cpython-313.pyc +0 -0
  351. package/codex-lens/src/codexlens/parsers/__pycache__/treesitter_parser.cpython-313.pyc +0 -0
  352. package/codex-lens/src/codexlens/parsers/encoding.py +202 -0
  353. package/codex-lens/src/codexlens/parsers/factory.py +256 -0
  354. package/codex-lens/src/codexlens/parsers/tokenizer.py +98 -0
  355. package/codex-lens/src/codexlens/parsers/treesitter_parser.py +335 -0
  356. package/codex-lens/src/codexlens/search/__init__.py +15 -0
  357. package/codex-lens/src/codexlens/search/__pycache__/__init__.cpython-313.pyc +0 -0
  358. package/codex-lens/src/codexlens/search/__pycache__/chain_search.cpython-313.pyc +0 -0
  359. package/codex-lens/src/codexlens/search/__pycache__/enrichment.cpython-313.pyc +0 -0
  360. package/codex-lens/src/codexlens/search/__pycache__/hybrid_search.cpython-313.pyc +0 -0
  361. package/codex-lens/src/codexlens/search/__pycache__/query_parser.cpython-313.pyc +0 -0
  362. package/codex-lens/src/codexlens/search/__pycache__/ranking.cpython-313.pyc +0 -0
  363. package/codex-lens/src/codexlens/search/chain_search.py +647 -0
  364. package/codex-lens/src/codexlens/search/enrichment.py +150 -0
  365. package/codex-lens/src/codexlens/search/hybrid_search.py +313 -0
  366. package/codex-lens/src/codexlens/search/query_parser.py +242 -0
  367. package/codex-lens/src/codexlens/search/ranking.py +274 -0
  368. package/codex-lens/src/codexlens/semantic/__init__.py +39 -0
  369. package/codex-lens/src/codexlens/semantic/__pycache__/__init__.cpython-313.pyc +0 -0
  370. package/codex-lens/src/codexlens/semantic/__pycache__/ann_index.cpython-313.pyc +0 -0
  371. package/codex-lens/src/codexlens/semantic/__pycache__/chunker.cpython-313.pyc +0 -0
  372. package/codex-lens/src/codexlens/semantic/__pycache__/code_extractor.cpython-313.pyc +0 -0
  373. package/codex-lens/src/codexlens/semantic/__pycache__/embedder.cpython-313.pyc +0 -0
  374. package/codex-lens/src/codexlens/semantic/__pycache__/graph_analyzer.cpython-313.pyc +0 -0
  375. package/codex-lens/src/codexlens/semantic/__pycache__/llm_enhancer.cpython-313.pyc +0 -0
  376. package/codex-lens/src/codexlens/semantic/__pycache__/vector_store.cpython-313.pyc +0 -0
  377. package/codex-lens/src/codexlens/semantic/ann_index.py +414 -0
  378. package/codex-lens/src/codexlens/semantic/chunker.py +448 -0
  379. package/codex-lens/src/codexlens/semantic/code_extractor.py +274 -0
  380. package/codex-lens/src/codexlens/semantic/embedder.py +185 -0
  381. package/codex-lens/src/codexlens/semantic/vector_store.py +955 -0
  382. package/codex-lens/src/codexlens/storage/__init__.py +29 -0
  383. package/codex-lens/src/codexlens/storage/__pycache__/__init__.cpython-313.pyc +0 -0
  384. package/codex-lens/src/codexlens/storage/__pycache__/dir_index.cpython-313.pyc +0 -0
  385. package/codex-lens/src/codexlens/storage/__pycache__/file_cache.cpython-313.pyc +0 -0
  386. package/codex-lens/src/codexlens/storage/__pycache__/index_tree.cpython-313.pyc +0 -0
  387. package/codex-lens/src/codexlens/storage/__pycache__/migration_manager.cpython-313.pyc +0 -0
  388. package/codex-lens/src/codexlens/storage/__pycache__/path_mapper.cpython-313.pyc +0 -0
  389. package/codex-lens/src/codexlens/storage/__pycache__/registry.cpython-313.pyc +0 -0
  390. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_store.cpython-313.pyc +0 -0
  391. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_utils.cpython-313.pyc +0 -0
  392. package/codex-lens/src/codexlens/storage/dir_index.py +1850 -0
  393. package/codex-lens/src/codexlens/storage/file_cache.py +32 -0
  394. package/codex-lens/src/codexlens/storage/index_tree.py +776 -0
  395. package/codex-lens/src/codexlens/storage/migration_manager.py +154 -0
  396. package/codex-lens/src/codexlens/storage/migrations/__init__.py +1 -0
  397. package/codex-lens/src/codexlens/storage/migrations/__pycache__/__init__.cpython-313.pyc +0 -0
  398. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_001_normalize_keywords.cpython-313.pyc +0 -0
  399. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_002_add_token_metadata.cpython-313.pyc +0 -0
  400. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_003_code_relationships.cpython-313.pyc +0 -0
  401. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_004_dual_fts.cpython-313.pyc +0 -0
  402. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_005_cleanup_unused_fields.cpython-313.pyc +0 -0
  403. package/codex-lens/src/codexlens/storage/migrations/migration_001_normalize_keywords.py +123 -0
  404. package/codex-lens/src/codexlens/storage/migrations/migration_002_add_token_metadata.py +48 -0
  405. package/codex-lens/src/codexlens/storage/migrations/migration_004_dual_fts.py +232 -0
  406. package/codex-lens/src/codexlens/storage/migrations/migration_005_cleanup_unused_fields.py +196 -0
  407. package/codex-lens/src/codexlens/storage/path_mapper.py +274 -0
  408. package/codex-lens/src/codexlens/storage/registry.py +670 -0
  409. package/codex-lens/src/codexlens/storage/sqlite_store.py +576 -0
  410. package/codex-lens/src/codexlens/storage/sqlite_utils.py +64 -0
  411. package/package.json +4 -1
  412. package/.claude/commands/memory/tech-research.md +0 -477
  413. package/.claude/scripts/classify-folders.sh +0 -39
  414. package/.claude/scripts/convert_tokens_to_css.sh +0 -229
  415. package/.claude/scripts/detect_changed_modules.sh +0 -161
  416. package/.claude/scripts/discover-design-files.sh +0 -87
  417. package/.claude/scripts/extract-animations.js +0 -243
  418. package/.claude/scripts/extract-computed-styles.js +0 -118
  419. package/.claude/scripts/extract-layout-structure.js +0 -411
  420. package/.claude/scripts/generate_module_docs.sh +0 -717
  421. package/.claude/scripts/get_modules_by_depth.sh +0 -170
  422. package/.claude/scripts/ui-generate-preview.sh +0 -395
  423. package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
  424. package/.claude/scripts/update_module_claude.sh +0 -337
  425. package/.claude/workflows/context-search-strategy.md +0 -77
  426. package/.claude/workflows/intelligent-tools-strategy.md +0 -662
  427. package/ccw/src/cli.js +0 -119
  428. package/ccw/src/commands/install.js +0 -324
  429. package/ccw/src/commands/tool.js +0 -138
  430. package/ccw/src/core/data-aggregator.js +0 -409
  431. package/ccw/src/core/server.js +0 -2063
  432. package/ccw/src/core/session-scanner.js +0 -235
  433. package/ccw/src/tools/classify-folders.js +0 -204
  434. package/ccw/src/tools/detect-changed-modules.js +0 -288
  435. package/ccw/src/tools/edit-file.js +0 -266
  436. package/ccw/src/tools/index.js +0 -176
  437. package/ccw/src/utils/file-utils.js +0 -48
@@ -0,0 +1,625 @@
1
+ /**
2
+ * History Importer - Import Claude Code history into memory store
3
+ * Supports global history.jsonl and project session JSONL files
4
+ *
5
+ * Usage:
6
+ * ```typescript
7
+ * const importer = new HistoryImporter('path/to/database.db');
8
+ *
9
+ * // Import global history (incremental)
10
+ * const globalResult = await importer.importGlobalHistory();
11
+ * console.log(`Imported ${globalResult.imported} entries`);
12
+ *
13
+ * // Import all sessions for a project
14
+ * const projectResult = await importer.importProjectSessions('D--Claude-dms3');
15
+ * console.log(`Imported ${projectResult.imported} messages from project sessions`);
16
+ *
17
+ * // Import specific session
18
+ * const sessionResult = await importer.importSession('path/to/session.jsonl');
19
+ *
20
+ * // Get import status
21
+ * const status = importer.getImportStatus();
22
+ * console.log(`Total imported: ${status.totalImported}`);
23
+ *
24
+ * importer.close();
25
+ * ```
26
+ */
27
+
28
+ import { createReadStream, existsSync, readdirSync, statSync } from 'fs';
29
+ import { createInterface } from 'readline';
30
+ import { join, basename } from 'path';
31
+ import { createHash } from 'crypto';
32
+ import Database from 'better-sqlite3';
33
+
34
+ // Type definitions
35
+ interface GlobalHistoryEntry {
36
+ display: string;
37
+ pastedContents: object;
38
+ timestamp: number;
39
+ project: string;
40
+ sessionId: string;
41
+ }
42
+
43
+ interface SessionEntry {
44
+ type: 'user' | 'assistant' | 'file-history-snapshot';
45
+ message?: {
46
+ role: 'user' | 'assistant';
47
+ content: string | ContentBlock[];
48
+ model?: string;
49
+ usage?: {
50
+ input_tokens: number;
51
+ output_tokens: number;
52
+ cache_creation_input_tokens?: number;
53
+ cache_read_input_tokens?: number;
54
+ };
55
+ };
56
+ sessionId: string;
57
+ timestamp: string;
58
+ cwd?: string;
59
+ gitBranch?: string;
60
+ todos?: any[];
61
+ uuid?: string;
62
+ parentUuid?: string;
63
+ }
64
+
65
+ interface ContentBlock {
66
+ type: 'text' | 'thinking' | 'tool_use' | 'tool_result';
67
+ text?: string;
68
+ thinking?: string;
69
+ name?: string;
70
+ input?: object;
71
+ content?: string;
72
+ id?: string;
73
+ }
74
+
75
+ export interface ImportResult {
76
+ imported: number;
77
+ skipped: number;
78
+ errors: number;
79
+ }
80
+
81
+ export interface ImportStatus {
82
+ lastGlobalImport?: string;
83
+ lastSessionImport?: string;
84
+ totalImported: number;
85
+ sessions: Map<string, { imported: number; lastUpdate: string }>;
86
+ }
87
+
88
+ /**
89
+ * History Importer for Claude Code
90
+ */
91
+ export class HistoryImporter {
92
+ private db: Database.Database;
93
+ private status: ImportStatus;
94
+
95
+ constructor(dbPath: string) {
96
+ this.db = new Database(dbPath);
97
+ this.db.pragma('journal_mode = WAL');
98
+ this.status = {
99
+ totalImported: 0,
100
+ sessions: new Map()
101
+ };
102
+ this.initSchema();
103
+ }
104
+
105
+ /**
106
+ * Initialize database schema for conversation history
107
+ * NOTE: Schema aligned with MemoryStore for seamless importing
108
+ */
109
+ private initSchema(): void {
110
+ this.db.exec(`
111
+ -- Conversations table (aligned with MemoryStore schema)
112
+ CREATE TABLE IF NOT EXISTS conversations (
113
+ id TEXT PRIMARY KEY,
114
+ source TEXT DEFAULT 'ccw',
115
+ external_id TEXT,
116
+ project_name TEXT,
117
+ git_branch TEXT,
118
+ created_at TEXT NOT NULL,
119
+ updated_at TEXT NOT NULL,
120
+ quality_score INTEGER,
121
+ turn_count INTEGER DEFAULT 0,
122
+ prompt_preview TEXT
123
+ );
124
+
125
+ -- Messages table (aligned with MemoryStore schema)
126
+ CREATE TABLE IF NOT EXISTS messages (
127
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
128
+ conversation_id TEXT NOT NULL,
129
+ role TEXT NOT NULL CHECK(role IN ('user', 'assistant', 'system')),
130
+ content_text TEXT,
131
+ content_json TEXT,
132
+ timestamp TEXT NOT NULL,
133
+ token_count INTEGER,
134
+ FOREIGN KEY (conversation_id) REFERENCES conversations(id) ON DELETE CASCADE
135
+ );
136
+
137
+ -- Tool calls table (aligned with MemoryStore schema)
138
+ CREATE TABLE IF NOT EXISTS tool_calls (
139
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
140
+ message_id INTEGER NOT NULL,
141
+ tool_name TEXT NOT NULL,
142
+ tool_args TEXT,
143
+ tool_output TEXT,
144
+ status TEXT,
145
+ duration_ms INTEGER,
146
+ FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE CASCADE
147
+ );
148
+
149
+ -- Import tracking table
150
+ CREATE TABLE IF NOT EXISTS import_metadata (
151
+ key TEXT PRIMARY KEY,
152
+ value TEXT,
153
+ updated_at TEXT
154
+ );
155
+
156
+ -- Deduplication table (hash-based)
157
+ CREATE TABLE IF NOT EXISTS message_hashes (
158
+ hash TEXT PRIMARY KEY,
159
+ message_id TEXT NOT NULL,
160
+ created_at TEXT NOT NULL
161
+ );
162
+
163
+ -- Indexes (aligned with MemoryStore)
164
+ CREATE INDEX IF NOT EXISTS idx_conversations_created ON conversations(created_at DESC);
165
+ CREATE INDEX IF NOT EXISTS idx_conversations_updated ON conversations(updated_at DESC);
166
+ CREATE INDEX IF NOT EXISTS idx_messages_conversation ON messages(conversation_id);
167
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_message ON tool_calls(message_id);
168
+ `);
169
+ }
170
+
171
+ /**
172
+ * Import from global history.jsonl (incremental)
173
+ */
174
+ async importGlobalHistory(historyPath?: string): Promise<ImportResult> {
175
+ const path = historyPath || join(process.env.USERPROFILE || process.env.HOME || '', '.claude', 'history.jsonl');
176
+
177
+ if (!existsSync(path)) {
178
+ return { imported: 0, skipped: 0, errors: 0 };
179
+ }
180
+
181
+ const result: ImportResult = { imported: 0, skipped: 0, errors: 0 };
182
+ const lastImportTime = this.getLastImportTime('global_history');
183
+
184
+ const fileStream = createReadStream(path, { encoding: 'utf8' });
185
+ const rl = createInterface({ input: fileStream, crlfDelay: Infinity });
186
+
187
+ const batch: GlobalHistoryEntry[] = [];
188
+ const BATCH_SIZE = 100;
189
+
190
+ for await (const line of rl) {
191
+ if (!line.trim()) continue;
192
+
193
+ try {
194
+ const entry: GlobalHistoryEntry = JSON.parse(line);
195
+
196
+ // Skip if already imported
197
+ if (lastImportTime && entry.timestamp <= new Date(lastImportTime).getTime()) {
198
+ result.skipped++;
199
+ continue;
200
+ }
201
+
202
+ batch.push(entry);
203
+
204
+ if (batch.length >= BATCH_SIZE) {
205
+ const batchResult = this.insertGlobalHistoryBatch(batch);
206
+ result.imported += batchResult.imported;
207
+ result.skipped += batchResult.skipped;
208
+ result.errors += batchResult.errors;
209
+ batch.length = 0;
210
+ }
211
+ } catch (err) {
212
+ result.errors++;
213
+ console.error(`Failed to parse line: ${(err as Error).message}`);
214
+ }
215
+ }
216
+
217
+ // Process remaining batch
218
+ if (batch.length > 0) {
219
+ const batchResult = this.insertGlobalHistoryBatch(batch);
220
+ result.imported += batchResult.imported;
221
+ result.skipped += batchResult.skipped;
222
+ result.errors += batchResult.errors;
223
+ }
224
+
225
+ if (result.imported > 0) {
226
+ this.updateLastImportTime('global_history');
227
+ }
228
+
229
+ this.status.lastGlobalImport = new Date().toISOString();
230
+ this.status.totalImported += result.imported;
231
+
232
+ return result;
233
+ }
234
+
235
+ /**
236
+ * Import full session from projects folder
237
+ */
238
+ async importSession(sessionFilePath: string): Promise<ImportResult> {
239
+ if (!existsSync(sessionFilePath)) {
240
+ return { imported: 0, skipped: 0, errors: 0 };
241
+ }
242
+
243
+ const result: ImportResult = { imported: 0, skipped: 0, errors: 0 };
244
+ const sessionId = basename(sessionFilePath, '.jsonl');
245
+
246
+ const fileStream = createReadStream(sessionFilePath, { encoding: 'utf8' });
247
+ const rl = createInterface({ input: fileStream, crlfDelay: Infinity });
248
+
249
+ const messages: SessionEntry[] = [];
250
+ let conversationMetadata: any = {};
251
+
252
+ for await (const line of rl) {
253
+ if (!line.trim()) continue;
254
+
255
+ try {
256
+ const entry: SessionEntry = JSON.parse(line);
257
+
258
+ if (entry.type === 'user' || entry.type === 'assistant') {
259
+ messages.push(entry);
260
+
261
+ // Extract metadata from first message
262
+ if (messages.length === 1) {
263
+ conversationMetadata = {
264
+ sessionId: entry.sessionId,
265
+ cwd: entry.cwd,
266
+ gitBranch: entry.gitBranch
267
+ };
268
+ }
269
+ }
270
+ } catch (err) {
271
+ result.errors++;
272
+ console.error(`Failed to parse session line: ${(err as Error).message}`);
273
+ }
274
+ }
275
+
276
+ if (messages.length > 0) {
277
+ const importResult = this.insertSessionMessages(sessionId, messages, conversationMetadata);
278
+ result.imported = importResult.imported;
279
+ result.skipped = importResult.skipped;
280
+ result.errors += importResult.errors;
281
+ }
282
+
283
+ this.status.lastSessionImport = new Date().toISOString();
284
+ this.status.totalImported += result.imported;
285
+ this.status.sessions.set(sessionId, {
286
+ imported: result.imported,
287
+ lastUpdate: new Date().toISOString()
288
+ });
289
+
290
+ return result;
291
+ }
292
+
293
+ /**
294
+ * Scan and import all sessions for a project
295
+ */
296
+ async importProjectSessions(projectName: string): Promise<ImportResult> {
297
+ const projectsDir = join(process.env.USERPROFILE || process.env.HOME || '', '.claude', 'projects');
298
+ const projectDir = join(projectsDir, projectName);
299
+
300
+ if (!existsSync(projectDir)) {
301
+ return { imported: 0, skipped: 0, errors: 0 };
302
+ }
303
+
304
+ const result: ImportResult = { imported: 0, skipped: 0, errors: 0 };
305
+ const sessionFiles = readdirSync(projectDir).filter(f => f.endsWith('.jsonl'));
306
+
307
+ for (const sessionFile of sessionFiles) {
308
+ const sessionPath = join(projectDir, sessionFile);
309
+ const sessionResult = await this.importSession(sessionPath);
310
+
311
+ result.imported += sessionResult.imported;
312
+ result.skipped += sessionResult.skipped;
313
+ result.errors += sessionResult.errors;
314
+ }
315
+
316
+ return result;
317
+ }
318
+
319
+ /**
320
+ * Get import status
321
+ */
322
+ getImportStatus(): ImportStatus {
323
+ return { ...this.status };
324
+ }
325
+
326
+ /**
327
+ * Insert global history batch
328
+ */
329
+ private insertGlobalHistoryBatch(entries: GlobalHistoryEntry[]): ImportResult {
330
+ const result: ImportResult = { imported: 0, skipped: 0, errors: 0 };
331
+
332
+ const upsertConversation = this.db.prepare(`
333
+ INSERT INTO conversations (id, source, external_id, project_name, created_at, updated_at, turn_count, prompt_preview)
334
+ VALUES (@id, @source, @external_id, @project_name, @created_at, @updated_at, 1, @prompt_preview)
335
+ ON CONFLICT(id) DO UPDATE SET
336
+ updated_at = @updated_at,
337
+ turn_count = turn_count + 1,
338
+ prompt_preview = @prompt_preview
339
+ `);
340
+
341
+ const upsertMessage = this.db.prepare(`
342
+ INSERT INTO messages (conversation_id, role, content_text, timestamp)
343
+ VALUES (@conversation_id, 'user', @content_text, @timestamp)
344
+ `);
345
+
346
+ const insertHash = this.db.prepare(`
347
+ INSERT OR IGNORE INTO message_hashes (hash, message_id, created_at)
348
+ VALUES (@hash, @message_id, @created_at)
349
+ `);
350
+
351
+ const transaction = this.db.transaction(() => {
352
+ for (const entry of entries) {
353
+ try {
354
+ const timestamp = new Date(entry.timestamp).toISOString();
355
+ const hash = this.generateHash(entry.sessionId, timestamp, entry.display);
356
+
357
+ // Check if hash exists
358
+ const existing = this.db.prepare('SELECT message_id FROM message_hashes WHERE hash = ?').get(hash);
359
+ if (existing) {
360
+ result.skipped++;
361
+ continue;
362
+ }
363
+
364
+ // Insert conversation (using MemoryStore-compatible fields)
365
+ upsertConversation.run({
366
+ id: entry.sessionId,
367
+ source: 'global_history',
368
+ external_id: entry.sessionId,
369
+ project_name: entry.project,
370
+ created_at: timestamp,
371
+ updated_at: timestamp,
372
+ prompt_preview: entry.display.substring(0, 100)
373
+ });
374
+
375
+ // Insert message (using MemoryStore-compatible fields)
376
+ const insertResult = upsertMessage.run({
377
+ conversation_id: entry.sessionId,
378
+ content_text: entry.display,
379
+ timestamp
380
+ });
381
+
382
+ // Insert hash (using actual message ID from insert)
383
+ insertHash.run({
384
+ hash,
385
+ message_id: String(insertResult.lastInsertRowid),
386
+ created_at: timestamp
387
+ });
388
+
389
+ result.imported++;
390
+ } catch (err) {
391
+ result.errors++;
392
+ console.error(`Failed to insert entry: ${(err as Error).message}`);
393
+ }
394
+ }
395
+ });
396
+
397
+ transaction();
398
+ return result;
399
+ }
400
+
401
+ /**
402
+ * Insert session messages
403
+ */
404
+ private insertSessionMessages(
405
+ sessionId: string,
406
+ messages: SessionEntry[],
407
+ metadata: any
408
+ ): ImportResult {
409
+ const result: ImportResult = { imported: 0, skipped: 0, errors: 0 };
410
+
411
+ const upsertConversation = this.db.prepare(`
412
+ INSERT INTO conversations (id, source, external_id, project_name, git_branch, created_at, updated_at, turn_count, prompt_preview)
413
+ VALUES (@id, @source, @external_id, @project_name, @git_branch, @created_at, @updated_at, @turn_count, @prompt_preview)
414
+ ON CONFLICT(id) DO UPDATE SET
415
+ updated_at = @updated_at,
416
+ turn_count = @turn_count,
417
+ prompt_preview = @prompt_preview
418
+ `);
419
+
420
+ const upsertMessage = this.db.prepare(`
421
+ INSERT INTO messages (conversation_id, role, content_text, content_json, timestamp, token_count)
422
+ VALUES (@conversation_id, @role, @content_text, @content_json, @timestamp, @token_count)
423
+ `);
424
+
425
+ const insertToolCall = this.db.prepare(`
426
+ INSERT INTO tool_calls (message_id, tool_name, tool_args, tool_output, status)
427
+ VALUES (@message_id, @tool_name, @tool_args, @tool_output, @status)
428
+ `);
429
+
430
+ const insertHash = this.db.prepare(`
431
+ INSERT OR IGNORE INTO message_hashes (hash, message_id, created_at)
432
+ VALUES (@hash, @message_id, @created_at)
433
+ `);
434
+
435
+ const transaction = this.db.transaction(() => {
436
+ const firstMessage = messages[0];
437
+ const lastMessage = messages[messages.length - 1];
438
+ const promptPreview = firstMessage?.message
439
+ ? this.extractTextContent(firstMessage.message.content).substring(0, 100)
440
+ : '';
441
+
442
+ // Insert conversation FIRST (before messages, for foreign key constraint)
443
+ upsertConversation.run({
444
+ id: sessionId,
445
+ source: 'session_file',
446
+ external_id: sessionId,
447
+ project_name: metadata.cwd || null,
448
+ git_branch: metadata.gitBranch || null,
449
+ created_at: firstMessage.timestamp,
450
+ updated_at: lastMessage.timestamp,
451
+ turn_count: 0,
452
+ prompt_preview: promptPreview
453
+ });
454
+
455
+ for (const msg of messages) {
456
+ if (!msg.message) continue;
457
+
458
+ try {
459
+ const content = this.extractTextContent(msg.message.content);
460
+ const hash = this.generateHash(sessionId, msg.timestamp, content);
461
+
462
+ // Check if hash exists
463
+ const existing = this.db.prepare('SELECT message_id FROM message_hashes WHERE hash = ?').get(hash);
464
+ if (existing) {
465
+ result.skipped++;
466
+ continue;
467
+ }
468
+
469
+ // Calculate total tokens
470
+ const inputTokens = msg.message.usage?.input_tokens || 0;
471
+ const outputTokens = msg.message.usage?.output_tokens || 0;
472
+ const totalTokens = inputTokens + outputTokens;
473
+
474
+ // Store content as JSON if complex, otherwise as text
475
+ const contentJson = typeof msg.message.content === 'object'
476
+ ? JSON.stringify(msg.message.content)
477
+ : null;
478
+
479
+ // Insert message (using MemoryStore-compatible fields)
480
+ const insertResult = upsertMessage.run({
481
+ conversation_id: sessionId,
482
+ role: msg.message.role,
483
+ content_text: content,
484
+ content_json: contentJson,
485
+ timestamp: msg.timestamp,
486
+ token_count: totalTokens
487
+ });
488
+
489
+ const messageId = insertResult.lastInsertRowid as number;
490
+
491
+ // Extract and insert tool calls
492
+ const toolCalls = this.extractToolCalls(msg.message.content);
493
+ for (const tool of toolCalls) {
494
+ insertToolCall.run({
495
+ message_id: messageId,
496
+ tool_name: tool.name,
497
+ tool_args: JSON.stringify(tool.input),
498
+ tool_output: tool.result || null,
499
+ status: 'success'
500
+ });
501
+ }
502
+
503
+ // Insert hash (using actual message ID from insert)
504
+ insertHash.run({
505
+ hash,
506
+ message_id: String(messageId),
507
+ created_at: msg.timestamp
508
+ });
509
+
510
+ result.imported++;
511
+ } catch (err) {
512
+ result.errors++;
513
+ console.error(`Failed to insert message: ${(err as Error).message}`);
514
+ }
515
+ }
516
+
517
+ // Update conversation with final counts
518
+ upsertConversation.run({
519
+ id: sessionId,
520
+ source: 'session_file',
521
+ external_id: sessionId,
522
+ project_name: metadata.cwd || null,
523
+ git_branch: metadata.gitBranch || null,
524
+ created_at: firstMessage.timestamp,
525
+ updated_at: lastMessage.timestamp,
526
+ turn_count: result.imported,
527
+ prompt_preview: promptPreview
528
+ });
529
+ });
530
+
531
+ transaction();
532
+ return result;
533
+ }
534
+
535
+ /**
536
+ * Extract text content from message content
537
+ */
538
+ private extractTextContent(content: string | ContentBlock[]): string {
539
+ if (typeof content === 'string') {
540
+ return content;
541
+ }
542
+
543
+ return content
544
+ .filter(block => block.type === 'text' || block.type === 'thinking')
545
+ .map(block => block.text || block.thinking || '')
546
+ .join('\n\n');
547
+ }
548
+
549
+ /**
550
+ * Extract tool calls from content blocks
551
+ */
552
+ private extractToolCalls(content: string | ContentBlock[]): Array<{
553
+ id?: string;
554
+ name: string;
555
+ input?: object;
556
+ result?: string;
557
+ }> {
558
+ if (typeof content === 'string') {
559
+ return [];
560
+ }
561
+
562
+ const toolCalls: Array<{ id?: string; name: string; input?: object; result?: string }> = [];
563
+ const toolResultMap = new Map<string, string>();
564
+
565
+ // First pass: collect tool results
566
+ for (const block of content) {
567
+ if (block.type === 'tool_result' && block.id) {
568
+ toolResultMap.set(block.id, block.content || '');
569
+ }
570
+ }
571
+
572
+ // Second pass: collect tool uses with their results
573
+ for (const block of content) {
574
+ if (block.type === 'tool_use' && block.name) {
575
+ toolCalls.push({
576
+ id: block.id,
577
+ name: block.name,
578
+ input: block.input,
579
+ result: block.id ? toolResultMap.get(block.id) : undefined
580
+ });
581
+ }
582
+ }
583
+
584
+ return toolCalls;
585
+ }
586
+
587
+ /**
588
+ * Generate SHA256 hash for deduplication
589
+ */
590
+ private generateHash(sessionId: string, timestamp: string, content: string): string {
591
+ const data = `${sessionId}:${timestamp}:${content}`;
592
+ return createHash('sha256').update(data).digest('hex');
593
+ }
594
+
595
+ /**
596
+ * Get last import time for a source
597
+ */
598
+ private getLastImportTime(source: string): string | null {
599
+ const result = this.db.prepare('SELECT value FROM import_metadata WHERE key = ?').get(source) as any;
600
+ return result?.value || null;
601
+ }
602
+
603
+ /**
604
+ * Update last import time
605
+ */
606
+ private updateLastImportTime(source: string): void {
607
+ const now = new Date().toISOString();
608
+ this.db.prepare(`
609
+ INSERT INTO import_metadata (key, value, updated_at)
610
+ VALUES (@key, @value, @updated_at)
611
+ ON CONFLICT(key) DO UPDATE SET value = @value, updated_at = @updated_at
612
+ `).run({
613
+ key: source,
614
+ value: now,
615
+ updated_at: now
616
+ });
617
+ }
618
+
619
+ /**
620
+ * Close database connection
621
+ */
622
+ close(): void {
623
+ this.db.close();
624
+ }
625
+ }