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,89 @@
1
+ /**
2
+ * OpenCode Adapter Configuration
3
+ *
4
+ * Native OpenCode configuration for integrating Squish as MCP server.
5
+ * OpenCode uses similar MCP configuration to Claude Code.
6
+ */
7
+
8
+ import { AgentAdapter, AgentType, AgentConfig } from '../types.js';
9
+
10
+ const ADAPTER_CONFIG: AgentConfig = {
11
+ agentId: 'opencode',
12
+ name: 'OpenCode',
13
+ type: 'opencode',
14
+ mcp: {
15
+ command: 'node',
16
+ args: ['dist/core/commands/mcp-server.js'],
17
+ env: { NODE_ENV: 'production' }
18
+ },
19
+ hooks: {},
20
+ settings: {
21
+ squish: {
22
+ autoCapture: true,
23
+ captureTools: ['Read', 'Write', 'Edit', 'Bash'],
24
+ contextLimit: 5,
25
+ }
26
+ }
27
+ };
28
+
29
+ /**
30
+ * Register OpenCode adapter
31
+ */
32
+ export function registerOpenCodeAdapter(): void {
33
+ const { registerAdapter } = require('../index.js');
34
+
35
+ const adapter: AgentAdapter = {
36
+ id: 'opencode',
37
+ type: 'opencode',
38
+ name: 'OpenCode',
39
+ version: 'latest',
40
+
41
+ getSessionContext: async (input) => {
42
+ const { getRecent } = await import('../../memory/memories.js');
43
+ const recent = await getRecent(input.project, 5);
44
+ const memories = recent.map((m, i) =>
45
+ `${i + 1}. [${m.type}] ${m.content?.substring(0, 100)}`
46
+ ).join('\n');
47
+
48
+ return { mode: input.mode, project: input.project, memories, count: recent.length };
49
+ },
50
+
51
+ recordObservation: async (input) => {
52
+ const { createLearning } = await import('../../ingestion/learnings.js');
53
+ const memory = await createLearning({
54
+ type: 'insight',
55
+ content: `[${input.toolName}] ${JSON.stringify(input.toolInput).substring(0, 200)}`,
56
+ action: input.toolName,
57
+ project: input.project,
58
+ autoLink: false,
59
+ });
60
+
61
+ return { memoryId: memory.id, category: 'other', content: `Recorded: ${input.toolName}` };
62
+ },
63
+
64
+ getTimeline: async (query, depth, limit) => {
65
+ const { getTimeline } = await require('../timeline.js');
66
+ return getTimeline(query, depth, limit);
67
+ },
68
+
69
+ shouldCaptureTool: (toolName) => {
70
+ return ['Read', 'Write', 'Edit', 'Bash', 'grep', 'Glob', 'Task'].includes(toolName);
71
+ },
72
+
73
+ getNativeConfig: () => ADAPTER_CONFIG,
74
+ };
75
+
76
+ registerAdapter(adapter);
77
+ }
78
+
79
+ /** OpenCode settings.json format */
80
+ export const OPENCODE_SETTINGS = {
81
+ "mcpServers": {
82
+ "squish": {
83
+ "command": "node",
84
+ "args": ["dist/core/commands/mcp-server.js"]
85
+ }
86
+ }
87
+ };
88
+
89
+ export default ADAPTER_CONFIG;
@@ -0,0 +1,90 @@
1
+ /**
2
+ * Windsurf Adapter Configuration
3
+ *
4
+ * Native Windsurf configuration for integrating Squish as MCP server.
5
+ * Windsurf uses .windsurf/config.json format.
6
+ */
7
+
8
+ import { AgentAdapter, AgentType, AgentConfig } from '../types.js';
9
+
10
+ const ADAPTER_CONFIG: AgentConfig = {
11
+ agentId: 'windsurf',
12
+ name: 'Windsurf',
13
+ type: 'windsurf',
14
+ mcp: {
15
+ command: 'node',
16
+ args: ['dist/core/commands/mcp-server.js'],
17
+ env: { NODE_ENV: 'production' }
18
+ },
19
+ hooks: {},
20
+ settings: {
21
+ squish: {
22
+ autoCapture: true,
23
+ captureTools: ['Read', 'Write', 'Edit', 'Bash'],
24
+ contextLimit: 5,
25
+ }
26
+ }
27
+ };
28
+
29
+ /**
30
+ * Register Windsurf adapter
31
+ */
32
+ export function registerWindsurfAdapter(): void {
33
+ const { registerAdapter } = require('../index.js');
34
+
35
+ const adapter: AgentAdapter = {
36
+ id: 'windsurf',
37
+ type: 'windsurf',
38
+ name: 'Windsurf',
39
+ version: 'latest',
40
+
41
+ getSessionContext: async (input) => {
42
+ const { getRecent } = await import('../../memory/memories.js');
43
+ const recent = await getRecent(input.project, 5);
44
+ const memories = recent.map((m, i) =>
45
+ `${i + 1}. [${m.type}] ${m.content?.substring(0, 100)}`
46
+ ).join('\n');
47
+
48
+ return { mode: input.mode, project: input.project, memories, count: recent.length };
49
+ },
50
+
51
+ recordObservation: async (input) => {
52
+ const { createLearning } = await import('../../ingestion/learnings.js');
53
+ const memory = await createLearning({
54
+ type: 'insight',
55
+ content: `[${input.toolName}] ${JSON.stringify(input.toolInput).substring(0, 200)}`,
56
+ action: input.toolName,
57
+ project: input.project,
58
+ autoLink: false,
59
+ });
60
+
61
+ return { memoryId: memory.id, category: 'other', content: `Recorded: ${input.toolName}` };
62
+ },
63
+
64
+ getTimeline: async (query, depth, limit) => {
65
+ const { getTimeline } = await require('../timeline.js');
66
+ return getTimeline(query, depth, limit);
67
+ },
68
+
69
+ shouldCaptureTool: (toolName) => {
70
+ return ['Read', 'Write', 'Edit', 'Bash', 'grep', 'Glob', 'Task'].includes(toolName);
71
+ },
72
+
73
+ getNativeConfig: () => ADAPTER_CONFIG,
74
+ };
75
+
76
+ registerAdapter(adapter);
77
+ }
78
+
79
+ /** Windsurf config.json format */
80
+ export const WINDSURF_CONFIG = {
81
+ "mcpServers": {
82
+ "squish": {
83
+ "command": "node",
84
+ "args": ["dist/core/commands/mcp-server.js"],
85
+ "env": { "NODE_ENV": "production" }
86
+ }
87
+ }
88
+ };
89
+
90
+ export default ADAPTER_CONFIG;
@@ -0,0 +1,102 @@
1
+ /**
2
+ * UAM Adapter Registry
3
+ *
4
+ * Manages agent adapters for Universal Agent Memory.
5
+ * Provides functions to register, retrieve, and list adapters.
6
+ */
7
+
8
+ import { AgentAdapter, AgentType, AgentConfig } from './types.js';
9
+ import { logger } from '../logger.js';
10
+
11
+ /** In-memory adapter registry */
12
+ const adapters = new Map<AgentType, AgentAdapter>();
13
+
14
+ /** Config directory for agent configs */
15
+ let configDir: string | null = null;
16
+
17
+ /**
18
+ * Register an adapter
19
+ */
20
+ export function registerAdapter(adapter: AgentAdapter): void {
21
+ if (adapters.has(adapter.type)) {
22
+ logger.warn(`[Adapters] Overriding existing adapter for type: ${adapter.type}`);
23
+ }
24
+ adapters.set(adapter.type, adapter);
25
+ logger.info(`[Adapters] Registered adapter: ${adapter.name} (${adapter.type})`);
26
+ }
27
+
28
+ /**
29
+ * Get adapter by type
30
+ */
31
+ export function getAdapter(type: AgentType): AgentAdapter | undefined {
32
+ return adapters.get(type);
33
+ }
34
+
35
+ /**
36
+ * List all registered adapters
37
+ */
38
+ export function listAdapters(): AgentAdapter[] {
39
+ return Array.from(adapters.values());
40
+ }
41
+
42
+ /**
43
+ * Load all adapters from config directory
44
+ */
45
+ export async function loadAllAdapters(dir: string): Promise<void> {
46
+ configDir = dir;
47
+ logger.info(`[Adapters] Loading adapters from: ${dir}`);
48
+
49
+ // Dynamic import and register adapters
50
+ try {
51
+ const { registerClaudeCodeAdapter } = await import('./config/claude-code.js');
52
+ const { registerOpenCodeAdapter } = await import('./config/opencode.js');
53
+ const { registerCursorAdapter } = await import('./config/cursor.js');
54
+ const { registerWindsurfAdapter } = await import('./config/windsurf.js');
55
+
56
+ registerClaudeCodeAdapter();
57
+ registerOpenCodeAdapter();
58
+ registerCursorAdapter();
59
+ registerWindsurfAdapter();
60
+
61
+ logger.info(`[Adapters] Loaded ${listAdapters().length} adapters`);
62
+ } catch (error) {
63
+ logger.error(`[Adapters] Failed to load adapters:`, error);
64
+ }
65
+ }
66
+
67
+ /**
68
+ * Get adapter config by type (for native integration)
69
+ */
70
+ export function getAdapterConfig(type: AgentType): AgentConfig | undefined {
71
+ const adapter = adapters.get(type);
72
+ return adapter?.getNativeConfig();
73
+ }
74
+
75
+ /**
76
+ * List all adapter configs (for MCP tool)
77
+ */
78
+ export function listAdapterConfigs(): AgentConfig[] {
79
+ return listAdapters().map(a => a.getNativeConfig());
80
+ }
81
+
82
+ /**
83
+ * Check if an adapter is registered
84
+ */
85
+ export function hasAdapter(type: AgentType): boolean {
86
+ return adapters.has(type);
87
+ }
88
+
89
+ /**
90
+ * Clear all adapters (mainly for testing)
91
+ */
92
+ export function clearAdapters(): void {
93
+ adapters.clear();
94
+ logger.info('[Adapters] Cleared all adapters');
95
+ }
96
+
97
+ // Re-export types
98
+ export * from './types.js';
99
+ export type { AgentAdapter, AgentType, AgentConfig };
100
+ export type { SessionContextInput, SessionContextOutput } from './types.js';
101
+ export type { ToolObservationInput, ToolObservationOutput } from './types.js';
102
+ export type { TimelineDepth } from './types.js';
@@ -0,0 +1,116 @@
1
+ /**
2
+ * Timeline Implementation - 3-Layer Progressive Disclosure
3
+ *
4
+ * Provides timeline functionality for MCP tool with depth parameter.
5
+ * Layer 1 (index): ~50 tokens - memory titles only
6
+ * Layer 2 (timeline): ~200 tokens - titles + timestamps + tags
7
+ * Layer 3 (detail): ~2000 tokens - full content
8
+ */
9
+
10
+ import { search as searchMemories, getMemory } from '../memory/memories.js';
11
+ import { TimelineDepth } from './types.js';
12
+
13
+ /** Token estimates per layer */
14
+ const LAYER_TOKENS = {
15
+ index: 50,
16
+ timeline: 200,
17
+ detail: 2000,
18
+ };
19
+
20
+ /**
21
+ * Get timeline with progressive disclosure
22
+ */
23
+ export async function getTimeline(
24
+ query: string,
25
+ depth: TimelineDepth = 'index',
26
+ limit: number = 10,
27
+ project?: string
28
+ ): Promise<{
29
+ results: unknown[];
30
+ layer: TimelineDepth;
31
+ tokenEstimate: number;
32
+ query: string;
33
+ }> {
34
+ // First get search results
35
+ const memories = await searchMemories({
36
+ query,
37
+ project,
38
+ limit,
39
+ });
40
+
41
+ // Format based on depth
42
+ let results: unknown[];
43
+ let tokenEstimate = 0;
44
+
45
+ switch (depth) {
46
+ case 'index':
47
+ results = memories.map(m => ({
48
+ id: m.id,
49
+ title: `${m.type}: ${m.content?.substring(0, 60)}...`,
50
+ }));
51
+ tokenEstimate = results.length * (LAYER_TOKENS.index / Math.max(results.length, 1));
52
+ break;
53
+
54
+ case 'timeline':
55
+ results = memories.map(m => ({
56
+ id: m.id,
57
+ type: m.type,
58
+ content: m.content?.substring(0, 100),
59
+ tags: m.tags,
60
+ createdAt: m.createdAt,
61
+ }));
62
+ tokenEstimate = results.length * (LAYER_TOKENS.timeline / Math.max(results.length, 1));
63
+ break;
64
+
65
+ case 'detail':
66
+ results = await Promise.all(
67
+ memories.map(async (m) => {
68
+ const full = await getMemory(m.id);
69
+ return full || { id: m.id, error: 'Not found' };
70
+ })
71
+ );
72
+ tokenEstimate = LAYER_TOKENS.detail;
73
+ break;
74
+ }
75
+
76
+ return {
77
+ results,
78
+ layer: depth,
79
+ tokenEstimate,
80
+ query,
81
+ };
82
+ }
83
+
84
+ /**
85
+ * Get memory by ID with optional depth
86
+ */
87
+ export async function getMemoryTimeline(
88
+ memoryId: string,
89
+ depth: TimelineDepth = 'detail'
90
+ ): Promise<unknown> {
91
+ const memory = await getMemory(memoryId);
92
+
93
+ if (!memory) {
94
+ return { error: 'Memory not found', id: memoryId };
95
+ }
96
+
97
+ if (depth === 'index') {
98
+ return {
99
+ id: memory.id,
100
+ title: `${memory.type}: ${memory.content?.substring(0, 60)}...`,
101
+ };
102
+ }
103
+
104
+ if (depth === 'timeline') {
105
+ return {
106
+ id: memory.id,
107
+ type: memory.type,
108
+ content: memory.content?.substring(0, 100),
109
+ tags: memory.tags,
110
+ createdAt: memory.createdAt,
111
+ };
112
+ }
113
+
114
+ // detail
115
+ return memory;
116
+ }
@@ -0,0 +1,166 @@
1
+ /**
2
+ * UAM Adapter Types
3
+ *
4
+ * Universal Agent Memory - Adapter interfaces for different AI coding agents.
5
+ * Each agent (Claude Code, OpenCode, Cursor, Windsurf, etc.) has its own adapter
6
+ * that defines how to integrate with Squish.
7
+ *
8
+ * Key concepts:
9
+ * - Agent adapters use native config formats where possible
10
+ * - Hooks are explicit (calls squish_learn) not hidden
11
+ * - Uses existing MCP tools (squish_context, squish_learn)
12
+ */
13
+
14
+ import { z } from 'zod';
15
+
16
+ /** Agent types supported by UAM */
17
+ export type AgentType = 'claude-code' | 'opencode' | 'cursor' | 'windsurf' | 'codex' | 'generic';
18
+
19
+ /** Hook event types */
20
+ export type HookEvent =
21
+ | 'onSessionStart'
22
+ | 'onSessionEnd'
23
+ | 'onToolCall'
24
+ | 'onPreCompact'
25
+ | 'onPostCompact';
26
+
27
+ /** 3-Layer depth for progressive disclosure */
28
+ export type TimelineDepth = 'index' | 'timeline' | 'detail';
29
+
30
+ /** Memory context for session start */
31
+ export interface SessionContextInput {
32
+ project: string;
33
+ mode: 'startup' | 'resume' | 'compact';
34
+ sessionId?: string;
35
+ }
36
+
37
+ export interface SessionContextOutput {
38
+ mode: string;
39
+ project: string;
40
+ memories: string;
41
+ count: number;
42
+ snapshot?: { id: string; content: string } | null;
43
+ }
44
+
45
+ /** Tool call observation */
46
+ export interface ToolObservationInput {
47
+ toolName: string;
48
+ toolInput: Record<string, unknown>;
49
+ toolResult: unknown;
50
+ project: string;
51
+ sessionId?: string;
52
+ }
53
+
54
+ export interface ToolObservationOutput {
55
+ memoryId: string;
56
+ category: 'reading' | 'modification' | 'commit' | 'testing' | 'command' | 'search' | 'planning' | 'other';
57
+ content: string;
58
+ }
59
+
60
+ /** Agent configuration */
61
+ export interface AgentConfig {
62
+ /** Unique agent identifier */
63
+ agentId: string;
64
+ /** Human-readable name */
65
+ name: string;
66
+ /** Agent type */
67
+ type: AgentType;
68
+ /** MCP server configuration */
69
+ mcp: {
70
+ command: string;
71
+ args: string[];
72
+ env?: Record<string, string>;
73
+ };
74
+ /** Hook patterns (native format for each agent) */
75
+ hooks?: {
76
+ sessionStart?: unknown;
77
+ sessionEnd?: unknown;
78
+ preCompact?: unknown;
79
+ postToolUse?: unknown;
80
+ };
81
+ /** Custom settings */
82
+ settings?: Record<string, unknown>;
83
+ }
84
+
85
+ /** Agent adapter interface */
86
+ export interface AgentAdapter {
87
+ /** Unique adapter ID */
88
+ id: string;
89
+ /** Agent type */
90
+ type: AgentType;
91
+ /** Human-readable name */
92
+ name: string;
93
+ /** Agent version (if known) */
94
+ version?: string;
95
+
96
+ /** Get session context for injection */
97
+ getSessionContext(input: SessionContextInput): Promise<SessionContextOutput>;
98
+
99
+ /** Record a tool observation */
100
+ recordObservation(input: ToolObservationInput): Promise<ToolObservationOutput>;
101
+
102
+ /** Get timeline (3-layer progressive disclosure) */
103
+ getTimeline(query: string, depth: TimelineDepth, limit: number): Promise<unknown[]>;
104
+
105
+ /** Check if this agent should capture this tool */
106
+ shouldCaptureTool(toolName: string): boolean;
107
+
108
+ /** Get raw config for native integration */
109
+ getNativeConfig(): AgentConfig;
110
+ }
111
+
112
+ /** Registry for managing adapters */
113
+ export interface AdapterRegistry {
114
+ /** Register a new adapter */
115
+ register(adapter: AgentAdapter): void;
116
+
117
+ /** Get adapter by type */
118
+ get(type: AgentType): AgentAdapter | undefined;
119
+
120
+ /** List all registered adapters */
121
+ list(): AgentAdapter[];
122
+
123
+ /** Load all adapters from config directory */
124
+ loadAll(configDir: string): Promise<void>;
125
+ }
126
+
127
+ /** Zod schemas for validation */
128
+ export const SessionContextInputSchema = z.object({
129
+ project: z.string(),
130
+ mode: z.enum(['startup', 'resume', 'compact']).default('startup'),
131
+ sessionId: z.string().optional(),
132
+ });
133
+
134
+ export const ToolObservationInputSchema = z.object({
135
+ toolName: z.string(),
136
+ toolInput: z.record(z.string(), z.unknown()),
137
+ toolResult: z.unknown(),
138
+ project: z.string(),
139
+ sessionId: z.string().optional(),
140
+ });
141
+
142
+ export const TimelineInputSchema = z.object({
143
+ query: z.string(),
144
+ depth: z.enum(['index', 'timeline', 'detail']).default('index'),
145
+ limit: z.number().min(1).max(100).default(10),
146
+ });
147
+
148
+ /** Tool category mapping */
149
+ export const TOOL_CATEGORIES: Record<string, ToolObservationOutput['category']> = {
150
+ 'Read': 'reading',
151
+ 'Write': 'modification',
152
+ 'Edit': 'modification',
153
+ 'Bash': 'command',
154
+ 'grep': 'search',
155
+ 'Glob': 'search',
156
+ 'TodoWrite': 'planning',
157
+ 'TodoRead': 'planning',
158
+ 'Task': 'planning',
159
+ };
160
+
161
+ /**
162
+ * Categorize a tool call
163
+ */
164
+ export function categorizeTool(toolName: string): ToolObservationOutput['category'] {
165
+ return TOOL_CATEGORIES[toolName] || 'other';
166
+ }
@@ -0,0 +1,140 @@
1
+ /**
2
+ * Agent Preferences - Accumulate and retrieve agent preferences from learnings
3
+ * Enables agents to learn and evolve over time
4
+ */
5
+
6
+ import { getDb } from '../db/index.js';
7
+ import { eq, and } from 'drizzle-orm';
8
+ import { logger } from './logger.js';
9
+
10
+ /**
11
+ * Extract preference from learning content
12
+ * E.g., "Prefer bun over npm" -> key: "prefer_bun", value: "bun"
13
+ */
14
+ function extractPreference(content: string): { key: string; value: string } | null {
15
+ const patterns = [
16
+ // "Prefer X over Y"
17
+ /prefer\s+(\w+)\s+over\s+(\w+)/i,
18
+ // "Always use X"
19
+ /always\s+use\s+(\w+)/i,
20
+ // "Use X instead of Y"
21
+ /use\s+(\w+)\s+instead\s+of\s+(\w+)/i,
22
+ // "X is better than Y"
23
+ /(\w+)\s+is\s+better\s+than\s+(\w+)/i,
24
+ // "Don't use X"
25
+ /(?:don't|do not|never)\s+use\s+(\w+)/i,
26
+ ];
27
+
28
+ for (const pattern of patterns) {
29
+ const match = content.match(pattern);
30
+ if (match) {
31
+ const key = `prefer_${match[1].toLowerCase()}`;
32
+ const value = match[2]?.toLowerCase() || 'true';
33
+ return { key, value };
34
+ }
35
+ }
36
+
37
+ return null;
38
+ }
39
+
40
+ /**
41
+ * Update agent preference from a learning
42
+ */
43
+ export async function updateAgentPreference(
44
+ projectId: string,
45
+ content: string,
46
+ sourceMemoryId?: string
47
+ ): Promise<void> {
48
+ const preference = extractPreference(content);
49
+ if (!preference) return;
50
+
51
+ try {
52
+ const db = getDb();
53
+
54
+ // Check if preference exists
55
+ const existing = await db.query.agentPreferences?.findFirst(
56
+ and(
57
+ eq(db.schema.agentPreferences.projectId, projectId),
58
+ eq(db.schema.agentPreferences.key, preference.key)
59
+ )
60
+ ).catch(() => null);
61
+
62
+ if (existing) {
63
+ // Update existing preference
64
+ await db.update(db.schema.agentPreferences)
65
+ .set({
66
+ value: preference.value,
67
+ sourceMemoryId: sourceMemoryId,
68
+ usageCount: (existing.usageCount ?? 0) + 1,
69
+ lastUpdated: Math.floor(Date.now() / 1000)
70
+ })
71
+ .where(eq(db.schema.agentPreferences.id, existing.id))
72
+ .catch(() => {
73
+ // Fallback for SQLite which uses different table name
74
+ const sqlite = (db as any)._?.sqlite || (db as any);
75
+ if (sqlite) {
76
+ sqlite.prepare(`
77
+ UPDATE agent_preferences
78
+ SET value = ?, source_memory_id = ?, usage_count = usage_count + 1, last_updated = ?
79
+ WHERE project_id = ? AND key = ?
80
+ `).run(preference.value, sourceMemoryId, Math.floor(Date.now() / 1000), projectId, preference.key);
81
+ }
82
+ });
83
+ logger.info(`[AgentPrefs] Updated preference: ${preference.key} = ${preference.value}`);
84
+ } else {
85
+ // Insert new preference
86
+ const id = `pref_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`;
87
+ await db.insert(db.schema.agentPreferences)
88
+ .values({
89
+ id,
90
+ projectId,
91
+ key: preference.key,
92
+ value: preference.value,
93
+ sourceMemoryId: sourceMemoryId ?? null,
94
+ confidence: 0.5,
95
+ usageCount: 1
96
+ })
97
+ .catch(() => {
98
+ // Fallback for SQLite
99
+ const sqlite = (db as any)._?.sqlite || (db as any);
100
+ if (sqlite) {
101
+ sqlite.prepare(`
102
+ INSERT INTO agent_preferences (id, project_id, key, value, source_memory_id, confidence, usage_count)
103
+ VALUES (?, ?, ?, ?, ?, 0.5, 1)
104
+ `).run(id, projectId, preference.key, preference.value, sourceMemoryId ?? null);
105
+ }
106
+ });
107
+ logger.info(`[AgentPrefs] Created preference: ${preference.key} = ${preference.value}`);
108
+ }
109
+ } catch (error) {
110
+ logger.warn(`[AgentPrefs] Failed to update preference:`, error);
111
+ }
112
+ }
113
+
114
+ /**
115
+ * Get all agent preferences for a project
116
+ */
117
+ export async function getAgentPreferences(projectId: string): Promise<Array<{key: string; value: string}>> {
118
+ try {
119
+ const db = getDb();
120
+ const results = await db.query.agentPreferences?.findMany({
121
+ where: eq(db.schema.agentPreferences.projectId, projectId)
122
+ }).catch(() => []);
123
+
124
+ if (results && results.length > 0) {
125
+ return results.map(p => ({ key: p.key, value: p.value }));
126
+ }
127
+
128
+ // Fallback for SQLite
129
+ const sqlite = (db as any)._?.sqlite || (db as any);
130
+ if (sqlite) {
131
+ const rows = sqlite.prepare('SELECT key, value FROM agent_preferences WHERE project_id = ?').all(projectId) as Array<{key: string; value: string}>;
132
+ return rows.map(r => ({ key: r.key, value: r.value }));
133
+ }
134
+
135
+ return [];
136
+ } catch (error) {
137
+ logger.warn(`[AgentPrefs] Failed to get preferences:`, error);
138
+ return [];
139
+ }
140
+ }