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,51 @@
|
|
|
1
|
+
import { randomBytes, pbkdf2Sync, createCipheriv, createDecipheriv } from 'crypto';
|
|
2
|
+
import { config } from '../../config.js';
|
|
3
|
+
|
|
4
|
+
const ITERATIONS = 100_000;
|
|
5
|
+
const KEY_LEN = 32; // 256-bit
|
|
6
|
+
const DIGEST = 'sha256';
|
|
7
|
+
const ALGO = 'aes-256-gcm';
|
|
8
|
+
const NONCE_LEN = 12;
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Derive a symmetric key from the passphrase and a salt.
|
|
12
|
+
* The salt is stored in `.squish/salt` (plain text) and is generated once.
|
|
13
|
+
*/
|
|
14
|
+
function getSalt(): Buffer {
|
|
15
|
+
const fs = require('fs');
|
|
16
|
+
const path = require('path');
|
|
17
|
+
const saltPath = path.join(config.dataDir, 'salt');
|
|
18
|
+
if (!fs.existsSync(saltPath)) {
|
|
19
|
+
const salt = randomBytes(16).toString('hex');
|
|
20
|
+
fs.writeFileSync(saltPath, salt);
|
|
21
|
+
return Buffer.from(salt, 'hex');
|
|
22
|
+
}
|
|
23
|
+
return Buffer.from(fs.readFileSync(saltPath, 'utf-8'), 'hex');
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function deriveKey(passphrase: string): Buffer {
|
|
27
|
+
const salt = getSalt();
|
|
28
|
+
return pbkdf2Sync(passphrase, salt, ITERATIONS, KEY_LEN, DIGEST);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function encrypt(plain: string, passphrase?: string): { ciphertext: string; nonce: string } {
|
|
32
|
+
const key = deriveKey(passphrase || config.encryptionPassphrase);
|
|
33
|
+
const nonce = randomBytes(NONCE_LEN);
|
|
34
|
+
const cipher = createCipheriv(ALGO, key, nonce);
|
|
35
|
+
const encrypted = Buffer.concat([cipher.update(plain, 'utf8'), cipher.final()]);
|
|
36
|
+
const tag = cipher.getAuthTag();
|
|
37
|
+
const ciphertext = Buffer.concat([encrypted, tag]).toString('base64');
|
|
38
|
+
return { ciphertext, nonce: nonce.toString('base64') };
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export function decrypt(ciphertext: string, nonceB64: string, passphrase?: string): string {
|
|
42
|
+
const key = deriveKey(passphrase || config.encryptionPassphrase);
|
|
43
|
+
const nonce = Buffer.from(nonceB64, 'base64');
|
|
44
|
+
const data = Buffer.from(ciphertext, 'base64');
|
|
45
|
+
const tag = data.slice(data.length - 16);
|
|
46
|
+
const encrypted = data.slice(0, data.length - 16);
|
|
47
|
+
const decipher = createDecipheriv(ALGO, key, nonce);
|
|
48
|
+
decipher.setAuthTag(tag);
|
|
49
|
+
const decrypted = Buffer.concat([decipher.update(encrypted), decipher.final()]);
|
|
50
|
+
return decrypted.toString('utf8');
|
|
51
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Memory Governance
|
|
3
|
+
* Implements protection, pinning, and immutability rules
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { eq } from 'drizzle-orm';
|
|
7
|
+
import { getDb } from '../../db/index.js';
|
|
8
|
+
import { getSchema } from '../../db/schema.js';
|
|
9
|
+
import { config } from '../../config.js';
|
|
10
|
+
import { performMemoryOperation } from '../utils/memory-operations.js';
|
|
11
|
+
import { logger } from '../logger.js';
|
|
12
|
+
import { createDatabaseClient } from '../storage/database.js';
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Mark a memory as protected (cannot be evicted)
|
|
16
|
+
*/
|
|
17
|
+
export async function protectMemory(memoryId: string, reason: string): Promise<void> {
|
|
18
|
+
await performMemoryOperation(memoryId, {
|
|
19
|
+
name: 'protecting memory',
|
|
20
|
+
updates: {
|
|
21
|
+
isProtected: true,
|
|
22
|
+
metadata: { protectionReason: reason, protectedAt: new Date().toISOString() },
|
|
23
|
+
},
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Pin a memory for automatic injection into context
|
|
29
|
+
*/
|
|
30
|
+
export async function pinMemory(memoryId: string): Promise<void> {
|
|
31
|
+
const db = createDatabaseClient(await getDb());
|
|
32
|
+
const schema = await getSchema();
|
|
33
|
+
|
|
34
|
+
await db
|
|
35
|
+
.update(schema.memories)
|
|
36
|
+
.set({
|
|
37
|
+
isPinned: true,
|
|
38
|
+
importanceScore: 100,
|
|
39
|
+
lastImportanceRecalc: new Date(),
|
|
40
|
+
})
|
|
41
|
+
.where(eq(schema.memories.id, memoryId));
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Unpin a memory
|
|
46
|
+
*/
|
|
47
|
+
export async function unpinMemory(memoryId: string): Promise<void> {
|
|
48
|
+
const db = createDatabaseClient(await getDb());
|
|
49
|
+
const schema = await getSchema();
|
|
50
|
+
|
|
51
|
+
await db
|
|
52
|
+
.update(schema.memories)
|
|
53
|
+
.set({
|
|
54
|
+
isPinned: false,
|
|
55
|
+
lastImportanceRecalc: new Date(),
|
|
56
|
+
})
|
|
57
|
+
.where(eq(schema.memories.id, memoryId));
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Get all pinned memories for auto-injection into context
|
|
74
|
+
* Works regardless of governance settings - pinned memories should always be accessible
|
|
75
|
+
*/
|
|
76
|
+
export async function getPinnedMemories(projectId?: string): Promise<any[]> {
|
|
77
|
+
try {
|
|
78
|
+
const db = await getDb();
|
|
79
|
+
const schema = await getSchema();
|
|
80
|
+
|
|
81
|
+
let query = (db as any)
|
|
82
|
+
.select()
|
|
83
|
+
.from(schema.memories)
|
|
84
|
+
.where(eq(schema.memories.isPinned, true))
|
|
85
|
+
.limit(50);
|
|
86
|
+
|
|
87
|
+
return await query;
|
|
88
|
+
} catch (error) {
|
|
89
|
+
logger.error('Error retrieving pinned memories', error);
|
|
90
|
+
return [];
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* Get pinned memories formatted for context injection
|
|
96
|
+
*/
|
|
97
|
+
export async function getPinnedMemoriesForContext(projectId?: string): Promise<string[]> {
|
|
98
|
+
const pinned = await getPinnedMemories(projectId);
|
|
99
|
+
return pinned.map(m =>
|
|
100
|
+
`[Pinned] ${m.content?.substring(0, 500) || '(no content)'}`
|
|
101
|
+
);
|
|
102
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Privacy System
|
|
3
|
+
* Handles private tag filtering, secret detection, and path-based exclusions
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { detectSecrets } from './secret-detector.js';
|
|
7
|
+
|
|
8
|
+
type PrivacyMode = 'strict' | 'moderate' | 'off';
|
|
9
|
+
|
|
10
|
+
const DENY_PATTERNS = [
|
|
11
|
+
'.env', '.env.local', '.env.*.local',
|
|
12
|
+
'*.key', '*.pem',
|
|
13
|
+
'secrets.json', 'credentials.json',
|
|
14
|
+
'.aws', '.ssh', 'node_modules/.env'
|
|
15
|
+
];
|
|
16
|
+
|
|
17
|
+
const PRIVATE_TAG_REGEX = /<(private|secret)>[\s\S]*?<\/\1>/gi;
|
|
18
|
+
|
|
19
|
+
export function hasPrivateTags(content: string): boolean {
|
|
20
|
+
return PRIVATE_TAG_REGEX.test(content);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function stripPrivateTags(content: string): string {
|
|
24
|
+
return content.replace(/<private>[\s\S]*?<\/private>/gi, '[PRIVATE]')
|
|
25
|
+
.replace(/<secret>[\s\S]*?<\/secret>/gi, '[SECRET]');
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function isInDenyList(filePath: string): boolean {
|
|
29
|
+
const normalized = filePath.toLowerCase().replace(/\\/g, '/');
|
|
30
|
+
|
|
31
|
+
for (const pattern of DENY_PATTERNS) {
|
|
32
|
+
if (normalized === pattern) return true;
|
|
33
|
+
if (pattern.startsWith('*.') && normalized.endsWith(pattern.substring(1))) return true;
|
|
34
|
+
if (pattern.endsWith('/*')) {
|
|
35
|
+
const dir = pattern.slice(0, -2);
|
|
36
|
+
if (normalized.includes(`/${dir}/`) || normalized.startsWith(`${dir}/`)) return true;
|
|
37
|
+
}
|
|
38
|
+
if (normalized.includes(pattern)) return true;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export async function shouldStore(
|
|
45
|
+
content: string,
|
|
46
|
+
filePath?: string,
|
|
47
|
+
privacyMode: PrivacyMode = 'moderate'
|
|
48
|
+
): Promise<boolean> {
|
|
49
|
+
if (privacyMode === 'off') return true;
|
|
50
|
+
if (hasPrivateTags(content)) return false;
|
|
51
|
+
if (filePath && isInDenyList(filePath)) return false;
|
|
52
|
+
if (detectSecrets(content).length > 0) return false;
|
|
53
|
+
|
|
54
|
+
return true;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export async function applyPrivacyFilters<T extends { content?: string; details?: Record<string, unknown> }>(
|
|
58
|
+
items: T[]
|
|
59
|
+
): Promise<T[]> {
|
|
60
|
+
const results: T[] = [];
|
|
61
|
+
|
|
62
|
+
for (const item of items) {
|
|
63
|
+
const content = item.content || JSON.stringify(item.details || '');
|
|
64
|
+
if (await shouldStore(content)) {
|
|
65
|
+
results.push(item);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return results;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export function sanitizeForStorage(content: string, privacyMode: PrivacyMode): string {
|
|
73
|
+
let sanitized = stripPrivateTags(content);
|
|
74
|
+
|
|
75
|
+
if (privacyMode !== 'off') {
|
|
76
|
+
const secrets = detectSecrets(sanitized);
|
|
77
|
+
for (let i = secrets.length - 1; i >= 0; i--) {
|
|
78
|
+
const secret = secrets[i];
|
|
79
|
+
sanitized = sanitized.substring(0, secret.start) + '[REDACTED]' + sanitized.substring(secret.end);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
sanitized = sanitized
|
|
83
|
+
.replace(/([a-zA-Z0-9_-]*[Aa]pi[_-]?[Kk]ey|Bearer\s+[^\s]+)/g, '[API_KEY]')
|
|
84
|
+
.replace(/(token|jwt|bearer)[\s=:]+[^\s,]+/gi, '[TOKEN]');
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return sanitized;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
export interface PrivacyReport {
|
|
91
|
+
isPrivate: boolean;
|
|
92
|
+
hasSecrets: boolean;
|
|
93
|
+
secretCount: number;
|
|
94
|
+
secretTypes: string[];
|
|
95
|
+
shouldStore: boolean;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
export async function analyzePrivacy(content: string, filePath?: string): Promise<PrivacyReport> {
|
|
99
|
+
const secrets = detectSecrets(content);
|
|
100
|
+
|
|
101
|
+
return {
|
|
102
|
+
isPrivate: hasPrivateTags(content),
|
|
103
|
+
hasSecrets: secrets.length > 0,
|
|
104
|
+
secretCount: secrets.length,
|
|
105
|
+
secretTypes: [...new Set(secrets.map(s => s.type))],
|
|
106
|
+
shouldStore: await shouldStore(content, filePath)
|
|
107
|
+
};
|
|
108
|
+
}
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Secret Detector
|
|
3
|
+
* Pattern-based detection of API keys, tokens, and credentials
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export interface SecretMatch {
|
|
7
|
+
type: string;
|
|
8
|
+
pattern: string;
|
|
9
|
+
value: string;
|
|
10
|
+
start: number;
|
|
11
|
+
end: number;
|
|
12
|
+
confidence: 'high' | 'medium' | 'low';
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
type Confidence = SecretMatch['confidence'];
|
|
16
|
+
|
|
17
|
+
interface PatternConfig {
|
|
18
|
+
pattern: RegExp;
|
|
19
|
+
type: string;
|
|
20
|
+
confidence: Confidence;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const HIGH_CONFIDENCE_PATTERNS: PatternConfig[] = [
|
|
24
|
+
{ pattern: /(?:AKIA|ASIA)[0-9A-Z]{16}/g, type: 'aws_access_key', confidence: 'high' },
|
|
25
|
+
{ pattern: /aws_secret_access_key\s*=\s*[^\s]+/gi, type: 'aws_secret_key', confidence: 'high' },
|
|
26
|
+
{ pattern: /-----BEGIN (?:RSA|OPENSSH|PRIVATE) KEY-----/g, type: 'ssh_private_key', confidence: 'high' },
|
|
27
|
+
{ pattern: /-----BEGIN PGP PRIVATE KEY BLOCK-----/g, type: 'gpg_private_key', confidence: 'high' },
|
|
28
|
+
{ pattern: /AIza[0-9A-Za-z\-_]{35}/g, type: 'google_api_key', confidence: 'high' },
|
|
29
|
+
{ pattern: /gh[pousr]_[A-Za-z0-9_]{36,255}/g, type: 'github_token', confidence: 'high' },
|
|
30
|
+
{ pattern: /xox[baprs]-[0-9]{10,13}-[0-9]{10,13}-[A-Za-z0-9_-]{24,34}/g, type: 'slack_token', confidence: 'high' },
|
|
31
|
+
{ pattern: /sk_live_[0-9a-zA-Z]{20,}/g, type: 'stripe_api_key', confidence: 'high' },
|
|
32
|
+
{ pattern: /(mongodb|postgres|mysql|redis|cassandra):\/\/[^\s]+@[^\s]+/gi, type: 'database_url', confidence: 'high' }
|
|
33
|
+
];
|
|
34
|
+
|
|
35
|
+
const MEDIUM_CONFIDENCE_PATTERNS: PatternConfig[] = [
|
|
36
|
+
{ pattern: /(?:api[_-]?key|apikey|api_secret|secret_key)\s*[:=]\s*['""`]?[A-Za-z0-9_-]{20,}['""`]?/gi, type: 'api_key', confidence: 'medium' },
|
|
37
|
+
{ pattern: /bearer\s+[A-Za-z0-9._-]+/gi, type: 'bearer_token', confidence: 'medium' },
|
|
38
|
+
{ pattern: /eyJ[A-Za-z0-9_-]+\.eyJ[A-Za-z0-9_-]+\.[A-Za-z0-9_-]+/g, type: 'jwt_token', confidence: 'medium' },
|
|
39
|
+
{ pattern: /basic\s+[A-Za-z0-9+/=]+/gi, type: 'basic_auth', confidence: 'medium' },
|
|
40
|
+
{ pattern: /(?:password|passwd|pwd)\s*[:=]\s*['""`]?[^\s'""`]+['""`]?/gi, type: 'password', confidence: 'medium' },
|
|
41
|
+
{ pattern: /(?:connection_string|connstr|conn_string)\s*[:=]\s*['""`][^'""`]+['""`]/gi, type: 'connection_string', confidence: 'medium' }
|
|
42
|
+
];
|
|
43
|
+
|
|
44
|
+
const LOW_CONFIDENCE_PATTERNS: PatternConfig[] = [
|
|
45
|
+
{ pattern: /\b(?:token|auth_token|access_token|refresh_token)\b/gi, type: 'token_reference', confidence: 'low' },
|
|
46
|
+
{ pattern: /\b(?:secret|secret_key|api_secret|client_secret)\b/gi, type: 'secret_reference', confidence: 'low' },
|
|
47
|
+
{ pattern: /\b(?:credential|cred|credentials)\b/gi, type: 'credential_reference', confidence: 'low' }
|
|
48
|
+
];
|
|
49
|
+
|
|
50
|
+
const CONFIDENCE_LEVELS: Record<Confidence, number> = { high: 3, medium: 2, low: 1 };
|
|
51
|
+
|
|
52
|
+
function findMatches(text: string, config: PatternConfig): SecretMatch[] {
|
|
53
|
+
const regex = new RegExp(config.pattern.source, config.pattern.flags);
|
|
54
|
+
const matches: SecretMatch[] = [];
|
|
55
|
+
let match;
|
|
56
|
+
|
|
57
|
+
while ((match = regex.exec(text)) !== null) {
|
|
58
|
+
matches.push({
|
|
59
|
+
type: config.type,
|
|
60
|
+
pattern: config.pattern.source,
|
|
61
|
+
value: match[0],
|
|
62
|
+
start: match.index,
|
|
63
|
+
end: match.index + match[0].length,
|
|
64
|
+
confidence: config.confidence
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
return matches;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
export function detectSecrets(text: string, minConfidence: Confidence = 'medium'): SecretMatch[] {
|
|
72
|
+
const minLevel = CONFIDENCE_LEVELS[minConfidence];
|
|
73
|
+
const matches: SecretMatch[] = [];
|
|
74
|
+
|
|
75
|
+
for (const config of HIGH_CONFIDENCE_PATTERNS) {
|
|
76
|
+
matches.push(...findMatches(text, config));
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
if (minLevel <= 2) {
|
|
80
|
+
for (const config of MEDIUM_CONFIDENCE_PATTERNS) {
|
|
81
|
+
matches.push(...findMatches(text, config));
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
if (minLevel <= 1) {
|
|
86
|
+
for (const config of LOW_CONFIDENCE_PATTERNS) {
|
|
87
|
+
matches.push(...findMatches(text, config));
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
return matches;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
export function likelyContainsSecrets(text: string): boolean {
|
|
95
|
+
return detectSecrets(text, 'high').length > 0;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
export function getSecretSummary(text: string): { count: number; types: string[] } {
|
|
99
|
+
const secrets = detectSecrets(text);
|
|
100
|
+
return {
|
|
101
|
+
count: secrets.length,
|
|
102
|
+
types: [...new Set(secrets.map(s => s.type))]
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
export function redactSecrets(text: string, replacement = '[REDACTED]'): string {
|
|
107
|
+
const secrets = detectSecrets(text, 'high');
|
|
108
|
+
let result = text;
|
|
109
|
+
|
|
110
|
+
const sorted = [...secrets].sort((a, b) => b.start - a.start);
|
|
111
|
+
for (const secret of sorted) {
|
|
112
|
+
result = result.substring(0, secret.start) + replacement + result.substring(secret.end);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
return result;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
export const SECRET_PATTERNS = {
|
|
119
|
+
high: HIGH_CONFIDENCE_PATTERNS,
|
|
120
|
+
medium: MEDIUM_CONFIDENCE_PATTERNS,
|
|
121
|
+
low: LOW_CONFIDENCE_PATTERNS
|
|
122
|
+
};
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
/** Session Auto-Load - Automatically load context when MCP session initializes */
|
|
2
|
+
|
|
3
|
+
import { logger } from '../logger.js';
|
|
4
|
+
import { config } from '../../config.js';
|
|
5
|
+
import { initializeCoreMemory, getCoreMemory } from '../ingestion/core-memory.js';
|
|
6
|
+
import { search } from '../memory/memories.js';
|
|
7
|
+
import { getProjectContext } from '../context/context.js';
|
|
8
|
+
import { getOrCreateProject } from '../projects.js';
|
|
9
|
+
import { AutoLoadConfig, AutoLoadResult, DEFAULT_AUTO_LOAD_CONFIG } from './types.js';
|
|
10
|
+
import { estimateTokens } from '../context/context-window.js';
|
|
11
|
+
import { getLatestProjectWorkingSetSummary } from './working-set.js';
|
|
12
|
+
import { loadHotCache, getHotCacheSummary } from '../hot-cache.js';
|
|
13
|
+
|
|
14
|
+
export async function performAutoLoad(
|
|
15
|
+
projectPath: string,
|
|
16
|
+
customConfig?: Partial<AutoLoadConfig>
|
|
17
|
+
): Promise<AutoLoadResult> {
|
|
18
|
+
const startTime = Date.now();
|
|
19
|
+
const cfg: AutoLoadConfig = { ...DEFAULT_AUTO_LOAD_CONFIG, ...customConfig };
|
|
20
|
+
const result: AutoLoadResult = {
|
|
21
|
+
coreMemoryLoaded: false,
|
|
22
|
+
memoriesLoaded: 0,
|
|
23
|
+
projectContextLoaded: false,
|
|
24
|
+
tokensUsed: 0,
|
|
25
|
+
duration: 0,
|
|
26
|
+
warnings: [],
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
if (!cfg.enabled) {
|
|
30
|
+
result.warnings.push('Auto-load disabled by configuration');
|
|
31
|
+
result.duration = Date.now() - startTime;
|
|
32
|
+
return result;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
try {
|
|
36
|
+
const project = await getOrCreateProject(projectPath);
|
|
37
|
+
if (!project) {
|
|
38
|
+
result.warnings.push(`Failed to create/find project: ${projectPath}`);
|
|
39
|
+
result.duration = Date.now() - startTime;
|
|
40
|
+
return result;
|
|
41
|
+
}
|
|
42
|
+
const projectId = project.id;
|
|
43
|
+
|
|
44
|
+
try {
|
|
45
|
+
const workingSetSummary = await getLatestProjectWorkingSetSummary(projectPath);
|
|
46
|
+
if (workingSetSummary) {
|
|
47
|
+
result.tokensUsed += estimateTokens(workingSetSummary);
|
|
48
|
+
}
|
|
49
|
+
} catch (error) {
|
|
50
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
51
|
+
result.warnings.push(`Failed to load session working set: ${msg}`);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// NEW: Load persistent hot cache (Karpathy-style, survives restart)
|
|
55
|
+
if (cfg.includeCoreMemory) {
|
|
56
|
+
try {
|
|
57
|
+
const hotCacheSummary = await getHotCacheSummary({ projectPath, maxEntries: 5 });
|
|
58
|
+
if (hotCacheSummary) {
|
|
59
|
+
result.tokensUsed += estimateTokens(hotCacheSummary);
|
|
60
|
+
logger.info('[AutoLoad] Hot cache loaded from persistent storage');
|
|
61
|
+
}
|
|
62
|
+
} catch (error) {
|
|
63
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
64
|
+
result.warnings.push(`Failed to load hot cache: ${msg}`);
|
|
65
|
+
logger.warn(`[AutoLoad] Failed to load hot cache: ${msg}`);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
if (cfg.includeCoreMemory) {
|
|
70
|
+
try {
|
|
71
|
+
await initializeCoreMemory(projectId);
|
|
72
|
+
const coreMemory = await getCoreMemory(projectId);
|
|
73
|
+
if (coreMemory) {
|
|
74
|
+
result.coreMemoryLoaded = true;
|
|
75
|
+
const coreMemoryText = Object.values(coreMemory).join('\n');
|
|
76
|
+
result.tokensUsed += estimateTokens(coreMemoryText);
|
|
77
|
+
logger.info(`[AutoLoad] Core memory loaded for project ${projectId}`);
|
|
78
|
+
}
|
|
79
|
+
} catch (error) {
|
|
80
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
81
|
+
result.warnings.push(`Failed to load core memory: ${msg}`);
|
|
82
|
+
logger.warn(`[AutoLoad] Failed to load core memory: ${msg}`);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
if (cfg.includeRecentMemories && cfg.recentMemoryCount > 0) {
|
|
87
|
+
try {
|
|
88
|
+
const recentMemories = await search({
|
|
89
|
+
query: '',
|
|
90
|
+
project: projectPath,
|
|
91
|
+
limit: cfg.recentMemoryCount,
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
if (recentMemories && recentMemories.length > 0) {
|
|
95
|
+
const importantMemories = recentMemories.filter(
|
|
96
|
+
(m: any) => (m.importanceScore ?? 50) >= cfg.importanceThreshold
|
|
97
|
+
);
|
|
98
|
+
|
|
99
|
+
result.memoriesLoaded = importantMemories.length;
|
|
100
|
+
|
|
101
|
+
for (const memory of importantMemories) {
|
|
102
|
+
result.tokensUsed += estimateTokens(memory.content || '');
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
logger.info(`[AutoLoad] Loaded ${result.memoriesLoaded} high-importance memories`);
|
|
106
|
+
}
|
|
107
|
+
} catch (error) {
|
|
108
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
109
|
+
result.warnings.push(`Failed to load recent memories: ${msg}`);
|
|
110
|
+
logger.warn(`[AutoLoad] Failed to load recent memories: ${msg}`);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
if (cfg.includeProjectContext) {
|
|
115
|
+
try {
|
|
116
|
+
const context = await getProjectContext({
|
|
117
|
+
project: projectPath,
|
|
118
|
+
include: ['memories', 'observations'],
|
|
119
|
+
limit: 5,
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
if (context) {
|
|
123
|
+
result.projectContextLoaded = true;
|
|
124
|
+
result.tokensUsed += estimateTokens(JSON.stringify(context));
|
|
125
|
+
logger.info('[AutoLoad] Project context loaded');
|
|
126
|
+
}
|
|
127
|
+
} catch (error) {
|
|
128
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
129
|
+
result.warnings.push(`Failed to load project context: ${msg}`);
|
|
130
|
+
logger.warn(`[AutoLoad] Failed to load project context: ${msg}`);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
result.duration = Date.now() - startTime;
|
|
135
|
+
logger.info(`[AutoLoad] Complete: ${result.memoriesLoaded} memories, ${result.tokensUsed} tokens, ${result.duration}ms`);
|
|
136
|
+
|
|
137
|
+
return result;
|
|
138
|
+
} catch (error) {
|
|
139
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
140
|
+
result.warnings.push(`Auto-load failed: ${msg}`);
|
|
141
|
+
result.duration = Date.now() - startTime;
|
|
142
|
+
logger.error(`[AutoLoad] Failed: ${msg}`);
|
|
143
|
+
return result;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
export function getAutoLoadConfig(): AutoLoadConfig {
|
|
148
|
+
return {
|
|
149
|
+
enabled: config.sessionAutoLoadEnabled,
|
|
150
|
+
includeCoreMemory: true,
|
|
151
|
+
includeRecentMemories: true,
|
|
152
|
+
recentMemoryCount: config.sessionAutoLoadRecentCount,
|
|
153
|
+
importanceThreshold: config.sessionAutoLoadImportanceThreshold,
|
|
154
|
+
includeProjectContext: true,
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
export function shouldAutoLoad(): boolean {
|
|
159
|
+
return config.sessionAutoLoadEnabled;
|
|
160
|
+
}
|