squish-memory 1.1.5 → 1.2.0

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 (499) hide show
  1. package/.env.example +32 -16
  2. package/CHANGELOG.md +147 -0
  3. package/README.md +120 -78
  4. package/{scripts → bin}/dependency-manager.mjs +217 -217
  5. package/{scripts → bin}/detect-clients.mjs +78 -78
  6. package/bin/install-interactive.mjs +321 -0
  7. package/bin/squish-mcp.mjs +46 -0
  8. package/bin/squish.mjs +33 -0
  9. package/config/mcp-migration-map.json +1 -6
  10. package/config/mcp-mode-semantics.json +19 -23
  11. package/config/mcp-remote-auth.json +3 -26
  12. package/config/mcp-universal.schema.json +5 -35
  13. package/config/settings.json +107 -52
  14. package/config.js +5 -0
  15. package/config.ts +218 -0
  16. package/core/adapters/config/claude-code.ts +133 -0
  17. package/core/adapters/config/cursor.ts +90 -0
  18. package/core/adapters/config/opencode.ts +89 -0
  19. package/core/adapters/config/windsurf.ts +90 -0
  20. package/core/adapters/index.ts +102 -0
  21. package/core/adapters/timeline.ts +116 -0
  22. package/core/adapters/types.ts +166 -0
  23. package/core/agent-preferences.ts +140 -0
  24. package/core/algorithms/analytics/token-estimator.ts +216 -0
  25. package/core/algorithms/detection/hash-filters.ts +260 -0
  26. package/core/algorithms/detection/semantic-ranker.ts +194 -0
  27. package/core/algorithms/detection/two-stage-detector.ts +421 -0
  28. package/core/algorithms/handlers/approve-merge.ts +215 -0
  29. package/core/algorithms/handlers/detect-duplicates.ts +192 -0
  30. package/core/algorithms/handlers/get-stats.ts +132 -0
  31. package/core/algorithms/handlers/list-proposals.ts +130 -0
  32. package/core/algorithms/handlers/preview-merge.ts +139 -0
  33. package/core/algorithms/handlers/reject-merge.ts +93 -0
  34. package/core/algorithms/handlers/reverse-merge.ts +155 -0
  35. package/core/algorithms/index.ts +39 -0
  36. package/core/algorithms/operations/cache-maintenance.ts +182 -0
  37. package/core/algorithms/safety/safety-checks.ts +256 -0
  38. package/core/algorithms/strategies/merge-strategies.ts +381 -0
  39. package/core/algorithms/types.ts +140 -0
  40. package/core/algorithms/utils/response-builder.ts +61 -0
  41. package/core/associations.ts +363 -0
  42. package/core/beliefs/decay.ts +289 -0
  43. package/core/beliefs/extractor.ts +131 -0
  44. package/core/beliefs/store.ts +557 -0
  45. package/core/beliefs/types.ts +38 -0
  46. package/core/commands/mcp-server.ts +5 -0
  47. package/core/compression.ts +177 -0
  48. package/core/config.js +2 -0
  49. package/core/consolidation.ts +330 -0
  50. package/core/context/agent-context.ts +388 -0
  51. package/core/context/context-paging.ts +449 -0
  52. package/core/context/context-window.ts +234 -0
  53. package/core/context/context.ts +35 -0
  54. package/core/embeddings/embeddings.ts +616 -0
  55. package/core/embeddings/google-multimodal.ts +200 -0
  56. package/{dist/core/local-embeddings.js → core/embeddings/local-embeddings.ts} +12 -11
  57. package/core/embeddings/qmd-client.ts +495 -0
  58. package/core/embeddings/transformers-local.ts +261 -0
  59. package/core/embeddings.js +4 -0
  60. package/core/error-handling.ts +206 -0
  61. package/core/external +219 -0
  62. package/core/graph/entity-deduplicator.ts +232 -0
  63. package/core/graph/graph-builder.ts +257 -0
  64. package/core/graph/graph-traversal.ts +490 -0
  65. package/core/graph/index.ts +24 -0
  66. package/core/graph/llm-entity-extractor.ts +402 -0
  67. package/core/graph/multi-hop-retrieval.ts +317 -0
  68. package/core/graph/relationship-extractor.ts +465 -0
  69. package/core/hooks/agent-hooks.ts +653 -0
  70. package/core/hooks/auto-tagger.ts +149 -0
  71. package/core/hooks/capture-filter.ts +169 -0
  72. package/core/hot-cache.ts +388 -0
  73. package/core/index.ts +10 -0
  74. package/core/ingestion/agent-memory.ts +167 -0
  75. package/core/ingestion/core-memory.ts +326 -0
  76. package/core/ingestion/learnings.ts +260 -0
  77. package/core/ingestion/signal-engine.ts +266 -0
  78. package/core/integrations/obsidian-vault.ts +197 -0
  79. package/core/layers/generator.ts +115 -0
  80. package/core/lib/db-client.ts +168 -0
  81. package/core/lib/parse-embedding.ts +59 -0
  82. package/core/lib/schemas.ts +102 -0
  83. package/core/lib/types.ts +49 -0
  84. package/core/lib/utils.ts +151 -0
  85. package/core/lib/validation.ts +180 -0
  86. package/core/lifecycle.ts +353 -0
  87. package/core/logger.ts +59 -0
  88. package/core/memory/bridge-discovery.ts +395 -0
  89. package/core/memory/categorizer.ts +390 -0
  90. package/core/memory/conflict-detector.ts +62 -0
  91. package/core/memory/consolidation.ts +372 -0
  92. package/core/memory/context-collector.ts +75 -0
  93. package/core/memory/contradiction-resolver.ts +494 -0
  94. package/core/memory/edit-workflow.ts +174 -0
  95. package/core/memory/entity-extractor.ts +426 -0
  96. package/core/memory/entity-resolver.ts +89 -0
  97. package/core/memory/explain.ts +112 -0
  98. package/core/memory/fact-deriver.ts +300 -0
  99. package/core/memory/fact-extractor.ts +120 -0
  100. package/core/memory/feedback-tracker.ts +200 -0
  101. package/core/memory/hooks.ts +230 -0
  102. package/core/memory/hybrid-retrieval.ts +65 -0
  103. package/core/memory/hybrid-scorer.ts +325 -0
  104. package/core/memory/hybrid-search.ts +748 -0
  105. package/core/memory/importance.ts +319 -0
  106. package/core/memory/index.ts +11 -0
  107. package/core/memory/loader.ts +178 -0
  108. package/core/memory/markdown/markdown-storage.ts +318 -0
  109. package/core/memory/memories.ts +565 -0
  110. package/core/memory/memory-lifecycle.ts +51 -0
  111. package/core/memory/memory-manager.ts +53 -0
  112. package/core/memory/migrate.ts +173 -0
  113. package/core/memory/normalization.ts +30 -0
  114. package/core/memory/path-strengthener.ts +211 -0
  115. package/core/memory/progressive-disclosure.ts +392 -0
  116. package/core/memory/query-processor.ts +130 -0
  117. package/core/memory/query-rewriter.ts +153 -0
  118. package/core/memory/response-analyzer.ts +81 -0
  119. package/core/memory/retrieval-feedback.ts +276 -0
  120. package/core/memory/serialization.ts +83 -0
  121. package/core/memory/stale-cleaner.ts +147 -0
  122. package/core/memory/stats.ts +181 -0
  123. package/core/memory/telemetry.ts +392 -0
  124. package/core/memory/temporal-facts.ts +356 -0
  125. package/core/memory/temporal-parser.ts +477 -0
  126. package/core/memory/trigger-detector.ts +104 -0
  127. package/core/memory/write-gate.ts +288 -0
  128. package/core/places/index.ts +14 -0
  129. package/core/places/memory-places.ts +339 -0
  130. package/core/places/places.ts +406 -0
  131. package/core/places/rules.ts +308 -0
  132. package/core/places/walking.ts +192 -0
  133. package/core/projects +89 -0
  134. package/core/projects.ts +131 -0
  135. package/core/redis.ts +82 -0
  136. package/core/responses.ts +187 -0
  137. package/core/runtime/trust-report.ts +195 -0
  138. package/core/runtime/trust-state.ts +360 -0
  139. package/core/scheduler/cron-scheduler.ts +581 -0
  140. package/core/scheduler/heartbeat.ts +91 -0
  141. package/core/scheduler/index.ts +8 -0
  142. package/core/scheduler/job-runner.ts +197 -0
  143. package/core/search/conversations.ts +166 -0
  144. package/core/search/entities.ts +46 -0
  145. package/core/search/folder-context.ts +154 -0
  146. package/core/search/graph-boost.ts +22 -0
  147. package/core/search/index.ts +4 -0
  148. package/core/search/qmd-wrapper.ts +84 -0
  149. package/core/security/encrypt.ts +51 -0
  150. package/core/security/governance.ts +102 -0
  151. package/core/security/privacy.ts +108 -0
  152. package/core/security/secret-detector.ts +122 -0
  153. package/core/session/auto-load.ts +160 -0
  154. package/core/session/entity-tracker.ts +363 -0
  155. package/core/session/index.ts +7 -0
  156. package/core/session/reference-resolver.ts +158 -0
  157. package/core/session/self-iteration-job.ts +478 -0
  158. package/core/session/session-hooks.ts +69 -0
  159. package/core/session/types.ts +36 -0
  160. package/core/session/working-set.ts +275 -0
  161. package/core/snapshots/cleanup.ts +13 -0
  162. package/core/snapshots/comparison.ts +59 -0
  163. package/core/snapshots/creation.ts +139 -0
  164. package/core/snapshots/retrieval.ts +44 -0
  165. package/core/snapshots/stats.ts +63 -0
  166. package/core/storage/cache.ts +241 -0
  167. package/core/storage/database.ts +23 -0
  168. package/core/summarization/cleanup.ts +13 -0
  169. package/core/summarization/queries.ts +32 -0
  170. package/core/summarization/stats.ts +64 -0
  171. package/core/summarization/strategies.ts +52 -0
  172. package/core/summarization.ts +248 -0
  173. package/core/temporal-facts.ts +244 -0
  174. package/core/tracing/collector.ts +470 -0
  175. package/core/tracing/visualizer.ts +195 -0
  176. package/core/utils/cleanup-operations.ts +50 -0
  177. package/core/utils/content-extraction.ts +95 -0
  178. package/core/utils/filter-builder.ts +56 -0
  179. package/core/utils/history-traversal.ts +63 -0
  180. package/core/utils/memory-operations.ts +56 -0
  181. package/core/utils/query-operations.ts +83 -0
  182. package/core/utils/summarization-helpers.ts +45 -0
  183. package/core/utils/temporal-queries.ts +39 -0
  184. package/core/utils/vector-operations.ts +135 -0
  185. package/core/utils/version-management.ts +74 -0
  186. package/core/worker.ts +324 -0
  187. package/db/adapter.ts +215 -0
  188. package/db/bootstrap.ts +1055 -0
  189. package/db/drizzle/migrations/0000_needy_cerebro.sql +402 -0
  190. package/db/drizzle/migrations/meta/0000_snapshot.json +3451 -0
  191. package/db/drizzle/migrations/meta/_journal.json +13 -0
  192. package/db/drizzle/schema-sqlite.ts +1032 -0
  193. package/db/drizzle/schema.ts +1128 -0
  194. package/db/drizzle.config.ts +12 -0
  195. package/db/index.ts +83 -0
  196. package/db/init.sql +5 -0
  197. package/db/migrations/associations.ts +35 -0
  198. package/db/migrations/beliefs.ts +89 -0
  199. package/db/migrations/core-memory.ts +35 -0
  200. package/db/migrations/fts.ts +59 -0
  201. package/db/migrations/index.ts +54 -0
  202. package/db/migrations/indexes.ts +36 -0
  203. package/db/migrations/learnings.ts +34 -0
  204. package/db/migrations/maintenance.ts +68 -0
  205. package/db/migrations/memories.ts +22 -0
  206. package/db/migrations/memory-places.ts +35 -0
  207. package/db/migrations/places.ts +49 -0
  208. package/db/migrations/projects.ts +21 -0
  209. package/db/migrations/tier-conversion.ts +24 -0
  210. package/db/neon.ts +22 -0
  211. package/db/schema/beliefs.ts +50 -0
  212. package/db/schema/generator.ts +159 -0
  213. package/db/schema/index.ts +58 -0
  214. package/db/schema/learnings.ts +32 -0
  215. package/db/schema/memories.ts +83 -0
  216. package/db/schema/projects.ts +33 -0
  217. package/db/schema.ts +13 -0
  218. package/db/supabase.ts +27 -0
  219. package/dist/config.d.ts +40 -17
  220. package/dist/config.js +150 -198
  221. package/dist/core/adapters/types.d.ts +13 -33
  222. package/dist/core/adapters/types.js +1 -1
  223. package/dist/core/agent-preferences.d.ts +16 -0
  224. package/dist/core/agent-preferences.js +124 -0
  225. package/dist/core/algorithms/safety/safety-checks.d.ts +1 -5
  226. package/dist/core/algorithms/types.d.ts +0 -8
  227. package/dist/core/associations.d.ts +3 -1
  228. package/dist/core/associations.js +37 -1
  229. package/dist/core/beliefs/decay.d.ts +27 -0
  230. package/dist/core/beliefs/decay.js +217 -0
  231. package/dist/core/beliefs/extractor.d.ts +9 -0
  232. package/dist/core/beliefs/extractor.js +113 -0
  233. package/dist/core/beliefs/store.d.ts +46 -0
  234. package/dist/core/beliefs/store.js +466 -0
  235. package/dist/core/beliefs/types.d.ts +28 -0
  236. package/dist/core/beliefs/types.js +2 -0
  237. package/dist/core/commands/mcp-server.d.ts +0 -1
  238. package/dist/core/commands/mcp-server.js +4 -737
  239. package/dist/core/commands/remember.d.ts +24 -0
  240. package/dist/core/commands/remember.js +144 -0
  241. package/dist/core/{toon.d.ts → compression.d.ts} +6 -4
  242. package/dist/core/{toon.js → compression.js} +8 -8
  243. package/dist/core/context/agent-context.js +1 -1
  244. package/dist/core/embeddings/embeddings.d.ts +29 -0
  245. package/dist/core/embeddings/embeddings.js +546 -0
  246. package/dist/core/embeddings/google-multimodal.js +6 -2
  247. package/dist/core/{local-embeddings.d.ts → embeddings/local-embeddings.d.ts} +1 -1
  248. package/dist/core/embeddings/local-embeddings.js +11 -0
  249. package/dist/core/embeddings/qmd-client.js +1 -1
  250. package/dist/core/embeddings/transformers-local.d.ts +64 -0
  251. package/dist/core/embeddings/transformers-local.js +213 -0
  252. package/dist/core/embeddings.d.ts +1 -28
  253. package/dist/core/embeddings.js +2 -453
  254. package/dist/core/graph/entity-deduplicator.d.ts +24 -0
  255. package/dist/core/graph/entity-deduplicator.js +183 -0
  256. package/dist/core/graph/graph-builder.d.ts +46 -0
  257. package/dist/core/graph/graph-builder.js +174 -0
  258. package/dist/core/graph/graph-traversal.d.ts +80 -0
  259. package/dist/core/graph/graph-traversal.js +315 -0
  260. package/dist/core/graph/index.d.ts +19 -0
  261. package/dist/core/graph/index.js +13 -0
  262. package/dist/core/graph/llm-entity-extractor.d.ts +49 -0
  263. package/dist/core/graph/llm-entity-extractor.js +313 -0
  264. package/dist/core/graph/multi-hop-retrieval.d.ts +48 -0
  265. package/dist/core/graph/multi-hop-retrieval.js +215 -0
  266. package/dist/core/graph/relationship-extractor.d.ts +48 -0
  267. package/dist/core/graph/relationship-extractor.js +351 -0
  268. package/dist/core/hooks/agent-hooks.d.ts +10 -1
  269. package/dist/core/hooks/agent-hooks.js +301 -24
  270. package/dist/core/hot-cache.d.ts +86 -0
  271. package/dist/core/hot-cache.js +285 -0
  272. package/dist/core/index.d.ts +9 -9
  273. package/dist/core/index.js +9 -12
  274. package/dist/core/ingestion/core-memory.d.ts +2 -2
  275. package/dist/core/ingestion/core-memory.js +3 -3
  276. package/dist/core/ingestion/learnings.js +3 -0
  277. package/dist/core/ingestion/signal-engine.d.ts +41 -0
  278. package/dist/core/ingestion/signal-engine.js +201 -0
  279. package/dist/core/{obsidian-vault.d.ts → integrations/obsidian-vault.d.ts} +2 -1
  280. package/dist/core/{obsidian-vault.js → integrations/obsidian-vault.js} +69 -7
  281. package/dist/core/lib/parse-embedding.d.ts +9 -0
  282. package/dist/core/lib/parse-embedding.js +58 -0
  283. package/dist/core/lib/schemas.d.ts +57 -54
  284. package/dist/core/lib/types.d.ts +45 -0
  285. package/dist/core/lib/types.js +6 -0
  286. package/dist/core/lib/utils.d.ts +4 -0
  287. package/dist/core/lib/utils.js +55 -0
  288. package/dist/core/lifecycle.d.ts +0 -1
  289. package/dist/core/lifecycle.js +13 -23
  290. package/dist/core/logger.d.ts +1 -0
  291. package/dist/core/logger.js +14 -8
  292. package/dist/core/mcp/tools.d.ts +0 -2
  293. package/dist/core/mcp/tools.js +0 -87
  294. package/dist/core/mcp/types.d.ts +25 -253
  295. package/dist/core/mcp/types.js +2 -2
  296. package/dist/core/memory/categorizer.js +1 -0
  297. package/dist/core/memory/consolidation.js +2 -28
  298. package/dist/core/memory/entity-extractor.d.ts +4 -0
  299. package/dist/core/memory/entity-extractor.js +30 -16
  300. package/dist/core/memory/explain.d.ts +18 -0
  301. package/dist/core/memory/explain.js +92 -0
  302. package/dist/core/memory/fact-deriver.d.ts +31 -0
  303. package/dist/core/memory/fact-deriver.js +236 -0
  304. package/dist/core/memory/hybrid-retrieval.d.ts +14 -16
  305. package/dist/core/memory/hybrid-retrieval.js +25 -127
  306. package/dist/core/memory/hybrid-scorer.js +6 -23
  307. package/dist/core/memory/hybrid-search.d.ts +10 -7
  308. package/dist/core/memory/hybrid-search.js +458 -221
  309. package/dist/core/memory/importance.d.ts +0 -17
  310. package/dist/core/memory/importance.js +1 -58
  311. package/dist/core/memory/index.d.ts +1 -0
  312. package/dist/core/memory/index.js +1 -0
  313. package/dist/core/memory/memories.d.ts +13 -17
  314. package/dist/core/memory/memories.js +78 -75
  315. package/dist/core/memory/memory-lifecycle.d.ts +2 -2
  316. package/dist/core/memory/memory-lifecycle.js +10 -18
  317. package/dist/core/memory/normalization.d.ts +1 -16
  318. package/dist/core/memory/path-strengthener.d.ts +39 -0
  319. package/dist/core/memory/path-strengthener.js +150 -0
  320. package/dist/core/memory/query-processor.js +37 -3
  321. package/dist/core/memory/retrieval-feedback.d.ts +70 -0
  322. package/dist/core/memory/retrieval-feedback.js +213 -0
  323. package/dist/core/memory/stale-cleaner.d.ts +26 -0
  324. package/dist/core/memory/stale-cleaner.js +97 -0
  325. package/dist/core/memory/stats.d.ts +10 -0
  326. package/dist/core/memory/stats.js +8 -3
  327. package/dist/core/memory/trigger-detector.d.ts +8 -1
  328. package/dist/core/memory/trigger-detector.js +42 -5
  329. package/dist/core/places/index.d.ts +1 -1
  330. package/dist/core/places/index.js +1 -1
  331. package/dist/core/places/places.d.ts +13 -13
  332. package/dist/core/places/places.js +27 -27
  333. package/dist/core/places/rules.js +23 -23
  334. package/dist/core/places/walking.d.ts +3 -3
  335. package/dist/core/places/walking.js +7 -7
  336. package/dist/core/projects.js +8 -0
  337. package/dist/core/runtime/trust-report.d.ts +102 -0
  338. package/dist/core/runtime/trust-report.js +107 -0
  339. package/dist/core/runtime/trust-state.d.ts +12 -0
  340. package/dist/core/runtime/trust-state.js +309 -0
  341. package/dist/core/scheduler/cron-scheduler.d.ts +1 -1
  342. package/dist/core/scheduler/cron-scheduler.js +164 -3
  343. package/dist/core/scheduler/job-runner.js +1 -1
  344. package/dist/core/search/qmd-wrapper.d.ts +36 -0
  345. package/dist/core/search/qmd-wrapper.js +58 -0
  346. package/dist/core/session/auto-load.js +28 -3
  347. package/dist/core/session/entity-tracker.d.ts +62 -0
  348. package/dist/core/session/entity-tracker.js +287 -0
  349. package/dist/core/session/reference-resolver.d.ts +26 -0
  350. package/dist/core/session/reference-resolver.js +121 -0
  351. package/dist/core/session/self-iteration-job.d.ts +15 -0
  352. package/dist/core/session/self-iteration-job.js +163 -58
  353. package/dist/core/session/working-set.d.ts +50 -0
  354. package/dist/core/session/working-set.js +212 -0
  355. package/dist/core/snapshots/creation.d.ts +2 -8
  356. package/dist/core/snapshots/creation.js +3 -12
  357. package/dist/core/utils/summarization-helpers.d.ts +0 -4
  358. package/dist/core/utils/summarization-helpers.js +1 -6
  359. package/dist/db/bootstrap.d.ts +2 -0
  360. package/dist/db/bootstrap.js +229 -280
  361. package/dist/db/drizzle/schema-sqlite.d.ts +702 -1
  362. package/dist/db/drizzle/schema-sqlite.js +83 -4
  363. package/dist/db/drizzle/schema.d.ts +653 -1
  364. package/dist/db/drizzle/schema.js +93 -4
  365. package/dist/db/migrations/associations.d.ts +6 -0
  366. package/dist/db/migrations/associations.js +29 -0
  367. package/dist/db/migrations/beliefs.d.ts +10 -0
  368. package/dist/db/migrations/beliefs.js +76 -0
  369. package/dist/db/migrations/core-memory.d.ts +6 -0
  370. package/dist/db/migrations/core-memory.js +29 -0
  371. package/dist/db/migrations/fts.d.ts +6 -0
  372. package/dist/db/migrations/fts.js +52 -0
  373. package/dist/db/migrations/index.d.ts +25 -0
  374. package/dist/db/migrations/index.js +51 -0
  375. package/dist/db/migrations/indexes.d.ts +6 -0
  376. package/dist/db/migrations/indexes.js +30 -0
  377. package/dist/db/migrations/learnings.d.ts +7 -0
  378. package/dist/db/migrations/learnings.js +26 -0
  379. package/dist/db/migrations/maintenance.d.ts +6 -0
  380. package/dist/db/migrations/maintenance.js +61 -0
  381. package/dist/db/migrations/memories.d.ts +7 -0
  382. package/dist/db/migrations/memories.js +16 -0
  383. package/dist/db/migrations/memory-places.d.ts +6 -0
  384. package/dist/db/migrations/memory-places.js +29 -0
  385. package/dist/db/migrations/places.d.ts +6 -0
  386. package/dist/db/migrations/places.js +43 -0
  387. package/dist/db/migrations/projects.d.ts +3 -0
  388. package/dist/db/migrations/projects.js +13 -0
  389. package/dist/db/migrations/tier-conversion.d.ts +7 -0
  390. package/dist/db/migrations/tier-conversion.js +20 -0
  391. package/dist/db/schema/beliefs.d.ts +9 -0
  392. package/dist/db/schema/beliefs.js +46 -0
  393. package/dist/db/schema/generator.d.ts +38 -0
  394. package/dist/db/schema/generator.js +108 -0
  395. package/dist/db/schema/index.d.ts +19 -20
  396. package/dist/db/schema/index.js +25 -79
  397. package/dist/db/schema/learnings.d.ts +7 -0
  398. package/dist/db/schema/learnings.js +30 -0
  399. package/dist/db/schema/memories.d.ts +7 -0
  400. package/dist/db/schema/memories.js +81 -0
  401. package/dist/db/schema/projects.d.ts +4 -0
  402. package/dist/db/schema/projects.js +31 -0
  403. package/dist/packages/mcp/src/index.d.ts +3 -0
  404. package/dist/packages/mcp/src/index.js +733 -0
  405. package/mcp.json.example +8 -11
  406. package/package.json +57 -76
  407. package/packages/cli/package.json +22 -0
  408. package/packages/cli/src/commands/clean.ts +68 -0
  409. package/packages/cli/src/commands/context.ts +79 -0
  410. package/packages/cli/src/commands/doctor.ts +357 -0
  411. package/packages/cli/src/commands/forget.ts +72 -0
  412. package/packages/cli/src/commands/health.ts +36 -0
  413. package/packages/cli/src/commands/inspect.ts +41 -0
  414. package/packages/cli/src/commands/link.ts +50 -0
  415. package/packages/cli/src/commands/migrate.ts +93 -0
  416. package/packages/cli/src/commands/recall.ts +99 -0
  417. package/packages/cli/src/commands/recent.ts +57 -0
  418. package/packages/cli/src/commands/remember.ts +139 -0
  419. package/packages/cli/src/commands/run.ts +58 -0
  420. package/packages/cli/src/commands/stale.ts +43 -0
  421. package/packages/cli/src/commands/stats.ts +42 -0
  422. package/packages/cli/src/index.ts +57 -0
  423. package/packages/cli/tsconfig.json +24 -0
  424. package/packages/mcp/package.json +26 -0
  425. package/packages/mcp/src/index.ts +877 -0
  426. package/packages/mcp/tsconfig.json +20 -0
  427. package/skills/squish-memory/SKILL.md +38 -35
  428. package/skills/squish-memory/{scripts/install.sh → install.sh} +1 -1
  429. package/skills/squish-memory/references/claude-desktop.json +12 -0
  430. package/skills/squish-memory/references/openclaw.json +13 -0
  431. package/skills/squish-memory/references/opencode.json +14 -0
  432. package/config/hooks/claude-code-hooks.json +0 -39
  433. package/config/hooks/cursor-hooks.json +0 -30
  434. package/config/hooks/opencode-hooks.json +0 -30
  435. package/config/hooks/windsurf-hooks.json +0 -30
  436. package/config/mcp-cli-fallback-policy.json +0 -22
  437. package/config/mcp.json +0 -38
  438. package/config/plugin-manifest.json +0 -101
  439. package/config/plugin-manifest.schema.json +0 -244
  440. package/config/plugin.json +0 -32
  441. package/config/remote-memory-policy.json +0 -32
  442. package/core/commands/context-paging.md +0 -51
  443. package/core/commands/context-status.md +0 -22
  444. package/core/commands/context.md +0 -5
  445. package/core/commands/core-memory.md +0 -56
  446. package/core/commands/health.md +0 -5
  447. package/core/commands/init.md +0 -39
  448. package/core/commands/merge.md +0 -113
  449. package/core/commands/recall.md +0 -5
  450. package/core/commands/remember.md +0 -11
  451. package/core/commands/search.md +0 -10
  452. package/dist/core/commands/managed-sync.d.ts +0 -10
  453. package/dist/core/commands/managed-sync.js +0 -64
  454. package/dist/core/external-folder/index.d.ts +0 -102
  455. package/dist/core/external-folder/index.js +0 -294
  456. package/dist/core/namespaces/index.d.ts +0 -71
  457. package/dist/core/namespaces/index.js +0 -305
  458. package/dist/core/namespaces/uri-parser.d.ts +0 -31
  459. package/dist/core/namespaces/uri-parser.js +0 -74
  460. package/dist/core/search/qmd-search.d.ts +0 -61
  461. package/dist/core/search/qmd-search.js +0 -178
  462. package/dist/core/session-hooks/self-iteration-job.d.ts +0 -20
  463. package/dist/core/session-hooks/self-iteration-job.js +0 -282
  464. package/dist/core/session-hooks/session-hooks.d.ts +0 -18
  465. package/dist/core/session-hooks/session-hooks.js +0 -58
  466. package/dist/core/snapshots.d.ts +0 -29
  467. package/dist/core/snapshots.js +0 -220
  468. package/dist/core/sync/qmd-sync.d.ts +0 -94
  469. package/dist/core/sync/qmd-sync.js +0 -201
  470. package/dist/index.d.ts +0 -7
  471. package/dist/index.js +0 -1677
  472. package/dist/vendor/sql.js/sql-wasm.wasm +0 -0
  473. package/dist/webui/server.d.ts +0 -5
  474. package/dist/webui/server.js +0 -642
  475. package/generated/mcp/manifest.json +0 -23
  476. package/generated/mcp/mcp-servers.json +0 -25
  477. package/generated/mcp/mcporter.json +0 -34
  478. package/generated/mcp/openclaw-memory-qmd.json +0 -17
  479. package/generated/mcp/runtime.json +0 -12
  480. package/scripts/README.md +0 -60
  481. package/scripts/build-release.sh +0 -36
  482. package/scripts/check-secrets.js +0 -132
  483. package/scripts/copy-runtime-assets.mjs +0 -26
  484. package/scripts/generate-mcp.mjs +0 -264
  485. package/scripts/github-release.sh +0 -77
  486. package/scripts/init-dirs.mjs +0 -13
  487. package/scripts/install-claude-code.sh +0 -85
  488. package/scripts/install-cursor.sh +0 -56
  489. package/scripts/install-hooks.sh +0 -73
  490. package/scripts/install-interactive.mjs +0 -357
  491. package/scripts/install-opencode.sh +0 -75
  492. package/scripts/install-plugin.mjs +0 -415
  493. package/scripts/install-windsurf.sh +0 -67
  494. package/scripts/remote-preflight.mjs +0 -62
  495. package/scripts/squish-fallback.mjs +0 -132
  496. package/scripts/test-interactive.mjs +0 -131
  497. package/scripts/verify-mcp.mjs +0 -214
  498. package/skills/squish-memory/scripts/install.mjs +0 -335
  499. package/skills/squish-memory/write_skill.js +0 -2
