claude-memory-layer 1.0.27 → 1.0.28

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 (329) hide show
  1. package/.env.example +7 -0
  2. package/AGENTS.md +11 -0
  3. package/README.md +184 -41
  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 +9 -5
  56. package/scripts/build.ts +25 -8
  57. package/scripts/generate-session-qrels.ts +126 -0
  58. package/scripts/replay-retrieval-benchmark.ts +69 -0
  59. package/specs/thin-core-refactor/context.md +275 -0
  60. package/specs/thin-core-refactor/plan.md +536 -0
  61. package/specs/thin-core-refactor/spec.md +465 -0
  62. package/src/adapters/claude/capture/index.ts +3 -0
  63. package/src/adapters/claude/context/index.ts +3 -0
  64. package/src/adapters/claude/hooks/index.ts +21 -0
  65. package/src/adapters/claude/hooks/post-tool-use.ts +239 -0
  66. package/src/adapters/claude/hooks/prompt-injection-policy.ts +104 -0
  67. package/src/adapters/claude/hooks/semantic-daemon-client.ts +209 -0
  68. package/src/adapters/claude/hooks/semantic-daemon.ts +283 -0
  69. package/src/adapters/claude/hooks/session-end.ts +59 -0
  70. package/src/adapters/claude/hooks/session-start.ts +73 -0
  71. package/src/adapters/claude/hooks/stop.ts +128 -0
  72. package/src/adapters/claude/hooks/user-prompt-submit.ts +361 -0
  73. package/src/adapters/claude/index.ts +4 -0
  74. package/src/adapters/claude/transcript/index.ts +4 -0
  75. package/src/adapters/claude/transcript/transcript-reader.ts +57 -0
  76. package/src/adapters/claude/transcript/turn-reconstructor.ts +65 -0
  77. package/src/apps/cli/claude-settings-hooks.ts +138 -0
  78. package/src/apps/cli/codex-import-runner.ts +125 -0
  79. package/src/apps/cli/codex-validation-output.ts +95 -0
  80. package/src/apps/cli/hermes-import-runner.ts +130 -0
  81. package/src/apps/cli/hermes-validation-output.ts +91 -0
  82. package/src/apps/cli/index.ts +1731 -0
  83. package/src/apps/cli/mcp-install.ts +106 -0
  84. package/src/apps/cli/retrieval-disclosure-output.ts +196 -0
  85. package/src/apps/dashboard/assets/js/bootstrap.js +244 -0
  86. package/src/apps/dashboard/assets/js/chat.js +373 -0
  87. package/src/apps/dashboard/assets/js/disclosure.js +232 -0
  88. package/src/apps/dashboard/assets/js/modals.js +298 -0
  89. package/src/apps/dashboard/assets/js/overview.js +655 -0
  90. package/src/apps/dashboard/assets/js/state.js +72 -0
  91. package/src/apps/dashboard/assets/js/views.js +468 -0
  92. package/src/{ui → apps/dashboard}/index.html +43 -1
  93. package/src/apps/dashboard/index.ts +3 -0
  94. package/src/{ui → apps/dashboard}/style.css +222 -0
  95. package/src/apps/index.ts +5 -0
  96. package/src/apps/server/api/chat.ts +244 -0
  97. package/src/apps/server/api/citations.ts +105 -0
  98. package/src/apps/server/api/events.ts +137 -0
  99. package/src/apps/server/api/health.ts +53 -0
  100. package/src/apps/server/api/index.ts +26 -0
  101. package/src/apps/server/api/projects.ts +74 -0
  102. package/src/apps/server/api/search.ts +184 -0
  103. package/src/apps/server/api/sessions.ts +115 -0
  104. package/src/apps/server/api/stats.ts +723 -0
  105. package/src/apps/server/api/turns.ts +143 -0
  106. package/src/apps/server/api/utils.ts +65 -0
  107. package/src/apps/server/index.ts +111 -0
  108. package/src/cli/index.ts +2 -1311
  109. package/src/cli/retrieval-disclosure-output.ts +2 -0
  110. package/src/compat/index.ts +5 -0
  111. package/src/core/derive/fact-deriver.ts +170 -0
  112. package/src/core/derive/index.ts +2 -0
  113. package/src/core/derive/summary-deriver.ts +76 -0
  114. package/src/core/embedder.ts +4 -152
  115. package/src/core/engine/embedding-maintenance-service.ts +187 -0
  116. package/src/core/engine/endless-memory-services.ts +4 -0
  117. package/src/core/engine/index.ts +19 -0
  118. package/src/core/engine/memory-engine-services.ts +170 -0
  119. package/src/core/engine/memory-ingest-service.ts +317 -0
  120. package/src/core/engine/memory-query-service.ts +173 -0
  121. package/src/core/engine/memory-runtime-service.ts +162 -0
  122. package/src/core/engine/memory-service-composition.ts +231 -0
  123. package/src/core/engine/retrieval-analytics-service.ts +181 -0
  124. package/src/core/engine/retrieval-disclosure-service.ts +420 -0
  125. package/src/core/engine/retrieval-orchestrator.ts +377 -0
  126. package/src/core/engine/retrieval-services.ts +176 -0
  127. package/src/core/engine/shared-memory-services.ts +4 -0
  128. package/src/core/entity-repo.ts +1 -3
  129. package/src/core/event-store.ts +3 -3
  130. package/src/core/evidence-aligner.ts +2 -2
  131. package/src/core/external-market-context.ts +582 -0
  132. package/src/core/graduation.ts +2 -3
  133. package/src/core/index.ts +21 -0
  134. package/src/core/matcher.ts +2 -4
  135. package/src/core/model/memory-fact.ts +30 -0
  136. package/src/core/model/memory-rule.ts +14 -0
  137. package/src/core/model/memory-summary.ts +21 -0
  138. package/src/core/model/raw-event.ts +28 -0
  139. package/src/core/model/retrieval-result.ts +35 -0
  140. package/src/core/privacy/filter.ts +21 -10
  141. package/src/core/product-validation-matrix.ts +314 -0
  142. package/src/core/progressive-retriever.ts +1 -2
  143. package/src/core/registry/project-path.ts +54 -0
  144. package/src/core/registry/session-registry.ts +69 -0
  145. package/src/core/replay-evaluator.ts +625 -0
  146. package/src/core/retrieval-benchmark.ts +117 -0
  147. package/src/core/retrieval-quality.ts +109 -0
  148. package/src/core/retriever.ts +53 -15
  149. package/src/core/session-qrels.ts +360 -0
  150. package/src/core/shared-event-store.ts +1 -1
  151. package/src/core/sqlite-event-store.ts +35 -11
  152. package/src/core/task/blocker-resolver.ts +2 -2
  153. package/src/core/task/task-resolver.ts +0 -1
  154. package/src/core/vector-outbox.ts +1 -10
  155. package/src/core/vector-worker.ts +1 -1
  156. package/src/extensions/endless-memory/endless-memory-services.ts +350 -0
  157. package/src/extensions/endless-memory/index.ts +1 -0
  158. package/src/extensions/index.ts +5 -0
  159. package/src/extensions/mcp/handlers.ts +960 -0
  160. package/src/extensions/mcp/index.ts +48 -0
  161. package/src/extensions/mcp/tools.ts +252 -0
  162. package/src/extensions/shared-memory/index.ts +1 -0
  163. package/src/extensions/shared-memory/shared-memory-services.ts +211 -0
  164. package/src/extensions/vector/embedder.ts +197 -0
  165. package/src/extensions/vector/index.ts +1 -0
  166. package/src/hooks/post-tool-use.ts +3 -236
  167. package/src/hooks/semantic-daemon-client.ts +1 -208
  168. package/src/hooks/semantic-daemon.ts +6 -271
  169. package/src/hooks/session-end.ts +4 -79
  170. package/src/hooks/session-start.ts +4 -73
  171. package/src/hooks/stop.ts +3 -173
  172. package/src/hooks/user-prompt-submit.ts +3 -338
  173. package/src/index.ts +13 -0
  174. package/src/mcp/handlers.ts +2 -212
  175. package/src/mcp/index.ts +3 -46
  176. package/src/mcp/tools.ts +2 -78
  177. package/src/server/api/chat.ts +2 -244
  178. package/src/server/api/citations.ts +2 -105
  179. package/src/server/api/events.ts +2 -137
  180. package/src/server/api/health.ts +2 -53
  181. package/src/server/api/index.ts +2 -26
  182. package/src/server/api/projects.ts +2 -74
  183. package/src/server/api/search.ts +2 -102
  184. package/src/server/api/sessions.ts +2 -115
  185. package/src/server/api/stats.ts +2 -724
  186. package/src/server/api/turns.ts +2 -143
  187. package/src/server/api/utils.ts +2 -46
  188. package/src/server/index.ts +2 -100
  189. package/src/services/bootstrap-organizer.ts +46 -26
  190. package/src/services/codex-session-history-importer.ts +521 -29
  191. package/src/services/hermes-session-history-importer.ts +733 -0
  192. package/src/services/memory-service-config.ts +36 -0
  193. package/src/services/memory-service-registry.ts +150 -0
  194. package/src/services/memory-service.ts +211 -1325
  195. package/src/services/session-history-importer.ts +58 -14
  196. package/tests/README.md +23 -0
  197. package/tests/adapters/claude/claude-semantic-daemon-adapter.test.ts +54 -0
  198. package/tests/adapters/claude/claude-transcript-reconstructor.test.ts +98 -0
  199. package/tests/adapters/claude-hook-prompt-injection-policy.test.ts +99 -0
  200. package/tests/apps/app-layer-boundary.test.ts +48 -0
  201. package/tests/apps/claude-settings-hooks.test.ts +107 -0
  202. package/tests/apps/cli-disclosure-output.test.ts +212 -0
  203. package/tests/apps/codex-import-runner.test.ts +99 -0
  204. package/tests/apps/codex-validation-output.test.ts +100 -0
  205. package/tests/apps/hermes-import-runner.test.ts +99 -0
  206. package/tests/apps/mcp-install-command.test.ts +59 -0
  207. package/tests/apps/package-build-entrypoints.test.ts +30 -0
  208. package/tests/apps/search-api-disclosure.test.ts +162 -0
  209. package/tests/apps/stats-api-lightweight.test.ts +67 -0
  210. package/tests/apps/ui-disclosure-output.test.ts +140 -0
  211. package/tests/{bootstrap-organizer.test.ts → core/bootstrap-organizer.test.ts} +1 -1
  212. package/tests/{canonical-key.test.ts → core/canonical-key.test.ts} +1 -1
  213. package/tests/core/codex-session-history-importer-validation.test.ts +185 -0
  214. package/tests/{consolidation-worker.test.ts → core/consolidation-worker.test.ts} +2 -2
  215. package/tests/core/embedding-maintenance-service.test.ts +282 -0
  216. package/tests/{evidence-aligner.test.ts → core/evidence-aligner.test.ts} +1 -1
  217. package/tests/core/external-market-context.test.ts +209 -0
  218. package/tests/core/fact-deriver.test.ts +79 -0
  219. package/tests/core/hermes-session-history-importer-validation.test.ts +609 -0
  220. package/tests/{ingest-interceptor.test.ts → core/ingest-interceptor.test.ts} +1 -1
  221. package/tests/{markdown-mirror.test.ts → core/markdown-mirror.test.ts} +2 -2
  222. package/tests/{matcher.test.ts → core/matcher.test.ts} +1 -1
  223. package/tests/{md-mirror.test.ts → core/md-mirror.test.ts} +2 -2
  224. package/tests/core/memory-engine-services.test.ts +240 -0
  225. package/tests/core/memory-ingest-service.test.ts +296 -0
  226. package/tests/core/memory-query-service.test.ts +129 -0
  227. package/tests/core/memory-runtime-service.test.ts +201 -0
  228. package/tests/core/memory-service-composition.test.ts +192 -0
  229. package/tests/core/memory-service-config.test.ts +41 -0
  230. package/tests/core/memory-service-facade.test.ts +30 -0
  231. package/tests/core/memory-service-registry.test.ts +206 -0
  232. package/tests/core/product-validation-matrix.test.ts +61 -0
  233. package/tests/core/project-registry.test.ts +78 -0
  234. package/tests/core/replay-evaluator.test.ts +181 -0
  235. package/tests/core/retrieval-analytics-service.test.ts +210 -0
  236. package/tests/core/retrieval-benchmark.test.ts +93 -0
  237. package/tests/core/retrieval-disclosure-service.test.ts +264 -0
  238. package/tests/core/retrieval-orchestrator.test.ts +403 -0
  239. package/tests/core/retrieval-quality.test.ts +31 -0
  240. package/tests/core/retrieval-services.test.ts +185 -0
  241. package/tests/{retriever-fallback-chain.test.ts → core/retriever-fallback-chain.test.ts} +3 -3
  242. package/tests/{retriever-strategy-scope.test.ts → core/retriever-strategy-scope.test.ts} +70 -3
  243. package/tests/{retriever.memu-adoption.test.ts → core/retriever.memu-adoption.test.ts} +3 -3
  244. package/tests/core/session-history-importer-filter.test.ts +78 -0
  245. package/tests/core/session-qrels.test.ts +250 -0
  246. package/tests/{sqlite-event-store-replication.test.ts → core/sqlite-event-store-replication.test.ts} +36 -1
  247. package/tests/core/summary-deriver.test.ts +66 -0
  248. package/tests/extensions/embedder-warning-suppression.test.ts +53 -0
  249. package/tests/extensions/endless-memory-extension-boundary.test.ts +17 -0
  250. package/tests/extensions/endless-memory-services.test.ts +325 -0
  251. package/tests/extensions/mcp-context-tools.test.ts +905 -0
  252. package/tests/extensions/mcp-extension-boundary.test.ts +21 -0
  253. package/tests/extensions/mcp-package-build.test.ts +22 -0
  254. package/tests/extensions/mcp-project-aware-tools.test.ts +102 -0
  255. package/tests/extensions/shared-memory-extension-boundary.test.ts +24 -0
  256. package/tests/extensions/shared-memory-services.test.ts +309 -0
  257. package/tests/extensions/vector-extension-boundary.test.ts +21 -0
  258. package/.claude/settings.local.json +0 -25
  259. package/.npm-cache/_cacache/content-v2/sha512/04/76/c098f88dfe584a2b80870bff7421b05d17d3d9ee1027f77772332a22d3f93a9a57101a2855107f6ad82077a818bba912b2bc317f2361b5ddb09ad284d9ce +0 -0
  260. package/.npm-cache/_cacache/content-v2/sha512/60/25/d2ecd39cfc7cab58351162814be77f935c6d6491c10c3745d456da7ddb2117ffd90c10e53fe3c0f1ed16b403307841543634504398b16ee4e6b6dd8e0c45 +0 -0
  261. package/.npm-cache/_cacache/index-v5/2b/9a/7f8f40206ed8a2e0a84efaa953ccaed1f5d001e14b931083f2e7a0738007 +0 -2
  262. package/.npm-cache/_cacache/index-v5/2e/d9/fcfa5c6a6abdc2a3644ab84a95936047298c465a2f47ee03db8f7fe1e946 +0 -3
  263. package/.npm-cache/_cacache/index-v5/a9/42/e519633356d12d3d2f19da66a8301016d496c8f5c3e0554124aaa62dc043 +0 -2
  264. package/.npm-cache/_logs/2026-02-26T12_04_52_729Z-debug-0.log +0 -256
  265. package/.npm-cache/_logs/2026-02-26T12_05_36_835Z-debug-0.log +0 -18
  266. package/.npm-cache/_logs/2026-02-26T12_05_45_982Z-debug-0.log +0 -32
  267. package/.npm-cache/_logs/2026-02-26T12_05_48_515Z-debug-0.log +0 -260
  268. package/.npm-cache/_logs/2026-02-26T12_05_53_567Z-debug-0.log +0 -69
  269. package/.npm-cache/_update-notifier-last-checked +0 -0
  270. package/bootstrap-kb/decisions/decisions.md +0 -244
  271. package/bootstrap-kb/glossary/glossary.md +0 -46
  272. package/bootstrap-kb/modules/.claude-plugin.md +0 -22
  273. package/bootstrap-kb/modules/agents.md.md +0 -15
  274. package/bootstrap-kb/modules/claude.md.md +0 -15
  275. package/bootstrap-kb/modules/context.md.md +0 -15
  276. package/bootstrap-kb/modules/docs.md +0 -18
  277. package/bootstrap-kb/modules/handoff.md.md +0 -15
  278. package/bootstrap-kb/modules/package-lock.json.md +0 -15
  279. package/bootstrap-kb/modules/package.json.md +0 -15
  280. package/bootstrap-kb/modules/plan.md.md +0 -15
  281. package/bootstrap-kb/modules/readme.md.md +0 -15
  282. package/bootstrap-kb/modules/scripts.md +0 -26
  283. package/bootstrap-kb/modules/spec.md.md +0 -15
  284. package/bootstrap-kb/modules/specs.md +0 -20
  285. package/bootstrap-kb/modules/src.md +0 -51
  286. package/bootstrap-kb/modules/tests.md +0 -42
  287. package/bootstrap-kb/modules/tsconfig.json.md +0 -15
  288. package/bootstrap-kb/modules/vitest.config.ts.md +0 -15
  289. package/bootstrap-kb/overview/overview.md +0 -40
  290. package/bootstrap-kb/sources/manifest.json +0 -950
  291. package/bootstrap-kb/sources/manifest.md +0 -227
  292. package/bootstrap-kb/timeline/timeline.md +0 -57
  293. package/claude-memory-layer-1.0.14.tgz +0 -0
  294. package/d.sh +0 -3
  295. package/deploy.sh +0 -3
  296. package/dist/ui/app.js +0 -2101
  297. package/memory/.claude-plugin/commands/2026-02-25.md +0 -263
  298. package/memory/_index.md +0 -419
  299. package/memory/agent_response/uncategorized/2026-02-26.md +0 -176
  300. package/memory/agent_response/uncategorized/2026-03-03.md +0 -14
  301. package/memory/agent_response/uncategorized/2026-03-04.md +0 -1421
  302. package/memory/agent_response/uncategorized/2026-03-05.md +0 -157
  303. package/memory/default/uncategorized/2026-02-25.md +0 -4839
  304. package/memory/session_summary/uncategorized/2026-02-26.md +0 -13
  305. package/memory/session_summary/uncategorized/2026-03-03.md +0 -5
  306. package/memory/session_summary/uncategorized/2026-03-04.md +0 -50
  307. package/memory/specs/20260207-dashboard-upgrade/2026-02-25.md +0 -142
  308. package/memory/specs/citations-system/2026-02-25.md +0 -1121
  309. package/memory/specs/endless-mode/2026-02-25.md +0 -1392
  310. package/memory/specs/entity-edge-model/2026-02-25.md +0 -1263
  311. package/memory/specs/evidence-aligner-v2/2026-02-25.md +0 -1028
  312. package/memory/specs/mcp-desktop-integration/2026-02-25.md +0 -1334
  313. package/memory/specs/post-tool-use-hook/2026-02-25.md +0 -1164
  314. package/memory/specs/private-tags/2026-02-25.md +0 -1057
  315. package/memory/specs/progressive-disclosure/2026-02-25.md +0 -1436
  316. package/memory/specs/task-entity-system/2026-02-25.md +0 -924
  317. package/memory/specs/vector-outbox-v2/2026-02-25.md +0 -1510
  318. package/memory/specs/web-viewer-ui/2026-02-25.md +0 -1709
  319. package/memory/tool_observation/uncategorized/2026-02-26.md +0 -209
  320. package/memory/tool_observation/uncategorized/2026-03-03.md +0 -21
  321. package/memory/tool_observation/uncategorized/2026-03-04.md +0 -1033
  322. package/memory/tool_observation/uncategorized/2026-03-05.md +0 -33
  323. package/memory/user_prompt/uncategorized/2026-02-26.md +0 -25
  324. package/memory/user_prompt/uncategorized/2026-03-04.md +0 -634
  325. package/memory/user_prompt/uncategorized/2026-03-05.md +0 -6
  326. package/specs/optional-duckdb/context.md +0 -77
  327. package/specs/optional-duckdb/plan.md +0 -142
  328. package/specs/optional-duckdb/spec.md +0 -35
  329. package/src/ui/app.js +0 -2101
