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,978 @@
1
+ /**
2
+ * Memory Store - SQLite Storage Backend
3
+ * Provides persistent storage for memory module with entity tracking, associations, and conversation history
4
+ */
5
+
6
+ import Database from 'better-sqlite3';
7
+ import { existsSync, mkdirSync } from 'fs';
8
+ import { join } from 'path';
9
+ import { StoragePaths, ensureStorageDir, getProjectId } from '../config/storage-paths.js';
10
+
11
+ // Types
12
+ export interface Entity {
13
+ id?: number;
14
+ type: 'file' | 'module' | 'topic' | 'url';
15
+ value: string;
16
+ normalized_value: string;
17
+ first_seen_at: string;
18
+ last_seen_at: string;
19
+ metadata?: string;
20
+ }
21
+
22
+ export interface AccessLog {
23
+ id?: number;
24
+ entity_id: number;
25
+ action: 'read' | 'write' | 'mention';
26
+ session_id?: string;
27
+ timestamp: string;
28
+ context_summary?: string;
29
+ }
30
+
31
+ export interface EntityStats {
32
+ entity_id: number;
33
+ read_count: number;
34
+ write_count: number;
35
+ mention_count: number;
36
+ heat_score: number;
37
+ }
38
+
39
+ export interface Association {
40
+ source_id: number;
41
+ target_id: number;
42
+ weight: number;
43
+ last_interaction_at?: string;
44
+ }
45
+
46
+ export interface PromptHistory {
47
+ id?: number;
48
+ session_id: string;
49
+ project_path?: string;
50
+ prompt_text?: string;
51
+ context_summary?: string;
52
+ timestamp: number;
53
+ hash?: string;
54
+ quality_score?: number;
55
+ intent_label?: string;
56
+ }
57
+
58
+ export interface PromptPattern {
59
+ id?: number;
60
+ pattern_type?: string;
61
+ frequency: number;
62
+ example_ids?: string;
63
+ last_detected?: number;
64
+ }
65
+
66
+ export interface Conversation {
67
+ id: string;
68
+ source?: string;
69
+ external_id?: string;
70
+ project_name?: string;
71
+ git_branch?: string;
72
+ created_at: string;
73
+ updated_at: string;
74
+ quality_score?: number;
75
+ turn_count: number;
76
+ prompt_preview?: string;
77
+ }
78
+
79
+ export interface Message {
80
+ id?: number;
81
+ conversation_id: string;
82
+ role: 'user' | 'assistant' | 'system';
83
+ content_text?: string;
84
+ content_json?: string;
85
+ timestamp: string;
86
+ token_count?: number;
87
+ }
88
+
89
+ export interface ToolCall {
90
+ id?: number;
91
+ message_id: number;
92
+ tool_name: string;
93
+ // NOTE: Naming inconsistency - using tool_args/tool_output vs tool_input/tool_result in HistoryImporter
94
+ // Kept for backward compatibility with existing databases
95
+ tool_args?: string;
96
+ tool_output?: string;
97
+ status?: string;
98
+ duration_ms?: number;
99
+ }
100
+
101
+ export interface HotEntity extends Entity {
102
+ stats: EntityStats;
103
+ }
104
+
105
+ export interface EntityWithAssociations extends Entity {
106
+ associations: Array<{
107
+ target: Entity;
108
+ weight: number;
109
+ last_interaction_at?: string;
110
+ }>;
111
+ }
112
+
113
+ /**
114
+ * Memory Store using SQLite
115
+ */
116
+ export class MemoryStore {
117
+ private db: Database.Database;
118
+ private dbPath: string;
119
+ private projectPath: string;
120
+
121
+ constructor(projectPath: string) {
122
+ this.projectPath = projectPath;
123
+ // Use centralized storage path
124
+ const paths = StoragePaths.project(projectPath);
125
+ const memoryDir = paths.memory;
126
+ ensureStorageDir(memoryDir);
127
+
128
+ this.dbPath = paths.memoryDb;
129
+ this.db = new Database(this.dbPath);
130
+ this.db.pragma('journal_mode = WAL');
131
+ this.db.pragma('synchronous = NORMAL');
132
+
133
+ this.initDatabase();
134
+ this.migrateSchema();
135
+ }
136
+
137
+ /**
138
+ * Initialize database schema
139
+ */
140
+ private initDatabase(): void {
141
+ this.db.exec(`
142
+ -- Entity table
143
+ CREATE TABLE IF NOT EXISTS entities (
144
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
145
+ type TEXT NOT NULL,
146
+ value TEXT NOT NULL,
147
+ normalized_value TEXT NOT NULL,
148
+ first_seen_at TEXT NOT NULL,
149
+ last_seen_at TEXT NOT NULL,
150
+ metadata TEXT,
151
+ UNIQUE(type, normalized_value)
152
+ );
153
+
154
+ -- Access logs table
155
+ CREATE TABLE IF NOT EXISTS access_logs (
156
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
157
+ entity_id INTEGER NOT NULL,
158
+ action TEXT NOT NULL,
159
+ session_id TEXT,
160
+ timestamp TEXT NOT NULL,
161
+ context_summary TEXT,
162
+ FOREIGN KEY (entity_id) REFERENCES entities(id) ON DELETE CASCADE
163
+ );
164
+
165
+ -- Entity statistics table
166
+ CREATE TABLE IF NOT EXISTS entity_stats (
167
+ entity_id INTEGER PRIMARY KEY,
168
+ read_count INTEGER DEFAULT 0,
169
+ write_count INTEGER DEFAULT 0,
170
+ mention_count INTEGER DEFAULT 0,
171
+ heat_score REAL DEFAULT 0,
172
+ FOREIGN KEY (entity_id) REFERENCES entities(id) ON DELETE CASCADE
173
+ );
174
+
175
+ -- Associations table
176
+ CREATE TABLE IF NOT EXISTS associations (
177
+ source_id INTEGER NOT NULL,
178
+ target_id INTEGER NOT NULL,
179
+ weight INTEGER DEFAULT 0,
180
+ last_interaction_at TEXT,
181
+ PRIMARY KEY (source_id, target_id),
182
+ FOREIGN KEY (source_id) REFERENCES entities(id) ON DELETE CASCADE,
183
+ FOREIGN KEY (target_id) REFERENCES entities(id) ON DELETE CASCADE
184
+ );
185
+
186
+ -- Prompt history table
187
+ CREATE TABLE IF NOT EXISTS prompt_history (
188
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
189
+ session_id TEXT NOT NULL,
190
+ project_path TEXT,
191
+ prompt_text TEXT,
192
+ context_summary TEXT,
193
+ timestamp INTEGER,
194
+ hash TEXT UNIQUE,
195
+ quality_score INTEGER,
196
+ intent_label TEXT
197
+ );
198
+
199
+ -- Prompt patterns table
200
+ CREATE TABLE IF NOT EXISTS prompt_patterns (
201
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
202
+ pattern_type TEXT,
203
+ frequency INTEGER,
204
+ example_ids TEXT,
205
+ last_detected INTEGER
206
+ );
207
+
208
+ -- Conversations table
209
+ CREATE TABLE IF NOT EXISTS conversations (
210
+ id TEXT PRIMARY KEY,
211
+ source TEXT DEFAULT 'ccw',
212
+ external_id TEXT,
213
+ project_name TEXT,
214
+ git_branch TEXT,
215
+ created_at TEXT,
216
+ updated_at TEXT,
217
+ quality_score INTEGER,
218
+ turn_count INTEGER,
219
+ prompt_preview TEXT
220
+ );
221
+
222
+ -- Messages table
223
+ CREATE TABLE IF NOT EXISTS messages (
224
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
225
+ conversation_id TEXT NOT NULL,
226
+ role TEXT NOT NULL,
227
+ content_text TEXT,
228
+ content_json TEXT,
229
+ timestamp TEXT NOT NULL,
230
+ token_count INTEGER,
231
+ FOREIGN KEY (conversation_id) REFERENCES conversations(id) ON DELETE CASCADE
232
+ );
233
+
234
+ -- Tool calls table
235
+ CREATE TABLE IF NOT EXISTS tool_calls (
236
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
237
+ message_id INTEGER NOT NULL,
238
+ tool_name TEXT NOT NULL,
239
+ tool_args TEXT,
240
+ tool_output TEXT,
241
+ status TEXT,
242
+ duration_ms INTEGER,
243
+ FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE CASCADE
244
+ );
245
+
246
+ -- Indexes for efficient queries
247
+ CREATE INDEX IF NOT EXISTS idx_entities_type ON entities(type);
248
+ CREATE INDEX IF NOT EXISTS idx_entities_normalized ON entities(normalized_value);
249
+ CREATE INDEX IF NOT EXISTS idx_entities_last_seen ON entities(last_seen_at DESC);
250
+ CREATE INDEX IF NOT EXISTS idx_access_logs_entity ON access_logs(entity_id);
251
+ CREATE INDEX IF NOT EXISTS idx_access_logs_timestamp ON access_logs(timestamp DESC);
252
+ CREATE INDEX IF NOT EXISTS idx_access_logs_session ON access_logs(session_id);
253
+ CREATE INDEX IF NOT EXISTS idx_entity_stats_heat ON entity_stats(heat_score DESC);
254
+ CREATE INDEX IF NOT EXISTS idx_associations_source ON associations(source_id);
255
+ CREATE INDEX IF NOT EXISTS idx_associations_target ON associations(target_id);
256
+ CREATE INDEX IF NOT EXISTS idx_prompt_history_session ON prompt_history(session_id);
257
+ CREATE INDEX IF NOT EXISTS idx_prompt_history_timestamp ON prompt_history(timestamp DESC);
258
+ CREATE INDEX IF NOT EXISTS idx_conversations_created ON conversations(created_at DESC);
259
+ CREATE INDEX IF NOT EXISTS idx_conversations_updated ON conversations(updated_at DESC);
260
+ CREATE INDEX IF NOT EXISTS idx_messages_conversation ON messages(conversation_id);
261
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_message ON tool_calls(message_id);
262
+
263
+ -- Full-text search for prompt history
264
+ CREATE VIRTUAL TABLE IF NOT EXISTS prompt_history_fts USING fts5(
265
+ prompt_text,
266
+ context_summary,
267
+ content='prompt_history',
268
+ content_rowid='id'
269
+ );
270
+
271
+ -- Triggers to keep FTS index updated
272
+ CREATE TRIGGER IF NOT EXISTS prompt_history_ai AFTER INSERT ON prompt_history BEGIN
273
+ INSERT INTO prompt_history_fts(rowid, prompt_text, context_summary)
274
+ VALUES (new.id, new.prompt_text, new.context_summary);
275
+ END;
276
+
277
+ CREATE TRIGGER IF NOT EXISTS prompt_history_ad AFTER DELETE ON prompt_history BEGIN
278
+ INSERT INTO prompt_history_fts(prompt_history_fts, rowid, prompt_text, context_summary)
279
+ VALUES('delete', old.id, old.prompt_text, old.context_summary);
280
+ END;
281
+
282
+ CREATE TRIGGER IF NOT EXISTS prompt_history_au AFTER UPDATE ON prompt_history BEGIN
283
+ INSERT INTO prompt_history_fts(prompt_history_fts, rowid, prompt_text, context_summary)
284
+ VALUES('delete', old.id, old.prompt_text, old.context_summary);
285
+ INSERT INTO prompt_history_fts(rowid, prompt_text, context_summary)
286
+ VALUES (new.id, new.prompt_text, new.context_summary);
287
+ END;
288
+ `);
289
+ }
290
+
291
+ /**
292
+ * Migrate schema for existing databases
293
+ */
294
+ private migrateSchema(): void {
295
+ try {
296
+ // Check if hierarchical storage columns exist in conversations table
297
+ const tableInfo = this.db.prepare('PRAGMA table_info(conversations)').all() as Array<{ name: string }>;
298
+ const hasProjectRoot = tableInfo.some(col => col.name === 'project_root');
299
+ const hasRelativePath = tableInfo.some(col => col.name === 'relative_path');
300
+
301
+ // Add hierarchical storage support columns
302
+ if (!hasProjectRoot) {
303
+ console.log('[Memory Store] Migrating database: adding project_root column for hierarchical storage...');
304
+ this.db.exec(`
305
+ ALTER TABLE conversations ADD COLUMN project_root TEXT;
306
+ `);
307
+ try {
308
+ this.db.exec(`CREATE INDEX IF NOT EXISTS idx_conversations_project_root ON conversations(project_root);`);
309
+ } catch (indexErr) {
310
+ console.warn('[Memory Store] Project root index creation warning:', (indexErr as Error).message);
311
+ }
312
+ console.log('[Memory Store] Migration complete: project_root column added');
313
+ }
314
+
315
+ if (!hasRelativePath) {
316
+ console.log('[Memory Store] Migrating database: adding relative_path column for hierarchical storage...');
317
+ this.db.exec(`
318
+ ALTER TABLE conversations ADD COLUMN relative_path TEXT;
319
+ `);
320
+ console.log('[Memory Store] Migration complete: relative_path column added');
321
+ }
322
+
323
+ // Add missing timestamp index for messages table (for time-based queries)
324
+ try {
325
+ const indexExists = this.db.prepare(`
326
+ SELECT name FROM sqlite_master
327
+ WHERE type='index' AND name='idx_messages_timestamp'
328
+ `).get();
329
+
330
+ if (!indexExists) {
331
+ console.log('[Memory Store] Adding missing timestamp index to messages table...');
332
+ this.db.exec(`CREATE INDEX IF NOT EXISTS idx_messages_timestamp ON messages(timestamp DESC);`);
333
+ console.log('[Memory Store] Migration complete: messages timestamp index added');
334
+ }
335
+ } catch (indexErr) {
336
+ console.warn('[Memory Store] Messages timestamp index creation warning:', (indexErr as Error).message);
337
+ }
338
+ } catch (err) {
339
+ console.error('[Memory Store] Migration error:', (err as Error).message);
340
+ // Don't throw - allow the store to continue working with existing schema
341
+ }
342
+ }
343
+
344
+ /**
345
+ * Upsert an entity
346
+ */
347
+ upsertEntity(entity: Entity): number {
348
+ const stmt = this.db.prepare(`
349
+ INSERT INTO entities (type, value, normalized_value, first_seen_at, last_seen_at, metadata)
350
+ VALUES (@type, @value, @normalized_value, @first_seen_at, @last_seen_at, @metadata)
351
+ ON CONFLICT(type, normalized_value) DO UPDATE SET
352
+ value = @value,
353
+ last_seen_at = @last_seen_at,
354
+ metadata = @metadata
355
+ RETURNING id
356
+ `);
357
+
358
+ const result = stmt.get({
359
+ type: entity.type,
360
+ value: entity.value,
361
+ normalized_value: entity.normalized_value,
362
+ first_seen_at: entity.first_seen_at,
363
+ last_seen_at: entity.last_seen_at,
364
+ metadata: entity.metadata || null
365
+ }) as { id: number };
366
+
367
+ return result.id;
368
+ }
369
+
370
+ /**
371
+ * Get entity by type and normalized value
372
+ */
373
+ getEntity(type: string, normalizedValue: string): Entity | null {
374
+ const stmt = this.db.prepare(`
375
+ SELECT * FROM entities WHERE type = ? AND normalized_value = ?
376
+ `);
377
+ return stmt.get(type, normalizedValue) as Entity | null;
378
+ }
379
+
380
+ /**
381
+ * Get entity by ID
382
+ */
383
+ getEntityById(id: number): Entity | null {
384
+ const stmt = this.db.prepare(`SELECT * FROM entities WHERE id = ?`);
385
+ return stmt.get(id) as Entity | null;
386
+ }
387
+
388
+ /**
389
+ * Get hot entities (by heat score)
390
+ */
391
+ getHotEntities(limit: number = 20): HotEntity[] {
392
+ const stmt = this.db.prepare(`
393
+ SELECT e.*, s.read_count, s.write_count, s.mention_count, s.heat_score
394
+ FROM entities e
395
+ INNER JOIN entity_stats s ON e.id = s.entity_id
396
+ ORDER BY s.heat_score DESC
397
+ LIMIT ?
398
+ `);
399
+
400
+ const rows = stmt.all(limit) as any[];
401
+ return rows.map(row => ({
402
+ id: row.id,
403
+ type: row.type,
404
+ value: row.value,
405
+ normalized_value: row.normalized_value,
406
+ first_seen_at: row.first_seen_at,
407
+ last_seen_at: row.last_seen_at,
408
+ metadata: row.metadata,
409
+ stats: {
410
+ entity_id: row.id,
411
+ read_count: row.read_count,
412
+ write_count: row.write_count,
413
+ mention_count: row.mention_count,
414
+ heat_score: row.heat_score
415
+ }
416
+ }));
417
+ }
418
+
419
+ /**
420
+ * Log entity access
421
+ */
422
+ logAccess(log: AccessLog): void {
423
+ const stmt = this.db.prepare(`
424
+ INSERT INTO access_logs (entity_id, action, session_id, timestamp, context_summary)
425
+ VALUES (@entity_id, @action, @session_id, @timestamp, @context_summary)
426
+ `);
427
+
428
+ stmt.run({
429
+ entity_id: log.entity_id,
430
+ action: log.action,
431
+ session_id: log.session_id || null,
432
+ timestamp: log.timestamp,
433
+ context_summary: log.context_summary || null
434
+ });
435
+ }
436
+
437
+ /**
438
+ * Get recent access logs for an entity
439
+ */
440
+ getRecentAccess(entityId: number, limit: number = 50): AccessLog[] {
441
+ const stmt = this.db.prepare(`
442
+ SELECT * FROM access_logs
443
+ WHERE entity_id = ?
444
+ ORDER BY timestamp DESC
445
+ LIMIT ?
446
+ `);
447
+ return stmt.all(entityId, limit) as AccessLog[];
448
+ }
449
+
450
+ /**
451
+ * Update entity statistics
452
+ */
453
+ updateStats(entityId: number, action: 'read' | 'write' | 'mention'): void {
454
+ const upsertStmt = this.db.prepare(`
455
+ INSERT INTO entity_stats (entity_id, read_count, write_count, mention_count, heat_score)
456
+ VALUES (@entity_id, 0, 0, 0, 0)
457
+ ON CONFLICT(entity_id) DO NOTHING
458
+ `);
459
+
460
+ upsertStmt.run({ entity_id: entityId });
461
+
462
+ const field = `${action}_count`;
463
+ const updateStmt = this.db.prepare(`
464
+ UPDATE entity_stats
465
+ SET ${field} = ${field} + 1
466
+ WHERE entity_id = ?
467
+ `);
468
+
469
+ updateStmt.run(entityId);
470
+ }
471
+
472
+ /**
473
+ * Get entity statistics
474
+ */
475
+ getStats(entityId: number): EntityStats | null {
476
+ const stmt = this.db.prepare(`SELECT * FROM entity_stats WHERE entity_id = ?`);
477
+ return stmt.get(entityId) as EntityStats | null;
478
+ }
479
+
480
+ /**
481
+ * Calculate and update heat score for an entity
482
+ */
483
+ calculateHeatScore(entityId: number): number {
484
+ const stats = this.getStats(entityId);
485
+ if (!stats) return 0;
486
+
487
+ const now = Date.now();
488
+ const logs = this.getRecentAccess(entityId, 100);
489
+
490
+ let recencyScore = 0;
491
+ for (const log of logs) {
492
+ const ageMs = now - new Date(log.timestamp).getTime();
493
+ const ageDays = ageMs / (1000 * 60 * 60 * 24);
494
+ const decay = Math.exp(-ageDays / 7); // 7-day half-life
495
+ recencyScore += decay;
496
+ }
497
+
498
+ const heatScore = (
499
+ stats.read_count * 1 +
500
+ stats.write_count * 3 +
501
+ stats.mention_count * 2 +
502
+ recencyScore * 5
503
+ );
504
+
505
+ const updateStmt = this.db.prepare(`
506
+ UPDATE entity_stats SET heat_score = ? WHERE entity_id = ?
507
+ `);
508
+ updateStmt.run(heatScore, entityId);
509
+
510
+ return heatScore;
511
+ }
512
+
513
+ /**
514
+ * Record association between entities
515
+ */
516
+ recordAssociation(sourceId: number, targetId: number, timestamp?: string): void {
517
+ const stmt = this.db.prepare(`
518
+ INSERT INTO associations (source_id, target_id, weight, last_interaction_at)
519
+ VALUES (@source_id, @target_id, 1, @last_interaction_at)
520
+ ON CONFLICT(source_id, target_id) DO UPDATE SET
521
+ weight = weight + 1,
522
+ last_interaction_at = @last_interaction_at
523
+ `);
524
+
525
+ stmt.run({
526
+ source_id: sourceId,
527
+ target_id: targetId,
528
+ last_interaction_at: timestamp || new Date().toISOString()
529
+ });
530
+ }
531
+
532
+ /**
533
+ * Get associations for an entity
534
+ */
535
+ getAssociations(entityId: number, limit: number = 20): EntityWithAssociations['associations'] {
536
+ const stmt = this.db.prepare(`
537
+ SELECT e.*, a.weight, a.last_interaction_at
538
+ FROM associations a
539
+ INNER JOIN entities e ON a.target_id = e.id
540
+ WHERE a.source_id = ?
541
+ ORDER BY a.weight DESC
542
+ LIMIT ?
543
+ `);
544
+
545
+ const rows = stmt.all(entityId, limit) as any[];
546
+ return rows.map(row => ({
547
+ target: {
548
+ id: row.id,
549
+ type: row.type,
550
+ value: row.value,
551
+ normalized_value: row.normalized_value,
552
+ first_seen_at: row.first_seen_at,
553
+ last_seen_at: row.last_seen_at,
554
+ metadata: row.metadata
555
+ },
556
+ weight: row.weight,
557
+ last_interaction_at: row.last_interaction_at
558
+ }));
559
+ }
560
+
561
+ /**
562
+ * Save prompt to history
563
+ */
564
+ savePrompt(prompt: PromptHistory): number {
565
+ const stmt = this.db.prepare(`
566
+ INSERT INTO prompt_history (session_id, project_path, prompt_text, context_summary, timestamp, hash, quality_score, intent_label)
567
+ VALUES (@session_id, @project_path, @prompt_text, @context_summary, @timestamp, @hash, @quality_score, @intent_label)
568
+ ON CONFLICT(hash) DO UPDATE SET
569
+ quality_score = @quality_score,
570
+ intent_label = @intent_label
571
+ RETURNING id
572
+ `);
573
+
574
+ const result = stmt.get({
575
+ session_id: prompt.session_id,
576
+ project_path: prompt.project_path || null,
577
+ prompt_text: prompt.prompt_text || null,
578
+ context_summary: prompt.context_summary || null,
579
+ timestamp: prompt.timestamp,
580
+ hash: prompt.hash || null,
581
+ quality_score: prompt.quality_score || null,
582
+ intent_label: prompt.intent_label || null
583
+ }) as { id: number };
584
+
585
+ return result.id;
586
+ }
587
+
588
+ /**
589
+ * Get prompt history for a session
590
+ */
591
+ getPromptHistory(sessionId: string, limit: number = 50): PromptHistory[] {
592
+ const stmt = this.db.prepare(`
593
+ SELECT * FROM prompt_history
594
+ WHERE session_id = ?
595
+ ORDER BY timestamp DESC
596
+ LIMIT ?
597
+ `);
598
+ return stmt.all(sessionId, limit) as PromptHistory[];
599
+ }
600
+
601
+ /**
602
+ * Search prompts by text
603
+ */
604
+ searchPrompts(query: string, limit: number = 20): PromptHistory[] {
605
+ const stmt = this.db.prepare(`
606
+ SELECT ph.* FROM prompt_history ph
607
+ INNER JOIN prompt_history_fts fts ON fts.rowid = ph.id
608
+ WHERE prompt_history_fts MATCH ?
609
+ ORDER BY ph.timestamp DESC
610
+ LIMIT ?
611
+ `);
612
+ return stmt.all(query, limit) as PromptHistory[];
613
+ }
614
+
615
+ /**
616
+ * Save or update a conversation
617
+ */
618
+ saveConversation(conversation: Conversation): void {
619
+ const stmt = this.db.prepare(`
620
+ INSERT INTO conversations (id, source, external_id, project_name, git_branch, created_at, updated_at, quality_score, turn_count, prompt_preview, project_root, relative_path)
621
+ VALUES (@id, @source, @external_id, @project_name, @git_branch, @created_at, @updated_at, @quality_score, @turn_count, @prompt_preview, @project_root, @relative_path)
622
+ ON CONFLICT(id) DO UPDATE SET
623
+ updated_at = @updated_at,
624
+ quality_score = @quality_score,
625
+ turn_count = @turn_count,
626
+ prompt_preview = @prompt_preview,
627
+ project_root = @project_root,
628
+ relative_path = @relative_path
629
+ `);
630
+
631
+ stmt.run({
632
+ id: conversation.id,
633
+ source: conversation.source || 'ccw',
634
+ external_id: conversation.external_id || null,
635
+ project_name: conversation.project_name || null,
636
+ git_branch: conversation.git_branch || null,
637
+ created_at: conversation.created_at,
638
+ updated_at: conversation.updated_at,
639
+ quality_score: conversation.quality_score || null,
640
+ turn_count: conversation.turn_count,
641
+ prompt_preview: conversation.prompt_preview || null,
642
+ project_root: this.projectPath,
643
+ relative_path: null // For future hierarchical tracking
644
+ });
645
+ }
646
+
647
+ /**
648
+ * Get conversations
649
+ */
650
+ getConversations(limit: number = 50, offset: number = 0): Conversation[] {
651
+ const stmt = this.db.prepare(`
652
+ SELECT * FROM conversations
653
+ ORDER BY updated_at DESC
654
+ LIMIT ? OFFSET ?
655
+ `);
656
+ return stmt.all(limit, offset) as Conversation[];
657
+ }
658
+
659
+ /**
660
+ * Get conversation by ID
661
+ */
662
+ getConversation(id: string): Conversation | null {
663
+ const stmt = this.db.prepare(`SELECT * FROM conversations WHERE id = ?`);
664
+ return stmt.get(id) as Conversation | null;
665
+ }
666
+
667
+ /**
668
+ * Save message
669
+ */
670
+ saveMessage(message: Message): number {
671
+ const stmt = this.db.prepare(`
672
+ INSERT INTO messages (conversation_id, role, content_text, content_json, timestamp, token_count)
673
+ VALUES (@conversation_id, @role, @content_text, @content_json, @timestamp, @token_count)
674
+ RETURNING id
675
+ `);
676
+
677
+ const result = stmt.get({
678
+ conversation_id: message.conversation_id,
679
+ role: message.role,
680
+ content_text: message.content_text || null,
681
+ content_json: message.content_json || null,
682
+ timestamp: message.timestamp,
683
+ token_count: message.token_count || null
684
+ }) as { id: number };
685
+
686
+ return result.id;
687
+ }
688
+
689
+ /**
690
+ * Get messages for a conversation
691
+ */
692
+ getMessages(conversationId: string): Message[] {
693
+ const stmt = this.db.prepare(`
694
+ SELECT * FROM messages
695
+ WHERE conversation_id = ?
696
+ ORDER BY timestamp ASC
697
+ `);
698
+ return stmt.all(conversationId) as Message[];
699
+ }
700
+
701
+ /**
702
+ * Save tool call
703
+ */
704
+ saveToolCall(toolCall: ToolCall): number {
705
+ const stmt = this.db.prepare(`
706
+ INSERT INTO tool_calls (message_id, tool_name, tool_args, tool_output, status, duration_ms)
707
+ VALUES (@message_id, @tool_name, @tool_args, @tool_output, @status, @duration_ms)
708
+ RETURNING id
709
+ `);
710
+
711
+ const result = stmt.get({
712
+ message_id: toolCall.message_id,
713
+ tool_name: toolCall.tool_name,
714
+ tool_args: toolCall.tool_args || null,
715
+ tool_output: toolCall.tool_output || null,
716
+ status: toolCall.status || null,
717
+ duration_ms: toolCall.duration_ms || null
718
+ }) as { id: number };
719
+
720
+ return result.id;
721
+ }
722
+
723
+ /**
724
+ * Get tool calls for a message
725
+ */
726
+ getToolCalls(messageId: number): ToolCall[] {
727
+ const stmt = this.db.prepare(`
728
+ SELECT * FROM tool_calls
729
+ WHERE message_id = ?
730
+ `);
731
+ return stmt.all(messageId) as ToolCall[];
732
+ }
733
+
734
+ /**
735
+ * Close database connection
736
+ */
737
+ close(): void {
738
+ this.db.close();
739
+ }
740
+ }
741
+
742
+ // Singleton instance cache - keyed by normalized project ID for consistency
743
+ const storeCache = new Map<string, MemoryStore>();
744
+
745
+ /**
746
+ * Get or create a store instance for a project
747
+ * Uses normalized project ID as cache key to handle path casing differences
748
+ */
749
+ export function getMemoryStore(projectPath: string): MemoryStore {
750
+ // Use getProjectId to normalize path for consistent cache key
751
+ const cacheKey = getProjectId(projectPath);
752
+
753
+ if (!storeCache.has(cacheKey)) {
754
+ storeCache.set(cacheKey, new MemoryStore(projectPath));
755
+ }
756
+ return storeCache.get(cacheKey)!;
757
+ }
758
+
759
+ /**
760
+ * Get aggregated stats from parent and all child projects
761
+ * @param projectPath - Parent project path
762
+ * @returns Aggregated statistics from all projects
763
+ */
764
+ export async function getAggregatedStats(projectPath: string): Promise<{
765
+ entities: number;
766
+ prompts: number;
767
+ conversations: number;
768
+ total: number;
769
+ projects: Array<{ path: string; stats: { entities: number; prompts: number; conversations: number } }>;
770
+ }> {
771
+ const { scanChildProjectsAsync } = await import('../config/storage-paths.js');
772
+ const childProjects = await scanChildProjectsAsync(projectPath);
773
+
774
+ const projectStats: Array<{ path: string; stats: { entities: number; prompts: number; conversations: number } }> = [];
775
+ let totalEntities = 0;
776
+ let totalPrompts = 0;
777
+ let totalConversations = 0;
778
+
779
+ // Get parent stats
780
+ try {
781
+ const parentStore = getMemoryStore(projectPath);
782
+ const db = (parentStore as any).db;
783
+
784
+ const entityCount = (db.prepare('SELECT COUNT(*) as count FROM entities').get() as { count: number }).count;
785
+ const promptCount = (db.prepare('SELECT COUNT(*) as count FROM prompt_history').get() as { count: number }).count;
786
+ const conversationCount = (db.prepare('SELECT COUNT(*) as count FROM conversations').get() as { count: number }).count;
787
+
788
+ projectStats.push({
789
+ path: projectPath,
790
+ stats: { entities: entityCount, prompts: promptCount, conversations: conversationCount }
791
+ });
792
+ totalEntities += entityCount;
793
+ totalPrompts += promptCount;
794
+ totalConversations += conversationCount;
795
+ } catch (error) {
796
+ if (process.env.DEBUG) {
797
+ console.error(`[Memory Store] Failed to get stats for parent ${projectPath}:`, error);
798
+ }
799
+ }
800
+
801
+ // Get child stats
802
+ for (const child of childProjects) {
803
+ try {
804
+ const childStore = getMemoryStore(child.projectPath);
805
+ const db = (childStore as any).db;
806
+
807
+ const entityCount = (db.prepare('SELECT COUNT(*) as count FROM entities').get() as { count: number }).count;
808
+ const promptCount = (db.prepare('SELECT COUNT(*) as count FROM prompt_history').get() as { count: number }).count;
809
+ const conversationCount = (db.prepare('SELECT COUNT(*) as count FROM conversations').get() as { count: number }).count;
810
+
811
+ projectStats.push({
812
+ path: child.relativePath,
813
+ stats: { entities: entityCount, prompts: promptCount, conversations: conversationCount }
814
+ });
815
+ totalEntities += entityCount;
816
+ totalPrompts += promptCount;
817
+ totalConversations += conversationCount;
818
+ } catch (error) {
819
+ if (process.env.DEBUG) {
820
+ console.error(`[Memory Store] Failed to get stats for child ${child.projectPath}:`, error);
821
+ }
822
+ }
823
+ }
824
+
825
+ return {
826
+ entities: totalEntities,
827
+ prompts: totalPrompts,
828
+ conversations: totalConversations,
829
+ total: totalEntities + totalPrompts + totalConversations,
830
+ projects: projectStats
831
+ };
832
+ }
833
+
834
+ /**
835
+ * Get aggregated entities from parent and all child projects
836
+ * @param projectPath - Parent project path
837
+ * @param options - Query options
838
+ * @returns Combined entities from all projects with source information
839
+ */
840
+ export async function getAggregatedEntities(
841
+ projectPath: string,
842
+ options: { type?: string; limit?: number; offset?: number } = {}
843
+ ): Promise<Array<HotEntity & { sourceProject?: string }>> {
844
+ const { scanChildProjectsAsync } = await import('../config/storage-paths.js');
845
+ const childProjects = await scanChildProjectsAsync(projectPath);
846
+
847
+ const limit = options.limit || 50;
848
+ const offset = options.offset || 0;
849
+ const allEntities: Array<HotEntity & { sourceProject?: string }> = [];
850
+
851
+ // Get parent entities - apply LIMIT at SQL level
852
+ try {
853
+ const parentStore = getMemoryStore(projectPath);
854
+ const db = (parentStore as any).db;
855
+
856
+ let query = 'SELECT * FROM entities';
857
+ const params: any[] = [];
858
+
859
+ if (options.type) {
860
+ query += ' WHERE type = ?';
861
+ params.push(options.type);
862
+ }
863
+
864
+ query += ' ORDER BY last_seen_at DESC LIMIT ?';
865
+ params.push(limit);
866
+
867
+ const stmt = db.prepare(query);
868
+ const parentEntities = stmt.all(...params) as Entity[];
869
+ allEntities.push(...parentEntities.map((e: Entity) => ({ ...e, stats: {} as EntityStats, sourceProject: projectPath })));
870
+ } catch (error) {
871
+ if (process.env.DEBUG) {
872
+ console.error(`[Memory Store] Failed to get entities for parent ${projectPath}:`, error);
873
+ }
874
+ }
875
+
876
+ // Get child entities - apply LIMIT to each child
877
+ for (const child of childProjects) {
878
+ try {
879
+ const childStore = getMemoryStore(child.projectPath);
880
+ const db = (childStore as any).db;
881
+
882
+ let query = 'SELECT * FROM entities';
883
+ const params: any[] = [];
884
+
885
+ if (options.type) {
886
+ query += ' WHERE type = ?';
887
+ params.push(options.type);
888
+ }
889
+
890
+ query += ' ORDER BY last_seen_at DESC LIMIT ?';
891
+ params.push(limit);
892
+
893
+ const stmt = db.prepare(query);
894
+ const childEntities = stmt.all(...params) as Entity[];
895
+ allEntities.push(...childEntities.map((e: Entity) => ({ ...e, stats: {} as EntityStats, sourceProject: child.relativePath })));
896
+ } catch (error) {
897
+ if (process.env.DEBUG) {
898
+ console.error(`[Memory Store] Failed to get entities for child ${child.projectPath}:`, error);
899
+ }
900
+ }
901
+ }
902
+
903
+ // Sort by last_seen_at and apply final limit with offset
904
+ allEntities.sort((a, b) => {
905
+ const aTime = a.last_seen_at ? new Date(a.last_seen_at).getTime() : 0;
906
+ const bTime = b.last_seen_at ? new Date(b.last_seen_at).getTime() : 0;
907
+ return bTime - aTime;
908
+ });
909
+
910
+ return allEntities.slice(offset, offset + limit);
911
+ }
912
+
913
+ /**
914
+ * Get aggregated prompts from parent and all child projects
915
+ * @param projectPath - Parent project path
916
+ * @param limit - Maximum number of prompts to return
917
+ * @returns Combined prompts from all projects with source information
918
+ */
919
+ export async function getAggregatedPrompts(
920
+ projectPath: string,
921
+ limit: number = 50
922
+ ): Promise<Array<PromptHistory & { sourceProject?: string }>> {
923
+ const { scanChildProjectsAsync } = await import('../config/storage-paths.js');
924
+ const childProjects = await scanChildProjectsAsync(projectPath);
925
+
926
+ const allPrompts: Array<PromptHistory & { sourceProject?: string }> = [];
927
+
928
+ // Get parent prompts - use direct SQL query with LIMIT
929
+ try {
930
+ const parentStore = getMemoryStore(projectPath);
931
+ const db = (parentStore as any).db;
932
+
933
+ const stmt = db.prepare('SELECT * FROM prompt_history ORDER BY timestamp DESC LIMIT ?');
934
+ const parentPrompts = stmt.all(limit) as PromptHistory[];
935
+ allPrompts.push(...parentPrompts.map((p: PromptHistory) => ({ ...p, sourceProject: projectPath })));
936
+ } catch (error) {
937
+ if (process.env.DEBUG) {
938
+ console.error(`[Memory Store] Failed to get prompts for parent ${projectPath}:`, error);
939
+ }
940
+ }
941
+
942
+ // Get child prompts - apply LIMIT to each child to reduce memory footprint
943
+ for (const child of childProjects) {
944
+ try {
945
+ const childStore = getMemoryStore(child.projectPath);
946
+ const db = (childStore as any).db;
947
+
948
+ const stmt = db.prepare('SELECT * FROM prompt_history ORDER BY timestamp DESC LIMIT ?');
949
+ const childPrompts = stmt.all(limit) as PromptHistory[];
950
+ allPrompts.push(...childPrompts.map((p: PromptHistory) => ({ ...p, sourceProject: child.relativePath })));
951
+ } catch (error) {
952
+ if (process.env.DEBUG) {
953
+ console.error(`[Memory Store] Failed to get prompts for child ${child.projectPath}:`, error);
954
+ }
955
+ }
956
+ }
957
+
958
+ // Sort by timestamp and apply final limit
959
+ allPrompts.sort((a, b) => {
960
+ const aTime = a.timestamp ? new Date(a.timestamp).getTime() : 0;
961
+ const bTime = b.timestamp ? new Date(b.timestamp).getTime() : 0;
962
+ return bTime - aTime;
963
+ });
964
+
965
+ return allPrompts.slice(0, limit);
966
+ }
967
+
968
+ /**
969
+ * Close all store instances
970
+ */
971
+ export function closeAllStores(): void {
972
+ for (const store of storeCache.values()) {
973
+ store.close();
974
+ }
975
+ storeCache.clear();
976
+ }
977
+
978
+ export default MemoryStore;