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.
- package/.env.example +32 -16
- package/CHANGELOG.md +147 -0
- package/README.md +120 -78
- package/{scripts → bin}/dependency-manager.mjs +217 -217
- package/{scripts → bin}/detect-clients.mjs +78 -78
- package/bin/install-interactive.mjs +321 -0
- package/bin/squish-mcp.mjs +46 -0
- package/bin/squish.mjs +33 -0
- package/config/mcp-migration-map.json +1 -6
- package/config/mcp-mode-semantics.json +19 -23
- package/config/mcp-remote-auth.json +3 -26
- package/config/mcp-universal.schema.json +5 -35
- package/config/settings.json +107 -52
- package/config.js +5 -0
- package/config.ts +218 -0
- package/core/adapters/config/claude-code.ts +133 -0
- package/core/adapters/config/cursor.ts +90 -0
- package/core/adapters/config/opencode.ts +89 -0
- package/core/adapters/config/windsurf.ts +90 -0
- package/core/adapters/index.ts +102 -0
- package/core/adapters/timeline.ts +116 -0
- package/core/adapters/types.ts +166 -0
- package/core/agent-preferences.ts +140 -0
- package/core/algorithms/analytics/token-estimator.ts +216 -0
- package/core/algorithms/detection/hash-filters.ts +260 -0
- package/core/algorithms/detection/semantic-ranker.ts +194 -0
- package/core/algorithms/detection/two-stage-detector.ts +421 -0
- package/core/algorithms/handlers/approve-merge.ts +215 -0
- package/core/algorithms/handlers/detect-duplicates.ts +192 -0
- package/core/algorithms/handlers/get-stats.ts +132 -0
- package/core/algorithms/handlers/list-proposals.ts +130 -0
- package/core/algorithms/handlers/preview-merge.ts +139 -0
- package/core/algorithms/handlers/reject-merge.ts +93 -0
- package/core/algorithms/handlers/reverse-merge.ts +155 -0
- package/core/algorithms/index.ts +39 -0
- package/core/algorithms/operations/cache-maintenance.ts +182 -0
- package/core/algorithms/safety/safety-checks.ts +256 -0
- package/core/algorithms/strategies/merge-strategies.ts +381 -0
- package/core/algorithms/types.ts +140 -0
- package/core/algorithms/utils/response-builder.ts +61 -0
- package/core/associations.ts +363 -0
- package/core/beliefs/decay.ts +289 -0
- package/core/beliefs/extractor.ts +131 -0
- package/core/beliefs/store.ts +557 -0
- package/core/beliefs/types.ts +38 -0
- package/core/commands/mcp-server.ts +5 -0
- package/core/compression.ts +177 -0
- package/core/config.js +2 -0
- package/core/consolidation.ts +330 -0
- package/core/context/agent-context.ts +388 -0
- package/core/context/context-paging.ts +449 -0
- package/core/context/context-window.ts +234 -0
- package/core/context/context.ts +35 -0
- package/core/embeddings/embeddings.ts +616 -0
- package/core/embeddings/google-multimodal.ts +200 -0
- package/{dist/core/local-embeddings.js → core/embeddings/local-embeddings.ts} +12 -11
- package/core/embeddings/qmd-client.ts +495 -0
- package/core/embeddings/transformers-local.ts +261 -0
- package/core/embeddings.js +4 -0
- package/core/error-handling.ts +206 -0
- package/core/external +219 -0
- package/core/graph/entity-deduplicator.ts +232 -0
- package/core/graph/graph-builder.ts +257 -0
- package/core/graph/graph-traversal.ts +490 -0
- package/core/graph/index.ts +24 -0
- package/core/graph/llm-entity-extractor.ts +402 -0
- package/core/graph/multi-hop-retrieval.ts +317 -0
- package/core/graph/relationship-extractor.ts +465 -0
- package/core/hooks/agent-hooks.ts +653 -0
- package/core/hooks/auto-tagger.ts +149 -0
- package/core/hooks/capture-filter.ts +169 -0
- package/core/hot-cache.ts +388 -0
- package/core/index.ts +10 -0
- package/core/ingestion/agent-memory.ts +167 -0
- package/core/ingestion/core-memory.ts +326 -0
- package/core/ingestion/learnings.ts +260 -0
- package/core/ingestion/signal-engine.ts +266 -0
- package/core/integrations/obsidian-vault.ts +197 -0
- package/core/layers/generator.ts +115 -0
- package/core/lib/db-client.ts +168 -0
- package/core/lib/parse-embedding.ts +59 -0
- package/core/lib/schemas.ts +102 -0
- package/core/lib/types.ts +49 -0
- package/core/lib/utils.ts +151 -0
- package/core/lib/validation.ts +180 -0
- package/core/lifecycle.ts +353 -0
- package/core/logger.ts +59 -0
- package/core/memory/bridge-discovery.ts +395 -0
- package/core/memory/categorizer.ts +390 -0
- package/core/memory/conflict-detector.ts +62 -0
- package/core/memory/consolidation.ts +372 -0
- package/core/memory/context-collector.ts +75 -0
- package/core/memory/contradiction-resolver.ts +494 -0
- package/core/memory/edit-workflow.ts +174 -0
- package/core/memory/entity-extractor.ts +426 -0
- package/core/memory/entity-resolver.ts +89 -0
- package/core/memory/explain.ts +112 -0
- package/core/memory/fact-deriver.ts +300 -0
- package/core/memory/fact-extractor.ts +120 -0
- package/core/memory/feedback-tracker.ts +200 -0
- package/core/memory/hooks.ts +230 -0
- package/core/memory/hybrid-retrieval.ts +65 -0
- package/core/memory/hybrid-scorer.ts +325 -0
- package/core/memory/hybrid-search.ts +748 -0
- package/core/memory/importance.ts +319 -0
- package/core/memory/index.ts +11 -0
- package/core/memory/loader.ts +178 -0
- package/core/memory/markdown/markdown-storage.ts +318 -0
- package/core/memory/memories.ts +565 -0
- package/core/memory/memory-lifecycle.ts +51 -0
- package/core/memory/memory-manager.ts +53 -0
- package/core/memory/migrate.ts +173 -0
- package/core/memory/normalization.ts +30 -0
- package/core/memory/path-strengthener.ts +211 -0
- package/core/memory/progressive-disclosure.ts +392 -0
- package/core/memory/query-processor.ts +130 -0
- package/core/memory/query-rewriter.ts +153 -0
- package/core/memory/response-analyzer.ts +81 -0
- package/core/memory/retrieval-feedback.ts +276 -0
- package/core/memory/serialization.ts +83 -0
- package/core/memory/stale-cleaner.ts +147 -0
- package/core/memory/stats.ts +181 -0
- package/core/memory/telemetry.ts +392 -0
- package/core/memory/temporal-facts.ts +356 -0
- package/core/memory/temporal-parser.ts +477 -0
- package/core/memory/trigger-detector.ts +104 -0
- package/core/memory/write-gate.ts +288 -0
- package/core/places/index.ts +14 -0
- package/core/places/memory-places.ts +339 -0
- package/core/places/places.ts +406 -0
- package/core/places/rules.ts +308 -0
- package/core/places/walking.ts +192 -0
- package/core/projects +89 -0
- package/core/projects.ts +131 -0
- package/core/redis.ts +82 -0
- package/core/responses.ts +187 -0
- package/core/runtime/trust-report.ts +195 -0
- package/core/runtime/trust-state.ts +360 -0
- package/core/scheduler/cron-scheduler.ts +581 -0
- package/core/scheduler/heartbeat.ts +91 -0
- package/core/scheduler/index.ts +8 -0
- package/core/scheduler/job-runner.ts +197 -0
- package/core/search/conversations.ts +166 -0
- package/core/search/entities.ts +46 -0
- package/core/search/folder-context.ts +154 -0
- package/core/search/graph-boost.ts +22 -0
- package/core/search/index.ts +4 -0
- package/core/search/qmd-wrapper.ts +84 -0
- package/core/security/encrypt.ts +51 -0
- package/core/security/governance.ts +102 -0
- package/core/security/privacy.ts +108 -0
- package/core/security/secret-detector.ts +122 -0
- package/core/session/auto-load.ts +160 -0
- package/core/session/entity-tracker.ts +363 -0
- package/core/session/index.ts +7 -0
- package/core/session/reference-resolver.ts +158 -0
- package/core/session/self-iteration-job.ts +478 -0
- package/core/session/session-hooks.ts +69 -0
- package/core/session/types.ts +36 -0
- package/core/session/working-set.ts +275 -0
- package/core/snapshots/cleanup.ts +13 -0
- package/core/snapshots/comparison.ts +59 -0
- package/core/snapshots/creation.ts +139 -0
- package/core/snapshots/retrieval.ts +44 -0
- package/core/snapshots/stats.ts +63 -0
- package/core/storage/cache.ts +241 -0
- package/core/storage/database.ts +23 -0
- package/core/summarization/cleanup.ts +13 -0
- package/core/summarization/queries.ts +32 -0
- package/core/summarization/stats.ts +64 -0
- package/core/summarization/strategies.ts +52 -0
- package/core/summarization.ts +248 -0
- package/core/temporal-facts.ts +244 -0
- package/core/tracing/collector.ts +470 -0
- package/core/tracing/visualizer.ts +195 -0
- package/core/utils/cleanup-operations.ts +50 -0
- package/core/utils/content-extraction.ts +95 -0
- package/core/utils/filter-builder.ts +56 -0
- package/core/utils/history-traversal.ts +63 -0
- package/core/utils/memory-operations.ts +56 -0
- package/core/utils/query-operations.ts +83 -0
- package/core/utils/summarization-helpers.ts +45 -0
- package/core/utils/temporal-queries.ts +39 -0
- package/core/utils/vector-operations.ts +135 -0
- package/core/utils/version-management.ts +74 -0
- package/core/worker.ts +324 -0
- package/db/adapter.ts +215 -0
- package/db/bootstrap.ts +1055 -0
- package/db/drizzle/migrations/0000_needy_cerebro.sql +402 -0
- package/db/drizzle/migrations/meta/0000_snapshot.json +3451 -0
- package/db/drizzle/migrations/meta/_journal.json +13 -0
- package/db/drizzle/schema-sqlite.ts +1032 -0
- package/db/drizzle/schema.ts +1128 -0
- package/db/drizzle.config.ts +12 -0
- package/db/index.ts +83 -0
- package/db/init.sql +5 -0
- package/db/migrations/associations.ts +35 -0
- package/db/migrations/beliefs.ts +89 -0
- package/db/migrations/core-memory.ts +35 -0
- package/db/migrations/fts.ts +59 -0
- package/db/migrations/index.ts +54 -0
- package/db/migrations/indexes.ts +36 -0
- package/db/migrations/learnings.ts +34 -0
- package/db/migrations/maintenance.ts +68 -0
- package/db/migrations/memories.ts +22 -0
- package/db/migrations/memory-places.ts +35 -0
- package/db/migrations/places.ts +49 -0
- package/db/migrations/projects.ts +21 -0
- package/db/migrations/tier-conversion.ts +24 -0
- package/db/neon.ts +22 -0
- package/db/schema/beliefs.ts +50 -0
- package/db/schema/generator.ts +159 -0
- package/db/schema/index.ts +58 -0
- package/db/schema/learnings.ts +32 -0
- package/db/schema/memories.ts +83 -0
- package/db/schema/projects.ts +33 -0
- package/db/schema.ts +13 -0
- package/db/supabase.ts +27 -0
- package/dist/config.d.ts +40 -17
- package/dist/config.js +150 -198
- package/dist/core/adapters/types.d.ts +13 -33
- package/dist/core/adapters/types.js +1 -1
- package/dist/core/agent-preferences.d.ts +16 -0
- package/dist/core/agent-preferences.js +124 -0
- package/dist/core/algorithms/safety/safety-checks.d.ts +1 -5
- package/dist/core/algorithms/types.d.ts +0 -8
- package/dist/core/associations.d.ts +3 -1
- package/dist/core/associations.js +37 -1
- package/dist/core/beliefs/decay.d.ts +27 -0
- package/dist/core/beliefs/decay.js +217 -0
- package/dist/core/beliefs/extractor.d.ts +9 -0
- package/dist/core/beliefs/extractor.js +113 -0
- package/dist/core/beliefs/store.d.ts +46 -0
- package/dist/core/beliefs/store.js +466 -0
- package/dist/core/beliefs/types.d.ts +28 -0
- package/dist/core/beliefs/types.js +2 -0
- package/dist/core/commands/mcp-server.d.ts +0 -1
- package/dist/core/commands/mcp-server.js +4 -737
- package/dist/core/commands/remember.d.ts +24 -0
- package/dist/core/commands/remember.js +144 -0
- package/dist/core/{toon.d.ts → compression.d.ts} +6 -4
- package/dist/core/{toon.js → compression.js} +8 -8
- package/dist/core/context/agent-context.js +1 -1
- package/dist/core/embeddings/embeddings.d.ts +29 -0
- package/dist/core/embeddings/embeddings.js +546 -0
- package/dist/core/embeddings/google-multimodal.js +6 -2
- package/dist/core/{local-embeddings.d.ts → embeddings/local-embeddings.d.ts} +1 -1
- package/dist/core/embeddings/local-embeddings.js +11 -0
- package/dist/core/embeddings/qmd-client.js +1 -1
- package/dist/core/embeddings/transformers-local.d.ts +64 -0
- package/dist/core/embeddings/transformers-local.js +213 -0
- package/dist/core/embeddings.d.ts +1 -28
- package/dist/core/embeddings.js +2 -453
- package/dist/core/graph/entity-deduplicator.d.ts +24 -0
- package/dist/core/graph/entity-deduplicator.js +183 -0
- package/dist/core/graph/graph-builder.d.ts +46 -0
- package/dist/core/graph/graph-builder.js +174 -0
- package/dist/core/graph/graph-traversal.d.ts +80 -0
- package/dist/core/graph/graph-traversal.js +315 -0
- package/dist/core/graph/index.d.ts +19 -0
- package/dist/core/graph/index.js +13 -0
- package/dist/core/graph/llm-entity-extractor.d.ts +49 -0
- package/dist/core/graph/llm-entity-extractor.js +313 -0
- package/dist/core/graph/multi-hop-retrieval.d.ts +48 -0
- package/dist/core/graph/multi-hop-retrieval.js +215 -0
- package/dist/core/graph/relationship-extractor.d.ts +48 -0
- package/dist/core/graph/relationship-extractor.js +351 -0
- package/dist/core/hooks/agent-hooks.d.ts +10 -1
- package/dist/core/hooks/agent-hooks.js +301 -24
- package/dist/core/hot-cache.d.ts +86 -0
- package/dist/core/hot-cache.js +285 -0
- package/dist/core/index.d.ts +9 -9
- package/dist/core/index.js +9 -12
- package/dist/core/ingestion/core-memory.d.ts +2 -2
- package/dist/core/ingestion/core-memory.js +3 -3
- package/dist/core/ingestion/learnings.js +3 -0
- package/dist/core/ingestion/signal-engine.d.ts +41 -0
- package/dist/core/ingestion/signal-engine.js +201 -0
- package/dist/core/{obsidian-vault.d.ts → integrations/obsidian-vault.d.ts} +2 -1
- package/dist/core/{obsidian-vault.js → integrations/obsidian-vault.js} +69 -7
- package/dist/core/lib/parse-embedding.d.ts +9 -0
- package/dist/core/lib/parse-embedding.js +58 -0
- package/dist/core/lib/schemas.d.ts +57 -54
- package/dist/core/lib/types.d.ts +45 -0
- package/dist/core/lib/types.js +6 -0
- package/dist/core/lib/utils.d.ts +4 -0
- package/dist/core/lib/utils.js +55 -0
- package/dist/core/lifecycle.d.ts +0 -1
- package/dist/core/lifecycle.js +13 -23
- package/dist/core/logger.d.ts +1 -0
- package/dist/core/logger.js +14 -8
- package/dist/core/mcp/tools.d.ts +0 -2
- package/dist/core/mcp/tools.js +0 -87
- package/dist/core/mcp/types.d.ts +25 -253
- package/dist/core/mcp/types.js +2 -2
- package/dist/core/memory/categorizer.js +1 -0
- package/dist/core/memory/consolidation.js +2 -28
- package/dist/core/memory/entity-extractor.d.ts +4 -0
- package/dist/core/memory/entity-extractor.js +30 -16
- package/dist/core/memory/explain.d.ts +18 -0
- package/dist/core/memory/explain.js +92 -0
- package/dist/core/memory/fact-deriver.d.ts +31 -0
- package/dist/core/memory/fact-deriver.js +236 -0
- package/dist/core/memory/hybrid-retrieval.d.ts +14 -16
- package/dist/core/memory/hybrid-retrieval.js +25 -127
- package/dist/core/memory/hybrid-scorer.js +6 -23
- package/dist/core/memory/hybrid-search.d.ts +10 -7
- package/dist/core/memory/hybrid-search.js +458 -221
- package/dist/core/memory/importance.d.ts +0 -17
- package/dist/core/memory/importance.js +1 -58
- package/dist/core/memory/index.d.ts +1 -0
- package/dist/core/memory/index.js +1 -0
- package/dist/core/memory/memories.d.ts +13 -17
- package/dist/core/memory/memories.js +78 -75
- package/dist/core/memory/memory-lifecycle.d.ts +2 -2
- package/dist/core/memory/memory-lifecycle.js +10 -18
- package/dist/core/memory/normalization.d.ts +1 -16
- package/dist/core/memory/path-strengthener.d.ts +39 -0
- package/dist/core/memory/path-strengthener.js +150 -0
- package/dist/core/memory/query-processor.js +37 -3
- package/dist/core/memory/retrieval-feedback.d.ts +70 -0
- package/dist/core/memory/retrieval-feedback.js +213 -0
- package/dist/core/memory/stale-cleaner.d.ts +26 -0
- package/dist/core/memory/stale-cleaner.js +97 -0
- package/dist/core/memory/stats.d.ts +10 -0
- package/dist/core/memory/stats.js +8 -3
- package/dist/core/memory/trigger-detector.d.ts +8 -1
- package/dist/core/memory/trigger-detector.js +42 -5
- package/dist/core/places/index.d.ts +1 -1
- package/dist/core/places/index.js +1 -1
- package/dist/core/places/places.d.ts +13 -13
- package/dist/core/places/places.js +27 -27
- package/dist/core/places/rules.js +23 -23
- package/dist/core/places/walking.d.ts +3 -3
- package/dist/core/places/walking.js +7 -7
- package/dist/core/projects.js +8 -0
- package/dist/core/runtime/trust-report.d.ts +102 -0
- package/dist/core/runtime/trust-report.js +107 -0
- package/dist/core/runtime/trust-state.d.ts +12 -0
- package/dist/core/runtime/trust-state.js +309 -0
- package/dist/core/scheduler/cron-scheduler.d.ts +1 -1
- package/dist/core/scheduler/cron-scheduler.js +164 -3
- package/dist/core/scheduler/job-runner.js +1 -1
- package/dist/core/search/qmd-wrapper.d.ts +36 -0
- package/dist/core/search/qmd-wrapper.js +58 -0
- package/dist/core/session/auto-load.js +28 -3
- package/dist/core/session/entity-tracker.d.ts +62 -0
- package/dist/core/session/entity-tracker.js +287 -0
- package/dist/core/session/reference-resolver.d.ts +26 -0
- package/dist/core/session/reference-resolver.js +121 -0
- package/dist/core/session/self-iteration-job.d.ts +15 -0
- package/dist/core/session/self-iteration-job.js +163 -58
- package/dist/core/session/working-set.d.ts +50 -0
- package/dist/core/session/working-set.js +212 -0
- package/dist/core/snapshots/creation.d.ts +2 -8
- package/dist/core/snapshots/creation.js +3 -12
- package/dist/core/utils/summarization-helpers.d.ts +0 -4
- package/dist/core/utils/summarization-helpers.js +1 -6
- package/dist/db/bootstrap.d.ts +2 -0
- package/dist/db/bootstrap.js +229 -280
- package/dist/db/drizzle/schema-sqlite.d.ts +702 -1
- package/dist/db/drizzle/schema-sqlite.js +83 -4
- package/dist/db/drizzle/schema.d.ts +653 -1
- package/dist/db/drizzle/schema.js +93 -4
- package/dist/db/migrations/associations.d.ts +6 -0
- package/dist/db/migrations/associations.js +29 -0
- package/dist/db/migrations/beliefs.d.ts +10 -0
- package/dist/db/migrations/beliefs.js +76 -0
- package/dist/db/migrations/core-memory.d.ts +6 -0
- package/dist/db/migrations/core-memory.js +29 -0
- package/dist/db/migrations/fts.d.ts +6 -0
- package/dist/db/migrations/fts.js +52 -0
- package/dist/db/migrations/index.d.ts +25 -0
- package/dist/db/migrations/index.js +51 -0
- package/dist/db/migrations/indexes.d.ts +6 -0
- package/dist/db/migrations/indexes.js +30 -0
- package/dist/db/migrations/learnings.d.ts +7 -0
- package/dist/db/migrations/learnings.js +26 -0
- package/dist/db/migrations/maintenance.d.ts +6 -0
- package/dist/db/migrations/maintenance.js +61 -0
- package/dist/db/migrations/memories.d.ts +7 -0
- package/dist/db/migrations/memories.js +16 -0
- package/dist/db/migrations/memory-places.d.ts +6 -0
- package/dist/db/migrations/memory-places.js +29 -0
- package/dist/db/migrations/places.d.ts +6 -0
- package/dist/db/migrations/places.js +43 -0
- package/dist/db/migrations/projects.d.ts +3 -0
- package/dist/db/migrations/projects.js +13 -0
- package/dist/db/migrations/tier-conversion.d.ts +7 -0
- package/dist/db/migrations/tier-conversion.js +20 -0
- package/dist/db/schema/beliefs.d.ts +9 -0
- package/dist/db/schema/beliefs.js +46 -0
- package/dist/db/schema/generator.d.ts +38 -0
- package/dist/db/schema/generator.js +108 -0
- package/dist/db/schema/index.d.ts +19 -20
- package/dist/db/schema/index.js +25 -79
- package/dist/db/schema/learnings.d.ts +7 -0
- package/dist/db/schema/learnings.js +30 -0
- package/dist/db/schema/memories.d.ts +7 -0
- package/dist/db/schema/memories.js +81 -0
- package/dist/db/schema/projects.d.ts +4 -0
- package/dist/db/schema/projects.js +31 -0
- package/dist/packages/mcp/src/index.d.ts +3 -0
- package/dist/packages/mcp/src/index.js +733 -0
- package/mcp.json.example +8 -11
- package/package.json +57 -76
- package/packages/cli/package.json +22 -0
- package/packages/cli/src/commands/clean.ts +68 -0
- package/packages/cli/src/commands/context.ts +79 -0
- package/packages/cli/src/commands/doctor.ts +357 -0
- package/packages/cli/src/commands/forget.ts +72 -0
- package/packages/cli/src/commands/health.ts +36 -0
- package/packages/cli/src/commands/inspect.ts +41 -0
- package/packages/cli/src/commands/link.ts +50 -0
- package/packages/cli/src/commands/migrate.ts +93 -0
- package/packages/cli/src/commands/recall.ts +99 -0
- package/packages/cli/src/commands/recent.ts +57 -0
- package/packages/cli/src/commands/remember.ts +139 -0
- package/packages/cli/src/commands/run.ts +58 -0
- package/packages/cli/src/commands/stale.ts +43 -0
- package/packages/cli/src/commands/stats.ts +42 -0
- package/packages/cli/src/index.ts +57 -0
- package/packages/cli/tsconfig.json +24 -0
- package/packages/mcp/package.json +26 -0
- package/packages/mcp/src/index.ts +877 -0
- package/packages/mcp/tsconfig.json +20 -0
- package/skills/squish-memory/SKILL.md +38 -35
- package/skills/squish-memory/{scripts/install.sh → install.sh} +1 -1
- package/skills/squish-memory/references/claude-desktop.json +12 -0
- package/skills/squish-memory/references/openclaw.json +13 -0
- package/skills/squish-memory/references/opencode.json +14 -0
- package/config/hooks/claude-code-hooks.json +0 -39
- package/config/hooks/cursor-hooks.json +0 -30
- package/config/hooks/opencode-hooks.json +0 -30
- package/config/hooks/windsurf-hooks.json +0 -30
- package/config/mcp-cli-fallback-policy.json +0 -22
- package/config/mcp.json +0 -38
- package/config/plugin-manifest.json +0 -101
- package/config/plugin-manifest.schema.json +0 -244
- package/config/plugin.json +0 -32
- package/config/remote-memory-policy.json +0 -32
- package/core/commands/context-paging.md +0 -51
- package/core/commands/context-status.md +0 -22
- package/core/commands/context.md +0 -5
- package/core/commands/core-memory.md +0 -56
- package/core/commands/health.md +0 -5
- package/core/commands/init.md +0 -39
- package/core/commands/merge.md +0 -113
- package/core/commands/recall.md +0 -5
- package/core/commands/remember.md +0 -11
- package/core/commands/search.md +0 -10
- package/dist/core/commands/managed-sync.d.ts +0 -10
- package/dist/core/commands/managed-sync.js +0 -64
- package/dist/core/external-folder/index.d.ts +0 -102
- package/dist/core/external-folder/index.js +0 -294
- package/dist/core/namespaces/index.d.ts +0 -71
- package/dist/core/namespaces/index.js +0 -305
- package/dist/core/namespaces/uri-parser.d.ts +0 -31
- package/dist/core/namespaces/uri-parser.js +0 -74
- package/dist/core/search/qmd-search.d.ts +0 -61
- package/dist/core/search/qmd-search.js +0 -178
- package/dist/core/session-hooks/self-iteration-job.d.ts +0 -20
- package/dist/core/session-hooks/self-iteration-job.js +0 -282
- package/dist/core/session-hooks/session-hooks.d.ts +0 -18
- package/dist/core/session-hooks/session-hooks.js +0 -58
- package/dist/core/snapshots.d.ts +0 -29
- package/dist/core/snapshots.js +0 -220
- package/dist/core/sync/qmd-sync.d.ts +0 -94
- package/dist/core/sync/qmd-sync.js +0 -201
- package/dist/index.d.ts +0 -7
- package/dist/index.js +0 -1677
- package/dist/vendor/sql.js/sql-wasm.wasm +0 -0
- package/dist/webui/server.d.ts +0 -5
- package/dist/webui/server.js +0 -642
- package/generated/mcp/manifest.json +0 -23
- package/generated/mcp/mcp-servers.json +0 -25
- package/generated/mcp/mcporter.json +0 -34
- package/generated/mcp/openclaw-memory-qmd.json +0 -17
- package/generated/mcp/runtime.json +0 -12
- package/scripts/README.md +0 -60
- package/scripts/build-release.sh +0 -36
- package/scripts/check-secrets.js +0 -132
- package/scripts/copy-runtime-assets.mjs +0 -26
- package/scripts/generate-mcp.mjs +0 -264
- package/scripts/github-release.sh +0 -77
- package/scripts/init-dirs.mjs +0 -13
- package/scripts/install-claude-code.sh +0 -85
- package/scripts/install-cursor.sh +0 -56
- package/scripts/install-hooks.sh +0 -73
- package/scripts/install-interactive.mjs +0 -357
- package/scripts/install-opencode.sh +0 -75
- package/scripts/install-plugin.mjs +0 -415
- package/scripts/install-windsurf.sh +0 -67
- package/scripts/remote-preflight.mjs +0 -62
- package/scripts/squish-fallback.mjs +0 -132
- package/scripts/test-interactive.mjs +0 -131
- package/scripts/verify-mcp.mjs +0 -214
- package/skills/squish-memory/scripts/install.mjs +0 -335
- 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
|
+
}
|