@@ -0,0 +1,53 @@
1
+ /**
2
+ * Health API
3
+ * Operational health checks including outbox backlog/failures
4
+ */
5
+
6
+ import { Hono } from 'hono';
7
+ import { getServiceFromQuery } from './utils.js';
8
+
9
+ export const healthRouter = new Hono();
10
+
11
+ // GET /api/health
12
+ healthRouter.get('/', async (c) => {
13
+ const memoryService = getServiceFromQuery(c);
14
+ try {
15
+ await memoryService.initialize();
16
+
17
+ const [stats, outbox] = await Promise.all([
18
+ memoryService.getStats(),
19
+ memoryService.getOutboxStats()
20
+ ]);
21
+
22
+ const outboxPending = outbox.embedding.pending + outbox.vector.pending;
23
+ const outboxFailed = outbox.embedding.failed + outbox.vector.failed;
24
+
25
+ const status = outboxFailed > 0 ? 'needs-attention' : 'ok';
26
+
27
+ return c.json({
28
+ status,
29
+ timestamp: new Date().toISOString(),
30
+ storage: {
31
+ totalEvents: stats.totalEvents,
32
+ vectorCount: stats.vectorCount
33
+ },
34
+ outbox: {
35
+ embedding: outbox.embedding,
36
+ vector: outbox.vector,
37
+ totals: {
38
+ pending: outboxPending,
39
+ failed: outboxFailed
40
+ }
41
+ },
42
+ levelStats: stats.levelStats
43
+ });
44
+ } catch (error) {
45
+ return c.json({
46
+ status: 'error',
47
+ timestamp: new Date().toISOString(),
48
+ error: (error as Error).message
49
+ }, 500);
50
+ } finally {
51
+ await memoryService.shutdown();
52
+ }
53
+ });
@@ -0,0 +1,26 @@
1
+ /**
2
+ * API Router
3
+ * Central router for all API endpoints
4
+ */
5
+
6
+ import { Hono } from 'hono';
7
+ import { sessionsRouter } from './sessions.js';
8
+ import { eventsRouter } from './events.js';
9
+ import { searchRouter } from './search.js';
10
+ import { statsRouter } from './stats.js';
11
+ import { citationsRouter } from './citations.js';
12
+ import { turnsRouter } from './turns.js';
13
+ import { projectsRouter } from './projects.js';
14
+ import { chatRouter } from './chat.js';
15
+ import { healthRouter } from './health.js';
16
+
17
+ export const apiRouter = new Hono()
18
+ .route('/sessions', sessionsRouter)
19
+ .route('/events', eventsRouter)
20
+ .route('/search', searchRouter)
21
+ .route('/stats', statsRouter)
22
+ .route('/citations', citationsRouter)
23
+ .route('/turns', turnsRouter)
24
+ .route('/projects', projectsRouter)
25
+ .route('/chat', chatRouter)
26
+ .route('/health', healthRouter);
@@ -0,0 +1,74 @@
1
+ /**
2
+ * Projects API
3
+ * Endpoints for listing available projects
4
+ */
5
+
6
+ import { Hono } from 'hono';
7
+ import * as fs from 'fs';
8
+ import * as path from 'path';
9
+ import * as os from 'os';
10
+ import { loadSessionRegistry } from '../../../services/memory-service.js';
11
+
12
+ export const projectsRouter = new Hono();
13
+
14
+ // GET /api/projects - List available projects
15
+ projectsRouter.get('/', async (c) => {
16
+ try {
17
+ const projectsDir = path.join(os.homedir(), '.claude-code', 'memory', 'projects');
18
+
19
+ if (!fs.existsSync(projectsDir)) {
20
+ return c.json({ projects: [] });
21
+ }
22
+
23
+ // Read project directories
24
+ const projectHashes = fs.readdirSync(projectsDir)
25
+ .filter(name => {
26
+ const fullPath = path.join(projectsDir, name);
27
+ return fs.statSync(fullPath).isDirectory();
28
+ });
29
+
30
+ // Load session registry to map hashes to project paths
31
+ const registry = loadSessionRegistry();
32
+ const hashToPath = new Map<string, string>();
33
+ for (const entry of Object.values(registry.sessions)) {
34
+ if (!hashToPath.has(entry.projectHash)) {
35
+ hashToPath.set(entry.projectHash, entry.projectPath);
36
+ }
37
+ }
38
+
39
+ // Build project list
40
+ const projects = projectHashes.map(hash => {
41
+ const dirPath = path.join(projectsDir, hash);
42
+ const dbPath = path.join(dirPath, 'events.sqlite');
43
+ let dbSize = 0;
44
+ if (fs.existsSync(dbPath)) {
45
+ dbSize = fs.statSync(dbPath).size;
46
+ }
47
+
48
+ const projectPath = hashToPath.get(hash) || `unknown (${hash})`;
49
+
50
+ return {
51
+ hash,
52
+ projectPath,
53
+ projectName: path.basename(projectPath),
54
+ dbSize,
55
+ dbSizeHuman: formatBytes(dbSize)
56
+ };
57
+ });
58
+
59
+ // Sort by project name
60
+ projects.sort((a, b) => a.projectName.localeCompare(b.projectName));
61
+
62
+ return c.json({ projects });
63
+ } catch (error) {
64
+ return c.json({ projects: [], error: (error as Error).message }, 500);
65
+ }
66
+ });
67
+
68
+ function formatBytes(bytes: number): string {
69
+ if (bytes === 0) return '0 B';
70
+ const k = 1024;
71
+ const sizes = ['B', 'KB', 'MB', 'GB'];
72
+ const i = Math.floor(Math.log(bytes) / Math.log(k));
73
+ return parseFloat((bytes / Math.pow(k, i)).toFixed(1)) + ' ' + sizes[i];
74
+ }
@@ -0,0 +1,184 @@
1
+ /**
2
+ * Search API
3
+ * Endpoints for memory search
4
+ */
5
+
6
+ import { Hono } from 'hono';
7
+ import { getLightweightServiceFromQuery, getServiceFromQuery } from './utils.js';
8
+
9
+ export const searchRouter = new Hono();
10
+
11
+ interface SearchRequest {
12
+ query: string;
13
+ options?: {
14
+ topK?: number;
15
+ minScore?: number;
16
+ sessionId?: string;
17
+ eventType?: string;
18
+ };
19
+ }
20
+
21
+ interface DisclosureSearchRequest {
22
+ query: string;
23
+ options?: {
24
+ topK?: number;
25
+ minScore?: number;
26
+ sessionId?: string;
27
+ includeShared?: boolean;
28
+ adaptiveRerank?: boolean;
29
+ intentRewrite?: boolean;
30
+ projectScopeMode?: 'strict' | 'prefer' | 'global';
31
+ allowedProjectHashes?: string[];
32
+ strategy?: 'auto' | 'fast' | 'deep';
33
+ };
34
+ }
35
+
36
+ // POST /api/search - Search memories
37
+ searchRouter.post('/', async (c) => {
38
+ const memoryService = getServiceFromQuery(c);
39
+ try {
40
+ const body = await c.req.json<SearchRequest>();
41
+
42
+ if (!body.query) {
43
+ return c.json({ error: 'Query is required' }, 400);
44
+ }
45
+
46
+ await memoryService.initialize();
47
+
48
+ const startTime = Date.now();
49
+
50
+ const result = await memoryService.retrieveMemories(body.query, {
51
+ topK: body.options?.topK ?? 10,
52
+ minScore: body.options?.minScore ?? 0.7,
53
+ sessionId: body.options?.sessionId
54
+ });
55
+
56
+ const searchTime = Date.now() - startTime;
57
+
58
+ return c.json({
59
+ results: result.memories.map(m => ({
60
+ id: m.event.id,
61
+ eventType: m.event.eventType,
62
+ timestamp: m.event.timestamp,
63
+ sessionId: m.event.sessionId,
64
+ score: m.score,
65
+ content: m.event.content,
66
+ preview: m.event.content.slice(0, 200) + (m.event.content.length > 200 ? '...' : ''),
67
+ context: m.sessionContext
68
+ })),
69
+ meta: {
70
+ totalMatches: result.memories.length,
71
+ searchTime,
72
+ confidence: result.matchResult.confidence,
73
+ totalTokens: result.totalTokens
74
+ }
75
+ });
76
+ } catch (error) {
77
+ return c.json({ error: (error as Error).message }, 500);
78
+ } finally {
79
+ await memoryService.shutdown();
80
+ }
81
+ });
82
+
83
+ // POST /api/search/disclosure - Progressive disclosure search (Search layer)
84
+ searchRouter.post('/disclosure', async (c) => {
85
+ let memoryService: ReturnType<typeof getServiceFromQuery> | undefined;
86
+ try {
87
+ const body = await c.req.json<DisclosureSearchRequest>();
88
+
89
+ if (!body.query) {
90
+ return c.json({ error: 'Query is required' }, 400);
91
+ }
92
+
93
+ memoryService = body.options?.strategy === 'fast'
94
+ ? getLightweightServiceFromQuery(c)
95
+ : getServiceFromQuery(c);
96
+
97
+ await memoryService.initialize();
98
+ const result = await memoryService.searchDisclosure(body.query, body.options);
99
+ return c.json(result);
100
+ } catch (error) {
101
+ return c.json({ error: (error as Error).message }, 500);
102
+ } finally {
103
+ await memoryService?.shutdown();
104
+ }
105
+ });
106
+
107
+ // GET /api/search/disclosure/:resultId/expand - Expand a disclosure search result
108
+ searchRouter.get('/disclosure/:resultId/expand', async (c) => {
109
+ const memoryService = getServiceFromQuery(c);
110
+ try {
111
+ const resultId = c.req.param('resultId');
112
+ const rawWindowSize = c.req.query('windowSize');
113
+ const windowSize = rawWindowSize ? parseInt(rawWindowSize, 10) : undefined;
114
+ const result = await memoryService.expandDisclosure(
115
+ resultId,
116
+ Number.isFinite(windowSize) ? { windowSize } : undefined
117
+ );
118
+
119
+ if (!result) {
120
+ return c.json({ error: 'Expansion target not found' }, 404);
121
+ }
122
+
123
+ return c.json(result);
124
+ } catch (error) {
125
+ return c.json({ error: (error as Error).message }, 500);
126
+ } finally {
127
+ await memoryService.shutdown();
128
+ }
129
+ });
130
+
131
+ // GET /api/search/disclosure/:resultId/source - Resolve source for a disclosure search result
132
+ searchRouter.get('/disclosure/:resultId/source', async (c) => {
133
+ const memoryService = getServiceFromQuery(c);
134
+ try {
135
+ const resultId = c.req.param('resultId');
136
+ const result = await memoryService.sourceDisclosure(resultId);
137
+
138
+ if (!result) {
139
+ return c.json({ error: 'Source not found' }, 404);
140
+ }
141
+
142
+ return c.json(result);
143
+ } catch (error) {
144
+ return c.json({ error: (error as Error).message }, 500);
145
+ } finally {
146
+ await memoryService.shutdown();
147
+ }
148
+ });
149
+
150
+ // GET /api/search - Simple search via query param
151
+ searchRouter.get('/', async (c) => {
152
+ const query = c.req.query('q');
153
+
154
+ if (!query) {
155
+ return c.json({ error: 'Query parameter "q" is required' }, 400);
156
+ }
157
+
158
+ const topK = parseInt(c.req.query('topK') || '5', 10);
159
+ const memoryService = getServiceFromQuery(c);
160
+
161
+ try {
162
+ await memoryService.initialize();
163
+
164
+ const result = await memoryService.retrieveMemories(query, { topK });
165
+
166
+ return c.json({
167
+ results: result.memories.map(m => ({
168
+ id: m.event.id,
169
+ eventType: m.event.eventType,
170
+ timestamp: m.event.timestamp,
171
+ score: m.score,
172
+ preview: m.event.content.slice(0, 200) + (m.event.content.length > 200 ? '...' : '')
173
+ })),
174
+ meta: {
175
+ totalMatches: result.memories.length,
176
+ confidence: result.matchResult.confidence
177
+ }
178
+ });
179
+ } catch (error) {
180
+ return c.json({ error: (error as Error).message }, 500);
181
+ } finally {
182
+ await memoryService.shutdown();
183
+ }
184
+ });
@@ -0,0 +1,115 @@
1
+ /**
2
+ * Sessions API
3
+ * Endpoints for session management
4
+ */
5
+
6
+ import { Hono } from 'hono';
7
+ import { getServiceFromQuery } from './utils.js';
8
+
9
+ export const sessionsRouter = new Hono();
10
+
11
+ // GET /api/sessions - List all sessions
12
+ sessionsRouter.get('/', async (c) => {
13
+ const page = parseInt(c.req.query('page') || '1', 10);
14
+ const pageSize = parseInt(c.req.query('pageSize') || '20', 10);
15
+ const memoryService = getServiceFromQuery(c);
16
+
17
+ try {
18
+ await memoryService.initialize();
19
+
20
+ // Get recent events and extract sessions
21
+ const recentEvents = await memoryService.getRecentEvents(1000);
22
+
23
+ // Group by session
24
+ const sessionMap = new Map<string, {
25
+ id: string;
26
+ startedAt: Date;
27
+ eventCount: number;
28
+ lastEventAt: Date;
29
+ }>();
30
+
31
+ for (const event of recentEvents) {
32
+ const existing = sessionMap.get(event.sessionId);
33
+ if (!existing) {
34
+ sessionMap.set(event.sessionId, {
35
+ id: event.sessionId,
36
+ startedAt: event.timestamp,
37
+ eventCount: 1,
38
+ lastEventAt: event.timestamp
39
+ });
40
+ } else {
41
+ existing.eventCount++;
42
+ if (event.timestamp < existing.startedAt) {
43
+ existing.startedAt = event.timestamp;
44
+ }
45
+ if (event.timestamp > existing.lastEventAt) {
46
+ existing.lastEventAt = event.timestamp;
47
+ }
48
+ }
49
+ }
50
+
51
+ const sessions = Array.from(sessionMap.values())
52
+ .sort((a, b) => b.lastEventAt.getTime() - a.lastEventAt.getTime());
53
+
54
+ const total = sessions.length;
55
+ const start = (page - 1) * pageSize;
56
+ const end = start + pageSize;
57
+ const paginatedSessions = sessions.slice(start, end);
58
+
59
+ return c.json({
60
+ sessions: paginatedSessions,
61
+ total,
62
+ page,
63
+ pageSize,
64
+ hasMore: end < total
65
+ });
66
+ } catch (error) {
67
+ return c.json({ error: (error as Error).message }, 500);
68
+ } finally {
69
+ await memoryService.shutdown();
70
+ }
71
+ });
72
+
73
+ // GET /api/sessions/:id - Get session details
74
+ sessionsRouter.get('/:id', async (c) => {
75
+ const { id } = c.req.param();
76
+ const memoryService = getServiceFromQuery(c);
77
+
78
+ try {
79
+ await memoryService.initialize();
80
+
81
+ const events = await memoryService.getSessionHistory(id);
82
+
83
+ if (events.length === 0) {
84
+ return c.json({ error: 'Session not found' }, 404);
85
+ }
86
+
87
+ const session = {
88
+ id,
89
+ startedAt: events[0].timestamp,
90
+ endedAt: events[events.length - 1].timestamp,
91
+ eventCount: events.length
92
+ };
93
+
94
+ const eventsByType = {
95
+ user_prompt: events.filter(e => e.eventType === 'user_prompt').length,
96
+ agent_response: events.filter(e => e.eventType === 'agent_response').length,
97
+ tool_observation: events.filter(e => e.eventType === 'tool_observation').length
98
+ };
99
+
100
+ return c.json({
101
+ session,
102
+ events: events.slice(0, 100).map(e => ({
103
+ id: e.id,
104
+ eventType: e.eventType,
105
+ timestamp: e.timestamp,
106
+ preview: e.content.slice(0, 200) + (e.content.length > 200 ? '...' : '')
107
+ })),
108
+ stats: eventsByType
109
+ });
110
+ } catch (error) {
111
+ return c.json({ error: (error as Error).message }, 500);
112
+ } finally {
113
+ await memoryService.shutdown();
114
+ }
115
+ });