claude-memory-layer 1.0.27 → 1.0.29

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 (331) hide show
  1. package/.env.example +7 -0
  2. package/AGENTS.md +11 -0
  3. package/README.md +374 -49
  4. package/benchmarks/replay/anonymized-real-sessions.json +48 -0
  5. package/dist/cli/index.js +10097 -6003
  6. package/dist/cli/index.js.map +4 -4
  7. package/dist/core/index.js +9745 -5587
  8. package/dist/core/index.js.map +4 -4
  9. package/dist/hooks/post-tool-use.js +6545 -5270
  10. package/dist/hooks/post-tool-use.js.map +4 -4
  11. package/dist/hooks/semantic-daemon.js +6646 -5354
  12. package/dist/hooks/semantic-daemon.js.map +4 -4
  13. package/dist/hooks/session-end.js +6618 -5347
  14. package/dist/hooks/session-end.js.map +4 -4
  15. package/dist/hooks/session-start.js +6619 -5354
  16. package/dist/hooks/session-start.js.map +4 -4
  17. package/dist/hooks/stop.js +6614 -5325
  18. package/dist/hooks/stop.js.map +4 -4
  19. package/dist/hooks/user-prompt-submit.js +6702 -5356
  20. package/dist/hooks/user-prompt-submit.js.map +4 -4
  21. package/dist/index.js +13537 -0
  22. package/dist/index.js.map +7 -0
  23. package/dist/mcp/index.js +20770 -0
  24. package/dist/mcp/index.js.map +7 -0
  25. package/dist/server/api/index.js +6632 -5319
  26. package/dist/server/api/index.js.map +4 -4
  27. package/dist/server/index.js +6667 -5340
  28. package/dist/server/index.js.map +4 -4
  29. package/dist/services/memory-service.js +6568 -5350
  30. package/dist/services/memory-service.js.map +4 -4
  31. package/dist/ui/assets/js/bootstrap.js +244 -0
  32. package/dist/ui/assets/js/chat.js +373 -0
  33. package/dist/ui/assets/js/disclosure.js +232 -0
  34. package/dist/ui/assets/js/modals.js +298 -0
  35. package/dist/ui/assets/js/overview.js +655 -0
  36. package/dist/ui/assets/js/state.js +72 -0
  37. package/dist/ui/assets/js/views.js +468 -0
  38. package/dist/ui/index.html +43 -1
  39. package/dist/ui/index.ts +3 -0
  40. package/dist/ui/style.css +222 -0
  41. package/docs/ARCHITECTURE_COMPARISON_AND_RECOMMENDATIONS.md +627 -0
  42. package/docs/HERMES_MEMORY_INGESTION_ANALYSIS.md +440 -0
  43. package/docs/MEMORY_USEFULNESS_AUDIT.md +371 -0
  44. package/docs/MEMORY_USEFULNESS_AUDIT_RAW.json +80 -0
  45. package/docs/MEMSEARCH_PROJECT_STRUCTURE_ANALYSIS.md +333 -0
  46. package/docs/PRODUCT_VALIDATION_MATRIX.md +82 -0
  47. package/docs/PROJECT_STRUCTURE_ANALYSIS.md +421 -0
  48. package/docs/REFACTORING_MILESTONES_AND_ISSUES.md +501 -0
  49. package/docs/REFACTORING_PLAN_THIN_CORE.md +414 -0
  50. package/docs/REFERENCE_PROJECT_ANALYSES.md +25 -0
  51. package/docs/SUPERLOCALMEMORY_PROJECT_STRUCTURE_ANALYSIS.md +452 -0
  52. package/docs/TARGET_ARCHITECTURE_AND_FOLDER_STRUCTURE.md +446 -0
  53. package/docs/architecture/comparison-index.md +47 -0
  54. package/docs/reports/codex-real-data-validation-20260505T040447Z.md +46 -0
  55. package/package.json +12 -5
  56. package/scripts/build.ts +25 -8
  57. package/scripts/generate-session-qrels.ts +126 -0
  58. package/scripts/postinstall-embedding-backend.cjs +142 -0
  59. package/scripts/replay-retrieval-benchmark.ts +69 -0
  60. package/specs/thin-core-refactor/context.md +275 -0
  61. package/specs/thin-core-refactor/plan.md +536 -0
  62. package/specs/thin-core-refactor/spec.md +465 -0
  63. package/src/adapters/claude/capture/index.ts +3 -0
  64. package/src/adapters/claude/context/index.ts +3 -0
  65. package/src/adapters/claude/hooks/index.ts +21 -0
  66. package/src/adapters/claude/hooks/post-tool-use.ts +239 -0
  67. package/src/adapters/claude/hooks/prompt-injection-policy.ts +104 -0
  68. package/src/adapters/claude/hooks/semantic-daemon-client.ts +209 -0
  69. package/src/adapters/claude/hooks/semantic-daemon.ts +283 -0
  70. package/src/adapters/claude/hooks/session-end.ts +59 -0
  71. package/src/adapters/claude/hooks/session-start.ts +73 -0
  72. package/src/adapters/claude/hooks/stop.ts +128 -0
  73. package/src/adapters/claude/hooks/user-prompt-submit.ts +361 -0
  74. package/src/adapters/claude/index.ts +4 -0
  75. package/src/adapters/claude/transcript/index.ts +4 -0
  76. package/src/adapters/claude/transcript/transcript-reader.ts +57 -0
  77. package/src/adapters/claude/transcript/turn-reconstructor.ts +65 -0
  78. package/src/apps/cli/claude-settings-hooks.ts +138 -0
  79. package/src/apps/cli/codex-import-runner.ts +125 -0
  80. package/src/apps/cli/codex-validation-output.ts +95 -0
  81. package/src/apps/cli/hermes-import-runner.ts +130 -0
  82. package/src/apps/cli/hermes-validation-output.ts +91 -0
  83. package/src/apps/cli/index.ts +1731 -0
  84. package/src/apps/cli/mcp-install.ts +106 -0
  85. package/src/apps/cli/retrieval-disclosure-output.ts +196 -0
  86. package/src/apps/dashboard/assets/js/bootstrap.js +244 -0
  87. package/src/apps/dashboard/assets/js/chat.js +373 -0
  88. package/src/apps/dashboard/assets/js/disclosure.js +232 -0
  89. package/src/apps/dashboard/assets/js/modals.js +298 -0
  90. package/src/apps/dashboard/assets/js/overview.js +655 -0
  91. package/src/apps/dashboard/assets/js/state.js +72 -0
  92. package/src/apps/dashboard/assets/js/views.js +468 -0
  93. package/src/{ui → apps/dashboard}/index.html +43 -1
  94. package/src/apps/dashboard/index.ts +3 -0
  95. package/src/{ui → apps/dashboard}/style.css +222 -0
  96. package/src/apps/index.ts +5 -0
  97. package/src/apps/server/api/chat.ts +244 -0
  98. package/src/apps/server/api/citations.ts +105 -0
  99. package/src/apps/server/api/events.ts +137 -0
  100. package/src/apps/server/api/health.ts +53 -0
  101. package/src/apps/server/api/index.ts +26 -0
  102. package/src/apps/server/api/projects.ts +74 -0
  103. package/src/apps/server/api/search.ts +184 -0
  104. package/src/apps/server/api/sessions.ts +115 -0
  105. package/src/apps/server/api/stats.ts +723 -0
  106. package/src/apps/server/api/turns.ts +143 -0
  107. package/src/apps/server/api/utils.ts +65 -0
  108. package/src/apps/server/index.ts +111 -0
  109. package/src/cli/index.ts +2 -1311
  110. package/src/cli/retrieval-disclosure-output.ts +2 -0
  111. package/src/compat/index.ts +5 -0
  112. package/src/core/derive/fact-deriver.ts +170 -0
  113. package/src/core/derive/index.ts +2 -0
  114. package/src/core/derive/summary-deriver.ts +76 -0
  115. package/src/core/embedder.ts +4 -152
  116. package/src/core/engine/embedding-maintenance-service.ts +187 -0
  117. package/src/core/engine/endless-memory-services.ts +4 -0
  118. package/src/core/engine/index.ts +19 -0
  119. package/src/core/engine/memory-engine-services.ts +170 -0
  120. package/src/core/engine/memory-ingest-service.ts +317 -0
  121. package/src/core/engine/memory-query-service.ts +173 -0
  122. package/src/core/engine/memory-runtime-service.ts +162 -0
  123. package/src/core/engine/memory-service-composition.ts +231 -0
  124. package/src/core/engine/retrieval-analytics-service.ts +181 -0
  125. package/src/core/engine/retrieval-disclosure-service.ts +420 -0
  126. package/src/core/engine/retrieval-orchestrator.ts +377 -0
  127. package/src/core/engine/retrieval-services.ts +176 -0
  128. package/src/core/engine/shared-memory-services.ts +4 -0
  129. package/src/core/entity-repo.ts +1 -3
  130. package/src/core/event-store.ts +3 -3
  131. package/src/core/evidence-aligner.ts +2 -2
  132. package/src/core/external-market-context.ts +582 -0
  133. package/src/core/graduation.ts +2 -3
  134. package/src/core/index.ts +21 -0
  135. package/src/core/matcher.ts +2 -4
  136. package/src/core/model/memory-fact.ts +30 -0
  137. package/src/core/model/memory-rule.ts +14 -0
  138. package/src/core/model/memory-summary.ts +21 -0
  139. package/src/core/model/raw-event.ts +28 -0
  140. package/src/core/model/retrieval-result.ts +35 -0
  141. package/src/core/privacy/filter.ts +21 -10
  142. package/src/core/product-validation-matrix.ts +314 -0
  143. package/src/core/progressive-retriever.ts +1 -2
  144. package/src/core/registry/project-path.ts +54 -0
  145. package/src/core/registry/session-registry.ts +69 -0
  146. package/src/core/replay-evaluator.ts +625 -0
  147. package/src/core/retrieval-benchmark.ts +117 -0
  148. package/src/core/retrieval-quality.ts +109 -0
  149. package/src/core/retriever.ts +53 -15
  150. package/src/core/session-qrels.ts +360 -0
  151. package/src/core/shared-event-store.ts +1 -1
  152. package/src/core/sqlite-event-store.ts +35 -11
  153. package/src/core/task/blocker-resolver.ts +2 -2
  154. package/src/core/task/task-resolver.ts +0 -1
  155. package/src/core/vector-outbox.ts +1 -10
  156. package/src/core/vector-worker.ts +1 -1
  157. package/src/extensions/endless-memory/endless-memory-services.ts +350 -0
  158. package/src/extensions/endless-memory/index.ts +1 -0
  159. package/src/extensions/index.ts +5 -0
  160. package/src/extensions/mcp/handlers.ts +960 -0
  161. package/src/extensions/mcp/index.ts +48 -0
  162. package/src/extensions/mcp/tools.ts +252 -0
  163. package/src/extensions/shared-memory/index.ts +1 -0
  164. package/src/extensions/shared-memory/shared-memory-services.ts +211 -0
  165. package/src/extensions/vector/embedder.ts +197 -0
  166. package/src/extensions/vector/index.ts +1 -0
  167. package/src/hooks/post-tool-use.ts +3 -236
  168. package/src/hooks/semantic-daemon-client.ts +1 -208
  169. package/src/hooks/semantic-daemon.ts +6 -271
  170. package/src/hooks/session-end.ts +4 -79
  171. package/src/hooks/session-start.ts +4 -73
  172. package/src/hooks/stop.ts +3 -173
  173. package/src/hooks/user-prompt-submit.ts +3 -338
  174. package/src/index.ts +13 -0
  175. package/src/mcp/handlers.ts +2 -212
  176. package/src/mcp/index.ts +3 -46
  177. package/src/mcp/tools.ts +2 -78
  178. package/src/server/api/chat.ts +2 -244
  179. package/src/server/api/citations.ts +2 -105
  180. package/src/server/api/events.ts +2 -137
  181. package/src/server/api/health.ts +2 -53
  182. package/src/server/api/index.ts +2 -26
  183. package/src/server/api/projects.ts +2 -74
  184. package/src/server/api/search.ts +2 -102
  185. package/src/server/api/sessions.ts +2 -115
  186. package/src/server/api/stats.ts +2 -724
  187. package/src/server/api/turns.ts +2 -143
  188. package/src/server/api/utils.ts +2 -46
  189. package/src/server/index.ts +2 -100
  190. package/src/services/bootstrap-organizer.ts +46 -26
  191. package/src/services/codex-session-history-importer.ts +521 -29
  192. package/src/services/hermes-session-history-importer.ts +733 -0
  193. package/src/services/memory-service-config.ts +36 -0
  194. package/src/services/memory-service-registry.ts +150 -0
  195. package/src/services/memory-service.ts +211 -1325
  196. package/src/services/session-history-importer.ts +58 -14
  197. package/tests/README.md +23 -0
  198. package/tests/adapters/claude/claude-semantic-daemon-adapter.test.ts +54 -0
  199. package/tests/adapters/claude/claude-transcript-reconstructor.test.ts +98 -0
  200. package/tests/adapters/claude-hook-prompt-injection-policy.test.ts +99 -0
  201. package/tests/apps/app-layer-boundary.test.ts +48 -0
  202. package/tests/apps/claude-settings-hooks.test.ts +107 -0
  203. package/tests/apps/cli-disclosure-output.test.ts +212 -0
  204. package/tests/apps/codex-import-runner.test.ts +99 -0
  205. package/tests/apps/codex-validation-output.test.ts +100 -0
  206. package/tests/apps/hermes-import-runner.test.ts +99 -0
  207. package/tests/apps/mcp-install-command.test.ts +59 -0
  208. package/tests/apps/package-build-entrypoints.test.ts +30 -0
  209. package/tests/apps/postinstall-embedding-backend.test.ts +167 -0
  210. package/tests/apps/search-api-disclosure.test.ts +162 -0
  211. package/tests/apps/stats-api-lightweight.test.ts +67 -0
  212. package/tests/apps/ui-disclosure-output.test.ts +140 -0
  213. package/tests/{bootstrap-organizer.test.ts → core/bootstrap-organizer.test.ts} +1 -1
  214. package/tests/{canonical-key.test.ts → core/canonical-key.test.ts} +1 -1
  215. package/tests/core/codex-session-history-importer-validation.test.ts +185 -0
  216. package/tests/{consolidation-worker.test.ts → core/consolidation-worker.test.ts} +2 -2
  217. package/tests/core/embedding-maintenance-service.test.ts +282 -0
  218. package/tests/{evidence-aligner.test.ts → core/evidence-aligner.test.ts} +1 -1
  219. package/tests/core/external-market-context.test.ts +209 -0
  220. package/tests/core/fact-deriver.test.ts +79 -0
  221. package/tests/core/hermes-session-history-importer-validation.test.ts +609 -0
  222. package/tests/{ingest-interceptor.test.ts → core/ingest-interceptor.test.ts} +1 -1
  223. package/tests/{markdown-mirror.test.ts → core/markdown-mirror.test.ts} +2 -2
  224. package/tests/{matcher.test.ts → core/matcher.test.ts} +1 -1
  225. package/tests/{md-mirror.test.ts → core/md-mirror.test.ts} +2 -2
  226. package/tests/core/memory-engine-services.test.ts +240 -0
  227. package/tests/core/memory-ingest-service.test.ts +296 -0
  228. package/tests/core/memory-query-service.test.ts +129 -0
  229. package/tests/core/memory-runtime-service.test.ts +201 -0
  230. package/tests/core/memory-service-composition.test.ts +192 -0
  231. package/tests/core/memory-service-config.test.ts +41 -0
  232. package/tests/core/memory-service-facade.test.ts +30 -0
  233. package/tests/core/memory-service-registry.test.ts +206 -0
  234. package/tests/core/product-validation-matrix.test.ts +61 -0
  235. package/tests/core/project-registry.test.ts +78 -0
  236. package/tests/core/replay-evaluator.test.ts +181 -0
  237. package/tests/core/retrieval-analytics-service.test.ts +210 -0
  238. package/tests/core/retrieval-benchmark.test.ts +93 -0
  239. package/tests/core/retrieval-disclosure-service.test.ts +264 -0
  240. package/tests/core/retrieval-orchestrator.test.ts +403 -0
  241. package/tests/core/retrieval-quality.test.ts +31 -0
  242. package/tests/core/retrieval-services.test.ts +185 -0
  243. package/tests/{retriever-fallback-chain.test.ts → core/retriever-fallback-chain.test.ts} +3 -3
  244. package/tests/{retriever-strategy-scope.test.ts → core/retriever-strategy-scope.test.ts} +70 -3
  245. package/tests/{retriever.memu-adoption.test.ts → core/retriever.memu-adoption.test.ts} +3 -3
  246. package/tests/core/session-history-importer-filter.test.ts +78 -0
  247. package/tests/core/session-qrels.test.ts +250 -0
  248. package/tests/{sqlite-event-store-replication.test.ts → core/sqlite-event-store-replication.test.ts} +36 -1
  249. package/tests/core/summary-deriver.test.ts +66 -0
  250. package/tests/extensions/embedder-warning-suppression.test.ts +53 -0
  251. package/tests/extensions/endless-memory-extension-boundary.test.ts +17 -0
  252. package/tests/extensions/endless-memory-services.test.ts +325 -0
  253. package/tests/extensions/mcp-context-tools.test.ts +905 -0
  254. package/tests/extensions/mcp-extension-boundary.test.ts +21 -0
  255. package/tests/extensions/mcp-package-build.test.ts +22 -0
  256. package/tests/extensions/mcp-project-aware-tools.test.ts +102 -0
  257. package/tests/extensions/shared-memory-extension-boundary.test.ts +24 -0
  258. package/tests/extensions/shared-memory-services.test.ts +309 -0
  259. package/tests/extensions/vector-extension-boundary.test.ts +21 -0
  260. package/.claude/settings.local.json +0 -25
  261. package/.npm-cache/_cacache/content-v2/sha512/04/76/c098f88dfe584a2b80870bff7421b05d17d3d9ee1027f77772332a22d3f93a9a57101a2855107f6ad82077a818bba912b2bc317f2361b5ddb09ad284d9ce +0 -0
  262. package/.npm-cache/_cacache/content-v2/sha512/60/25/d2ecd39cfc7cab58351162814be77f935c6d6491c10c3745d456da7ddb2117ffd90c10e53fe3c0f1ed16b403307841543634504398b16ee4e6b6dd8e0c45 +0 -0
  263. package/.npm-cache/_cacache/index-v5/2b/9a/7f8f40206ed8a2e0a84efaa953ccaed1f5d001e14b931083f2e7a0738007 +0 -2
  264. package/.npm-cache/_cacache/index-v5/2e/d9/fcfa5c6a6abdc2a3644ab84a95936047298c465a2f47ee03db8f7fe1e946 +0 -3
  265. package/.npm-cache/_cacache/index-v5/a9/42/e519633356d12d3d2f19da66a8301016d496c8f5c3e0554124aaa62dc043 +0 -2
  266. package/.npm-cache/_logs/2026-02-26T12_04_52_729Z-debug-0.log +0 -256
  267. package/.npm-cache/_logs/2026-02-26T12_05_36_835Z-debug-0.log +0 -18
  268. package/.npm-cache/_logs/2026-02-26T12_05_45_982Z-debug-0.log +0 -32
  269. package/.npm-cache/_logs/2026-02-26T12_05_48_515Z-debug-0.log +0 -260
  270. package/.npm-cache/_logs/2026-02-26T12_05_53_567Z-debug-0.log +0 -69
  271. package/.npm-cache/_update-notifier-last-checked +0 -0
  272. package/bootstrap-kb/decisions/decisions.md +0 -244
  273. package/bootstrap-kb/glossary/glossary.md +0 -46
  274. package/bootstrap-kb/modules/.claude-plugin.md +0 -22
  275. package/bootstrap-kb/modules/agents.md.md +0 -15
  276. package/bootstrap-kb/modules/claude.md.md +0 -15
  277. package/bootstrap-kb/modules/context.md.md +0 -15
  278. package/bootstrap-kb/modules/docs.md +0 -18
  279. package/bootstrap-kb/modules/handoff.md.md +0 -15
  280. package/bootstrap-kb/modules/package-lock.json.md +0 -15
  281. package/bootstrap-kb/modules/package.json.md +0 -15
  282. package/bootstrap-kb/modules/plan.md.md +0 -15
  283. package/bootstrap-kb/modules/readme.md.md +0 -15
  284. package/bootstrap-kb/modules/scripts.md +0 -26
  285. package/bootstrap-kb/modules/spec.md.md +0 -15
  286. package/bootstrap-kb/modules/specs.md +0 -20
  287. package/bootstrap-kb/modules/src.md +0 -51
  288. package/bootstrap-kb/modules/tests.md +0 -42
  289. package/bootstrap-kb/modules/tsconfig.json.md +0 -15
  290. package/bootstrap-kb/modules/vitest.config.ts.md +0 -15
  291. package/bootstrap-kb/overview/overview.md +0 -40
  292. package/bootstrap-kb/sources/manifest.json +0 -950
  293. package/bootstrap-kb/sources/manifest.md +0 -227
  294. package/bootstrap-kb/timeline/timeline.md +0 -57
  295. package/claude-memory-layer-1.0.14.tgz +0 -0
  296. package/d.sh +0 -3
  297. package/deploy.sh +0 -3
  298. package/dist/ui/app.js +0 -2101
  299. package/memory/.claude-plugin/commands/2026-02-25.md +0 -263
  300. package/memory/_index.md +0 -419
  301. package/memory/agent_response/uncategorized/2026-02-26.md +0 -176
  302. package/memory/agent_response/uncategorized/2026-03-03.md +0 -14
  303. package/memory/agent_response/uncategorized/2026-03-04.md +0 -1421
  304. package/memory/agent_response/uncategorized/2026-03-05.md +0 -157
  305. package/memory/default/uncategorized/2026-02-25.md +0 -4839
  306. package/memory/session_summary/uncategorized/2026-02-26.md +0 -13
  307. package/memory/session_summary/uncategorized/2026-03-03.md +0 -5
  308. package/memory/session_summary/uncategorized/2026-03-04.md +0 -50
  309. package/memory/specs/20260207-dashboard-upgrade/2026-02-25.md +0 -142
  310. package/memory/specs/citations-system/2026-02-25.md +0 -1121
  311. package/memory/specs/endless-mode/2026-02-25.md +0 -1392
  312. package/memory/specs/entity-edge-model/2026-02-25.md +0 -1263
  313. package/memory/specs/evidence-aligner-v2/2026-02-25.md +0 -1028
  314. package/memory/specs/mcp-desktop-integration/2026-02-25.md +0 -1334
  315. package/memory/specs/post-tool-use-hook/2026-02-25.md +0 -1164
  316. package/memory/specs/private-tags/2026-02-25.md +0 -1057
  317. package/memory/specs/progressive-disclosure/2026-02-25.md +0 -1436
  318. package/memory/specs/task-entity-system/2026-02-25.md +0 -924
  319. package/memory/specs/vector-outbox-v2/2026-02-25.md +0 -1510
  320. package/memory/specs/web-viewer-ui/2026-02-25.md +0 -1709
  321. package/memory/tool_observation/uncategorized/2026-02-26.md +0 -209
  322. package/memory/tool_observation/uncategorized/2026-03-03.md +0 -21
  323. package/memory/tool_observation/uncategorized/2026-03-04.md +0 -1033
  324. package/memory/tool_observation/uncategorized/2026-03-05.md +0 -33
  325. package/memory/user_prompt/uncategorized/2026-02-26.md +0 -25
  326. package/memory/user_prompt/uncategorized/2026-03-04.md +0 -634
  327. package/memory/user_prompt/uncategorized/2026-03-05.md +0 -6
  328. package/specs/optional-duckdb/context.md +0 -77
  329. package/specs/optional-duckdb/plan.md +0 -142
  330. package/specs/optional-duckdb/spec.md +0 -35
  331. package/src/ui/app.js +0 -2101
