squish-memory 1.1.5 → 1.2.1
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 +44 -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/{dist/core/algorithms/index.js → core/algorithms/index.ts} +39 -26
- 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/{dist/core/lib/db-client.d.ts → core/lib/db-client.ts} +168 -114
- package/core/lib/parse-embedding.ts +59 -0
- package/{dist/core/lib/schemas.js → core/lib/schemas.ts} +102 -87
- 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/{dist/core/memory/index.js → core/memory/index.ts} +11 -10
- 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/{dist/core/places/index.js → core/places/index.ts} +12 -12
- 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 +590 -0
- package/core/scheduler/heartbeat.ts +91 -0
- package/{dist/core/scheduler/index.js → core/scheduler/index.ts} +8 -8
- 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/{dist/core/search/index.js → core/search/index.ts} +4 -5
- 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/{dist/core/session/index.js → core/session/index.ts} +7 -7
- 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/{dist/core/snapshots/cleanup.js → core/snapshots/cleanup.ts} +13 -12
- 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/{dist/core/summarization/cleanup.js → core/summarization/cleanup.ts} +13 -12
- 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/{dist/core/utils/vector-operations.js → core/utils/vector-operations.ts} +135 -129
- package/core/utils/version-management.ts +74 -0
- package/core/worker.ts +333 -0
- package/db/adapter.ts +215 -0
- package/{dist/db/bootstrap.js → db/bootstrap.ts} +388 -418
- 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/mcp.json.example +8 -11
- package/package.json +140 -159
- 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 +940 -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/config.d.ts +0 -83
- package/dist/config.js +0 -242
- package/dist/core/adapters/config/claude-code.d.ts +0 -45
- package/dist/core/adapters/config/claude-code.js +0 -113
- package/dist/core/adapters/config/cursor.d.ts +0 -26
- package/dist/core/adapters/config/cursor.js +0 -74
- package/dist/core/adapters/config/opencode.d.ts +0 -23
- package/dist/core/adapters/config/opencode.js +0 -73
- package/dist/core/adapters/config/windsurf.d.ts +0 -26
- package/dist/core/adapters/config/windsurf.js +0 -74
- package/dist/core/adapters/index.d.ts +0 -45
- package/dist/core/adapters/index.js +0 -84
- package/dist/core/adapters/scripts/install-adapter.d.ts +0 -19
- package/dist/core/adapters/scripts/install-adapter.js +0 -149
- package/dist/core/adapters/timeline.d.ts +0 -23
- package/dist/core/adapters/timeline.js +0 -88
- package/dist/core/adapters/types.d.ts +0 -157
- package/dist/core/adapters/types.js +0 -50
- package/dist/core/algorithms/analytics/token-estimator.d.ts +0 -50
- package/dist/core/algorithms/analytics/token-estimator.js +0 -154
- package/dist/core/algorithms/detection/hash-filters.d.ts +0 -47
- package/dist/core/algorithms/detection/hash-filters.js +0 -190
- package/dist/core/algorithms/detection/semantic-ranker.d.ts +0 -32
- package/dist/core/algorithms/detection/semantic-ranker.js +0 -118
- package/dist/core/algorithms/detection/two-stage-detector.d.ts +0 -52
- package/dist/core/algorithms/detection/two-stage-detector.js +0 -299
- package/dist/core/algorithms/handlers/approve-merge.d.ts +0 -22
- package/dist/core/algorithms/handlers/approve-merge.js +0 -179
- package/dist/core/algorithms/handlers/detect-duplicates.d.ts +0 -47
- package/dist/core/algorithms/handlers/detect-duplicates.js +0 -145
- package/dist/core/algorithms/handlers/get-stats.d.ts +0 -39
- package/dist/core/algorithms/handlers/get-stats.js +0 -88
- package/dist/core/algorithms/handlers/list-proposals.d.ts +0 -45
- package/dist/core/algorithms/handlers/list-proposals.js +0 -83
- package/dist/core/algorithms/handlers/preview-merge.d.ts +0 -39
- package/dist/core/algorithms/handlers/preview-merge.js +0 -93
- package/dist/core/algorithms/handlers/reject-merge.d.ts +0 -28
- package/dist/core/algorithms/handlers/reject-merge.js +0 -69
- package/dist/core/algorithms/handlers/reverse-merge.d.ts +0 -21
- package/dist/core/algorithms/handlers/reverse-merge.js +0 -121
- package/dist/core/algorithms/index.d.ts +0 -21
- package/dist/core/algorithms/operations/cache-maintenance.d.ts +0 -12
- package/dist/core/algorithms/operations/cache-maintenance.js +0 -157
- package/dist/core/algorithms/safety/safety-checks.d.ts +0 -22
- package/dist/core/algorithms/safety/safety-checks.js +0 -179
- package/dist/core/algorithms/strategies/merge-strategies.d.ts +0 -50
- package/dist/core/algorithms/strategies/merge-strategies.js +0 -288
- package/dist/core/algorithms/types.d.ts +0 -133
- package/dist/core/algorithms/types.js +0 -5
- package/dist/core/algorithms/utils/response-builder.d.ts +0 -28
- package/dist/core/algorithms/utils/response-builder.js +0 -37
- package/dist/core/associations.d.ts +0 -31
- package/dist/core/associations.js +0 -248
- package/dist/core/autosave.d.ts +0 -19
- package/dist/core/autosave.js +0 -16
- package/dist/core/commands/managed-sync.d.ts +0 -10
- package/dist/core/commands/managed-sync.js +0 -64
- package/dist/core/commands/mcp-server.d.ts +0 -3
- package/dist/core/commands/mcp-server.js +0 -739
- package/dist/core/consolidation.d.ts +0 -37
- package/dist/core/consolidation.js +0 -248
- package/dist/core/context/agent-context.d.ts +0 -106
- package/dist/core/context/agent-context.js +0 -274
- package/dist/core/context/context-paging.d.ts +0 -80
- package/dist/core/context/context-paging.js +0 -328
- package/dist/core/context/context-window.d.ts +0 -40
- package/dist/core/context/context-window.js +0 -177
- package/dist/core/context/context.d.ts +0 -7
- package/dist/core/context/context.js +0 -22
- package/dist/core/embeddings/google-multimodal.d.ts +0 -14
- package/dist/core/embeddings/google-multimodal.js +0 -142
- package/dist/core/embeddings/qmd-client.d.ts +0 -136
- package/dist/core/embeddings/qmd-client.js +0 -403
- package/dist/core/embeddings.d.ts +0 -29
- package/dist/core/embeddings.js +0 -454
- package/dist/core/error-handling.d.ts +0 -63
- package/dist/core/error-handling.js +0 -173
- package/dist/core/external-folder/index.d.ts +0 -102
- package/dist/core/external-folder/index.js +0 -294
- package/dist/core/hooks/agent-hooks.d.ts +0 -74
- package/dist/core/hooks/agent-hooks.js +0 -244
- package/dist/core/hooks/auto-tagger.d.ts +0 -19
- package/dist/core/hooks/auto-tagger.js +0 -155
- package/dist/core/hooks/capture-filter.d.ts +0 -41
- package/dist/core/hooks/capture-filter.js +0 -128
- package/dist/core/index.d.ts +0 -10
- package/dist/core/index.js +0 -14
- package/dist/core/ingestion/agent-memory.d.ts +0 -22
- package/dist/core/ingestion/agent-memory.js +0 -109
- package/dist/core/ingestion/core-memory.d.ts +0 -78
- package/dist/core/ingestion/core-memory.js +0 -226
- package/dist/core/ingestion/learnings.d.ts +0 -57
- package/dist/core/ingestion/learnings.js +0 -202
- package/dist/core/layers/generator.d.ts +0 -25
- package/dist/core/layers/generator.js +0 -76
- package/dist/core/lib/db-client.js +0 -130
- package/dist/core/lib/schemas.d.ts +0 -129
- package/dist/core/lib/utils.d.ts +0 -14
- package/dist/core/lib/utils.js +0 -90
- package/dist/core/lib/validation.d.ts +0 -38
- package/dist/core/lib/validation.js +0 -151
- package/dist/core/lifecycle.d.ts +0 -26
- package/dist/core/lifecycle.js +0 -302
- package/dist/core/local-embeddings.d.ts +0 -11
- package/dist/core/logger.d.ts +0 -16
- package/dist/core/logger.js +0 -40
- package/dist/core/mcp/client.d.ts +0 -17
- package/dist/core/mcp/client.js +0 -101
- package/dist/core/mcp/index.d.ts +0 -6
- package/dist/core/mcp/index.js +0 -6
- package/dist/core/mcp/server.d.ts +0 -18
- package/dist/core/mcp/server.js +0 -157
- package/dist/core/mcp/standalone-server.d.ts +0 -13
- package/dist/core/mcp/standalone-server.js +0 -46
- package/dist/core/mcp/tools.d.ts +0 -9
- package/dist/core/mcp/tools.js +0 -365
- package/dist/core/mcp/types.d.ts +0 -315
- package/dist/core/mcp/types.js +0 -48
- package/dist/core/memory/bridge-discovery.d.ts +0 -50
- package/dist/core/memory/bridge-discovery.js +0 -291
- package/dist/core/memory/categorizer.d.ts +0 -27
- package/dist/core/memory/categorizer.js +0 -305
- package/dist/core/memory/conflict-detector.d.ts +0 -7
- package/dist/core/memory/conflict-detector.js +0 -43
- package/dist/core/memory/consolidation.d.ts +0 -42
- package/dist/core/memory/consolidation.js +0 -303
- package/dist/core/memory/context-collector.d.ts +0 -10
- package/dist/core/memory/context-collector.js +0 -56
- package/dist/core/memory/contradiction-resolver.d.ts +0 -40
- package/dist/core/memory/contradiction-resolver.js +0 -368
- package/dist/core/memory/edit-workflow.d.ts +0 -19
- package/dist/core/memory/edit-workflow.js +0 -120
- package/dist/core/memory/entity-extractor.d.ts +0 -33
- package/dist/core/memory/entity-extractor.js +0 -336
- package/dist/core/memory/entity-resolver.d.ts +0 -23
- package/dist/core/memory/entity-resolver.js +0 -64
- package/dist/core/memory/fact-extractor.d.ts +0 -24
- package/dist/core/memory/fact-extractor.js +0 -89
- package/dist/core/memory/feedback-tracker.d.ts +0 -12
- package/dist/core/memory/feedback-tracker.js +0 -155
- package/dist/core/memory/hooks.d.ts +0 -88
- package/dist/core/memory/hooks.js +0 -174
- package/dist/core/memory/hybrid-retrieval.d.ts +0 -29
- package/dist/core/memory/hybrid-retrieval.js +0 -139
- package/dist/core/memory/hybrid-scorer.d.ts +0 -40
- package/dist/core/memory/hybrid-scorer.js +0 -284
- package/dist/core/memory/hybrid-search.d.ts +0 -20
- package/dist/core/memory/hybrid-search.js +0 -359
- package/dist/core/memory/importance.d.ts +0 -63
- package/dist/core/memory/importance.js +0 -298
- package/dist/core/memory/index.d.ts +0 -8
- package/dist/core/memory/loader.d.ts +0 -31
- package/dist/core/memory/loader.js +0 -141
- package/dist/core/memory/markdown/markdown-storage.d.ts +0 -72
- package/dist/core/memory/markdown/markdown-storage.js +0 -243
- package/dist/core/memory/memories.d.ts +0 -47
- package/dist/core/memory/memories.js +0 -449
- package/dist/core/memory/memory-lifecycle.d.ts +0 -8
- package/dist/core/memory/memory-lifecycle.js +0 -55
- package/dist/core/memory/memory-manager.d.ts +0 -15
- package/dist/core/memory/memory-manager.js +0 -46
- package/dist/core/memory/migrate.d.ts +0 -21
- package/dist/core/memory/migrate.js +0 -134
- package/dist/core/memory/normalization.d.ts +0 -22
- package/dist/core/memory/normalization.js +0 -26
- package/dist/core/memory/progressive-disclosure.d.ts +0 -43
- package/dist/core/memory/progressive-disclosure.js +0 -280
- package/dist/core/memory/query-processor.d.ts +0 -21
- package/dist/core/memory/query-processor.js +0 -72
- package/dist/core/memory/query-rewriter.d.ts +0 -13
- package/dist/core/memory/query-rewriter.js +0 -118
- package/dist/core/memory/response-analyzer.d.ts +0 -9
- package/dist/core/memory/response-analyzer.js +0 -61
- package/dist/core/memory/serialization.d.ts +0 -10
- package/dist/core/memory/serialization.js +0 -84
- package/dist/core/memory/stats.d.ts +0 -22
- package/dist/core/memory/stats.js +0 -138
- package/dist/core/memory/telemetry.d.ts +0 -69
- package/dist/core/memory/telemetry.js +0 -313
- package/dist/core/memory/temporal-facts.d.ts +0 -41
- package/dist/core/memory/temporal-facts.js +0 -283
- package/dist/core/memory/temporal-parser.d.ts +0 -32
- package/dist/core/memory/temporal-parser.js +0 -385
- package/dist/core/memory/trigger-detector.d.ts +0 -14
- package/dist/core/memory/trigger-detector.js +0 -42
- package/dist/core/memory/write-gate.d.ts +0 -54
- package/dist/core/memory/write-gate.js +0 -210
- 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/obsidian-vault.d.ts +0 -30
- package/dist/core/obsidian-vault.js +0 -94
- package/dist/core/places/index.d.ts +0 -14
- package/dist/core/places/memory-places.d.ts +0 -68
- package/dist/core/places/memory-places.js +0 -261
- package/dist/core/places/places.d.ts +0 -88
- package/dist/core/places/places.js +0 -314
- package/dist/core/places/rules.d.ts +0 -74
- package/dist/core/places/rules.js +0 -240
- package/dist/core/places/walking.d.ts +0 -56
- package/dist/core/places/walking.js +0 -121
- package/dist/core/projects.d.ts +0 -17
- package/dist/core/projects.js +0 -108
- package/dist/core/redis.d.ts +0 -11
- package/dist/core/redis.js +0 -69
- package/dist/core/responses.d.ts +0 -96
- package/dist/core/responses.js +0 -122
- package/dist/core/scheduler/cron-scheduler.d.ts +0 -32
- package/dist/core/scheduler/cron-scheduler.js +0 -332
- package/dist/core/scheduler/heartbeat.d.ts +0 -11
- package/dist/core/scheduler/heartbeat.js +0 -73
- package/dist/core/scheduler/index.d.ts +0 -8
- package/dist/core/scheduler/job-runner.d.ts +0 -11
- package/dist/core/scheduler/job-runner.js +0 -164
- package/dist/core/search/conversations.d.ts +0 -25
- package/dist/core/search/conversations.js +0 -110
- package/dist/core/search/entities.d.ts +0 -12
- package/dist/core/search/entities.js +0 -31
- package/dist/core/search/folder-context.d.ts +0 -25
- package/dist/core/search/folder-context.js +0 -119
- package/dist/core/search/graph-boost.d.ts +0 -7
- package/dist/core/search/graph-boost.js +0 -23
- package/dist/core/search/index.d.ts +0 -4
- package/dist/core/search/qmd-search.d.ts +0 -61
- package/dist/core/search/qmd-search.js +0 -178
- package/dist/core/security/encrypt.d.ts +0 -6
- package/dist/core/security/encrypt.js +0 -47
- package/dist/core/security/governance.d.ts +0 -26
- package/dist/core/security/governance.js +0 -79
- package/dist/core/security/privacy.d.ts +0 -23
- package/dist/core/security/privacy.js +0 -82
- package/dist/core/security/secret-detector.d.ts +0 -32
- package/dist/core/security/secret-detector.js +0 -88
- package/dist/core/session/auto-load.d.ts +0 -6
- package/dist/core/session/auto-load.js +0 -119
- package/dist/core/session/index.d.ts +0 -7
- package/dist/core/session/self-iteration-job.d.ts +0 -20
- package/dist/core/session/self-iteration-job.js +0 -282
- package/dist/core/session/session-hooks.d.ts +0 -18
- package/dist/core/session/session-hooks.js +0 -58
- package/dist/core/session/types.d.ts +0 -26
- package/dist/core/session/types.js +0 -10
- 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/cleanup.d.ts +0 -9
- package/dist/core/snapshots/comparison.d.ts +0 -19
- package/dist/core/snapshots/comparison.js +0 -43
- package/dist/core/snapshots/creation.d.ts +0 -19
- package/dist/core/snapshots/creation.js +0 -126
- package/dist/core/snapshots/retrieval.d.ts +0 -7
- package/dist/core/snapshots/retrieval.js +0 -41
- package/dist/core/snapshots/stats.d.ts +0 -11
- package/dist/core/snapshots/stats.js +0 -52
- package/dist/core/snapshots.d.ts +0 -29
- package/dist/core/snapshots.js +0 -220
- package/dist/core/storage/cache.d.ts +0 -13
- package/dist/core/storage/cache.js +0 -202
- package/dist/core/storage/database.d.ts +0 -12
- package/dist/core/storage/database.js +0 -12
- package/dist/core/summarization/cleanup.d.ts +0 -9
- package/dist/core/summarization/queries.d.ts +0 -9
- package/dist/core/summarization/queries.js +0 -28
- package/dist/core/summarization/stats.d.ts +0 -14
- package/dist/core/summarization/stats.js +0 -52
- package/dist/core/summarization/strategies.d.ts +0 -24
- package/dist/core/summarization/strategies.js +0 -28
- package/dist/core/summarization.d.ts +0 -37
- package/dist/core/summarization.js +0 -188
- package/dist/core/sync/qmd-sync.d.ts +0 -94
- package/dist/core/sync/qmd-sync.js +0 -201
- package/dist/core/temporal-facts.d.ts +0 -54
- package/dist/core/temporal-facts.js +0 -193
- package/dist/core/toon.d.ts +0 -43
- package/dist/core/toon.js +0 -160
- package/dist/core/tracing/collector.d.ts +0 -111
- package/dist/core/tracing/collector.js +0 -350
- package/dist/core/tracing/visualizer.d.ts +0 -32
- package/dist/core/tracing/visualizer.js +0 -165
- package/dist/core/utils/cleanup-operations.d.ts +0 -13
- package/dist/core/utils/cleanup-operations.js +0 -44
- package/dist/core/utils/content-extraction.d.ts +0 -19
- package/dist/core/utils/content-extraction.js +0 -75
- package/dist/core/utils/filter-builder.d.ts +0 -13
- package/dist/core/utils/filter-builder.js +0 -44
- package/dist/core/utils/history-traversal.d.ts +0 -13
- package/dist/core/utils/history-traversal.js +0 -50
- package/dist/core/utils/memory-operations.d.ts +0 -17
- package/dist/core/utils/memory-operations.js +0 -43
- package/dist/core/utils/query-operations.d.ts +0 -18
- package/dist/core/utils/query-operations.js +0 -65
- package/dist/core/utils/summarization-helpers.d.ts +0 -21
- package/dist/core/utils/summarization-helpers.js +0 -38
- package/dist/core/utils/temporal-queries.d.ts +0 -13
- package/dist/core/utils/temporal-queries.js +0 -27
- package/dist/core/utils/vector-operations.d.ts +0 -71
- package/dist/core/utils/version-management.d.ts +0 -9
- package/dist/core/utils/version-management.js +0 -61
- package/dist/core/worker.d.ts +0 -82
- package/dist/core/worker.js +0 -272
- package/dist/db/adapter.d.ts +0 -7
- package/dist/db/adapter.js +0 -175
- package/dist/db/bootstrap.d.ts +0 -9
- package/dist/db/drizzle/schema-sqlite.d.ts +0 -4837
- package/dist/db/drizzle/schema-sqlite.js +0 -684
- package/dist/db/drizzle/schema.d.ts +0 -4082
- package/dist/db/drizzle/schema.js +0 -770
- package/dist/db/drizzle.config.d.ts +0 -3
- package/dist/db/drizzle.config.js +0 -12
- package/dist/db/index.d.ts +0 -7
- package/dist/db/index.js +0 -89
- package/dist/db/neon.d.ts +0 -8
- package/dist/db/neon.js +0 -20
- package/dist/db/schema/index.d.ts +0 -40
- package/dist/db/schema/index.js +0 -105
- package/dist/db/schema/tables/context-sessions.d.ts +0 -9
- package/dist/db/schema/tables/context-sessions.js +0 -37
- package/dist/db/schema/tables/conversations.d.ts +0 -9
- package/dist/db/schema/tables/conversations.js +0 -47
- package/dist/db/schema/tables/core-memory.d.ts +0 -9
- package/dist/db/schema/tables/core-memory.js +0 -41
- package/dist/db/schema/tables/entities.d.ts +0 -9
- package/dist/db/schema/tables/entities.js +0 -39
- package/dist/db/schema/tables/entity-relations.d.ts +0 -9
- package/dist/db/schema/tables/entity-relations.js +0 -31
- package/dist/db/schema/tables/learnings.d.ts +0 -9
- package/dist/db/schema/tables/learnings.js +0 -66
- package/dist/db/schema/tables/memories.d.ts +0 -9
- package/dist/db/schema/tables/memories.js +0 -161
- package/dist/db/schema/tables/memory-associations.d.ts +0 -9
- package/dist/db/schema/tables/memory-associations.js +0 -39
- package/dist/db/schema/tables/memory-hash-cache.d.ts +0 -9
- package/dist/db/schema/tables/memory-hash-cache.js +0 -29
- package/dist/db/schema/tables/memory-merge-history.d.ts +0 -9
- package/dist/db/schema/tables/memory-merge-history.js +0 -33
- package/dist/db/schema/tables/memory-merge-proposals.d.ts +0 -9
- package/dist/db/schema/tables/memory-merge-proposals.js +0 -39
- package/dist/db/schema/tables/messages.d.ts +0 -9
- package/dist/db/schema/tables/messages.js +0 -41
- package/dist/db/schema/tables/namespaces.d.ts +0 -9
- package/dist/db/schema/tables/namespaces.js +0 -37
- package/dist/db/schema/tables/projects.d.ts +0 -9
- package/dist/db/schema/tables/projects.js +0 -31
- package/dist/db/schema/tables/users.d.ts +0 -9
- package/dist/db/schema/tables/users.js +0 -27
- package/dist/db/schema.d.ts +0 -3
- package/dist/db/schema.js +0 -11
- package/dist/db/supabase.d.ts +0 -9
- package/dist/db/supabase.js +0 -24
- 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,288 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Write Gate Enforcement
|
|
3
|
+
* Validates memories before writing to ensure quality, security, and consistency
|
|
4
|
+
* Integrates secret detection, trigger detection, and content validation
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { detectSecrets, redactSecrets, SecretMatch } from '../security/secret-detector.js';
|
|
8
|
+
import { detectMemorySignals, MemorySignals } from './trigger-detector.js';
|
|
9
|
+
import { resolveContradictions } from './contradiction-resolver.js';
|
|
10
|
+
import { supersedeOldTemporalFacts } from './temporal-facts.js';
|
|
11
|
+
import { logger } from '../logger.js';
|
|
12
|
+
|
|
13
|
+
export interface WriteGateResult {
|
|
14
|
+
allowed: boolean;
|
|
15
|
+
sanitized: boolean;
|
|
16
|
+
warnings: string[];
|
|
17
|
+
errors: string[];
|
|
18
|
+
metadata: {
|
|
19
|
+
secretsDetected: number;
|
|
20
|
+
signals: MemorySignals | null;
|
|
21
|
+
contradictions: {
|
|
22
|
+
found: boolean;
|
|
23
|
+
count: number;
|
|
24
|
+
};
|
|
25
|
+
temporalSupersession: {
|
|
26
|
+
count: number;
|
|
27
|
+
};
|
|
28
|
+
};
|
|
29
|
+
sanitizedContent?: string;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export interface WriteGateOptions {
|
|
33
|
+
allowSecrets?: boolean; // If true, redact but allow write
|
|
34
|
+
minContentLength?: number;
|
|
35
|
+
maxContentLength?: number;
|
|
36
|
+
projectId?: string;
|
|
37
|
+
skipContradictionCheck?: boolean;
|
|
38
|
+
skipTemporalSupersession?: boolean;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const DEFAULT_OPTIONS: WriteGateOptions = {
|
|
42
|
+
allowSecrets: false,
|
|
43
|
+
minContentLength: 5,
|
|
44
|
+
maxContentLength: 50000,
|
|
45
|
+
skipContradictionCheck: false,
|
|
46
|
+
skipTemporalSupersession: false,
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Main write gate function - validates and sanitizes content before write
|
|
51
|
+
*/
|
|
52
|
+
export async function enforceWriteGate(
|
|
53
|
+
content: string,
|
|
54
|
+
type: string,
|
|
55
|
+
options: WriteGateOptions = {}
|
|
56
|
+
): Promise<WriteGateResult> {
|
|
57
|
+
const opts = { ...DEFAULT_OPTIONS, ...options };
|
|
58
|
+
const result: WriteGateResult = {
|
|
59
|
+
allowed: true,
|
|
60
|
+
sanitized: false,
|
|
61
|
+
warnings: [],
|
|
62
|
+
errors: [],
|
|
63
|
+
metadata: {
|
|
64
|
+
secretsDetected: 0,
|
|
65
|
+
signals: null,
|
|
66
|
+
contradictions: { found: false, count: 0 },
|
|
67
|
+
temporalSupersession: { count: 0 },
|
|
68
|
+
},
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
let processedContent = content;
|
|
72
|
+
|
|
73
|
+
// 1. Content length validation
|
|
74
|
+
if (content.length < opts.minContentLength!) {
|
|
75
|
+
result.errors.push(`Content too short (min ${opts.minContentLength} characters)`);
|
|
76
|
+
result.allowed = false;
|
|
77
|
+
return result;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if (content.length > opts.maxContentLength!) {
|
|
81
|
+
result.errors.push(`Content too long (max ${opts.maxContentLength} characters)`);
|
|
82
|
+
result.allowed = false;
|
|
83
|
+
return result;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// 2. Secret detection
|
|
87
|
+
const secrets = detectSecrets(content, 'high');
|
|
88
|
+
result.metadata.secretsDetected = secrets.length;
|
|
89
|
+
|
|
90
|
+
if (secrets.length > 0) {
|
|
91
|
+
const secretTypes = [...new Set(secrets.map(s => s.type))];
|
|
92
|
+
|
|
93
|
+
if (!opts.allowSecrets) {
|
|
94
|
+
result.errors.push(`Potential secrets detected: ${secretTypes.join(', ')}`);
|
|
95
|
+
result.warnings.push('Content contains sensitive data that must be redacted');
|
|
96
|
+
result.allowed = false;
|
|
97
|
+
return result;
|
|
98
|
+
} else {
|
|
99
|
+
// Redact secrets but allow write
|
|
100
|
+
processedContent = redactSecrets(content);
|
|
101
|
+
result.sanitized = true;
|
|
102
|
+
result.sanitizedContent = processedContent;
|
|
103
|
+
result.warnings.push(`Redacted ${secrets.length} potential secrets: ${secretTypes.join(', ')}`);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// 3. Memory signals detection
|
|
108
|
+
const signals = detectMemorySignals(content);
|
|
109
|
+
result.metadata.signals = signals;
|
|
110
|
+
|
|
111
|
+
// Add warnings for high-priority signals
|
|
112
|
+
if (signals.priority === 'high') {
|
|
113
|
+
result.warnings.push('High-priority memory signal detected');
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
if (signals.implicit.correction) {
|
|
117
|
+
result.warnings.push('Content appears to be a correction - contradiction check recommended');
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
// 4. Contradiction detection (async, non-blocking for write)
|
|
121
|
+
if (!opts.skipContradictionCheck && signals.implicit.correction) {
|
|
122
|
+
try {
|
|
123
|
+
const contradictionResult = await resolveContradictions(
|
|
124
|
+
processedContent,
|
|
125
|
+
type,
|
|
126
|
+
opts.projectId
|
|
127
|
+
);
|
|
128
|
+
|
|
129
|
+
result.metadata.contradictions = {
|
|
130
|
+
found: contradictionResult.supersededIds.length > 0,
|
|
131
|
+
count: contradictionResult.supersededIds.length,
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
if (contradictionResult.supersededIds.length > 0) {
|
|
135
|
+
result.warnings.push(
|
|
136
|
+
`This memory supersedes ${contradictionResult.supersededIds.length} older memories`
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
} catch (error) {
|
|
140
|
+
logger.error('Contradiction check failed', error);
|
|
141
|
+
result.warnings.push('Contradiction check failed - proceeding with write');
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
// 5. Temporal fact handling
|
|
146
|
+
if (!opts.skipTemporalSupersession) {
|
|
147
|
+
try {
|
|
148
|
+
// This would be done after the memory is actually written, but we note it here
|
|
149
|
+
result.metadata.temporalSupersession = { count: 0 };
|
|
150
|
+
} catch (error) {
|
|
151
|
+
logger.error('Temporal check failed', error);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
// 6. Content quality checks
|
|
156
|
+
const qualityIssues = checkContentQuality(processedContent);
|
|
157
|
+
result.warnings.push(...qualityIssues);
|
|
158
|
+
|
|
159
|
+
return result;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/**
|
|
163
|
+
* Quick validation without async operations
|
|
164
|
+
* Use for fast pre-check before full validation
|
|
165
|
+
*/
|
|
166
|
+
export function quickValidate(content: string): {
|
|
167
|
+
valid: boolean;
|
|
168
|
+
errors: string[];
|
|
169
|
+
} {
|
|
170
|
+
const errors: string[] = [];
|
|
171
|
+
|
|
172
|
+
// Check for empty or whitespace-only content
|
|
173
|
+
if (!content || content.trim().length === 0) {
|
|
174
|
+
errors.push('Content is empty');
|
|
175
|
+
return { valid: false, errors };
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
// Check for minimum length
|
|
179
|
+
if (content.trim().length < 5) {
|
|
180
|
+
errors.push('Content is too short');
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Check for obvious secrets (high confidence only)
|
|
184
|
+
const secrets = detectSecrets(content, 'high');
|
|
185
|
+
if (secrets.length > 0) {
|
|
186
|
+
errors.push(`Content contains ${secrets.length} potential secrets`);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
return { valid: errors.length === 0, errors };
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
/**
|
|
193
|
+
* Check content quality and return warnings
|
|
194
|
+
*/
|
|
195
|
+
function checkContentQuality(content: string): string[] {
|
|
196
|
+
const warnings: string[] = [];
|
|
197
|
+
|
|
198
|
+
// Check for excessive repetition
|
|
199
|
+
const words = content.toLowerCase().split(/\s+/);
|
|
200
|
+
const wordCounts = new Map<string, number>();
|
|
201
|
+
|
|
202
|
+
for (const word of words) {
|
|
203
|
+
if (word.length > 3) { // Only check meaningful words
|
|
204
|
+
wordCounts.set(word, (wordCounts.get(word) || 0) + 1);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
for (const [word, count] of wordCounts.entries()) {
|
|
209
|
+
if (count > 10) {
|
|
210
|
+
warnings.push(`Excessive repetition detected: "${word}" appears ${count} times`);
|
|
211
|
+
break; // Only report one
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
// Check for very long lines (might be code or minified content)
|
|
216
|
+
const lines = content.split('\n');
|
|
217
|
+
for (const line of lines) {
|
|
218
|
+
if (line.length > 1000) {
|
|
219
|
+
warnings.push('Content contains very long lines - may not be human-readable');
|
|
220
|
+
break;
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
// Check for binary-like content
|
|
225
|
+
const binaryPattern = /[\x00-\x08\x0E-\x1F]/;
|
|
226
|
+
if (binaryPattern.test(content)) {
|
|
227
|
+
warnings.push('Content may contain binary data');
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return warnings;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
/**
|
|
234
|
+
* Sanitize content for storage
|
|
235
|
+
* Redacts secrets and normalizes whitespace
|
|
236
|
+
*/
|
|
237
|
+
export function sanitizeForStorage(content: string): string {
|
|
238
|
+
// Redact secrets
|
|
239
|
+
let sanitized = redactSecrets(content);
|
|
240
|
+
|
|
241
|
+
// Normalize excessive whitespace
|
|
242
|
+
sanitized = sanitized.replace(/\n{3,}/g, '\n\n');
|
|
243
|
+
sanitized = sanitized.replace(/ {2,}/g, ' ');
|
|
244
|
+
|
|
245
|
+
// Trim
|
|
246
|
+
sanitized = sanitized.trim();
|
|
247
|
+
|
|
248
|
+
return sanitized;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Calculate a quality score for content (0-100)
|
|
253
|
+
*/
|
|
254
|
+
export function calculateContentQualityScore(content: string): number {
|
|
255
|
+
let score = 100;
|
|
256
|
+
|
|
257
|
+
// Deduct for short content
|
|
258
|
+
if (content.length < 20) {
|
|
259
|
+
score -= 30;
|
|
260
|
+
} else if (content.length < 50) {
|
|
261
|
+
score -= 15;
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
// Deduct for secrets
|
|
265
|
+
const secrets = detectSecrets(content, 'high');
|
|
266
|
+
score -= secrets.length * 20;
|
|
267
|
+
|
|
268
|
+
// Deduct for repetition
|
|
269
|
+
const words = content.toLowerCase().split(/\s+/);
|
|
270
|
+
const uniqueWords = new Set(words.filter(w => w.length > 3));
|
|
271
|
+
const repetitionRatio = uniqueWords.size / Math.max(1, words.length);
|
|
272
|
+
if (repetitionRatio < 0.3) {
|
|
273
|
+
score -= 20;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// Bonus for structure (sentences, punctuation)
|
|
277
|
+
const sentences = content.split(/[.!?]+/).filter(s => s.trim().length > 0);
|
|
278
|
+
if (sentences.length > 1) {
|
|
279
|
+
score += 10;
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
// Bonus for reasonable length
|
|
283
|
+
if (content.length >= 50 && content.length <= 2000) {
|
|
284
|
+
score += 5;
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
return Math.max(0, Math.min(100, score));
|
|
288
|
+
}
|
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Places Module - Spatial memory organization
|
|
3
|
-
*
|
|
4
|
-
* Exports:
|
|
5
|
-
* - places.ts: CRUD operations for places
|
|
6
|
-
* - rules.ts: Auto-assignment rules engine
|
|
7
|
-
* - memory-places.ts: Memory-to-place assignments
|
|
8
|
-
* - walking.ts: Sequential walking through places
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
export * from './
|
|
12
|
-
export * from './
|
|
2
|
+
* Places Module - Spatial memory organization
|
|
3
|
+
*
|
|
4
|
+
* Exports:
|
|
5
|
+
* - places.ts: CRUD operations for places
|
|
6
|
+
* - rules.ts: Auto-assignment rules engine
|
|
7
|
+
* - memory-places.ts: Memory-to-place assignments
|
|
8
|
+
* - walking.ts: Sequential walking through places
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
export * from './places.js';
|
|
12
|
+
export * from './rules.js';
|
|
13
|
+
export * from './memory-places.js';
|
|
13
14
|
export * from './walking.js';
|
|
14
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1,339 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Memory-Place Assignments - Assign memories to places
|
|
3
|
+
*
|
|
4
|
+
* Handles the assignment of memories to places, both:
|
|
5
|
+
* - Auto-assignment via rules
|
|
6
|
+
* - Manual assignment by users
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { randomUUID } from 'crypto';
|
|
10
|
+
import { eq, and } from 'drizzle-orm';
|
|
11
|
+
import { getDb } from '../../db/index.js';
|
|
12
|
+
import { getSchema } from '../../db/schema.js';
|
|
13
|
+
import { logger } from '../logger.js';
|
|
14
|
+
import { getPlaceByType, updatePlaceMemoryCount } from './places.js';
|
|
15
|
+
import { findMatchingPlace } from './rules.js';
|
|
16
|
+
import type { PlaceType } from './places.js';
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Assign a memory to a place (auto or manual)
|
|
20
|
+
*/
|
|
21
|
+
export async function assignMemoryToPlace(params: {
|
|
22
|
+
memoryId: string;
|
|
23
|
+
placeId: string;
|
|
24
|
+
isManual?: boolean;
|
|
25
|
+
ruleId?: string;
|
|
26
|
+
}): Promise<boolean> {
|
|
27
|
+
const db = await getDb();
|
|
28
|
+
if (!db) return false;
|
|
29
|
+
|
|
30
|
+
const schema = await getSchema();
|
|
31
|
+
const sqliteDb = db as any;
|
|
32
|
+
const id = randomUUID();
|
|
33
|
+
|
|
34
|
+
// Check if already assigned
|
|
35
|
+
const existing = await sqliteDb.select()
|
|
36
|
+
.from(schema.memoryPlaces)
|
|
37
|
+
.where(eq(schema.memoryPlaces.memoryId, params.memoryId))
|
|
38
|
+
.limit(1);
|
|
39
|
+
|
|
40
|
+
if (existing.length > 0) {
|
|
41
|
+
// Update existing
|
|
42
|
+
await sqliteDb.update(schema.memoryPlaces)
|
|
43
|
+
.set({
|
|
44
|
+
placeId: params.placeId,
|
|
45
|
+
isManual: params.isManual ? 1 : 0,
|
|
46
|
+
ruleId: params.ruleId || null,
|
|
47
|
+
})
|
|
48
|
+
.where(eq(schema.memoryPlaces.memoryId, params.memoryId));
|
|
49
|
+
} else {
|
|
50
|
+
// Insert new
|
|
51
|
+
await sqliteDb.insert(schema.memoryPlaces).values({
|
|
52
|
+
id,
|
|
53
|
+
memoryId: params.memoryId,
|
|
54
|
+
placeId: params.placeId,
|
|
55
|
+
isManual: params.isManual ? 1 : 0,
|
|
56
|
+
ruleId: params.ruleId || null,
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Update memory's place reference
|
|
61
|
+
await sqliteDb.update(schema.memories)
|
|
62
|
+
.set({ placeId: params.placeId })
|
|
63
|
+
.where(eq(schema.memories.id, params.memoryId));
|
|
64
|
+
|
|
65
|
+
// Update place memory count
|
|
66
|
+
await updatePlaceMemoryCount(params.placeId);
|
|
67
|
+
|
|
68
|
+
logger.info(`[MemoryPlaces] Assigned memory ${params.memoryId} to place ${params.placeId}`);
|
|
69
|
+
return true;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Auto-assign a memory based on rules
|
|
74
|
+
*/
|
|
75
|
+
export async function autoAssignMemory(params: {
|
|
76
|
+
memoryId: string;
|
|
77
|
+
projectId: string;
|
|
78
|
+
toolName?: string;
|
|
79
|
+
content?: string;
|
|
80
|
+
tags?: string[];
|
|
81
|
+
memoryType?: string;
|
|
82
|
+
}): Promise<{ assigned: boolean; placeId?: string; placeType?: PlaceType }> {
|
|
83
|
+
const { memoryId, projectId, toolName, content, tags, memoryType } = params;
|
|
84
|
+
|
|
85
|
+
// Find matching place via rules
|
|
86
|
+
const placeType = await findMatchingPlace(projectId, {
|
|
87
|
+
toolName,
|
|
88
|
+
content,
|
|
89
|
+
tags,
|
|
90
|
+
memoryType,
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
if (!placeType) {
|
|
94
|
+
logger.info(`[MemoryPlaces] No matching rule for memory ${memoryId}`);
|
|
95
|
+
return { assigned: false };
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// Get the place
|
|
99
|
+
const place = await getPlaceByType(projectId, placeType);
|
|
100
|
+
if (!place) {
|
|
101
|
+
logger.warn(`[MemoryPlaces] Place not found: ${placeType}`);
|
|
102
|
+
return { assigned: false };
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
// Assign
|
|
106
|
+
const success = await assignMemoryToPlace({
|
|
107
|
+
memoryId,
|
|
108
|
+
placeId: place.id,
|
|
109
|
+
isManual: false,
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
return {
|
|
113
|
+
assigned: success,
|
|
114
|
+
placeId: place.id,
|
|
115
|
+
placeType,
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Manually assign a memory to a place
|
|
121
|
+
*/
|
|
122
|
+
export async function manualAssignMemory(params: {
|
|
123
|
+
memoryId: string;
|
|
124
|
+
projectId: string;
|
|
125
|
+
placeType: PlaceType;
|
|
126
|
+
}): Promise<boolean> {
|
|
127
|
+
const { memoryId, projectId, placeType } = params;
|
|
128
|
+
|
|
129
|
+
// Get the place by type
|
|
130
|
+
const place = await getPlaceByType(projectId, placeType);
|
|
131
|
+
if (!place) {
|
|
132
|
+
logger.warn(`[MemoryPlaces] Place not found: ${placeType}`);
|
|
133
|
+
return false;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
return assignMemoryToPlace({
|
|
137
|
+
memoryId,
|
|
138
|
+
placeId: place.id,
|
|
139
|
+
isManual: true,
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
/**
|
|
144
|
+
* Get place for a memory
|
|
145
|
+
*/
|
|
146
|
+
export async function getMemoryPlace(memoryId: string): Promise<string | null> {
|
|
147
|
+
const db = await getDb();
|
|
148
|
+
if (!db) return null;
|
|
149
|
+
|
|
150
|
+
const schema = await getSchema();
|
|
151
|
+
const sqliteDb = db as any;
|
|
152
|
+
|
|
153
|
+
const result = await sqliteDb.select()
|
|
154
|
+
.from(schema.memoryPlaces)
|
|
155
|
+
.where(eq(schema.memoryPlaces.memoryId, memoryId))
|
|
156
|
+
.limit(1);
|
|
157
|
+
|
|
158
|
+
return result.length > 0 ? result[0].place_id : null;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Get memories for a place
|
|
163
|
+
*/
|
|
164
|
+
export async function getPlaceMemories(placeId: string, limit: number = 50): Promise<string[]> {
|
|
165
|
+
const db = await getDb();
|
|
166
|
+
if (!db) return [];
|
|
167
|
+
|
|
168
|
+
const schema = await getSchema();
|
|
169
|
+
const sqliteDb = db as any;
|
|
170
|
+
|
|
171
|
+
const results = await sqliteDb.select({ memoryId: schema.memoryPlaces.memoryId })
|
|
172
|
+
.from(schema.memoryPlaces)
|
|
173
|
+
.where(eq(schema.memoryPlaces.placeId, placeId))
|
|
174
|
+
.limit(limit);
|
|
175
|
+
|
|
176
|
+
return results.map((r: any) => r.memoryId);
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Remove memory from place
|
|
181
|
+
*/
|
|
182
|
+
export async function removeMemoryFromPlace(memoryId: string): Promise<boolean> {
|
|
183
|
+
const db = await getDb();
|
|
184
|
+
if (!db) return false;
|
|
185
|
+
|
|
186
|
+
const schema = await getSchema();
|
|
187
|
+
const sqliteDb = db as any;
|
|
188
|
+
|
|
189
|
+
// Get the place before deleting
|
|
190
|
+
const existing = await sqliteDb.select()
|
|
191
|
+
.from(schema.memoryPlaces)
|
|
192
|
+
.where(eq(schema.memoryPlaces.memoryId, memoryId))
|
|
193
|
+
.limit(1);
|
|
194
|
+
|
|
195
|
+
if (existing.length > 0) {
|
|
196
|
+
const oldPlaceId = existing[0].place_id;
|
|
197
|
+
|
|
198
|
+
// Delete assignment
|
|
199
|
+
await sqliteDb.delete(schema.memoryPlaces)
|
|
200
|
+
.where(eq(schema.memoryPlaces.memoryId, memoryId));
|
|
201
|
+
|
|
202
|
+
// Clear memory's place reference
|
|
203
|
+
await sqliteDb.update(schema.memories)
|
|
204
|
+
.set({ placeId: null })
|
|
205
|
+
.where(eq(schema.memories.id, memoryId));
|
|
206
|
+
|
|
207
|
+
// Update old place memory count
|
|
208
|
+
await updatePlaceMemoryCount(oldPlaceId);
|
|
209
|
+
|
|
210
|
+
logger.info(`[MemoryPlaces] Removed memory ${memoryId} from place ${oldPlaceId}`);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
return true;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
/**
|
|
217
|
+
* Initialize memory-place for a project (ensures all memories without places get assigned)
|
|
218
|
+
*/
|
|
219
|
+
export async function initializeProjectPlaces(projectId: string): Promise<{
|
|
220
|
+
initialized: number;
|
|
221
|
+
assigned: number;
|
|
222
|
+
}> {
|
|
223
|
+
const db = await getDb();
|
|
224
|
+
if (!db) return { initialized: 0, assigned: 0 };
|
|
225
|
+
|
|
226
|
+
const schema = await getSchema();
|
|
227
|
+
const sqliteDb = db as any;
|
|
228
|
+
|
|
229
|
+
// Get all memories without a place_id
|
|
230
|
+
const memoriesWithoutPlace = await sqliteDb.select({ id: schema.memories.id })
|
|
231
|
+
.from(schema.memories)
|
|
232
|
+
.where(and(
|
|
233
|
+
eq(schema.memories.projectId, projectId),
|
|
234
|
+
));
|
|
235
|
+
|
|
236
|
+
let assigned = 0;
|
|
237
|
+
|
|
238
|
+
for (const mem of memoriesWithoutPlace) {
|
|
239
|
+
const result = await autoAssignMemory({
|
|
240
|
+
memoryId: mem.id,
|
|
241
|
+
projectId,
|
|
242
|
+
memoryType: 'observation',
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
if (result.assigned) assigned++;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
logger.info(`[MemoryPlaces] Initialized places for project ${projectId}: ${assigned} assigned`);
|
|
249
|
+
return { initialized: memoriesWithoutPlace.length, assigned };
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
/**
|
|
253
|
+
* Auto-archive old memories - move memories > 30 days from active places to Archive
|
|
254
|
+
* Keeps active places lean and organized
|
|
255
|
+
*/
|
|
256
|
+
export async function autoArchiveOldMemories(projectId: string, daysOld: number = 30): Promise<{
|
|
257
|
+
archived: number;
|
|
258
|
+
failed: number;
|
|
259
|
+
}> {
|
|
260
|
+
const db = await getDb();
|
|
261
|
+
if (!db) return { archived: 0, failed: 0 };
|
|
262
|
+
|
|
263
|
+
const schema = await getSchema();
|
|
264
|
+
const sqliteDb = db as any;
|
|
265
|
+
|
|
266
|
+
// Get the Archive place
|
|
267
|
+
const archivePlace = await getPlaceByType(projectId, 'archive');
|
|
268
|
+
if (!archivePlace) {
|
|
269
|
+
logger.warn(`[MemoryPlaces] Archive place not found for project ${projectId}`);
|
|
270
|
+
return { archived: 0, failed: 0 };
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
// Calculate cutoff date (Unix timestamp for SQLite)
|
|
274
|
+
const cutoffDate = new Date();
|
|
275
|
+
cutoffDate.setDate(cutoffDate.getDate() - daysOld);
|
|
276
|
+
const cutoffTimestamp = Math.floor(cutoffDate.getTime() / 1000);
|
|
277
|
+
|
|
278
|
+
// Get all active place IDs (not archive)
|
|
279
|
+
const allPlaces = await sqliteDb.select()
|
|
280
|
+
.from(schema.places)
|
|
281
|
+
.where(eq(schema.places.projectId, projectId));
|
|
282
|
+
|
|
283
|
+
const activePlaceIds = allPlaces
|
|
284
|
+
.filter((p: any) => p.place_type !== 'archive')
|
|
285
|
+
.map((p: any) => p.id);
|
|
286
|
+
|
|
287
|
+
if (activePlaceIds.length === 0) {
|
|
288
|
+
return { archived: 0, failed: 0 };
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
// Find memories in active places that are older than cutoff
|
|
292
|
+
// Use raw SQL for complex date comparison
|
|
293
|
+
const oldMemories = await sqliteDb.select({
|
|
294
|
+
memoryId: schema.memories.id,
|
|
295
|
+
placeId: schema.memoryPlaces.placeId,
|
|
296
|
+
createdAt: schema.memories.createdAt,
|
|
297
|
+
})
|
|
298
|
+
.from(schema.memories)
|
|
299
|
+
.innerJoin(schema.memoryPlaces, eq(schema.memories.id, schema.memoryPlaces.memoryId))
|
|
300
|
+
.where(
|
|
301
|
+
and(
|
|
302
|
+
eq(schema.memories.projectId, projectId),
|
|
303
|
+
// Memories in active places (not archive)
|
|
304
|
+
)
|
|
305
|
+
);
|
|
306
|
+
|
|
307
|
+
// Actually filter in JS for simplicity
|
|
308
|
+
const memoriesToArchive = oldMemories.filter((m: any) =>
|
|
309
|
+
activePlaceIds.includes(m.placeId) && m.createdAt < cutoffTimestamp
|
|
310
|
+
);
|
|
311
|
+
|
|
312
|
+
let archived = 0;
|
|
313
|
+
let failed = 0;
|
|
314
|
+
|
|
315
|
+
for (const mem of memoriesToArchive) {
|
|
316
|
+
try {
|
|
317
|
+
// Move to archive place
|
|
318
|
+
await sqliteDb.update(schema.memoryPlaces)
|
|
319
|
+
.set({ placeId: archivePlace.id })
|
|
320
|
+
.where(eq(schema.memoryPlaces.memoryId, mem.memoryId));
|
|
321
|
+
|
|
322
|
+
// Update the old place's memory count
|
|
323
|
+
await updatePlaceMemoryCount(mem.placeId);
|
|
324
|
+
|
|
325
|
+
archived++;
|
|
326
|
+
} catch (e) {
|
|
327
|
+
logger.warn(`[MemoryPlaces] Failed to archive memory ${mem.memoryId}: ${e}`);
|
|
328
|
+
failed++;
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
// Update archive place memory count
|
|
333
|
+
if (archived > 0) {
|
|
334
|
+
await updatePlaceMemoryCount(archivePlace.id);
|
|
335
|
+
logger.info(`[MemoryPlaces] Archived ${archived} old memories to Archive place`);
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
return { archived, failed };
|
|
339
|
+
}
|