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,197 @@
|
|
|
1
|
+
/** Obsidian Vault Integration
|
|
2
|
+
*
|
|
3
|
+
* Appends hot memories to Obsidian daily notes
|
|
4
|
+
* Format: YYYY-MM-DD.md with memory blocks
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { existsSync, mkdirSync, appendFileSync, readFileSync, readdirSync } from 'fs';
|
|
8
|
+
import { join } from 'path';
|
|
9
|
+
import { randomUUID } from 'crypto';
|
|
10
|
+
import { logger } from '../logger.js';
|
|
11
|
+
|
|
12
|
+
export interface ObsidianMemoryInput {
|
|
13
|
+
content: string;
|
|
14
|
+
id: string;
|
|
15
|
+
type: string;
|
|
16
|
+
tags: string[];
|
|
17
|
+
reasoning?: string;
|
|
18
|
+
memoryContext?: string;
|
|
19
|
+
examples?: string;
|
|
20
|
+
exceptions?: string;
|
|
21
|
+
source?: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Append a memory to the daily note in Obsidian vault
|
|
26
|
+
* Creates YYYY-MM-DD.md file if it doesn't exist
|
|
27
|
+
*/
|
|
28
|
+
export async function appendToObsidianVault(
|
|
29
|
+
memory: ObsidianMemoryInput,
|
|
30
|
+
vaultPath: string
|
|
31
|
+
): Promise<void> {
|
|
32
|
+
const today = new Date();
|
|
33
|
+
const year = today.getFullYear();
|
|
34
|
+
const month = String(today.getMonth() + 1).padStart(2, '0');
|
|
35
|
+
const day = String(today.getDate()).padStart(2, '0');
|
|
36
|
+
const dateStr = `${year}-${month}-${day}`;
|
|
37
|
+
|
|
38
|
+
const dailyNotePath = join(vaultPath, `${dateStr}.md`);
|
|
39
|
+
|
|
40
|
+
// Ensure vault directory exists
|
|
41
|
+
if (!existsSync(vaultPath)) {
|
|
42
|
+
mkdirSync(vaultPath, { recursive: true });
|
|
43
|
+
logger.info(`[Obsidian] Created vault directory: ${vaultPath}`);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Build memory block content
|
|
47
|
+
const memoryBlock = formatMemoryAsBlock(memory);
|
|
48
|
+
|
|
49
|
+
// Check if daily note exists, if not create with header
|
|
50
|
+
if (!existsSync(dailyNotePath)) {
|
|
51
|
+
const header = `# ${dateStr}\n\n## Squish Memories\n\n`;
|
|
52
|
+
appendFileSync(dailyNotePath, header, 'utf-8');
|
|
53
|
+
logger.info(`[Obsidian] Created daily note: ${dailyNotePath}`);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Append memory block
|
|
57
|
+
appendFileSync(dailyNotePath, memoryBlock + '\n\n', 'utf-8');
|
|
58
|
+
logger.info(`[Obsidian] Appended memory to ${dateStr}.md`);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Format memory as Obsidian-compatible block
|
|
63
|
+
*/
|
|
64
|
+
function formatMemoryAsBlock(memory: ObsidianMemoryInput): string {
|
|
65
|
+
const lines: string[] = [];
|
|
66
|
+
|
|
67
|
+
// Memory content as checkbox (can be toggled in Obsidian)
|
|
68
|
+
lines.push(`- [ ] "${memory.content}"`);
|
|
69
|
+
|
|
70
|
+
// Add ID reference for linking
|
|
71
|
+
lines.push(` - **id:** \`${memory.id}\``);
|
|
72
|
+
|
|
73
|
+
// Type and tags
|
|
74
|
+
if (memory.type) {
|
|
75
|
+
lines.push(` - **type:** ${memory.type}`);
|
|
76
|
+
}
|
|
77
|
+
if (memory.tags && memory.tags.length > 0) {
|
|
78
|
+
const tagList = memory.tags.map(t => `#${t.replace(/\s+/g, '-')}`).join(' ');
|
|
79
|
+
lines.push(` - **tags:** ${tagList}`);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Rich context fields
|
|
83
|
+
if (memory.source) {
|
|
84
|
+
lines.push(` - **source:** ${memory.source}`);
|
|
85
|
+
}
|
|
86
|
+
if (memory.reasoning) {
|
|
87
|
+
lines.push(` - **reasoning:** ${memory.reasoning}`);
|
|
88
|
+
}
|
|
89
|
+
if (memory.memoryContext) {
|
|
90
|
+
lines.push(` - **context:** ${memory.memoryContext}`);
|
|
91
|
+
}
|
|
92
|
+
if (memory.examples) {
|
|
93
|
+
lines.push(` - **examples:** ${memory.examples}`);
|
|
94
|
+
}
|
|
95
|
+
if (memory.exceptions) {
|
|
96
|
+
lines.push(` - **exceptions:** ${memory.exceptions}`);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return lines.join('\n');
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Check if Obsidian vault is connected and accessible
|
|
104
|
+
*/
|
|
105
|
+
export function isObsidianConnected(vaultPath: string): boolean {
|
|
106
|
+
if (!vaultPath || vaultPath === '') {
|
|
107
|
+
return false;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// Check if path exists or can be created
|
|
111
|
+
const parentDir = vaultPath.includes('/')
|
|
112
|
+
? vaultPath.split('/').slice(0, -1).join('/')
|
|
113
|
+
: vaultPath.split('\\').slice(0, -1).join('\\');
|
|
114
|
+
|
|
115
|
+
return existsSync(parentDir) || true; // Assume creatable if parent exists
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* Read existing memories from Obsidian vault (for import feature)
|
|
120
|
+
* Scans vault for .md files and extracts memory blocks
|
|
121
|
+
*/
|
|
122
|
+
export async function readFromVault(vaultPath: string): Promise<ObsidianMemoryInput[]> {
|
|
123
|
+
if (!existsSync(vaultPath)) {
|
|
124
|
+
logger.warn(`[Obsidian] Vault path does not exist: ${vaultPath}`);
|
|
125
|
+
return [];
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
const memories: ObsidianMemoryInput[] = [];
|
|
129
|
+
|
|
130
|
+
// Recursively find all .md files
|
|
131
|
+
const mdFiles = await findMarkdownFiles(vaultPath);
|
|
132
|
+
|
|
133
|
+
for (const filePath of mdFiles) {
|
|
134
|
+
try {
|
|
135
|
+
const content = readFileSync(filePath, 'utf-8');
|
|
136
|
+
const filename = filePath.split(/[/\\]/).pop() || '';
|
|
137
|
+
|
|
138
|
+
// Extract frontmatter if present
|
|
139
|
+
const frontmatterMatch = content.match(/^---\n([\s\S]*?)\n---/);
|
|
140
|
+
let tags: string[] = [];
|
|
141
|
+
|
|
142
|
+
if (frontmatterMatch) {
|
|
143
|
+
const tagMatch = frontmatterMatch[1].match(/tags:\s*\[([^\]]*)\]/);
|
|
144
|
+
if (tagMatch) {
|
|
145
|
+
tags = tagMatch[1].split(',').map((t: string) => t.trim()).filter(Boolean);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Extract content after frontmatter
|
|
150
|
+
const body = frontmatterMatch ? content.slice(frontmatterMatch[0].length).trim() : content;
|
|
151
|
+
|
|
152
|
+
// Skip empty files or very short ones
|
|
153
|
+
if (body.length < 10) continue;
|
|
154
|
+
|
|
155
|
+
memories.push({
|
|
156
|
+
id: randomUUID(),
|
|
157
|
+
content: body,
|
|
158
|
+
tags,
|
|
159
|
+
type: 'observation'
|
|
160
|
+
});
|
|
161
|
+
} catch (error) {
|
|
162
|
+
logger.warn(`[Obsidian] Failed to read file ${filePath}: ${error}`);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
logger.info(`[Obsidian] Found ${memories.length} memories in vault`);
|
|
167
|
+
return memories;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Recursively find all markdown files in a directory
|
|
172
|
+
*/
|
|
173
|
+
async function findMarkdownFiles(dir: string): Promise<string[]> {
|
|
174
|
+
const files: string[] = [];
|
|
175
|
+
|
|
176
|
+
try {
|
|
177
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
178
|
+
|
|
179
|
+
for (const entry of entries) {
|
|
180
|
+
const fullPath = join(dir, entry.name);
|
|
181
|
+
|
|
182
|
+
if (entry.isDirectory()) {
|
|
183
|
+
// Skip hidden directories
|
|
184
|
+
if (!entry.name.startsWith('.')) {
|
|
185
|
+
const subFiles = await findMarkdownFiles(fullPath);
|
|
186
|
+
files.push(...subFiles);
|
|
187
|
+
}
|
|
188
|
+
} else if (entry.isFile() && entry.name.endsWith('.md')) {
|
|
189
|
+
files.push(fullPath);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
} catch (error) {
|
|
193
|
+
logger.warn(`[Obsidian] Failed to read directory ${dir}: ${error}`);
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
return files;
|
|
197
|
+
}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
/** Layer Extractor - Creates L0/L1 truncated layers for token-efficient hierarchical retrieval */
|
|
2
|
+
|
|
3
|
+
import { getDb } from '../../db/index.js';
|
|
4
|
+
import { getSchema } from '../../db/schema.js';
|
|
5
|
+
|
|
6
|
+
export interface LayerGenerationConfig {
|
|
7
|
+
l0MaxLength?: number; // Default: 50 chars (truncated preview)
|
|
8
|
+
l1MaxLength?: number; // Default: 300 chars (first 2 sentences truncated)
|
|
9
|
+
autoGenerate?: boolean; // Default: true for content > 500 chars
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export interface LayerGenerationResult {
|
|
13
|
+
l0?: string; // Ultra-short truncated preview
|
|
14
|
+
l1?: string; // Short truncated preview (first 2 sentences)
|
|
15
|
+
l2?: string; // Full content (reference)
|
|
16
|
+
tokenCounts: { l0: number; l1: number; l2: number };
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
const DEFAULT_CONFIG: LayerGenerationConfig = {
|
|
20
|
+
l0MaxLength: 50,
|
|
21
|
+
l1MaxLength: 300,
|
|
22
|
+
autoGenerate: true,
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Extract L0/L1/L2 layers for content via truncation
|
|
27
|
+
*/
|
|
28
|
+
export async function extractLayers(
|
|
29
|
+
content: string,
|
|
30
|
+
config: LayerGenerationConfig = {}
|
|
31
|
+
): Promise<LayerGenerationResult> {
|
|
32
|
+
const l0MaxLength = config.l0MaxLength ?? DEFAULT_CONFIG.l0MaxLength;
|
|
33
|
+
const l1MaxLength = config.l1MaxLength ?? DEFAULT_CONFIG.l1MaxLength;
|
|
34
|
+
|
|
35
|
+
// Extract L0 and L1 in parallel
|
|
36
|
+
const [l0, l1] = await Promise.all([
|
|
37
|
+
extractL0Truncation(content, l0MaxLength!),
|
|
38
|
+
extractL1Truncation(content, l1MaxLength!),
|
|
39
|
+
]);
|
|
40
|
+
|
|
41
|
+
const l2Tokens = countTokens(content);
|
|
42
|
+
const l0Tokens = countTokens(l0 || '');
|
|
43
|
+
const l1Tokens = countTokens(l1 || '');
|
|
44
|
+
|
|
45
|
+
return {
|
|
46
|
+
l0: l0 ?? '',
|
|
47
|
+
l1: l1 ?? '',
|
|
48
|
+
l2: content,
|
|
49
|
+
tokenCounts: {
|
|
50
|
+
l0: l0Tokens,
|
|
51
|
+
l1: l1Tokens,
|
|
52
|
+
l2: l2Tokens,
|
|
53
|
+
},
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Extract L0: Ultra-short truncated preview (first sentence or chars)
|
|
59
|
+
*/
|
|
60
|
+
async function extractL0Truncation(
|
|
61
|
+
content: string,
|
|
62
|
+
maxLength: number
|
|
63
|
+
): Promise<string> {
|
|
64
|
+
// Simple truncation for L0 (first sentence or chars)
|
|
65
|
+
const trimmed = content.trim();
|
|
66
|
+
const sentences = trimmed.split(/[.!?]/).filter(s => s.trim().length > 0);
|
|
67
|
+
|
|
68
|
+
// Take first non-empty sentence and truncate if needed
|
|
69
|
+
let result = sentences.length > 0 ? sentences[0] : trimmed.substring(0, maxLength);
|
|
70
|
+
|
|
71
|
+
if (result.length > maxLength) {
|
|
72
|
+
result = result.substring(0, maxLength - 3) + '...';
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
return result.trim();
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Extract L1: Short truncated preview (first 2 meaningful sentences)
|
|
80
|
+
*/
|
|
81
|
+
async function extractL1Truncation(
|
|
82
|
+
content: string,
|
|
83
|
+
maxLength: number
|
|
84
|
+
): Promise<string> {
|
|
85
|
+
// Simple truncation for L1 (first 2 meaningful sentences)
|
|
86
|
+
const trimmed = content.trim();
|
|
87
|
+
const sentences = trimmed.split(/[.!?]/).filter(s => s.trim().length > 10);
|
|
88
|
+
|
|
89
|
+
// Take first meaningful sentences and truncate if needed
|
|
90
|
+
let result = sentences.length > 0
|
|
91
|
+
? sentences.slice(0, 2).join(' ').trim()
|
|
92
|
+
: trimmed.substring(0, maxLength);
|
|
93
|
+
|
|
94
|
+
if (result.length > maxLength) {
|
|
95
|
+
result = result.substring(0, maxLength - 3) + '...';
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
return result.trim();
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Count tokens (approximate, 4 chars = 1 token for English)
|
|
103
|
+
*/
|
|
104
|
+
function countTokens(text: string): number {
|
|
105
|
+
// Simple approximation: 4 chars per token
|
|
106
|
+
return Math.ceil(text.length / 4);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* Check if content should have layers auto-extracted
|
|
111
|
+
*/
|
|
112
|
+
export function shouldAutoGenerate(contentLength: number): boolean {
|
|
113
|
+
const threshold = 500;
|
|
114
|
+
return contentLength > threshold;
|
|
115
|
+
}
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Database Client & Schema Loading Abstraction
|
|
3
|
+
*
|
|
4
|
+
* Provides a unified interface for accessing both the database client and schema
|
|
5
|
+
* in a single call, eliminating repetitive patterns of:
|
|
6
|
+
* `const db = createDatabaseClient(await getDb()); const schema = await getSchema();`
|
|
7
|
+
*
|
|
8
|
+
* Features:
|
|
9
|
+
* - Single async call to get both db and schema
|
|
10
|
+
* - Consistent error handling with clear messages
|
|
11
|
+
* - Preserves schema caching
|
|
12
|
+
* - Provides raw connection for special cases
|
|
13
|
+
* - Helper `withDbClient` for functional programming patterns
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* ```typescript
|
|
17
|
+
* // Old pattern (repetitive)
|
|
18
|
+
* const db = createDatabaseClient(await getDb());
|
|
19
|
+
* const schema = await getSchema();
|
|
20
|
+
*
|
|
21
|
+
* // New pattern (unified)
|
|
22
|
+
* const { db, schema } = await getDbClient();
|
|
23
|
+
*
|
|
24
|
+
* // With helper
|
|
25
|
+
* const result = await withDbClient(async (client) => {
|
|
26
|
+
* return await client.db.select().from(client.schema.memories);
|
|
27
|
+
* });
|
|
28
|
+
* ```
|
|
29
|
+
*/
|
|
30
|
+
|
|
31
|
+
import { getDb } from '../../db/index.js';
|
|
32
|
+
import { getSchema } from '../../db/schema.js';
|
|
33
|
+
import { createDatabaseClient } from '../storage/database.js';
|
|
34
|
+
import type { DatabaseClient } from '../storage/database.js';
|
|
35
|
+
import type { SchemaModule } from '../../db/schema.js';
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Unified database client interface combining:
|
|
39
|
+
* - `db`: Wrapped database client with select/insert/update/delete methods
|
|
40
|
+
* - `schema`: Drizzle schema module with table definitions
|
|
41
|
+
* - `raw`: Raw underlying connection for special cases (mcp-server, index, etc.)
|
|
42
|
+
*/
|
|
43
|
+
export interface DbClient {
|
|
44
|
+
/**
|
|
45
|
+
* Wrapped database client with unified API
|
|
46
|
+
* Provides select(), insert(), update(), delete() methods
|
|
47
|
+
*/
|
|
48
|
+
db: DatabaseClient;
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Schema module containing table definitions
|
|
52
|
+
* Access tables via schema.tableName (e.g., schema.memories, schema.users)
|
|
53
|
+
*/
|
|
54
|
+
schema: SchemaModule;
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Raw underlying database connection
|
|
58
|
+
* Use for special cases that require direct access to the native driver
|
|
59
|
+
* (e.g., raw SQL queries, specific driver features)
|
|
60
|
+
*/
|
|
61
|
+
raw: any;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Internal cached schema to preserve getSchema() caching behavior
|
|
66
|
+
*/
|
|
67
|
+
let cachedSchema: SchemaModule | null = null;
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Get a unified database client with both db and schema.
|
|
71
|
+
*
|
|
72
|
+
* This function:
|
|
73
|
+
* 1. Calls getDb() to initialize/retrieve the database connection
|
|
74
|
+
* 2. Calls getSchema() to get the appropriate schema (with caching)
|
|
75
|
+
* 3. Wraps the db with createDatabaseClient() for unified API
|
|
76
|
+
* 4. Returns all three as a single DbClient object
|
|
77
|
+
*
|
|
78
|
+
* Error handling:
|
|
79
|
+
* - Wraps any database initialization errors with a clear message
|
|
80
|
+
* - Preserves the original error as `cause` property
|
|
81
|
+
*
|
|
82
|
+
* Schema caching:
|
|
83
|
+
* - Uses the same caching mechanism as getSchema()
|
|
84
|
+
* - Multiple calls to getDbClient() will return the same schema reference
|
|
85
|
+
*
|
|
86
|
+
* @returns Promise<DbClient> Unified client with db, schema, and raw
|
|
87
|
+
* @throws {Error} When database initialization fails, with cause attached
|
|
88
|
+
*
|
|
89
|
+
* @example
|
|
90
|
+
* ```typescript
|
|
91
|
+
* const { db, schema, raw } = await getDbClient();
|
|
92
|
+
*
|
|
93
|
+
* // Use db for queries
|
|
94
|
+
* const memories = await db.select().from(schema.memories).limit(10);
|
|
95
|
+
*
|
|
96
|
+
* // Use raw for special cases
|
|
97
|
+
* const client = (raw as any).$client;
|
|
98
|
+
* if (client && typeof client.exec === 'function') {
|
|
99
|
+
* client.exec('VACUUM');
|
|
100
|
+
* }
|
|
101
|
+
* ```
|
|
102
|
+
*/
|
|
103
|
+
export async function getDbClient(): Promise<DbClient> {
|
|
104
|
+
try {
|
|
105
|
+
// Get raw database connection
|
|
106
|
+
const rawDb = await getDb();
|
|
107
|
+
|
|
108
|
+
// Get schema (with caching)
|
|
109
|
+
if (!cachedSchema) {
|
|
110
|
+
cachedSchema = await getSchema();
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Create wrapped database client
|
|
114
|
+
const db = createDatabaseClient(rawDb);
|
|
115
|
+
|
|
116
|
+
return {
|
|
117
|
+
db,
|
|
118
|
+
schema: cachedSchema,
|
|
119
|
+
raw: rawDb,
|
|
120
|
+
};
|
|
121
|
+
} catch (error) {
|
|
122
|
+
const message =
|
|
123
|
+
error instanceof Error
|
|
124
|
+
? `Database initialization failed: ${error.message}`
|
|
125
|
+
: 'Database initialization failed';
|
|
126
|
+
|
|
127
|
+
// Wrap error with clear message while preserving cause
|
|
128
|
+
const wrappedError = new Error(message);
|
|
129
|
+
// @ts-ignore - TypeScript doesn't allow 'cause' on Error, but it's valid
|
|
130
|
+
wrappedError.cause = error;
|
|
131
|
+
|
|
132
|
+
throw wrappedError;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/**
|
|
137
|
+
* Execute an operation with a unified database client.
|
|
138
|
+
*
|
|
139
|
+
* Convenience wrapper that:
|
|
140
|
+
* 1. Calls getDbClient() to obtain client
|
|
141
|
+
* 2. Executes the provided operation
|
|
142
|
+
* 3. Ensures proper cleanup if needed (no-op currently, for future extension)
|
|
143
|
+
*
|
|
144
|
+
* @param operation - Async function that receives the DbClient and returns a result
|
|
145
|
+
* @returns Promise<T> The result returned by the operation
|
|
146
|
+
* @throws {Error} When database initialization or operation fails
|
|
147
|
+
*
|
|
148
|
+
* @example
|
|
149
|
+
* ```typescript
|
|
150
|
+
* const count = await withDbClient(async (client) => {
|
|
151
|
+
* const result = await client.db
|
|
152
|
+
* .select({ count: sql`count(*)` })
|
|
153
|
+
* .from(client.schema.memories);
|
|
154
|
+
* return result[0].count;
|
|
155
|
+
* });
|
|
156
|
+
* ```
|
|
157
|
+
*/
|
|
158
|
+
export async function withDbClient<T>(
|
|
159
|
+
operation: (client: DbClient) => Promise<T>
|
|
160
|
+
): Promise<T> {
|
|
161
|
+
const client = await getDbClient();
|
|
162
|
+
try {
|
|
163
|
+
return await operation(client);
|
|
164
|
+
} finally {
|
|
165
|
+
// Future: Add cleanup logic if needed (connection pooling, etc.)
|
|
166
|
+
// Currently getDb() handles connection pooling and reuse
|
|
167
|
+
}
|
|
168
|
+
}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parse embedding from various storage formats.
|
|
3
|
+
* Converted from 4 locations: hybrid-search.ts, consolidation.ts, memories.ts, entity-deduplicator.ts
|
|
4
|
+
*
|
|
5
|
+
* @param embeddingData - Input in Float32Array, Buffer, Uint8Array, JSON string, or Array format
|
|
6
|
+
* @returns Array of numbers or null if parsing fails
|
|
7
|
+
*/
|
|
8
|
+
export function parseEmbedding(embeddingData: unknown): number[] | null {
|
|
9
|
+
if (!embeddingData) return null;
|
|
10
|
+
|
|
11
|
+
// Handle Array directly (most common case)
|
|
12
|
+
if (Array.isArray(embeddingData)) {
|
|
13
|
+
return embeddingData;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
// Handle Float32Array directly
|
|
17
|
+
if (embeddingData instanceof Float32Array) {
|
|
18
|
+
return Array.from(embeddingData);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
// Handle Uint8Array and Buffer - try JSON first, then binary Float32
|
|
22
|
+
if (embeddingData instanceof Uint8Array || Buffer.isBuffer(embeddingData)) {
|
|
23
|
+
// Try to parse as JSON string first
|
|
24
|
+
try {
|
|
25
|
+
const json = JSON.parse(embeddingData.toString());
|
|
26
|
+
if (Array.isArray(json)) {
|
|
27
|
+
return json;
|
|
28
|
+
}
|
|
29
|
+
} catch {
|
|
30
|
+
// Not JSON, try binary Float32
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Try to parse as binary Float32Array
|
|
34
|
+
try {
|
|
35
|
+
const buffer = embeddingData.buffer;
|
|
36
|
+
const arrayBuffer = buffer instanceof ArrayBuffer
|
|
37
|
+
? buffer
|
|
38
|
+
: (buffer as unknown as ArrayBuffer);
|
|
39
|
+
const floatArray = new Float32Array(arrayBuffer);
|
|
40
|
+
return Array.from(floatArray);
|
|
41
|
+
} catch {
|
|
42
|
+
return null;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Handle JSON string
|
|
47
|
+
if (typeof embeddingData === 'string') {
|
|
48
|
+
try {
|
|
49
|
+
const parsed = JSON.parse(embeddingData);
|
|
50
|
+
if (Array.isArray(parsed)) {
|
|
51
|
+
return parsed;
|
|
52
|
+
}
|
|
53
|
+
} catch {
|
|
54
|
+
return null;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return null;
|
|
59
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared Zod schemas for input validation across Squish
|
|
3
|
+
* Provides consistent validation for common input types
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { z } from 'zod';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Limit schema: integer between 1 and 100, default 20
|
|
10
|
+
*/
|
|
11
|
+
export const limitSchema = z.number().int().min(1).max(100).default(20);
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Offset schema: integer >= 0, default 0
|
|
15
|
+
*/
|
|
16
|
+
export const offsetSchema = z.number().int().min(0).default(0);
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Project ID/path schema: non-empty string
|
|
20
|
+
*/
|
|
21
|
+
export const projectIdSchema = z.string().min(1);
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Memory ID schema: valid UUID
|
|
25
|
+
*/
|
|
26
|
+
export const memoryIdSchema = z.string().uuid();
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Pagination schema combining limit and offset
|
|
30
|
+
*/
|
|
31
|
+
export const paginationSchema = z.object({
|
|
32
|
+
limit: limitSchema,
|
|
33
|
+
offset: offsetSchema,
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Search query schema: non-empty string
|
|
38
|
+
*/
|
|
39
|
+
export const searchQuerySchema = z.string().min(1).trim();
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Memory type schema
|
|
43
|
+
*/
|
|
44
|
+
export const memoryTypeSchema = z.enum(['observation', 'fact', 'decision', 'context', 'preference', 'note']);
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Association type schema
|
|
48
|
+
*/
|
|
49
|
+
export const associationTypeSchema = z.enum(['relates_to', 'supports', 'contradicts', 'supersedes', 'duplicate']);
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Weight schema: number between 0 and 1
|
|
53
|
+
*/
|
|
54
|
+
export const weightSchema = z.number().min(0).max(1);
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Learning type schema
|
|
58
|
+
*/
|
|
59
|
+
export const learningTypeSchema = z.enum(['success', 'failure', 'fix', 'observation']);
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Observation type schema
|
|
63
|
+
*/
|
|
64
|
+
export const observationTypeSchema = z.enum(['tool_use', 'file_change', 'error', 'pattern', 'insight']);
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Confidence level schema
|
|
68
|
+
*/
|
|
69
|
+
export const confidenceLevelSchema = z.enum(['certain', 'speculative', 'outdated']);
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Search input schema (common for search operations)
|
|
73
|
+
*/
|
|
74
|
+
export const searchInputSchema = z.object({
|
|
75
|
+
query: searchQuerySchema,
|
|
76
|
+
limit: limitSchema.optional(),
|
|
77
|
+
offset: offsetSchema.optional(),
|
|
78
|
+
project: projectIdSchema.optional(),
|
|
79
|
+
type: memoryTypeSchema.optional(),
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Memory recall schema
|
|
84
|
+
*/
|
|
85
|
+
export const recallInputSchema = z.object({
|
|
86
|
+
memoryId: memoryIdSchema,
|
|
87
|
+
limit: limitSchema.optional(),
|
|
88
|
+
project: projectIdSchema.optional(),
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Tag operation schema
|
|
93
|
+
*/
|
|
94
|
+
export const tagOperationSchema = z.object({
|
|
95
|
+
action: z.enum(['add', 'remove']),
|
|
96
|
+
tag: z.string().min(1),
|
|
97
|
+
search: z.string().optional(),
|
|
98
|
+
olderThan: z.string().optional(),
|
|
99
|
+
type: memoryTypeSchema.optional(),
|
|
100
|
+
limit: limitSchema.optional(),
|
|
101
|
+
project: projectIdSchema.optional(),
|
|
102
|
+
});
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared type definitions for Squish Memory system
|
|
3
|
+
* Consolidated from core/memory/memories.ts and core/memory/normalization.ts
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
// Valid memory type values
|
|
7
|
+
export type MemoryType = 'observation' | 'fact' | 'decision' | 'context' | 'preference' | 'note' | 'task';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Confidence level for memory records
|
|
11
|
+
* - certain: High confidence, verified information
|
|
12
|
+
* - speculative: Low confidence, unverified or uncertain
|
|
13
|
+
* - outdated: Information that may no longer be accurate
|
|
14
|
+
*/
|
|
15
|
+
export type ConfidenceLevel = 'certain' | 'speculative' | 'outdated';
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Unified MemoryRecord interface
|
|
19
|
+
* Used across memories.ts and normalization.ts
|
|
20
|
+
*
|
|
21
|
+
* Note: type is string to avoid circular dependency with memories.ts
|
|
22
|
+
* Use MemoryType for type checking when needed
|
|
23
|
+
*/
|
|
24
|
+
export interface MemoryRecord {
|
|
25
|
+
id: string;
|
|
26
|
+
projectId?: string | null;
|
|
27
|
+
type: string; // Stored as string in DB (use MemoryType for valid values)
|
|
28
|
+
content: string;
|
|
29
|
+
summary?: string | null;
|
|
30
|
+
tags: string[];
|
|
31
|
+
metadata?: Record<string, unknown> | null;
|
|
32
|
+
createdAt?: string | null;
|
|
33
|
+
validFrom?: string | null;
|
|
34
|
+
validTo?: string | null;
|
|
35
|
+
recordedAt?: string | null;
|
|
36
|
+
similarity?: number;
|
|
37
|
+
importance?: number;
|
|
38
|
+
confidenceLevel?: ConfidenceLevel | null;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Result of a safety check
|
|
43
|
+
* Used to determine if a merge action can proceed
|
|
44
|
+
*/
|
|
45
|
+
export interface SafetyCheckResult {
|
|
46
|
+
passed: boolean;
|
|
47
|
+
warnings: string[];
|
|
48
|
+
blockers: string[]; // Hard failures
|
|
49
|
+
}
|