@@ -0,0 +1,125 @@
1
+ import {
2
+ createCodexSessionHistoryImporter,
3
+ type CodexSessionHistoryImporter,
4
+ type CodexSessionHistoryImporterOptions
5
+ } from '../../services/codex-session-history-importer.js';
6
+ import {
7
+ getDefaultMemoryService,
8
+ getMemoryServiceForProject,
9
+ type MemoryService
10
+ } from '../../services/memory-service.js';
11
+ import type { ImportOptions, ImportResult, ProgressEvent } from '../../services/session-history-importer.js';
12
+
13
+ export interface CodexImportCommandOptions {
14
+ project?: string;
15
+ session?: string;
16
+ all?: boolean;
17
+ limit?: string;
18
+ sessionLimit?: string;
19
+ force?: boolean;
20
+ verbose?: boolean;
21
+ sessionsDir?: string;
22
+ processEmbeddings?: boolean;
23
+ }
24
+
25
+ export interface CodexImportOutcome {
26
+ mode: 'project' | 'session' | 'all';
27
+ storageScope: 'project' | 'global';
28
+ projectPath?: string;
29
+ result: ImportResult;
30
+ embedCount: number;
31
+ }
32
+
33
+ export interface CodexImportRunnerDeps {
34
+ cwd: () => string;
35
+ getDefaultMemoryService: () => MemoryService;
36
+ getMemoryServiceForProject: (projectPath: string) => MemoryService;
37
+ createImporter: (
38
+ memoryService: MemoryService,
39
+ options?: CodexSessionHistoryImporterOptions
40
+ ) => Pick<CodexSessionHistoryImporter, 'importProject' | 'importAll' | 'importSessionFile'>;
41
+ onProgress?: (event: ProgressEvent) => void;
42
+ }
43
+
44
+ const realDeps: CodexImportRunnerDeps = {
45
+ cwd: () => process.cwd(),
46
+ getDefaultMemoryService,
47
+ getMemoryServiceForProject,
48
+ createImporter: createCodexSessionHistoryImporter
49
+ };
50
+
51
+ function parsePositiveInteger(value: string | undefined, name: string): number | undefined {
52
+ if (value === undefined) return undefined;
53
+ const normalized = value.trim();
54
+ if (!/^\d+$/.test(normalized)) {
55
+ throw new Error(`Invalid --${name}: expected a positive integer`);
56
+ }
57
+ const parsed = Number.parseInt(normalized, 10);
58
+ if (!Number.isSafeInteger(parsed) || parsed <= 0) {
59
+ throw new Error(`Invalid --${name}: expected a positive integer`);
60
+ }
61
+ return parsed;
62
+ }
63
+
64
+ function shouldUseGlobalStorage(options: CodexImportCommandOptions): boolean {
65
+ return options.all === true && !options.project && !options.session;
66
+ }
67
+
68
+ export async function runCodexImportOnce(
69
+ options: CodexImportCommandOptions,
70
+ deps: CodexImportRunnerDeps = realDeps
71
+ ): Promise<CodexImportOutcome> {
72
+ const targetProjectPath = options.project || deps.cwd();
73
+ const useGlobalStorage = shouldUseGlobalStorage(options);
74
+ const memoryService = useGlobalStorage
75
+ ? deps.getDefaultMemoryService()
76
+ : deps.getMemoryServiceForProject(targetProjectPath);
77
+ const importer = deps.createImporter(memoryService, { sessionsDir: options.sessionsDir });
78
+
79
+ await memoryService.initialize();
80
+ await memoryService.ensureEmbeddingModelForImport({ autoMigrate: true });
81
+
82
+ const importOptions: ImportOptions = {
83
+ limit: parsePositiveInteger(options.limit, 'limit'),
84
+ sessionLimit: parsePositiveInteger(options.sessionLimit, 'session-limit'),
85
+ force: options.force,
86
+ verbose: options.verbose,
87
+ onProgress: deps.onProgress
88
+ };
89
+
90
+ let mode: CodexImportOutcome['mode'];
91
+ let result: ImportResult;
92
+
93
+ try {
94
+ if (options.session) {
95
+ mode = 'session';
96
+ result = await importer.importSessionFile(options.session, {
97
+ ...importOptions,
98
+ projectPath: targetProjectPath
99
+ });
100
+ } else if (options.all) {
101
+ mode = 'all';
102
+ result = await importer.importAll(importOptions);
103
+ } else {
104
+ mode = 'project';
105
+ result = await importer.importProject(targetProjectPath, {
106
+ ...importOptions,
107
+ projectPath: targetProjectPath
108
+ });
109
+ }
110
+
111
+ const embedCount = options.processEmbeddings === false
112
+ ? 0
113
+ : await memoryService.processPendingEmbeddings();
114
+
115
+ return {
116
+ mode,
117
+ storageScope: useGlobalStorage ? 'global' : 'project',
118
+ projectPath: useGlobalStorage ? undefined : targetProjectPath,
119
+ result,
120
+ embedCount
121
+ };
122
+ } finally {
123
+ await memoryService.shutdown();
124
+ }
125
+ }
@@ -0,0 +1,95 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import type { CodexSessionValidationReport } from '../../services/codex-session-history-importer.js';
4
+
5
+ export type CodexValidationReportFormat = 'json' | 'markdown';
6
+
7
+ function formatNumber(value: number): string {
8
+ return Number.isFinite(value) ? value.toLocaleString('en-US') : String(value);
9
+ }
10
+
11
+ export function formatCodexValidationReport(
12
+ report: CodexSessionValidationReport,
13
+ format: CodexValidationReportFormat = 'markdown'
14
+ ): string {
15
+ if (format === 'json') {
16
+ return `${JSON.stringify(report, null, 2)}\n`;
17
+ }
18
+
19
+ const lines: string[] = [
20
+ '# Codex dry-run validation report',
21
+ '',
22
+ `Generated: ${report.generatedAt}`,
23
+ `Dry-run: ${report.dryRun ? 'yes' : 'no'}`,
24
+ `Will mutate memory: ${report.willMutate ? 'yes' : 'no'}`,
25
+ `Sessions directory: ${report.source.sessionsDir}`,
26
+ `Project filter: ${report.source.projectPath ?? '(none)'}`,
27
+ `Source paths: ${report.source.sourcePaths.join(', ')}`,
28
+ `Session limit: ${report.limits.sessionLimit ?? '(none)'}`,
29
+ `Max content chars: ${formatNumber(report.limits.maxContentChars)}`,
30
+ '',
31
+ '## Totals',
32
+ '',
33
+ `- Sessions scanned: ${formatNumber(report.totals.sessionsScanned)}`,
34
+ `- Sessions matched: ${formatNumber(report.totals.sessionsMatched)}`,
35
+ `- Files read: ${formatNumber(report.totals.filesRead)}`,
36
+ `- Records read: ${formatNumber(report.totals.recordsRead)}`,
37
+ `- Messages normalized: ${formatNumber(report.totals.messagesNormalized)}`,
38
+ `- Turns normalized: ${formatNumber(report.totals.turnsNormalized)}`,
39
+ `- User messages: ${formatNumber(report.totals.userMessages)}`,
40
+ `- Assistant messages: ${formatNumber(report.totals.assistantMessages)}`,
41
+ `- Malformed lines: ${formatNumber(report.totals.malformedLines)}`,
42
+ `- Skipped/unsupported records: ${formatNumber(report.totals.skippedUnsupportedRecords)}`,
43
+ `- Empty assistant messages: ${formatNumber(report.totals.emptyAssistantMessages)}`,
44
+ `- Truncated messages: ${formatNumber(report.totals.truncatedMessages)}`,
45
+ `- Missing project cwd: ${formatNumber(report.totals.missingProjectCwd)}`,
46
+ `- Warnings: ${formatNumber(report.totals.warnings)}`,
47
+ '',
48
+ '## Top projects',
49
+ '',
50
+ '| Project | Hash | Sessions | Messages | Turns | User | Assistant | Malformed | Skipped/unsupported | Truncated | Empty assistant |',
51
+ '| --- | --- | ---: | ---: | ---: | ---: | ---: | ---: | ---: | ---: | ---: |'
52
+ ];
53
+
54
+ if (report.topProjects.length === 0) {
55
+ lines.push('| (none) | - | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |');
56
+ } else {
57
+ for (const project of report.topProjects) {
58
+ lines.push([
59
+ `| ${project.pathLabel}`,
60
+ project.projectHash,
61
+ formatNumber(project.sessions),
62
+ formatNumber(project.messagesNormalized),
63
+ formatNumber(project.turnsNormalized),
64
+ formatNumber(project.userMessages),
65
+ formatNumber(project.assistantMessages),
66
+ formatNumber(project.malformedLines),
67
+ formatNumber(project.skippedUnsupportedRecords),
68
+ formatNumber(project.truncatedMessages),
69
+ `${formatNumber(project.emptyAssistantMessages)} |`
70
+ ].join(' | '));
71
+ }
72
+ }
73
+
74
+ lines.push('', '## Warnings', '');
75
+ if (report.warnings.length === 0) {
76
+ lines.push('- None');
77
+ } else {
78
+ for (const warning of report.warnings) {
79
+ lines.push(`- ${warning}`);
80
+ }
81
+ }
82
+
83
+ lines.push('', '_Aggregate counts only; transcript content is not included._', '');
84
+ return lines.join('\n');
85
+ }
86
+
87
+ export function writeCodexValidationReport(
88
+ outputPath: string,
89
+ report: CodexSessionValidationReport,
90
+ format: CodexValidationReportFormat = 'markdown'
91
+ ): void {
92
+ const dir = path.dirname(outputPath);
93
+ fs.mkdirSync(dir, { recursive: true });
94
+ fs.writeFileSync(outputPath, formatCodexValidationReport(report, format), 'utf8');
95
+ }
@@ -0,0 +1,130 @@
1
+ import {
2
+ createHermesSessionHistoryImporter,
3
+ type HermesSessionHistoryImporter,
4
+ type HermesSessionHistoryImporterOptions
5
+ } from '../../services/hermes-session-history-importer.js';
6
+ import {
7
+ getDefaultMemoryService,
8
+ getMemoryServiceForProject,
9
+ type MemoryService
10
+ } from '../../services/memory-service.js';
11
+ import type { ImportOptions, ImportResult, ProgressEvent } from '../../services/session-history-importer.js';
12
+
13
+ export interface HermesImportCommandOptions {
14
+ project?: string;
15
+ session?: string;
16
+ all?: boolean;
17
+ limit?: string;
18
+ sessionLimit?: string;
19
+ force?: boolean;
20
+ verbose?: boolean;
21
+ stateDb?: string;
22
+ stateDbPath?: string;
23
+ processEmbeddings?: boolean;
24
+ }
25
+
26
+ export interface HermesImportOutcome {
27
+ mode: 'project' | 'session' | 'all';
28
+ storageScope: 'project' | 'global';
29
+ projectPath?: string;
30
+ result: ImportResult;
31
+ embedCount: number;
32
+ }
33
+
34
+ export interface HermesImportRunnerDeps {
35
+ cwd: () => string;
36
+ getDefaultMemoryService: () => MemoryService;
37
+ getMemoryServiceForProject: (projectPath: string) => MemoryService;
38
+ createImporter: (
39
+ memoryService: MemoryService,
40
+ options?: HermesSessionHistoryImporterOptions
41
+ ) => Pick<HermesSessionHistoryImporter, 'importProject' | 'importAll' | 'importSession'>;
42
+ onProgress?: (event: ProgressEvent) => void;
43
+ }
44
+
45
+ const realDeps: HermesImportRunnerDeps = {
46
+ cwd: () => process.cwd(),
47
+ getDefaultMemoryService,
48
+ getMemoryServiceForProject,
49
+ createImporter: createHermesSessionHistoryImporter
50
+ };
51
+
52
+ function parsePositiveInteger(value: string | undefined, name: string): number | undefined {
53
+ if (value === undefined) return undefined;
54
+ const normalized = value.trim();
55
+ if (!/^\d+$/.test(normalized)) {
56
+ throw new Error(`Invalid --${name}: expected a positive integer`);
57
+ }
58
+ const parsed = Number.parseInt(normalized, 10);
59
+ if (!Number.isSafeInteger(parsed) || parsed <= 0) {
60
+ throw new Error(`Invalid --${name}: expected a positive integer`);
61
+ }
62
+ return parsed;
63
+ }
64
+
65
+ function shouldUseGlobalStorage(options: HermesImportCommandOptions): boolean {
66
+ return options.all === true && !options.project && !options.session;
67
+ }
68
+
69
+ function getStateDbPathOption(options: HermesImportCommandOptions): string | undefined {
70
+ return options.stateDbPath ?? options.stateDb;
71
+ }
72
+
73
+ export async function runHermesImportOnce(
74
+ options: HermesImportCommandOptions,
75
+ deps: HermesImportRunnerDeps = realDeps
76
+ ): Promise<HermesImportOutcome> {
77
+ const targetProjectPath = options.project || deps.cwd();
78
+ const useGlobalStorage = shouldUseGlobalStorage(options);
79
+ const memoryService = useGlobalStorage
80
+ ? deps.getDefaultMemoryService()
81
+ : deps.getMemoryServiceForProject(targetProjectPath);
82
+ const importer = deps.createImporter(memoryService, { stateDbPath: getStateDbPathOption(options) });
83
+
84
+ await memoryService.initialize();
85
+ await memoryService.ensureEmbeddingModelForImport({ autoMigrate: true });
86
+
87
+ const importOptions: ImportOptions = {
88
+ limit: parsePositiveInteger(options.limit, 'limit'),
89
+ sessionLimit: parsePositiveInteger(options.sessionLimit, 'session-limit'),
90
+ force: options.force,
91
+ verbose: options.verbose,
92
+ onProgress: deps.onProgress
93
+ };
94
+
95
+ let mode: HermesImportOutcome['mode'];
96
+ let result: ImportResult;
97
+
98
+ try {
99
+ if (options.session) {
100
+ mode = 'session';
101
+ result = await importer.importSession(options.session, {
102
+ ...importOptions,
103
+ projectPath: targetProjectPath
104
+ });
105
+ } else if (options.all) {
106
+ mode = 'all';
107
+ result = await importer.importAll(importOptions);
108
+ } else {
109
+ mode = 'project';
110
+ result = await importer.importProject(targetProjectPath, {
111
+ ...importOptions,
112
+ projectPath: targetProjectPath
113
+ });
114
+ }
115
+
116
+ const embedCount = options.processEmbeddings === false
117
+ ? 0
118
+ : await memoryService.processPendingEmbeddings();
119
+
120
+ return {
121
+ mode,
122
+ storageScope: useGlobalStorage ? 'global' : 'project',
123
+ projectPath: useGlobalStorage ? undefined : targetProjectPath,
124
+ result,
125
+ embedCount
126
+ };
127
+ } finally {
128
+ await memoryService.shutdown();
129
+ }
130
+ }
@@ -0,0 +1,91 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import type { HermesSessionValidationReport } from '../../services/hermes-session-history-importer.js';
4
+
5
+ export type HermesValidationReportFormat = 'json' | 'markdown';
6
+
7
+ function formatNumber(value: number): string {
8
+ return Number.isFinite(value) ? value.toLocaleString('en-US') : String(value);
9
+ }
10
+
11
+ export function formatHermesValidationReport(
12
+ report: HermesSessionValidationReport,
13
+ format: HermesValidationReportFormat = 'markdown'
14
+ ): string {
15
+ if (format === 'json') {
16
+ return `${JSON.stringify(report, null, 2)}\n`;
17
+ }
18
+
19
+ const lines: string[] = [
20
+ '# Hermes dry-run validation report',
21
+ '',
22
+ `Generated: ${report.generatedAt}`,
23
+ `Dry-run: ${report.dryRun ? 'yes' : 'no'}`,
24
+ `Will mutate memory: ${report.willMutate ? 'yes' : 'no'}`,
25
+ `State DB: ${report.source.stateDbPath}`,
26
+ `Project filter: ${report.source.projectPath ?? '(none)'}`,
27
+ `Source paths: ${report.source.sourcePaths.join(', ')}`,
28
+ `Session limit: ${report.limits.sessionLimit ?? '(none)'}`,
29
+ `Max content chars: ${formatNumber(report.limits.maxContentChars)}`,
30
+ '',
31
+ '## Totals',
32
+ '',
33
+ `- Sessions scanned: ${formatNumber(report.totals.sessionsScanned)}`,
34
+ `- Sessions matched: ${formatNumber(report.totals.sessionsMatched)}`,
35
+ `- Messages read: ${formatNumber(report.totals.messagesRead)}`,
36
+ `- Messages normalized: ${formatNumber(report.totals.messagesNormalized)}`,
37
+ `- Turns normalized: ${formatNumber(report.totals.turnsNormalized)}`,
38
+ `- User messages: ${formatNumber(report.totals.userMessages)}`,
39
+ `- Assistant messages: ${formatNumber(report.totals.assistantMessages)}`,
40
+ `- Skipped/unsupported messages: ${formatNumber(report.totals.skippedUnsupportedMessages)}`,
41
+ `- Empty assistant messages: ${formatNumber(report.totals.emptyAssistantMessages)}`,
42
+ `- Truncated messages: ${formatNumber(report.totals.truncatedMessages)}`,
43
+ `- Missing project context: ${formatNumber(report.totals.missingProjectContext)}`,
44
+ `- Warnings: ${formatNumber(report.totals.warnings)}`,
45
+ '',
46
+ '## Top sources',
47
+ '',
48
+ '| Source | Sessions | Messages | Turns | User | Assistant | Skipped/unsupported | Truncated | Empty assistant |',
49
+ '| --- | ---: | ---: | ---: | ---: | ---: | ---: | ---: | ---: |'
50
+ ];
51
+
52
+ if (report.topSources.length === 0) {
53
+ lines.push('| (none) | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |');
54
+ } else {
55
+ for (const source of report.topSources) {
56
+ lines.push([
57
+ `| ${source.source}`,
58
+ formatNumber(source.sessions),
59
+ formatNumber(source.messagesNormalized),
60
+ formatNumber(source.turnsNormalized),
61
+ formatNumber(source.userMessages),
62
+ formatNumber(source.assistantMessages),
63
+ formatNumber(source.skippedUnsupportedMessages),
64
+ formatNumber(source.truncatedMessages),
65
+ `${formatNumber(source.emptyAssistantMessages)} |`
66
+ ].join(' | '));
67
+ }
68
+ }
69
+
70
+ lines.push('', '## Warnings', '');
71
+ if (report.warnings.length === 0) {
72
+ lines.push('- None');
73
+ } else {
74
+ for (const warning of report.warnings) {
75
+ lines.push(`- ${warning}`);
76
+ }
77
+ }
78
+
79
+ lines.push('', '_Aggregate counts only; transcript content is not included._', '');
80
+ return lines.join('\n');
81
+ }
82
+
83
+ export function writeHermesValidationReport(
84
+ outputPath: string,
85
+ report: HermesSessionValidationReport,
86
+ format: HermesValidationReportFormat = 'markdown'
87
+ ): void {
88
+ const dir = path.dirname(outputPath);
89
+ fs.mkdirSync(dir, { recursive: true });
90
+ fs.writeFileSync(outputPath, formatHermesValidationReport(report, format), 'utf8');
91
+ }