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,616 @@
1
+ import { config } from '../config.js';
2
+ import { getGoogleMultimodalEmbedding, isMultimodalInput, MultimodalInput } from './google-multimodal.js';
3
+ import { logger } from '../logger.js';
4
+
5
+ // Lazy-import transformers to avoid loading unless requested
6
+ let transformersLocal: typeof import('./transformers-local.js') | null = null;
7
+ async function getTransformersLocal() {
8
+ if (!transformersLocal) {
9
+ transformersLocal = import('./transformers-local.js');
10
+ }
11
+ return transformersLocal;
12
+ }
13
+
14
+ export type EmbeddingProvider = 'local' | 'openai' | 'ollama' | 'lmstudio' | 'transformers' | 'google' | 'none' | 'auto';
15
+
16
+ function missingModelError(provider: string, envVar: string): Error {
17
+ return new Error(`Embedding provider "${provider}" requires ${envVar} to be set`);
18
+ }
19
+
20
+ function requireModel(provider: string, envVar: string, model: string): string {
21
+ if (!model.trim()) {
22
+ throw missingModelError(provider, envVar);
23
+ }
24
+ return model;
25
+ }
26
+
27
+ // Retry utility with exponential backoff
28
+ async function withRetry<T>(
29
+ fn: () => Promise<T>,
30
+ maxRetries: number = config.embeddingsMaxRetries,
31
+ baseDelayMs: number = config.embeddingsRetryDelayMs
32
+ ): Promise<T> {
33
+ let lastError: Error | undefined;
34
+
35
+ for (let attempt = 0; attempt < maxRetries; attempt++) {
36
+ try {
37
+ return await fn();
38
+ } catch (error) {
39
+ lastError = error as Error;
40
+
41
+ // Only retry on network errors (5xx, ECONNRESET, ETIMEDOUT, etc.)
42
+ if (error instanceof Error && shouldRetryError(error)) {
43
+ const delay = baseDelayMs * Math.pow(2, attempt) + Math.random() * baseDelayMs;
44
+ logger.debug(`Embedding request failed (attempt ${attempt + 1}/${maxRetries}), retrying in ${delay.toFixed(0)}ms`, { error: error as Error });
45
+ await new Promise(resolve => setTimeout(resolve, delay));
46
+ continue;
47
+ }
48
+
49
+ // Don't retry on 4xx errors or non-retryable errors
50
+ break;
51
+ }
52
+ }
53
+
54
+ throw lastError;
55
+ }
56
+
57
+ function shouldRetryError(error: Error): boolean {
58
+ const message = error.message.toLowerCase();
59
+
60
+ // Network errors that are typically transient
61
+ const retryablePatterns = [
62
+ 'econnreset',
63
+ 'etimedout',
64
+ 'econnrefused',
65
+ 'esocket',
66
+ 'network error',
67
+ 'fetch failed',
68
+ 'timeout',
69
+ 'request timed out',
70
+ 'service unavailable',
71
+ 'too many requests',
72
+ 'rate limit',
73
+ 'internal server error',
74
+ 'bad gateway',
75
+ 'gateway timeout',
76
+ ];
77
+
78
+ return retryablePatterns.some(pattern => message.includes(pattern));
79
+ }
80
+
81
+ // Timeout wrapper using AbortController
82
+ async function withTimeout<T>(
83
+ promise: Promise<T>,
84
+ timeoutMs: number
85
+ ): Promise<T> {
86
+ const controller = new AbortController();
87
+ const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
88
+
89
+ try {
90
+ return await promise;
91
+ } finally {
92
+ clearTimeout(timeoutId);
93
+ }
94
+ }
95
+
96
+ // Fetch wrapper that combines retry and timeout
97
+ async function fetchWithRetryAndTimeout(
98
+ url: string,
99
+ options: RequestInit,
100
+ timeoutMs: number = config.embeddingsTimeoutMs
101
+ ): Promise<Response> {
102
+ return withRetry(async () => {
103
+ return withTimeout(fetch(url, options), timeoutMs);
104
+ });
105
+ }
106
+
107
+ // Simple in-memory cache for embeddings (LRU with 1000 entries)
108
+ const embeddingCache = new Map<string, number[]>();
109
+ const MAX_CACHE_SIZE = 1000;
110
+
111
+ function getCacheKey(input: string, provider: string): string {
112
+ // Simple hash of input + provider
113
+ let hash = 0;
114
+ const str = input + provider;
115
+ for (let i = 0; i < str.length; i++) {
116
+ const char = str.charCodeAt(i);
117
+ hash = ((hash << 5) - hash) + char;
118
+ hash = hash & hash; // Convert to 32bit integer
119
+ }
120
+ return hash.toString();
121
+ }
122
+
123
+ function getCachedEmbedding(key: string): number[] | undefined {
124
+ return embeddingCache.get(key);
125
+ }
126
+
127
+ function setCachedEmbedding(key: string, embedding: number[]): void {
128
+ if (embeddingCache.size >= MAX_CACHE_SIZE) {
129
+ // Remove oldest entry (first one)
130
+ const firstKey = embeddingCache.keys().next().value;
131
+ if (firstKey) {
132
+ embeddingCache.delete(firstKey);
133
+ }
134
+ }
135
+ embeddingCache.set(key, embedding);
136
+ }
137
+
138
+ export async function getEmbedding(input: string | MultimodalInput): Promise<number[] | null> {
139
+ if (!input || (typeof input !== 'string' && !isMultimodalInput(input))) {
140
+ return null;
141
+ }
142
+
143
+ const provider = config.embeddingsProvider;
144
+ const cacheKey = typeof input === 'string'
145
+ ? getCacheKey(input, provider)
146
+ : getCacheKey(JSON.stringify(input), provider);
147
+
148
+ // Check cache first
149
+ const cached = getCachedEmbedding(cacheKey);
150
+ if (cached) {
151
+ return cached;
152
+ }
153
+
154
+ let result: number[] | null = null;
155
+
156
+ // Handle multimodal input
157
+ if (isMultimodalInput(input) && provider === 'google') {
158
+ requireModel('google', 'SQUISH_GOOGLE_EMBEDDING_MODEL', config.googleEmbeddingModel);
159
+ const multimodalResult = await getGoogleMultimodalEmbedding(input);
160
+ if (multimodalResult) {
161
+ result = multimodalResult.embedding;
162
+ }
163
+ }
164
+
165
+ // Handle text-only input
166
+ if (!result && typeof input === 'string') {
167
+ const textInput = input;
168
+
169
+ if (provider === 'none') {
170
+ result = null;
171
+ } else if (provider === 'google') {
172
+ requireModel('google', 'SQUISH_GOOGLE_EMBEDDING_MODEL', config.googleEmbeddingModel);
173
+ const multimodalResult = await getGoogleMultimodalEmbedding({ text: textInput });
174
+ result = multimodalResult?.embedding || null;
175
+ } else if (provider === 'openai') {
176
+ requireModel('openai', 'SQUISH_OPENAI_EMBEDDING_MODEL', config.openAiEmbeddingModel);
177
+ result = await getOpenAiEmbedding(textInput);
178
+ } else if (provider === 'ollama') {
179
+ requireModel('ollama', 'SQUISH_OLLAMA_EMBEDDING_MODEL', config.ollamaEmbeddingModel);
180
+ result = await getOllamaEmbedding(textInput);
181
+ } else if (provider === 'lmstudio') {
182
+ requireModel('lmstudio', 'SQUISH_LM_STUDIO_EMBEDDING_MODEL', config.lmStudioEmbeddingModel);
183
+ result = await getLmStudioEmbedding(textInput);
184
+ } else if (provider === 'transformers') {
185
+ requireModel('transformers', 'SQUISH_LOCAL_MODEL', config.transformersLocalModel);
186
+ try {
187
+ const mod = await getTransformersLocal();
188
+ result = await mod.getEmbedding(textInput);
189
+ } catch (error) {
190
+ logger.debug(`Transformers not available, falling back to TF-IDF: ${error}`);
191
+ }
192
+ // If transformers failed, use TF-IDF
193
+ if (!result) {
194
+ result = getLocalEmbedding(textInput);
195
+ }
196
+ } else if (provider === 'local') {
197
+ result = getLocalEmbedding(textInput);
198
+ } else {
199
+ // Auto mode: cloud -> transformers -> TF-IDF (smart fallback)
200
+ // Step 1: Try cloud providers
201
+ if (config.openAiApiKey && config.openAiEmbeddingModel) {
202
+ result = await getOpenAiEmbedding(textInput);
203
+ }
204
+ // Step 2: Try Ollama
205
+ if (!result && config.ollamaUrl && config.ollamaEmbeddingModel) {
206
+ result = await getOllamaEmbedding(textInput);
207
+ }
208
+ // Step 3: Try LM Studio
209
+ if (!result && config.lmStudioUrl && config.lmStudioEmbeddingModel) {
210
+ result = await getLmStudioEmbedding(textInput);
211
+ }
212
+ // Step 4: Try Transformers.js local
213
+ if (!result && config.transformersLocalModel) {
214
+ try {
215
+ const mod = await getTransformersLocal();
216
+ result = await mod.getEmbedding(textInput);
217
+ } catch {
218
+ // Transformers not available, continue to fallback
219
+ }
220
+ }
221
+ // Step 5: Fall back to TF-IDF (always works)
222
+ if (!result) {
223
+ result = getLocalEmbedding(textInput);
224
+ }
225
+ }
226
+ }
227
+
228
+ // Cache the result if valid
229
+ if (result) {
230
+ setCachedEmbedding(cacheKey, result);
231
+ }
232
+
233
+ return result;
234
+ }
235
+
236
+ /**
237
+ * Get embeddings for multiple inputs in parallel batches
238
+ * Processes inputs in batches to respect rate limits while parallelizing
239
+ */
240
+ export async function getBatchEmbeddings(
241
+ inputs: string[],
242
+ batchSize: number = 20
243
+ ): Promise<Array<number[] | null>> {
244
+ if (inputs.length === 0) return [];
245
+
246
+ const results: Array<number[] | null> = new Array(inputs.length).fill(null);
247
+ const provider = config.embeddingsProvider;
248
+
249
+ // Check cache for all inputs first
250
+ const uncachedIndices: number[] = [];
251
+ const uncachedInputs: string[] = [];
252
+
253
+ for (let i = 0; i < inputs.length; i++) {
254
+ const cacheKey = getCacheKey(inputs[i], provider);
255
+ const cached = getCachedEmbedding(cacheKey);
256
+ if (cached) {
257
+ results[i] = cached;
258
+ } else {
259
+ uncachedIndices.push(i);
260
+ uncachedInputs.push(inputs[i]);
261
+ }
262
+ }
263
+
264
+ // Process only uncached inputs in batches
265
+ for (let i = 0; i < uncachedInputs.length; i += batchSize) {
266
+ const batchEnd = Math.min(i + batchSize, uncachedInputs.length);
267
+ const batch = uncachedInputs.slice(i, batchEnd);
268
+ const indices = uncachedIndices.slice(i, batchEnd);
269
+
270
+ // Parallelize embeddings within batch using Promise.all
271
+ const batchResults = await Promise.all(
272
+ batch.map((input) => getEmbedding(input))
273
+ );
274
+
275
+ // Store results in correct positions and cache
276
+ for (let j = 0; j < batchResults.length; j++) {
277
+ results[indices[j]] = batchResults[j];
278
+ }
279
+ }
280
+
281
+ return results;
282
+ }
283
+
284
+ /**
285
+ * Clear the embedding cache
286
+ */
287
+ export function clearEmbeddingCache(): void {
288
+ embeddingCache.clear();
289
+ }
290
+
291
+ /**
292
+ * Get embedding cache statistics
293
+ */
294
+ export function getEmbeddingCacheStats(): { size: number; maxSize: number } {
295
+ return {
296
+ size: embeddingCache.size,
297
+ maxSize: MAX_CACHE_SIZE,
298
+ };
299
+ }
300
+
301
+ /**
302
+ * Local TF-IDF embedding using character n-grams and word hashing
303
+ * Creates a 768-dimensional vector for fast offline similarity.
304
+ * Fast, no API calls, works offline
305
+ */
306
+ function getLocalEmbedding(input: string): number[] | null {
307
+ if (!input || typeof input !== 'string') {
308
+ return null;
309
+ }
310
+
311
+ // Normalize text
312
+ const text = input.toLowerCase()
313
+ .replace(/[^a-z0-9\s]/g, ' ')
314
+ .replace(/\s+/g, ' ')
315
+ .trim();
316
+
317
+ if (text.length === 0) {
318
+ return null;
319
+ }
320
+
321
+ // Embedding dimensions
322
+ const dimensions = 768;
323
+ const vector: number[] = new Array(dimensions).fill(0);
324
+
325
+ // Character n-grams (3-5 grams for semantic similarity)
326
+ const ngrams = [3, 4, 5];
327
+ for (const n of ngrams) {
328
+ for (let i = 0; i <= text.length - n; i++) {
329
+ const gram = text.substring(i, i + n);
330
+ const hash = djb2Hash(gram);
331
+ const idx = Math.abs(hash) % dimensions;
332
+ vector[idx] += 1;
333
+ }
334
+ }
335
+
336
+ // Word-level hashing for semantic capture
337
+ const words = text.split(/\s+/).filter(w => w.length > 2);
338
+ for (const word of words) {
339
+ const hash = djb2Hash(word);
340
+ const idx = Math.abs(hash) % dimensions;
341
+ vector[idx] += 2; // Words weighted higher than n-grams
342
+
343
+ // Bigrams
344
+ if (words.length > 1) {
345
+ const idx2 = words.indexOf(word);
346
+ if (idx2 < words.length - 1) {
347
+ const bigram = `${word}_${words[idx2 + 1]}`;
348
+ const bigramHash = djb2Hash(bigram);
349
+ const bigramIdx = Math.abs(bigramHash) % dimensions;
350
+ vector[bigramIdx] += 3; // Bigrams weighted highest
351
+ }
352
+ }
353
+ }
354
+
355
+ // Apply TF-IDF-like scaling: square root to dampen high frequencies
356
+ for (let i = 0; i < dimensions; i++) {
357
+ if (vector[i] > 0) {
358
+ vector[i] = Math.sqrt(vector[i]);
359
+ }
360
+ }
361
+
362
+ // L2 normalize
363
+ const norm = Math.sqrt(vector.reduce((sum, val) => sum + val * val, 0));
364
+ if (norm > 0) {
365
+ for (let i = 0; i < dimensions; i++) {
366
+ vector[i] /= norm;
367
+ }
368
+ }
369
+
370
+ return vector;
371
+ }
372
+
373
+ /**
374
+ * DJB2 hash function - fast, good distribution
375
+ */
376
+ function djb2Hash(str: string): number {
377
+ let hash = 5381;
378
+ for (let i = 0; i < str.length; i++) {
379
+ hash = ((hash << 5) + hash) + str.charCodeAt(i);
380
+ }
381
+ return hash;
382
+ }
383
+
384
+ async function getOpenAiEmbedding(input: string): Promise<number[] | null> {
385
+ if (!config.openAiApiKey) return null;
386
+ if (!config.openAiEmbeddingModel) return null;
387
+
388
+ try {
389
+ const response = await fetchWithRetryAndTimeout(config.openAiApiUrl, {
390
+ method: 'POST',
391
+ headers: {
392
+ 'Content-Type': 'application/json',
393
+ Authorization: `Bearer ${config.openAiApiKey}`,
394
+ },
395
+ body: JSON.stringify({
396
+ model: config.openAiEmbeddingModel,
397
+ input,
398
+ }),
399
+ }, config.openAiTimeoutMs);
400
+
401
+ if (!response.ok) {
402
+ const message = await response.text();
403
+ logger.warn(`OpenAI embeddings failed: ${response.status} ${message}`);
404
+ return null; // Return null to allow fallback
405
+ }
406
+
407
+ const payload = await response.json() as { data?: Array<{ embedding: number[] }> };
408
+ const embedding = payload.data?.[0]?.embedding;
409
+ return embedding ?? null;
410
+ } catch (error) {
411
+ logger.warn('OpenAI embeddings error:', { error: error as Error });
412
+ return null; // Return null to allow fallback
413
+ }
414
+ }
415
+
416
+ async function getOllamaEmbedding(input: string): Promise<number[] | null> {
417
+ if (!config.ollamaEmbeddingModel) return null;
418
+
419
+ try {
420
+ const response = await fetchWithRetryAndTimeout(`${config.ollamaUrl}/api/embeddings`, {
421
+ method: 'POST',
422
+ headers: { 'Content-Type': 'application/json' },
423
+ body: JSON.stringify({
424
+ model: config.ollamaEmbeddingModel,
425
+ prompt: input,
426
+ }),
427
+ }, config.ollamaTimeoutMs);
428
+
429
+ if (!response.ok) {
430
+ const message = await response.text();
431
+ logger.warn(`Ollama embeddings failed: ${response.status} ${message}`);
432
+ return null; // Return null to allow fallback
433
+ }
434
+
435
+ const payload = await response.json() as { embedding?: number[] };
436
+ return payload.embedding ?? null;
437
+ } catch (error) {
438
+ logger.warn('Ollama embeddings error:', { error: error as Error });
439
+ return null; // Return null to allow fallback
440
+ }
441
+ }
442
+
443
+ // LM Studio uses OpenAI-compatible API
444
+ async function getLmStudioEmbedding(input: string): Promise<number[] | null> {
445
+ if (!config.lmStudioEmbeddingModel) return null;
446
+
447
+ try {
448
+ const response = await fetchWithRetryAndTimeout(`${config.lmStudioUrl}/v1/embeddings`, {
449
+ method: 'POST',
450
+ headers: { 'Content-Type': 'application/json' },
451
+ body: JSON.stringify({
452
+ model: config.lmStudioEmbeddingModel,
453
+ input: input,
454
+ }),
455
+ }, config.ollamaTimeoutMs); // Reuse Ollama timeout
456
+
457
+ if (!response.ok) {
458
+ const message = await response.text();
459
+ logger.warn(`LM Studio embeddings failed: ${response.status} ${message}`);
460
+ return null; // Return null to allow fallback
461
+ }
462
+
463
+ const payload = await response.json() as { data?: Array<{ embedding: number[] }> };
464
+ return payload.data?.[0]?.embedding ?? null;
465
+ } catch (error) {
466
+ logger.warn('LM Studio embeddings error:', { error: error as Error });
467
+ return null; // Return null to allow fallback
468
+ }
469
+ }
470
+
471
+ /**
472
+ * Check health of all configured embedding providers
473
+ * Returns availability and latency for each provider
474
+ */
475
+ export async function checkEmbeddingProviderHealth(): Promise<Map<string, { available: boolean; latencyMs?: number; error?: string }>> {
476
+ const results = new Map<string, { available: boolean; latencyMs?: number; error?: string }>();
477
+ const providers = ['local', 'openai', 'ollama', 'lmstudio', 'transformers', 'google', 'none', 'auto'] as const;
478
+
479
+ // Test local provider (always available)
480
+ results.set('local', { available: true, latencyMs: 0 });
481
+
482
+ // Test OpenAI if configured
483
+ if (config.openAiApiKey && config.openAiEmbeddingModel) {
484
+ const start = Date.now();
485
+ try {
486
+ const testInput = 'health check';
487
+ const embedding = await withRetry(
488
+ () => withTimeout(getOpenAiEmbedding(testInput), config.openAiTimeoutMs),
489
+ config.embeddingsMaxRetries,
490
+ config.embeddingsRetryDelayMs
491
+ );
492
+ const latency = Date.now() - start;
493
+ results.set('openai', {
494
+ available: embedding !== null && embedding.length > 0,
495
+ latencyMs: latency
496
+ });
497
+ } catch (error) {
498
+ results.set('openai', {
499
+ available: false,
500
+ error: (error as Error).message
501
+ });
502
+ }
503
+ } else {
504
+ results.set('openai', { available: false, error: 'Not configured' });
505
+ }
506
+
507
+ // Test Ollama if configured
508
+ if (config.ollamaUrl && config.ollamaEmbeddingModel) {
509
+ const start = Date.now();
510
+ try {
511
+ const testInput = 'health check';
512
+ const embedding = await withRetry(
513
+ () => withTimeout(getOllamaEmbedding(testInput), config.ollamaTimeoutMs),
514
+ config.embeddingsMaxRetries,
515
+ config.embeddingsRetryDelayMs
516
+ );
517
+ const latency = Date.now() - start;
518
+ results.set('ollama', {
519
+ available: embedding !== null && embedding.length > 0,
520
+ latencyMs: latency
521
+ });
522
+ } catch (error) {
523
+ results.set('ollama', {
524
+ available: false,
525
+ error: (error as Error).message
526
+ });
527
+ }
528
+ } else {
529
+ results.set('ollama', { available: false, error: 'Not configured' });
530
+ }
531
+
532
+ // Test LM Studio if configured
533
+ if (config.lmStudioUrl && config.lmStudioEmbeddingModel) {
534
+ const start = Date.now();
535
+ try {
536
+ const testInput = 'health check';
537
+ const embedding = await withRetry(
538
+ () => withTimeout(getLmStudioEmbedding(testInput), config.ollamaTimeoutMs),
539
+ config.embeddingsMaxRetries,
540
+ config.embeddingsRetryDelayMs
541
+ );
542
+ const latency = Date.now() - start;
543
+ results.set('lmstudio', {
544
+ available: embedding !== null && embedding.length > 0,
545
+ latencyMs: latency
546
+ });
547
+ } catch (error) {
548
+ results.set('lmstudio', {
549
+ available: false,
550
+ error: (error as Error).message
551
+ });
552
+ }
553
+ } else {
554
+ results.set('lmstudio', { available: false, error: 'Not configured' });
555
+ }
556
+
557
+ // Test Transformers.js local if requested
558
+ const transformersHealth = async () => {
559
+ try {
560
+ const mod = await getTransformersLocal();
561
+ const health = await mod.checkHealth();
562
+ return health;
563
+ } catch (error) {
564
+ return { available: false, error: (error as Error).message };
565
+ }
566
+ };
567
+
568
+ // Try to test transformers (library must be installed)
569
+ if (!config.transformersLocalModel) {
570
+ results.set('transformers', {
571
+ available: false,
572
+ error: 'Not configured',
573
+ });
574
+ } else try {
575
+ const start = Date.now();
576
+ const mod = await getTransformersLocal();
577
+ const health = await mod.checkHealth();
578
+ const latency = Date.now() - start;
579
+ results.set('transformers', {
580
+ available: health.available,
581
+ latencyMs: latency,
582
+ error: health.error,
583
+ });
584
+ } catch (error) {
585
+ results.set('transformers', {
586
+ available: false,
587
+ error: (error as Error).message,
588
+ });
589
+ }
590
+
591
+ // Test Google if configured
592
+ if ((config.googleCloudApiKey || config.googleCloudProject) && config.googleEmbeddingModel) {
593
+ const start = Date.now();
594
+ try {
595
+ const result = await withRetry(
596
+ () => withTimeout(getGoogleMultimodalEmbedding({ text: 'health check' }), config.googleTimeoutMs),
597
+ config.embeddingsMaxRetries,
598
+ config.embeddingsRetryDelayMs
599
+ );
600
+ const latency = Date.now() - start;
601
+ results.set('google', {
602
+ available: result !== null && result.embedding.length > 0,
603
+ latencyMs: latency
604
+ });
605
+ } catch (error) {
606
+ results.set('google', {
607
+ available: false,
608
+ error: (error as Error).message
609
+ });
610
+ }
611
+ } else {
612
+ results.set('google', { available: false, error: 'Not configured' });
613
+ }
614
+
615
+ return results;
616
+ }