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,66 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { SummaryDeriver } from '../../src/core/derive/summary-deriver.js';
4
+ import type { MemoryEvent } from '../../src/core/types.js';
5
+
6
+ function event(overrides: Partial<MemoryEvent>): MemoryEvent {
7
+ return {
8
+ id: '11111111-1111-4111-8111-111111111111',
9
+ eventType: 'user_prompt',
10
+ sessionId: 'session-1',
11
+ timestamp: new Date('2026-04-30T00:00:00.000Z'),
12
+ content: 'default content',
13
+ canonicalKey: 'default-content',
14
+ dedupeKey: 'session-1:default-content',
15
+ metadata: {},
16
+ ...overrides
17
+ };
18
+ }
19
+
20
+ describe('SummaryDeriver', () => {
21
+ it('derives the legacy rule-based session summary from events', () => {
22
+ const deriver = new SummaryDeriver();
23
+
24
+ const result = deriver.deriveSessionSummary([
25
+ event({ id: '11111111-1111-4111-8111-111111111111', eventType: 'user_prompt', content: '첫 번째 작업 요청\n자세한 설명', timestamp: new Date('2026-04-30T01:00:00.000Z') }),
26
+ event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response', content: '응답', timestamp: new Date('2026-04-30T01:01:00.000Z') }),
27
+ event({ id: '33333333-3333-4333-8333-333333333333', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:02:00.000Z'), metadata: { toolName: 'terminal', exitCode: 1 } }),
28
+ event({ id: '44444444-4444-4444-8444-444444444444', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:03:00.000Z'), metadata: { toolName: 'terminal', exitCode: 0 } }),
29
+ event({ id: '55555555-5555-4555-8555-555555555555', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:04:00.000Z'), metadata: { toolName: 'read_file', success: false } }),
30
+ event({ id: '66666666-6666-4666-8666-666666666666', eventType: 'user_prompt', content: '두 번째 요청', timestamp: new Date('2026-04-30T01:05:00.000Z') })
31
+ ]);
32
+
33
+ expect(result).toEqual({
34
+ text: '[2026-04-30] 2턴 세션. 주요 작업: 첫 번째 작업 요청 자세한 설명. 사용 툴: terminal, read_file. 오류 2건 발생',
35
+ metadata: { generated: 'rule-based', eventCount: 6 }
36
+ });
37
+ });
38
+
39
+ it('returns null for too-short sessions and sessions that already have summaries', () => {
40
+ const deriver = new SummaryDeriver();
41
+
42
+ expect(deriver.deriveSessionSummary([
43
+ event({ id: '11111111-1111-4111-8111-111111111111' }),
44
+ event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response' })
45
+ ])).toBeNull();
46
+
47
+ expect(deriver.deriveSessionSummary([
48
+ event({ id: '33333333-3333-4333-8333-333333333333' }),
49
+ event({ id: '44444444-4444-4444-8444-444444444444', eventType: 'agent_response' }),
50
+ event({ id: '55555555-5555-4555-8555-555555555555', eventType: 'session_summary', content: 'already summarized' })
51
+ ])).toBeNull();
52
+ });
53
+
54
+ it('limits the first prompt preview to keep summaries compact', () => {
55
+ const deriver = new SummaryDeriver();
56
+ const longPrompt = `${'a'.repeat(140)}\nextra`;
57
+
58
+ const result = deriver.deriveSessionSummary([
59
+ event({ id: '11111111-1111-4111-8111-111111111111', content: longPrompt }),
60
+ event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response', content: '응답' }),
61
+ event({ id: '33333333-3333-4333-8333-333333333333', eventType: 'tool_observation', content: '{}', metadata: { toolName: 'terminal' } })
62
+ ]);
63
+
64
+ expect(result?.text).toBe(`[2026-04-30] 1턴 세션. 주요 작업: ${'a'.repeat(120)}. 사용 툴: terminal`);
65
+ });
66
+ });
@@ -0,0 +1,53 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+
3
+ import {
4
+ isKnownBenignTransformersWarning,
5
+ withSuppressedKnownTransformersWarnings
6
+ } from '../../src/extensions/vector/embedder.js';
7
+
8
+ describe('Embedder warning suppression', () => {
9
+ it('recognizes known benign transformer warnings', () => {
10
+ expect(isKnownBenignTransformersWarning('Unknown model class "eurobert", attempting to construct from base class.')).toBe(true);
11
+ expect(isKnownBenignTransformersWarning('dtype not specified for "model". Using the default dtype (fp32).')).toBe(true);
12
+ expect(isKnownBenignTransformersWarning('serious model load failure')).toBe(false);
13
+ });
14
+
15
+ it('suppresses known transformer warnings but keeps unrelated warnings', async () => {
16
+ const warn = vi.spyOn(console, 'warn').mockImplementation(() => undefined);
17
+
18
+ await withSuppressedKnownTransformersWarnings(async () => {
19
+ console.warn('Unknown model class "eurobert", attempting to construct from base class.');
20
+ console.warn('dtype not specified for "model". Using the default dtype (fp32).');
21
+ console.warn('serious model load failure');
22
+ });
23
+
24
+ expect(warn).toHaveBeenCalledTimes(1);
25
+ expect(warn).toHaveBeenCalledWith('serious model load failure');
26
+ warn.mockRestore();
27
+ });
28
+
29
+ it('restores console.warn only after overlapping suppressions complete', async () => {
30
+ const warn = vi.spyOn(console, 'warn').mockImplementation(() => undefined);
31
+ const originalWarn = console.warn;
32
+ let releaseFirst!: () => void;
33
+ let releaseSecond!: () => void;
34
+
35
+ const first = withSuppressedKnownTransformersWarnings(async () => {
36
+ await new Promise<void>((resolve) => { releaseFirst = resolve; });
37
+ });
38
+ const suppressedWarn = console.warn;
39
+ const second = withSuppressedKnownTransformersWarnings(async () => {
40
+ await new Promise<void>((resolve) => { releaseSecond = resolve; });
41
+ });
42
+
43
+ expect(console.warn).toBe(suppressedWarn);
44
+ releaseFirst();
45
+ await first;
46
+ expect(console.warn).toBe(suppressedWarn);
47
+
48
+ releaseSecond();
49
+ await second;
50
+ expect(console.warn).toBe(originalWarn);
51
+ warn.mockRestore();
52
+ });
53
+ });
@@ -0,0 +1,17 @@
1
+ import { readFileSync } from 'node:fs';
2
+ import { describe, expect, it } from 'vitest';
3
+
4
+ import { createEndlessMemoryServices as createFromExtension } from '../../src/extensions/endless-memory/index.js';
5
+ import { createEndlessMemoryServices as createFromCompat } from '../../src/core/engine/endless-memory-services.js';
6
+
7
+ describe('endless-memory extension boundary', () => {
8
+ it('keeps endless memory implementation under extensions with an engine compatibility re-export', () => {
9
+ const compatSource = readFileSync('src/core/engine/endless-memory-services.ts', 'utf8');
10
+ const memoryServiceSource = readFileSync('src/services/memory-service.ts', 'utf8');
11
+
12
+ expect(createFromCompat).toBe(createFromExtension);
13
+ expect(compatSource).toContain("../../extensions/endless-memory/index.js");
14
+ expect(memoryServiceSource).not.toContain("../core/engine/endless-memory-services.js");
15
+ expect(memoryServiceSource).toContain("../extensions/endless-memory/index.js");
16
+ });
17
+ });
@@ -0,0 +1,325 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import {
4
+ createEndlessMemoryServices,
5
+ type EndlessMemoryServicesFactories
6
+ } from '../../src/core/engine/endless-memory-services.js';
7
+ import type { EventStore } from '../../src/core/event-store.js';
8
+ import type {
9
+ ConsolidatedMemory,
10
+ EndlessModeConfig,
11
+ MemoryEvent,
12
+ WorkingSet
13
+ } from '../../src/core/types.js';
14
+
15
+ function event(id: string): MemoryEvent {
16
+ return {
17
+ id,
18
+ sessionId: 'session-1',
19
+ eventType: 'agent_response',
20
+ content: `event ${id}`,
21
+ canonicalKey: `event/${id}`,
22
+ dedupeKey: `session-1:${id}`,
23
+ timestamp: new Date('2026-05-02T00:00:00.000Z'),
24
+ metadata: {}
25
+ };
26
+ }
27
+
28
+ function consolidated(memoryId: string): ConsolidatedMemory {
29
+ return {
30
+ memoryId,
31
+ summary: `summary ${memoryId}`,
32
+ topics: ['thin-core'],
33
+ sourceEvents: ['event-1'],
34
+ confidence: 0.9,
35
+ createdAt: new Date('2026-05-02T00:00:00.000Z'),
36
+ accessCount: 0
37
+ };
38
+ }
39
+
40
+ function makeHarness(options?: { savedMode?: 'session' | 'endless'; savedConfig?: EndlessModeConfig }) {
41
+ let initializeCalls = 0;
42
+ const starts: string[] = [];
43
+ const stops: string[] = [];
44
+ const activity: string[] = [];
45
+ const forced: string[] = [];
46
+ const added: Array<{ eventId: string; relevanceScore?: number }> = [];
47
+ const searches: Array<{ query: string; options?: { topK?: number } }> = [];
48
+ const getAllCalls: Array<{ limit?: number }> = [];
49
+ const marked: string[] = [];
50
+ const snapshots: Array<{ id: string; content: string; metadata?: { files?: string[]; entities?: string[] } }> = [];
51
+ const created: Array<{ kind: string; config?: EndlessModeConfig }> = [];
52
+ const setConfigCalls: Array<{ key: string; value: unknown }> = [];
53
+ const configValues: Record<string, unknown> = {};
54
+ if (options?.savedMode) configValues.mode = options.savedMode;
55
+ if (options?.savedConfig) configValues.config = options.savedConfig;
56
+
57
+ const workingSet: WorkingSet = {
58
+ recentEvents: [event('event-1')],
59
+ lastActivity: new Date('2026-05-02T00:00:00.000Z'),
60
+ continuityScore: 0.82
61
+ };
62
+ const lastConsolidation = new Date('2026-05-02T01:00:00.000Z');
63
+
64
+ const factories: EndlessMemoryServicesFactories = {
65
+ createWorkingSetStore: (_eventStore, config) => {
66
+ created.push({ kind: 'working-set', config });
67
+ return {
68
+ add: async (eventId, relevanceScore) => { added.push({ eventId, relevanceScore }); },
69
+ get: async () => workingSet,
70
+ count: async () => 7
71
+ };
72
+ },
73
+ createConsolidatedStore: () => {
74
+ created.push({ kind: 'consolidated' });
75
+ return {
76
+ search: async (query, searchOptions) => {
77
+ searches.push({ query, options: searchOptions });
78
+ return [consolidated('memory-1')];
79
+ },
80
+ getAll: async (getAllOptions) => {
81
+ getAllCalls.push({ limit: getAllOptions?.limit });
82
+ return [consolidated('memory-2')];
83
+ },
84
+ markAccessed: async (memoryId) => { marked.push(memoryId); },
85
+ count: async () => 3,
86
+ getLastConsolidationTime: async () => lastConsolidation
87
+ };
88
+ },
89
+ createConsolidationWorker: () => {
90
+ created.push({ kind: 'worker' });
91
+ return {
92
+ start: () => { starts.push('worker'); },
93
+ stop: () => { stops.push('worker'); },
94
+ recordActivity: () => { activity.push('worker'); },
95
+ forceRun: async () => {
96
+ forced.push('worker');
97
+ return 5;
98
+ }
99
+ };
100
+ },
101
+ createContinuityManager: (_eventStore, config) => {
102
+ created.push({ kind: 'continuity', config });
103
+ return {
104
+ createSnapshot: (id, content, metadata) => {
105
+ snapshots.push({ id, content, metadata });
106
+ return { id, content, metadata };
107
+ },
108
+ calculateScore: async () => ({ score: 0.91, transitionType: 'seamless' })
109
+ };
110
+ },
111
+ randomUUID: () => 'snapshot-1'
112
+ };
113
+
114
+ const configStore = {
115
+ getEndlessConfig: async (key: string) => configValues[key] ?? null,
116
+ setEndlessConfig: async (key: string, value: unknown) => {
117
+ setConfigCalls.push({ key, value });
118
+ configValues[key] = value;
119
+ }
120
+ };
121
+
122
+ const services = createEndlessMemoryServices({
123
+ eventStore: { marker: 'event-store' } as unknown as EventStore,
124
+ configStore,
125
+ initialize: async () => { initializeCalls += 1; },
126
+ factories
127
+ });
128
+
129
+ return {
130
+ services,
131
+ get initializeCalls() { return initializeCalls; },
132
+ starts,
133
+ stops,
134
+ activity,
135
+ forced,
136
+ added,
137
+ searches,
138
+ getAllCalls,
139
+ marked,
140
+ snapshots,
141
+ created,
142
+ setConfigCalls,
143
+ configValues,
144
+ lastConsolidation
145
+ };
146
+ }
147
+
148
+ describe('createEndlessMemoryServices', () => {
149
+ it('loads saved endless mode and starts the consolidation worker', async () => {
150
+ const harness = makeHarness({ savedMode: 'endless' });
151
+
152
+ await harness.services.initializeFromSavedMode();
153
+
154
+ expect(harness.services.getMode()).toBe('endless');
155
+ expect(harness.services.isEndlessModeActive()).toBe(true);
156
+ expect(harness.created.map((entry) => entry.kind)).toEqual([
157
+ 'working-set',
158
+ 'consolidated',
159
+ 'worker',
160
+ 'continuity'
161
+ ]);
162
+ expect(harness.starts).toEqual(['worker']);
163
+ });
164
+
165
+ it('persists mode changes and clears endless components when returning to session mode', async () => {
166
+ const harness = makeHarness();
167
+
168
+ await harness.services.setMode('endless');
169
+ await harness.services.addToWorkingSet('event-1', 0.7);
170
+ await expect(harness.services.getWorkingSet()).resolves.toMatchObject({ continuityScore: 0.82 });
171
+ await expect(harness.services.searchConsolidated('thin core', { topK: 2 })).resolves.toMatchObject([
172
+ { memoryId: 'memory-1' }
173
+ ]);
174
+
175
+ await harness.services.setMode('session');
176
+
177
+ expect(harness.initializeCalls).toBe(2);
178
+ expect(harness.setConfigCalls.map((call) => call.value)).toEqual(['endless', 'session']);
179
+ expect(harness.added).toEqual([{ eventId: 'event-1', relevanceScore: 0.7 }]);
180
+ expect(harness.searches).toEqual([{ query: 'thin core', options: { topK: 2 } }]);
181
+ expect(harness.stops).toEqual(['worker']);
182
+ expect(harness.services.getMode()).toBe('session');
183
+ await expect(harness.services.getWorkingSet()).resolves.toBeNull();
184
+ await expect(harness.services.searchConsolidated('thin core')).resolves.toEqual([]);
185
+ await expect(harness.services.forceConsolidation()).resolves.toBe(0);
186
+ });
187
+
188
+ it('formats no endless context while session mode is active', async () => {
189
+ await expect(makeHarness().services.formatEndlessContext('continue refactor')).resolves.toBe('');
190
+ });
191
+
192
+ it('formats endless context from continuity, working set, and consolidated memories', async () => {
193
+ const harness = makeHarness();
194
+ await harness.services.setMode('endless');
195
+ const initializeCallsAfterModeSwitch = harness.initializeCalls;
196
+
197
+ const formatted = await harness.services.formatEndlessContext('continue refactor');
198
+
199
+ expect(formatted).toContain('🔗 Context: seamless (score: 0.91)');
200
+ expect(formatted).toContain('## Recent Context (Working Set)');
201
+ expect(formatted).toContain('[agent_response] event event-1');
202
+ expect(formatted).toContain('## Related Knowledge (Consolidated)');
203
+ expect(formatted).toContain('thin-core: summary memory-1...');
204
+ expect(harness.searches).toEqual([
205
+ { query: 'continue refactor', options: { topK: 3 } }
206
+ ]);
207
+ expect(harness.snapshots).toEqual([
208
+ { id: 'snapshot-1', content: 'continue refactor', metadata: undefined }
209
+ ]);
210
+ expect(harness.initializeCalls).toBe(initializeCallsAfterModeSwitch);
211
+ });
212
+
213
+ it('delegates config, continuity, activity, consolidation, and status operations', async () => {
214
+ const harness = makeHarness();
215
+
216
+ await expect(harness.services.getEndlessConfig()).resolves.toMatchObject({
217
+ enabled: true,
218
+ workingSet: { maxEvents: 100 },
219
+ continuity: { minScoreForSeamless: 0.7 }
220
+ });
221
+ await harness.services.setEndlessConfig({
222
+ consolidation: {
223
+ triggerIntervalMs: 100,
224
+ triggerEventCount: 10,
225
+ triggerIdleMs: 50,
226
+ useLLMSummarization: true
227
+ }
228
+ });
229
+ await harness.services.setMode('endless');
230
+
231
+ await expect(harness.services.getConsolidatedMemories(4)).resolves.toMatchObject([
232
+ { memoryId: 'memory-2' }
233
+ ]);
234
+ await harness.services.markMemoryAccessed('memory-2');
235
+ await expect(harness.services.calculateContinuity('continue refactor', { files: ['src/a.ts'] })).resolves.toEqual({
236
+ score: 0.91,
237
+ transitionType: 'seamless'
238
+ });
239
+ harness.services.recordActivity();
240
+ await expect(harness.services.forceConsolidation()).resolves.toBe(5);
241
+ await expect(harness.services.getEndlessModeStatus()).resolves.toEqual({
242
+ mode: 'endless',
243
+ workingSetSize: 7,
244
+ continuityScore: 0.82,
245
+ consolidatedCount: 3,
246
+ lastConsolidation: harness.lastConsolidation
247
+ });
248
+ harness.services.shutdown();
249
+
250
+ expect(harness.getAllCalls).toEqual([{ limit: 4 }]);
251
+ expect(harness.marked).toEqual(['memory-2']);
252
+ expect(harness.snapshots).toEqual([{ id: 'snapshot-1', content: 'continue refactor', metadata: { files: ['src/a.ts'] } }]);
253
+ expect(harness.activity).toEqual(['worker']);
254
+ expect(harness.forced).toEqual(['worker']);
255
+ expect(harness.initializeCalls).toBe(2);
256
+ expect(harness.stops).toEqual(['worker']);
257
+ });
258
+
259
+ it('does not keep partial endless state when component startup fails', async () => {
260
+ const harness = makeHarness();
261
+ let failContinuityCreation = true;
262
+ const baseCreateContinuityManager = harness.created;
263
+
264
+ const factories: EndlessMemoryServicesFactories = {
265
+ createWorkingSetStore: (_eventStore, _config) => {
266
+ baseCreateContinuityManager.push({ kind: 'working-set' });
267
+ return {
268
+ add: async () => {},
269
+ get: async () => ({ recentEvents: [], lastActivity: new Date('2026-05-02T00:00:00.000Z'), continuityScore: 0.5 }),
270
+ count: async () => 0
271
+ };
272
+ },
273
+ createConsolidatedStore: () => {
274
+ baseCreateContinuityManager.push({ kind: 'consolidated' });
275
+ return {
276
+ search: async () => [],
277
+ getAll: async () => [],
278
+ markAccessed: async () => {},
279
+ count: async () => 0,
280
+ getLastConsolidationTime: async () => null
281
+ };
282
+ },
283
+ createConsolidationWorker: () => {
284
+ baseCreateContinuityManager.push({ kind: 'worker' });
285
+ return {
286
+ start: () => { harness.starts.push('worker'); },
287
+ stop: () => { harness.stops.push('worker'); },
288
+ recordActivity: () => {},
289
+ forceRun: async () => 1
290
+ };
291
+ },
292
+ createContinuityManager: () => {
293
+ baseCreateContinuityManager.push({ kind: 'continuity' });
294
+ if (failContinuityCreation) {
295
+ failContinuityCreation = false;
296
+ throw new Error('continuity unavailable');
297
+ }
298
+ return {
299
+ createSnapshot: () => ({}),
300
+ calculateScore: async () => ({ score: 0.5, transitionType: 'break' })
301
+ };
302
+ },
303
+ randomUUID: () => 'snapshot-1'
304
+ };
305
+
306
+ const services = createEndlessMemoryServices({
307
+ eventStore: { marker: 'event-store' } as unknown as EventStore,
308
+ configStore: {
309
+ getEndlessConfig: async () => null,
310
+ setEndlessConfig: async () => {}
311
+ },
312
+ initialize: async () => {},
313
+ factories
314
+ });
315
+
316
+ await expect(services.initializeEndlessMode()).rejects.toThrow('continuity unavailable');
317
+ expect(harness.starts).toEqual([]);
318
+ await expect(services.forceConsolidation()).resolves.toBe(0);
319
+
320
+ await expect(services.initializeEndlessMode()).resolves.toBeUndefined();
321
+ expect(harness.starts).toEqual(['worker']);
322
+ await expect(services.forceConsolidation()).resolves.toBe(1);
323
+ });
324
+
325
+ });