@@ -0,0 +1,565 @@
1
+ import { randomUUID } from 'crypto';
2
+ import { desc, eq } from 'drizzle-orm';
3
+ import { config } from '../../config.js';
4
+ import { logger } from '../../core/logger.js';
5
+ import { getOrCreateProject, requireProject } from '../../core/projects.js';
6
+ import { getEmbedding } from '../../core/embeddings.js';
7
+ import { normalizeTags, serializeTags, deserializeTags, serializeMetadata, deserializeMetadata } from '../../core/memory/serialization.js';
8
+ import { normalizeTimestamp, clampLimit, prepareEmbedding } from '../lib/utils.js';
9
+ import { validateUuid, requireUuid } from '../lib/validation.js';
10
+ import { cosineSimilarity } from '../utils/vector-operations.js';
11
+ import { hybridSearch as hybridSearchImpl } from './hybrid-search.js';
12
+ import { calculateImportance } from './importance.js';
13
+ import { detectMemorySignals, MemorySignals } from './trigger-detector.js';
14
+ import { resolveContradictions, applySupersession } from './contradiction-resolver.js';
15
+ import { encrypt, decrypt } from '../security/encrypt.js';
16
+ import { estimateTokens } from '../context/context-window.js';
17
+ import { getDbClient } from '../lib/db-client.js';
18
+ import { extractBeliefsFromMemory } from '../beliefs/extractor.js';
19
+ import { upsertBeliefsForMemory } from '../beliefs/store.js';
20
+ import { extractEntityNames } from './entity-extractor.js';
21
+ import { autoLinkByEntities } from '../associations.js';
22
+ import { addMemoryToGraph } from '../graph/graph-builder.js';
23
+ import { MemoryRecord, MemoryType } from '../lib/types.js';
24
+ import { parseEmbedding } from '../lib/parse-embedding.js';
25
+
26
+ // MemoryType and MemoryRecord imported from ../lib/types.js
27
+
28
+ export interface RememberInput {
29
+ content: string;
30
+ type?: MemoryType;
31
+ tags?: string[];
32
+ project?: string;
33
+ metadata?: Record<string, unknown>;
34
+ source?: string;
35
+ // Rich context fields (Agent 4 feedback)
36
+ reasoning?: string; // Why it's true/important
37
+ memoryContext?: string; // What triggered this memory
38
+ examples?: string; // When to apply this knowledge
39
+ exceptions?: string; // When NOT to apply
40
+ // Hot/Cold tier (replaces isHighRes)
41
+ tier?: 'hot' | 'cold'; // Memory tier: hot = active, cold = archived (simplified: warm removed)
42
+ // Namespace for grouping
43
+ namespaceId?: string; // Assign to namespace
44
+ // Session metadata for temporal queries (Task 1)
45
+ sessionId?: string; // Session identifier for linking memories
46
+ sessionStartTime?: string; // When this session started
47
+ toolName?: string; // Tool that generated this memory
48
+ }
49
+
50
+ export interface SearchInput {
51
+ query: string;
52
+ type?: MemoryType;
53
+ tags?: string[];
54
+ limit?: number;
55
+ project?: string;
56
+ // Place and session filters for unified search (Task 2, Task 3)
57
+ placeId?: string; // Filter by place
58
+ placeType?: string; // Filter by place type (inbox, wip, archive, etc.)
59
+ sessionId?: string; // Filter by session
60
+ sessionStartTime?: string; // Session start for temporal queries
61
+ }
62
+
63
+ // SearchResult extends the shared MemoryRecord from normalization.ts
64
+ export interface SearchResult extends MemoryRecord {
65
+ similarity: number;
66
+ }
67
+
68
+ export async function rememberMemory(input: RememberInput): Promise<MemoryRecord> {
69
+ const { db, schema } = await getDbClient();
70
+ const tags = normalizeTags(input.tags);
71
+ const project = await getOrCreateProject(input.project);
72
+ const embedding = await getEmbedding(input.content);
73
+ const id = randomUUID();
74
+ const signals = detectMemorySignals(input.content);
75
+ const type = input.type ?? signals.suggestedType;
76
+
77
+ const baseValues = {
78
+ id,
79
+ projectId: project?.id ?? null,
80
+ type,
81
+ content: input.content,
82
+ source: input.source ?? 'mcp',
83
+ };
84
+
85
+ // Calculate initial importance score
86
+ const importance = calculateImportance({
87
+ type,
88
+ createdAt: new Date().toISOString(),
89
+ accessCount: 0,
90
+ usageCount: 0,
91
+ isPinned: false,
92
+ isProtected: false,
93
+ isImmutable: false,
94
+ });
95
+
96
+ const embeddingValues = prepareEmbedding(embedding);
97
+
98
+ const tokensEstimate = estimateTokens(input.content);
99
+
100
+ let tagsValue = serializeTags(tags);
101
+ const enrichedMetadata: Record<string, unknown> = {
102
+ ...(input.metadata ?? {}),
103
+ // Rich context fields (Agent 4 feedback)
104
+ reasoning: input.reasoning,
105
+ memoryContext: input.memoryContext,
106
+ examples: input.examples,
107
+ exceptions: input.exceptions,
108
+ memorySignals: {
109
+ explicitTriggers: signals.explicitTriggers,
110
+ implicit: signals.implicit,
111
+ priority: signals.priority,
112
+ requiresConflictCheck: signals.implicit.correction,
113
+ },
114
+ // Session metadata for temporal queries (Task 1)
115
+ sessionMetadata: {
116
+ sessionId: input.sessionId,
117
+ sessionStartTime: input.sessionStartTime,
118
+ toolName: input.toolName,
119
+ },
120
+ };
121
+ let metadataValue = serializeMetadata(enrichedMetadata);
122
+
123
+ // Prepare fields for insertion, handling optional encryption
124
+ let insertValues: any = {
125
+ ...baseValues,
126
+ tags: tagsValue,
127
+ metadata: metadataValue,
128
+ ...embeddingValues,
129
+ importanceScore: importance.score,
130
+ lastImportanceRecalc: new Date(),
131
+ tokensEstimate,
132
+ createdAt: new Date(),
133
+ status: 'active',
134
+ tier: input.tier || 'hot', // Default to hot tier
135
+ };
136
+
137
+ // Add namespace if specified
138
+ if (input.namespaceId) {
139
+ insertValues.namespaceId = input.namespaceId;
140
+ }
141
+
142
+ // For cold tier, store original content in metadata
143
+ if (input.tier === 'cold') {
144
+ enrichedMetadata.originalContent = input.content;
145
+ }
146
+
147
+ if (config.clientEncryptionEnabled) {
148
+ const { ciphertext, nonce } = encrypt(input.content);
149
+ insertValues.encrypted_content = ciphertext;
150
+ insertValues.encryption_nonce = nonce;
151
+ insertValues.is_encrypted = true;
152
+ // Store empty placeholder for plain content
153
+ insertValues.content = '';
154
+ } else {
155
+ insertValues.content = input.content;
156
+ insertValues.is_encrypted = false;
157
+ }
158
+
159
+ await db.insert(schema.memories).values(insertValues);
160
+
161
+ if (project?.id) {
162
+ try {
163
+ const beliefs = extractBeliefsFromMemory({
164
+ memoryId: id,
165
+ content: input.content,
166
+ type,
167
+ metadata: enrichedMetadata,
168
+ });
169
+ if (beliefs.length > 0) {
170
+ await upsertBeliefsForMemory({
171
+ projectId: project.id,
172
+ memoryId: id,
173
+ beliefs,
174
+ });
175
+ }
176
+ } catch (beliefError) {
177
+ logger.warn(`[Beliefs] Failed to derive beliefs for memory ${id}: ${beliefError}`);
178
+ }
179
+ }
180
+
181
+ // Auto-link by entity overlap (synchronous - Task 5)
182
+ // Now memories are immediately findable via associations after storage
183
+ const entityNames = extractEntityNames(input.content);
184
+ if (entityNames.length > 0 && project?.id) {
185
+ try {
186
+ const linked = await autoLinkByEntities(id, entityNames, project.id);
187
+ if (linked > 0) logger.debug(`[AutoLink] Linked memory ${id} to ${linked} related memories`);
188
+ } catch (err) {
189
+ logger.debug(`[AutoLink] Failed: ${err}`);
190
+ }
191
+ }
192
+
193
+ // Build graph for this memory (auto-build if enabled)
194
+ // This populates the entity_entities and entity_relations tables
195
+ if (config.graphAutoBuild && project?.id) {
196
+ try {
197
+ const graphResult = await addMemoryToGraph(id, {
198
+ preferLLM: config.llmEnabled,
199
+ });
200
+ if (graphResult.entitiesCreated > 0 || graphResult.relationsCreated > 0) {
201
+ logger.debug(`[Graph] Built graph for memory ${id}: ${graphResult.entitiesCreated} entities, ${graphResult.relationsCreated} relations`);
202
+ }
203
+ } catch (graphError) {
204
+ logger.debug(`[Graph] Failed to build graph for memory ${id}: ${graphError}`);
205
+ }
206
+ }
207
+
208
+ // Append to Obsidian vault if enabled and hot tier (NEW)
209
+ if (config.obsidianEnabled && config.obsidianVaultPath && insertValues.tier === 'hot') {
210
+ try {
211
+ const { appendToObsidianVault } = await import('../integrations/obsidian-vault.js');
212
+ await appendToObsidianVault({
213
+ content: input.content,
214
+ id,
215
+ type,
216
+ tags,
217
+ reasoning: input.reasoning,
218
+ memoryContext: input.memoryContext,
219
+ examples: input.examples,
220
+ exceptions: input.exceptions,
221
+ source: input.source,
222
+ }, config.obsidianVaultPath);
223
+ } catch (error) {
224
+ logger.warn(`[Obsidian] Failed to append to vault: ${error}`);
225
+ }
226
+ }
227
+
228
+ // Resolve contradictions and supersede old memories (async, non-blocking)
229
+ resolveContradictions(input.content, type, project?.id)
230
+ .then(async (result) => {
231
+ if (result.supersededIds.length > 0) {
232
+ await applySupersession(id, result.supersededIds, result.confidence);
233
+ // Update metadata with contradiction resolution info
234
+ const updatedMetadata: Record<string, unknown> = {
235
+ ...enrichedMetadata,
236
+ contradictionResolution: {
237
+ supersededCount: result.supersededIds.length,
238
+ confidence: result.confidence,
239
+ reason: result.reason,
240
+ },
241
+ };
242
+ metadataValue = serializeMetadata(updatedMetadata);
243
+ }
244
+ })
245
+ .catch((error) => {
246
+ import('../logger.js').then(({ logger }) => {
247
+ logger?.debug?.(`Contradiction resolution failed: ${error}`);
248
+ });
249
+ });
250
+
251
+ // Sync to QMD if enabled (async, don't block)
252
+ const memoryRecord: MemoryRecord = {
253
+ id,
254
+ projectId: project?.id ?? null,
255
+ type,
256
+ content: input.content,
257
+ tags,
258
+ metadata: enrichedMetadata,
259
+ importance: importance.score as number,
260
+ };
261
+
262
+ return memoryRecord;
263
+ }
264
+
265
+ export async function getMemory(id: string, incrementAccess: boolean = true): Promise<MemoryRecord | null> {
266
+ try {
267
+ // Validate UUID
268
+ requireUuid(id);
269
+
270
+ const { db, schema } = await getDbClient();
271
+ const rows = await db.select().from(schema.memories).where(eq(schema.memories.id, id)).limit(1);
272
+ const row = rows[0];
273
+ if (!row) return null;
274
+
275
+ // Increment access count and update last accessed time
276
+ if (incrementAccess) {
277
+ await db.update(schema.memories)
278
+ .set({
279
+ accessCount: (row.accessCount ?? 0) + 1,
280
+ lastAccessedAt: new Date(),
281
+ })
282
+ .where(eq(schema.memories.id, id));
283
+ }
284
+
285
+ let content = row.content;
286
+ if (row.is_encrypted) {
287
+ try {
288
+ content = decrypt(row.encrypted_content, row.encryption_nonce);
289
+ } catch (e) {
290
+ console.warn('Failed to decrypt memory', e);
291
+ content = row.content; // fall back to stored content
292
+ }
293
+ }
294
+ const decryptedRow = { ...row, content };
295
+ return normalizeMemory(decryptedRow);
296
+ } catch (error: any) {
297
+ throw error;
298
+ }
299
+ }
300
+
301
+ export async function setConfidence(id: string, level: 'certain' | 'speculative' | 'outdated'): Promise<boolean> {
302
+ try {
303
+ // Validate UUID
304
+ requireUuid(id);
305
+
306
+ const { db, schema } = await getDbClient();
307
+ await db.update(schema.memories)
308
+ .set({ confidenceLevel: level, updatedAt: new Date() })
309
+ .where(eq(schema.memories.id, id));
310
+ return true;
311
+ } catch (error: any) {
312
+ throw error;
313
+ }
314
+ }
315
+
316
+ export async function getRecent(projectPath: string, limit: number): Promise<MemoryRecord[]> {
317
+ try {
318
+ const { db } = await getDbClient();
319
+ const sqlite = db.$client as any;
320
+ const project = await requireProject(projectPath);
321
+
322
+ // Use raw SQL to avoid drizzle column name issues
323
+ const rows = sqlite.prepare(`
324
+ SELECT * FROM memories
325
+ WHERE project_id = ?
326
+ ORDER BY created_at DESC
327
+ LIMIT ?
328
+ `).all(project.id, limit);
329
+
330
+ return rows.map((row: any) => normalizeMemory(row));
331
+ } catch (error: any) {
332
+ throw error;
333
+ }
334
+ }
335
+
336
+ export async function search(input: SearchInput): Promise<SearchResult[]> {
337
+ const limit = clampLimit(input.limit, 10, 1, 100);
338
+ const tags = normalizeTags(input.tags);
339
+
340
+ // Always use hybrid search for both SQLite and PostgreSQL
341
+ // Graph boost requires hybrid path for proper scoring
342
+ const dbResults = await hybridSearchImpl(input, { limit });
343
+
344
+ return dbResults.slice(0, limit);
345
+ }
346
+
347
+ // parseEmbedding imported from ../lib/parse-embedding.js
348
+
349
+ async function searchMemoriesSqlite(input: SearchInput, tags: string[], limit: number): Promise<SearchResult[]> {
350
+ const { db } = await getDbClient();
351
+ const sqlite = db.$client as any;
352
+
353
+ // Get embedding for the query (for semantic search)
354
+ const queryEmbedding = await getEmbedding(input.query);
355
+
356
+ // Build the base query
357
+ const conditions: string[] = [];
358
+ const params: any[] = [];
359
+
360
+ if (input.type) {
361
+ conditions.push('m.type = ?');
362
+ params.push(input.type);
363
+ }
364
+
365
+ if (tags.length) {
366
+ conditions.push('m.tags IS NOT NULL AND (' + tags.map(() => 'm.tags LIKE ?').join(' OR ') + ')');
367
+ params.push(...tags.map((tag) => `%${tag}%`));
368
+ }
369
+
370
+ let projectId: string | null = null;
371
+ if (input.project) {
372
+ const project = await requireProject(input.project);
373
+ projectId = project.id;
374
+ conditions.push('m.project_id = ?');
375
+ params.push(project.id);
376
+ }
377
+
378
+ const whereClause = conditions.length > 0 ? 'WHERE ' + conditions.join(' AND ') : '';
379
+
380
+ // Fetch memories with embeddings for semantic search
381
+ const fetchLimit = Math.max(limit * 3, 50); // Fetch more for re-ranking
382
+
383
+ const statement = sqlite.prepare(`
384
+ SELECT
385
+ m.id as id,
386
+ m.project_id as projectId,
387
+ m.type as type,
388
+ m.content as content,
389
+ m.summary as summary,
390
+ m.tags as tags,
391
+ m.metadata as metadata,
392
+ m.embedding as embedding,
393
+ m.embedding_json as embeddingJson,
394
+ m.created_at as createdAt
395
+ FROM memories m
396
+ ${whereClause}
397
+ ORDER BY m.created_at DESC
398
+ LIMIT ?
399
+ `);
400
+
401
+ const rows = statement.all(...params, fetchLimit) as Array<{
402
+ id: string;
403
+ projectId: string | null;
404
+ type: string;
405
+ content: string;
406
+ summary: string | null;
407
+ tags: string | null;
408
+ metadata: string | null;
409
+ embedding: any;
410
+ embeddingJson: any;
411
+ createdAt: string | null;
412
+ }>;
413
+
414
+ if (rows.length === 0) return [];
415
+
416
+ // If we have query embedding, do semantic ranking
417
+ if (queryEmbedding) {
418
+ const scored = rows.map((row) => {
419
+ let embedding = parseEmbedding(row.embedding);
420
+
421
+ // Fallback to embedding_json if embedding blob is null
422
+ if (!embedding && row.embeddingJson) {
423
+ embedding = parseEmbedding(row.embeddingJson);
424
+ }
425
+
426
+ const score = embedding ? cosineSimilarity(queryEmbedding, embedding) : 0;
427
+ return { row, score };
428
+ });
429
+
430
+ // Sort by similarity score (descending)
431
+ scored.sort((a, b) => b.score - a.score);
432
+
433
+ // Return top results WITH their similarity scores
434
+ return scored.slice(0, limit).map((item): SearchResult => ({
435
+ ...normalizeMemory(item.row),
436
+ similarity: item.score,
437
+ }));
438
+ }
439
+
440
+ // No embeddings available, return results with 0 similarity
441
+ return rows.slice(0, limit).map((row, i): SearchResult => ({
442
+ ...normalizeMemory(row),
443
+ similarity: 0,
444
+ }));
445
+ }
446
+
447
+ async function searchMemoriesPostgres(input: SearchInput, tags: string[], limit: number): Promise<SearchResult[]> {
448
+ const { db } = await getDbClient();
449
+ const values: Array<string | string[] | number[] | null> = [];
450
+ const whereParts: string[] = [];
451
+
452
+ values.push(`%${input.query}%`);
453
+ whereParts.push(`content ILIKE $1`);
454
+
455
+ if (input.type) {
456
+ values.push(input.type);
457
+ whereParts.push(`type = $${values.length}`);
458
+ }
459
+
460
+ if (tags.length) {
461
+ values.push(tags);
462
+ whereParts.push(`tags && $${values.length}::text[]`);
463
+ }
464
+
465
+ if (input.project) {
466
+ const project = await requireProject(input.project);
467
+ values.push(project.id);
468
+ whereParts.push(`project_id = $${values.length}`);
469
+ }
470
+
471
+ const whereClause = whereParts.length ? `WHERE ${whereParts.join(' AND ')}` : '';
472
+ const embedding = await getEmbedding(input.query);
473
+
474
+ if (embedding) {
475
+ const rows = await (db.$client as any).query(
476
+ `SELECT
477
+ id,
478
+ project_id as "projectId",
479
+ type,
480
+ content,
481
+ summary,
482
+ tags,
483
+ metadata,
484
+ created_at as "createdAt",
485
+ valid_from as "validFrom",
486
+ valid_to as "validTo",
487
+ recorded_at as "recordedAt"
488
+ FROM memories
489
+ ${whereClause}
490
+ ORDER BY created_at DESC
491
+ LIMIT $${values.length + 1}`,
492
+ [...values, limit]
493
+ );
494
+ return rows.rows.map((row: any): SearchResult => ({
495
+ ...normalizeMemory(row),
496
+ similarity: row.similarity ?? 0,
497
+ }));
498
+ }
499
+
500
+ const rows = await (db.$client as any).query(
501
+ `SELECT
502
+ id,
503
+ project_id as "projectId",
504
+ type,
505
+ content,
506
+ summary,
507
+ tags,
508
+ metadata,
509
+ created_at as "createdAt",
510
+ valid_from as "validFrom",
511
+ valid_to as "validTo",
512
+ recorded_at as "recordedAt"
513
+ FROM memories
514
+ ${whereClause}
515
+ ORDER BY created_at DESC
516
+ LIMIT $${values.length + 1}`,
517
+ [...values, limit]
518
+ );
519
+
520
+ return rows.rows.map((row: any): SearchResult => ({
521
+ ...normalizeMemory(row),
522
+ similarity: 0,
523
+ }));
524
+ }
525
+
526
+ function normalizeMemory(row: any): MemoryRecord {
527
+ const tags = deserializeTags(row.tags ?? null);
528
+ const metadata = deserializeMetadata(row.metadata ?? null);
529
+
530
+ const createdAtStr = normalizeTimestamp(row.createdAt ?? row.created_at);
531
+
532
+ return {
533
+ id: row.id,
534
+ projectId: row.projectId ?? row.project_id ?? null,
535
+ type: row.type,
536
+ content: row.content,
537
+ summary: row.summary ?? null,
538
+ tags,
539
+ metadata,
540
+ createdAt: createdAtStr,
541
+ validFrom: row.validFrom ?? row.valid_from ?? null,
542
+ validTo: row.validTo ?? row.valid_to ?? null,
543
+ recordedAt: row.recordedAt ?? row.recorded_at ?? null,
544
+ confidenceLevel: row.confidenceLevel ?? row.confidence_level ?? null,
545
+ };
546
+ }
547
+
548
+ /**
549
+ * Find similar memories to prevent duplicates
550
+ * Returns memories with similarity >= threshold
551
+ */
552
+ export async function findSimilarMemories(
553
+ content: string,
554
+ threshold: number = 0.85,
555
+ limit: number = 5
556
+ ): Promise<SearchResult[]> {
557
+ // Use search with high similarity
558
+ const results = await search({
559
+ query: content,
560
+ limit,
561
+ });
562
+
563
+ // Filter by similarity threshold
564
+ return results.filter(r => (r.similarity ?? 0) >= threshold);
565
+ }
@@ -0,0 +1,51 @@
1
+ import { getDb } from '../../db/index.js';
2
+ import { config } from '../../config.js';
3
+ import { memories } from '../../db/drizzle/schema.js';
4
+ import { eq } from 'drizzle-orm';
5
+ import { logger } from '../logger.js';
6
+
7
+ /** Promote a memory to a higher tier (cold -> hot only, simplified from warm hierarchy) */
8
+ export async function promoteTier(memoryId: string) {
9
+ const db = await getDb();
10
+ // @ts-ignore - drizzle overloads
11
+ const row = await db.select().from(memories).where(eq(memories.id, memoryId)).limit(1);
12
+ const mem = row[0];
13
+ if (!mem) return;
14
+ // Simplified: cold -> hot (no warm tier)
15
+ if (mem.tier === 'cold') {
16
+ await db.update(memories).set({ tier: 'hot' }).where(eq(memories.id, memoryId));
17
+ }
18
+ }
19
+
20
+ /** Demote a memory tier or expire it based on decay (simplified: hot -> cold only) */
21
+ export async function demoteTier(memoryId: string) {
22
+ const db = await getDb();
23
+ // @ts-ignore
24
+ const row = await db.select().from(memories).where(eq(memories.id, memoryId)).limit(1);
25
+ const mem = row[0];
26
+ if (!mem) return;
27
+ // Simplified: hot -> cold (no warm tier)
28
+ if (mem.tier === 'hot') {
29
+ await db.update(memories).set({ tier: 'cold' }).where(eq(memories.id, memoryId));
30
+ }
31
+ }
32
+
33
+ export async function expireMemory(memoryId: string) {
34
+ const db = await getDb();
35
+ // @ts-ignore
36
+ await db.update(memories).set({ status: 'expired' }).where(eq(memories.id, memoryId));
37
+ }
38
+
39
+ export async function markMerged(memoryId: string, targetId: string) {
40
+ const db = await getDb();
41
+ // @ts-ignore
42
+ await db.update(memories).set({ is_merged: true, merged_into_id: targetId }).where(eq(memories.id, memoryId));
43
+ }
44
+
45
+ export async function markSuperseded(memoryId: string) {
46
+ const db = await getDb();
47
+ // @ts-ignore
48
+ await db.update(memories).set({ is_merged: true }).where(eq(memories.id, memoryId));
49
+ }
50
+
51
+ logger.info('Memory lifecycle helpers loaded');
@@ -0,0 +1,53 @@
1
+ import { config } from '../../config.js';
2
+ import { logger } from '../../core/logger.js';
3
+
4
+ // Memory monitoring and optimization
5
+ class MemoryManager {
6
+ private lastGcTime = 0;
7
+ private gcIntervalMs = 5 * 60 * 1000; // 5 minutes
8
+
9
+ shouldGc(): boolean {
10
+ const now = Date.now();
11
+ if (now - this.lastGcTime > this.gcIntervalMs) {
12
+ const usage = process.memoryUsage();
13
+ const heapUsedMB = usage.heapUsed / 1024 / 1024;
14
+ const heapTotalMB = usage.heapTotal / 1024 / 1024;
15
+
16
+ // Trigger GC if heap usage is high (>80% of total)
17
+ if (heapUsedMB > heapTotalMB * 0.8) {
18
+ this.lastGcTime = now;
19
+ return true;
20
+ }
21
+ }
22
+ return false;
23
+ }
24
+
25
+ gc(): void {
26
+ if (global.gc && this.shouldGc()) {
27
+ if (process.env.NODE_ENV !== 'production') {
28
+ logger.info('Running garbage collection');
29
+ }
30
+ global.gc();
31
+ }
32
+ }
33
+
34
+ getStats() {
35
+ const usage = process.memoryUsage();
36
+ return {
37
+ heapUsed: Math.round(usage.heapUsed / 1024 / 1024),
38
+ heapTotal: Math.round(usage.heapTotal / 1024 / 1024),
39
+ external: Math.round(usage.external / 1024 / 1024),
40
+ rss: Math.round(usage.rss / 1024 / 1024),
41
+ };
42
+ }
43
+ }
44
+
45
+ export const memoryManager = new MemoryManager();
46
+
47
+ // Periodic memory monitoring (production-safe)
48
+ if (process.env.NODE_ENV !== 'production') {
49
+ setInterval(() => {
50
+ const stats = memoryManager.getStats();
51
+ logger.info(`Memory: ${stats.heapUsed}MB used, ${stats.heapTotal}MB total`);
52
+ }, 10 * 60 * 1000); // Every 10 minutes
53
+ }