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,478 @@
|
|
|
1
|
+
/** Self-Iteration Job - Auto-extract key facts from ended conversations
|
|
2
|
+
*
|
|
3
|
+
* Processes conversations to extract memories and generate summaries
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { sql, eq } from 'drizzle-orm';
|
|
7
|
+
import { registerJobHandler, type JobExecutionContext, type JobHandler } from '../scheduler/cron-scheduler.js';
|
|
8
|
+
import { getDb } from '../../db/index.js';
|
|
9
|
+
import { getSchema } from '../../db/schema.js';
|
|
10
|
+
import { getProjectById } from '../projects.js';
|
|
11
|
+
import { rememberMemory, search } from '../memory/memories.js';
|
|
12
|
+
import { serializeMetadata, deserializeMetadata } from '../memory/serialization.js';
|
|
13
|
+
import { logger } from '../logger.js';
|
|
14
|
+
import { generateExtractiveSummary, extractMessageContent } from '../utils/content-extraction.js';
|
|
15
|
+
import type { MemoryType } from '../lib/types.js';
|
|
16
|
+
|
|
17
|
+
export interface SelfIterationConfig {
|
|
18
|
+
enabled: boolean;
|
|
19
|
+
extractFacts: boolean;
|
|
20
|
+
generateSummaries: boolean;
|
|
21
|
+
consolidateMemories: boolean;
|
|
22
|
+
minMessageCount: number;
|
|
23
|
+
maxMessagesToProcess: number;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
const DEFAULT_CONFIG: SelfIterationConfig = {
|
|
27
|
+
enabled: true,
|
|
28
|
+
extractFacts: true,
|
|
29
|
+
generateSummaries: true,
|
|
30
|
+
consolidateMemories: true,
|
|
31
|
+
minMessageCount: 5,
|
|
32
|
+
maxMessagesToProcess: 50,
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
interface ConversationRow {
|
|
36
|
+
id: string;
|
|
37
|
+
sessionId: string;
|
|
38
|
+
projectId: string | null;
|
|
39
|
+
summary: string | null;
|
|
40
|
+
messageCount: number;
|
|
41
|
+
metadata: Record<string, unknown> | null;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export interface MessageRow {
|
|
45
|
+
id: string;
|
|
46
|
+
conversationId: string;
|
|
47
|
+
role: string;
|
|
48
|
+
content: string;
|
|
49
|
+
createdAt: Date;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
interface ExtractedFact {
|
|
53
|
+
content: string;
|
|
54
|
+
type: ExtractableMemoryType;
|
|
55
|
+
confidence: number;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
type ExtractableMemoryType = Extract<MemoryType, 'fact' | 'decision' | 'preference'>;
|
|
59
|
+
|
|
60
|
+
const MIN_CONFIDENCE_TO_STORE = 75;
|
|
61
|
+
const FORBIDDEN_BRAND_TERMS = new RegExp(
|
|
62
|
+
`\\b(?:${['mem' + '\\s*' + 'palace', 'mem' + '-' + 'palace', 'om' + 'ni'].join('|')})\\b`,
|
|
63
|
+
'i'
|
|
64
|
+
);
|
|
65
|
+
|
|
66
|
+
const LOW_SIGNAL_PATTERNS = [
|
|
67
|
+
/^\s*(?:thanks?|thank you|hello|hi|hey)\b/i,
|
|
68
|
+
/\b(?:can you|could you|please|help me|show me|explain|what is|how do i)\b/i,
|
|
69
|
+
/\b(?:i'll|i will|we will|let's|todo|task|goal|next step|follow up|remind me)\b/i,
|
|
70
|
+
/^\s*[-*]\s+\[[ x]\]/i,
|
|
71
|
+
];
|
|
72
|
+
|
|
73
|
+
const EXTRACTION_PATTERNS: Array<{
|
|
74
|
+
pattern: RegExp;
|
|
75
|
+
type: ExtractableMemoryType;
|
|
76
|
+
confidence: number;
|
|
77
|
+
prefix: string;
|
|
78
|
+
}> = [
|
|
79
|
+
{
|
|
80
|
+
pattern: /\b(?:i|we)\s+(?:decided|chose|picked|selected|settled on|agreed to use|agreed on)\b/i,
|
|
81
|
+
type: 'decision',
|
|
82
|
+
confidence: 90,
|
|
83
|
+
prefix: 'Decision',
|
|
84
|
+
},
|
|
85
|
+
{
|
|
86
|
+
pattern: /\b(?:final decision|decision)\s*:/i,
|
|
87
|
+
type: 'decision',
|
|
88
|
+
confidence: 90,
|
|
89
|
+
prefix: 'Decision',
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
pattern: /\b(?:i|we)\s+(?:prefer|like|dislike|hate|love)\b/i,
|
|
93
|
+
type: 'preference',
|
|
94
|
+
confidence: 85,
|
|
95
|
+
prefix: 'Preference',
|
|
96
|
+
},
|
|
97
|
+
{
|
|
98
|
+
pattern: /\b(?:my|our|the user's)\s+(?:preference|preferred)\b/i,
|
|
99
|
+
type: 'preference',
|
|
100
|
+
confidence: 85,
|
|
101
|
+
prefix: 'Preference',
|
|
102
|
+
},
|
|
103
|
+
{
|
|
104
|
+
pattern: /\b(?:remember|don't forget)\s+(?:that\s+)?\S+/i,
|
|
105
|
+
type: 'fact',
|
|
106
|
+
confidence: 80,
|
|
107
|
+
prefix: 'Fact',
|
|
108
|
+
},
|
|
109
|
+
{
|
|
110
|
+
pattern: /\b(?:project|repo|workspace|service|app)\s+\S+\s+(?:uses|runs on|depends on|requires|stores)\b/i,
|
|
111
|
+
type: 'fact',
|
|
112
|
+
confidence: 80,
|
|
113
|
+
prefix: 'Fact',
|
|
114
|
+
},
|
|
115
|
+
];
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Build fact extraction prompt
|
|
119
|
+
*/
|
|
120
|
+
function buildFactExtractionPrompt(messages: MessageRow[]): string {
|
|
121
|
+
const messagesText = messages.map(m => `[${m.role}]: ${m.content}`).join('\n\n');
|
|
122
|
+
|
|
123
|
+
return `Extract key facts, decisions, and user preferences from this conversation.
|
|
124
|
+
|
|
125
|
+
Format your response as JSON:
|
|
126
|
+
{
|
|
127
|
+
"extractedFacts": [
|
|
128
|
+
{
|
|
129
|
+
"content": "specific fact text",
|
|
130
|
+
"type": "fact|decision|preference|observation",
|
|
131
|
+
"confidence": 0-100
|
|
132
|
+
}
|
|
133
|
+
]
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
Only extract truly meaningful information that should be remembered:
|
|
137
|
+
- Facts that are useful for future work or decisions
|
|
138
|
+
- User preferences and choices
|
|
139
|
+
- Important decisions made
|
|
140
|
+
- Patterns or behaviors observed
|
|
141
|
+
|
|
142
|
+
Do NOT extract:
|
|
143
|
+
- Temporary conversational details
|
|
144
|
+
- Greetings or pleasantries
|
|
145
|
+
- Unless they indicate something significant
|
|
146
|
+
|
|
147
|
+
Conversation messages:
|
|
148
|
+
${messagesText}`;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Build summary generation prompt
|
|
153
|
+
*/
|
|
154
|
+
function buildSummaryPrompt(messages: MessageRow[]): string {
|
|
155
|
+
const messagesText = messages.map(m => `[${m.role}]: ${m.content}`).join('\n\n');
|
|
156
|
+
|
|
157
|
+
return `Generate a concise summary of this conversation (2-3 sentences, under 300 characters).
|
|
158
|
+
|
|
159
|
+
Focus on:
|
|
160
|
+
- What was accomplished
|
|
161
|
+
- Key decisions made
|
|
162
|
+
- Important context established
|
|
163
|
+
|
|
164
|
+
Conversation messages:
|
|
165
|
+
${messagesText}`;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Parse LLM fact extraction response
|
|
170
|
+
*/
|
|
171
|
+
function parseExtractedFacts(llmResponse: string): ExtractedFact[] {
|
|
172
|
+
try {
|
|
173
|
+
const jsonMatch = llmResponse.match(/\{[\s\S]*\}/);
|
|
174
|
+
if (!jsonMatch) return [];
|
|
175
|
+
|
|
176
|
+
const json = JSON.parse(jsonMatch[0]);
|
|
177
|
+
return json.extractedFacts || [];
|
|
178
|
+
} catch {
|
|
179
|
+
logger.warn('[SelfIteration] Failed to parse fact extraction response');
|
|
180
|
+
return [];
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
function normalizeCaptureKey(content: string): string {
|
|
185
|
+
return content
|
|
186
|
+
.toLowerCase()
|
|
187
|
+
.replace(/^(?:decision|preference|fact):\s*/i, '')
|
|
188
|
+
.replace(/['"`]/g, '')
|
|
189
|
+
.replace(/\s+/g, ' ')
|
|
190
|
+
.replace(/[^\w\s./:-]/g, '')
|
|
191
|
+
.trim();
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
function isLowSignalCapture(content: string): boolean {
|
|
195
|
+
const text = content.trim();
|
|
196
|
+
if (text.length < 20 || text.split(/\s+/).length < 4) return true;
|
|
197
|
+
if (text.endsWith('?')) return true;
|
|
198
|
+
if (FORBIDDEN_BRAND_TERMS.test(text)) return true;
|
|
199
|
+
return LOW_SIGNAL_PATTERNS.some((pattern) => pattern.test(text));
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
function splitCandidateStatements(content: string): string[] {
|
|
203
|
+
return content
|
|
204
|
+
.split(/\n+|(?<=[.!?])\s+/)
|
|
205
|
+
.map((part) => part.trim())
|
|
206
|
+
.filter(Boolean);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
export function extractDurableSelfIterationFacts(messages: MessageRow[]): ExtractedFact[] {
|
|
210
|
+
const captures = new Map<string, ExtractedFact>();
|
|
211
|
+
|
|
212
|
+
for (const msg of messages) {
|
|
213
|
+
if (msg.role !== 'user') continue;
|
|
214
|
+
|
|
215
|
+
for (const statement of splitCandidateStatements(msg.content)) {
|
|
216
|
+
if (isLowSignalCapture(statement)) continue;
|
|
217
|
+
|
|
218
|
+
for (const { pattern, type, confidence, prefix } of EXTRACTION_PATTERNS) {
|
|
219
|
+
if (!pattern.test(statement)) continue;
|
|
220
|
+
|
|
221
|
+
const normalizedStatement = statement.replace(/^(?:decision|preference|fact):\s*/i, '');
|
|
222
|
+
const content = `${prefix}: ${normalizedStatement}`;
|
|
223
|
+
const key = `${type}:${normalizeCaptureKey(content)}`;
|
|
224
|
+
const existing = captures.get(key);
|
|
225
|
+
if (!existing || confidence > existing.confidence) {
|
|
226
|
+
captures.set(key, { content, type, confidence });
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
return Array.from(captures.values()).filter((fact) => fact.confidence >= MIN_CONFIDENCE_TO_STORE);
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
async function hasSimilarSelfIterationMemory(fact: ExtractedFact, projectPath?: string): Promise<boolean> {
|
|
236
|
+
const matches = await search({
|
|
237
|
+
query: fact.content,
|
|
238
|
+
type: fact.type,
|
|
239
|
+
project: projectPath,
|
|
240
|
+
limit: 5,
|
|
241
|
+
});
|
|
242
|
+
const factKey = normalizeCaptureKey(fact.content);
|
|
243
|
+
|
|
244
|
+
return matches.some((match) => {
|
|
245
|
+
const matchMetadata = match.metadata ?? {};
|
|
246
|
+
if (matchMetadata.extractionMethod !== 'self-iteration') return false;
|
|
247
|
+
if ((match.similarity ?? 0) >= 0.92) return true;
|
|
248
|
+
return normalizeCaptureKey(match.content) === factKey;
|
|
249
|
+
});
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
/**
|
|
253
|
+
* Get conversations ready for self-iteration
|
|
254
|
+
*/
|
|
255
|
+
async function getConversationsForIteration(minMessageCount: number): Promise<ConversationRow[]> {
|
|
256
|
+
const db = await getDb();
|
|
257
|
+
if (!db) return [];
|
|
258
|
+
|
|
259
|
+
const schema = await getSchema();
|
|
260
|
+
const sqliteDb = db as any;
|
|
261
|
+
|
|
262
|
+
// Get conversations that:
|
|
263
|
+
// 1. Have ended (endedAt is not null)
|
|
264
|
+
// 2. Haven't been processed yet (metadata.selfIterationProcessed is not set)
|
|
265
|
+
// 3. Have enough messages (messageCount >= minMessageCount)
|
|
266
|
+
const conversations = await sqliteDb.select()
|
|
267
|
+
.from(schema.conversations)
|
|
268
|
+
.where(sql`${schema.conversations.endedAt} IS NOT NULL
|
|
269
|
+
AND ${schema.conversations.messageCount} >= ${minMessageCount}`)
|
|
270
|
+
.limit(10);
|
|
271
|
+
|
|
272
|
+
return (conversations as any[])
|
|
273
|
+
.map((conversation) => ({
|
|
274
|
+
...conversation,
|
|
275
|
+
metadata: deserializeMetadata(conversation.metadata ?? null),
|
|
276
|
+
}))
|
|
277
|
+
.filter((conversation) => !conversation.metadata?.selfIterationProcessed) as ConversationRow[];
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
/**
|
|
281
|
+
* Get messages for a conversation
|
|
282
|
+
*/
|
|
283
|
+
async function getConversationMessages(conversationId: string): Promise<MessageRow[]> {
|
|
284
|
+
const db = await getDb();
|
|
285
|
+
if (!db) return [];
|
|
286
|
+
|
|
287
|
+
const schema = await getSchema();
|
|
288
|
+
const sqliteDb = db as any;
|
|
289
|
+
|
|
290
|
+
const messages = await sqliteDb.select()
|
|
291
|
+
.from(schema.messages)
|
|
292
|
+
.where(eq(schema.messages.conversationId, conversationId))
|
|
293
|
+
.orderBy(schema.messages.createdAt);
|
|
294
|
+
|
|
295
|
+
return messages.map((m: any) => ({
|
|
296
|
+
id: m.id,
|
|
297
|
+
conversationId: m.conversation_id,
|
|
298
|
+
role: m.role,
|
|
299
|
+
content: m.content,
|
|
300
|
+
createdAt: new Date(m.createdAt),
|
|
301
|
+
})) as MessageRow[];
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
/**
|
|
305
|
+
* Mark conversation as processed
|
|
306
|
+
*/
|
|
307
|
+
async function markConversationProcessed(conversationId: string): Promise<void> {
|
|
308
|
+
const db = await getDb();
|
|
309
|
+
if (!db) return;
|
|
310
|
+
|
|
311
|
+
const schema = await getSchema();
|
|
312
|
+
const sqliteDb = db as any;
|
|
313
|
+
|
|
314
|
+
const rows = await sqliteDb.select()
|
|
315
|
+
.from(schema.conversations)
|
|
316
|
+
.where(eq(schema.conversations.id, conversationId))
|
|
317
|
+
.limit(1);
|
|
318
|
+
const existingMetadata = deserializeMetadata(rows[0]?.metadata ?? null) ?? {};
|
|
319
|
+
|
|
320
|
+
await sqliteDb.update(schema.conversations)
|
|
321
|
+
.set({
|
|
322
|
+
metadata: serializeMetadata({
|
|
323
|
+
...existingMetadata,
|
|
324
|
+
selfIterationProcessed: true,
|
|
325
|
+
selfIterationProcessedAt: new Date().toISOString(),
|
|
326
|
+
}),
|
|
327
|
+
})
|
|
328
|
+
.where(eq(schema.conversations.id, conversationId));
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
/**
|
|
332
|
+
* Process a single conversation
|
|
333
|
+
*/
|
|
334
|
+
async function processConversation(
|
|
335
|
+
conversation: ConversationRow,
|
|
336
|
+
config: SelfIterationConfig
|
|
337
|
+
): Promise<{ memoriesCreated: number; summariesCreated: number }> {
|
|
338
|
+
const messages = await getConversationMessages(conversation.id);
|
|
339
|
+
let memoriesCreated = 0;
|
|
340
|
+
let summariesCreated = 0;
|
|
341
|
+
|
|
342
|
+
// Limit messages if configured
|
|
343
|
+
const messagesToProcess = config.maxMessagesToProcess > 0
|
|
344
|
+
? messages.slice(-config.maxMessagesToProcess)
|
|
345
|
+
: messages;
|
|
346
|
+
|
|
347
|
+
if (messagesToProcess.length === 0) {
|
|
348
|
+
return { memoriesCreated, summariesCreated };
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// Extract facts using improved pattern matching
|
|
352
|
+
if (config.extractFacts) {
|
|
353
|
+
const extractedFacts = extractDurableSelfIterationFacts(messagesToProcess);
|
|
354
|
+
const project = conversation.projectId ? await getProjectById(conversation.projectId) : null;
|
|
355
|
+
const projectPath = project?.path;
|
|
356
|
+
|
|
357
|
+
// Store extracted facts as memories
|
|
358
|
+
for (const fact of extractedFacts) {
|
|
359
|
+
try {
|
|
360
|
+
if (await hasSimilarSelfIterationMemory(fact, projectPath)) {
|
|
361
|
+
logger.debug(`[SelfIteration] Suppressed duplicate memory: ${fact.type} - ${fact.content.substring(0, 50)}...`);
|
|
362
|
+
continue;
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
await rememberMemory({
|
|
366
|
+
content: fact.content,
|
|
367
|
+
type: fact.type,
|
|
368
|
+
project: projectPath,
|
|
369
|
+
metadata: {
|
|
370
|
+
extractionMethod: 'self-iteration',
|
|
371
|
+
confidence: fact.confidence,
|
|
372
|
+
conversationId: conversation.id,
|
|
373
|
+
sessionId: conversation.sessionId,
|
|
374
|
+
},
|
|
375
|
+
source: 'self-iteration',
|
|
376
|
+
});
|
|
377
|
+
memoriesCreated++;
|
|
378
|
+
logger.info(`[SelfIteration] Extracted memory: ${fact.type} - ${fact.content.substring(0, 50)}...`);
|
|
379
|
+
} catch (error) {
|
|
380
|
+
logger.error(`[SelfIteration] Failed to store memory:`, error);
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
// Generate summary
|
|
386
|
+
if (config.generateSummaries && messagesToProcess.length > 0) {
|
|
387
|
+
// Use extractive summarization for better results
|
|
388
|
+
const extracted = extractMessageContent(messagesToProcess);
|
|
389
|
+
const summary = generateExtractiveSummary(extracted);
|
|
390
|
+
|
|
391
|
+
// Update conversation with summary
|
|
392
|
+
const db = await getDb();
|
|
393
|
+
const schema = await getSchema();
|
|
394
|
+
const sqliteDb = db as any;
|
|
395
|
+
|
|
396
|
+
await sqliteDb.update(schema.conversations)
|
|
397
|
+
.set({ summary })
|
|
398
|
+
.where(eq(schema.conversations.id, conversation.id));
|
|
399
|
+
|
|
400
|
+
summariesCreated++;
|
|
401
|
+
logger.info(`[SelfIteration] Generated summary for conversation ${conversation.sessionId}`);
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
return { memoriesCreated, summariesCreated };
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
/**
|
|
408
|
+
* Register self-iteration job handler
|
|
409
|
+
*/
|
|
410
|
+
const selfIterationHandler: JobHandler = async (
|
|
411
|
+
context: JobExecutionContext
|
|
412
|
+
): Promise<{ recordsProcessed: number; summary: Record<string, unknown> }> => {
|
|
413
|
+
logger.info('[SelfIteration] Starting job');
|
|
414
|
+
|
|
415
|
+
const config = {
|
|
416
|
+
...DEFAULT_CONFIG,
|
|
417
|
+
...(context.config as Partial<SelfIterationConfig>),
|
|
418
|
+
};
|
|
419
|
+
|
|
420
|
+
if (!config.enabled) {
|
|
421
|
+
logger.info('[SelfIteration] Job disabled, skipping');
|
|
422
|
+
return { recordsProcessed: 0, summary: { status: 'disabled' } };
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
try {
|
|
426
|
+
const conversations = await getConversationsForIteration(
|
|
427
|
+
config.minMessageCount || DEFAULT_CONFIG.minMessageCount
|
|
428
|
+
);
|
|
429
|
+
|
|
430
|
+
if (conversations.length === 0) {
|
|
431
|
+
logger.info('[SelfIteration] No conversations to process');
|
|
432
|
+
return { recordsProcessed: 0, summary: { status: 'no_conversations' } };
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
let totalMemoriesCreated = 0;
|
|
436
|
+
let totalSummariesGenerated = 0;
|
|
437
|
+
let processedCount = 0;
|
|
438
|
+
|
|
439
|
+
for (const conversation of conversations) {
|
|
440
|
+
// Skip if already processed
|
|
441
|
+
if (conversation.metadata?.selfIterationProcessed) {
|
|
442
|
+
continue;
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
try {
|
|
446
|
+
const result = await processConversation(conversation, config);
|
|
447
|
+
totalMemoriesCreated += result.memoriesCreated;
|
|
448
|
+
totalSummariesGenerated += result.summariesCreated;
|
|
449
|
+
|
|
450
|
+
await markConversationProcessed(conversation.id);
|
|
451
|
+
processedCount++;
|
|
452
|
+
} catch (error) {
|
|
453
|
+
logger.error(`[SelfIteration] Failed to process conversation ${conversation.id}:`, error);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
logger.info(`[SelfIteration] Processed ${processedCount}/${conversations.length} conversations`);
|
|
458
|
+
logger.info(`[SelfIteration] Created ${totalMemoriesCreated} memories`);
|
|
459
|
+
logger.info(`[SelfIteration] Generated ${totalSummariesGenerated} summaries`);
|
|
460
|
+
|
|
461
|
+
return {
|
|
462
|
+
recordsProcessed: processedCount,
|
|
463
|
+
summary: {
|
|
464
|
+
memoriesCreated: totalMemoriesCreated,
|
|
465
|
+
summariesGenerated: totalSummariesGenerated,
|
|
466
|
+
totalConversations: conversations.length,
|
|
467
|
+
},
|
|
468
|
+
};
|
|
469
|
+
} catch (error) {
|
|
470
|
+
logger.error('[SelfIteration] Job failed:', error);
|
|
471
|
+
return {
|
|
472
|
+
recordsProcessed: 0,
|
|
473
|
+
summary: { error: error instanceof Error ? error.message : String(error) },
|
|
474
|
+
};
|
|
475
|
+
}
|
|
476
|
+
};
|
|
477
|
+
|
|
478
|
+
export { selfIterationHandler, DEFAULT_CONFIG };
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
/** Session Hooks - Lifecycle hooks for conversation management
|
|
2
|
+
*
|
|
3
|
+
* Provides functions to be called when sessions start/end
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { sql, eq } from 'drizzle-orm';
|
|
7
|
+
import { getDb } from '../../db/index.js';
|
|
8
|
+
import { getSchema } from '../../db/schema.js';
|
|
9
|
+
import { logger } from '../logger.js';
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Called when a session ends
|
|
13
|
+
* Marks the conversation as ended so self-iteration can process it
|
|
14
|
+
*/
|
|
15
|
+
export async function onSessionEnd(sessionId: string): Promise<void> {
|
|
16
|
+
logger.info(`[SessionHooks] Session ended: ${sessionId}`);
|
|
17
|
+
|
|
18
|
+
const db = await getDb();
|
|
19
|
+
if (!db) {
|
|
20
|
+
logger.warn('[SessionHooks] Database unavailable, cannot end session');
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const schema = await getSchema();
|
|
25
|
+
const sqliteDb = db as any;
|
|
26
|
+
|
|
27
|
+
// Mark conversation as ended
|
|
28
|
+
await sqliteDb.update(schema.conversations)
|
|
29
|
+
.set({ endedAt: new Date() })
|
|
30
|
+
.where(eq(schema.conversations.sessionId, sessionId));
|
|
31
|
+
|
|
32
|
+
logger.info(`[SessionHooks] Session ${sessionId} marked as ended`);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Get active sessions for a project
|
|
37
|
+
*/
|
|
38
|
+
export async function getActiveSessions(projectId: string): Promise<string[]> {
|
|
39
|
+
const db = await getDb();
|
|
40
|
+
if (!db) return [];
|
|
41
|
+
|
|
42
|
+
const schema = await getSchema();
|
|
43
|
+
const sqliteDb = db as any;
|
|
44
|
+
|
|
45
|
+
const sessions = await sqliteDb.select()
|
|
46
|
+
.from(schema.conversations)
|
|
47
|
+
.where(eq(schema.conversations.projectId, projectId))
|
|
48
|
+
.where(sql`${schema.conversations.endedAt} IS NULL`);
|
|
49
|
+
|
|
50
|
+
return sessions.map((s: any) => s.session_id);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* End all sessions for a project
|
|
55
|
+
*/
|
|
56
|
+
export async function endAllProjectSessions(projectId: string): Promise<number> {
|
|
57
|
+
const db = await getDb();
|
|
58
|
+
if (!db) return 0;
|
|
59
|
+
|
|
60
|
+
const schema = await getSchema();
|
|
61
|
+
const sqliteDb = db as any;
|
|
62
|
+
|
|
63
|
+
const result = await sqliteDb.update(schema.conversations)
|
|
64
|
+
.set({ endedAt: new Date() })
|
|
65
|
+
.where(eq(schema.conversations.projectId, projectId));
|
|
66
|
+
|
|
67
|
+
logger.info(`[SessionHooks] Ended ${result.rowCount} sessions for project ${projectId}`);
|
|
68
|
+
return result.rowCount || 0;
|
|
69
|
+
}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/** Session Auto-Load Types */
|
|
2
|
+
|
|
3
|
+
export interface AutoLoadConfig {
|
|
4
|
+
enabled: boolean;
|
|
5
|
+
includeCoreMemory: boolean;
|
|
6
|
+
includeRecentMemories: boolean;
|
|
7
|
+
recentMemoryCount: number;
|
|
8
|
+
importanceThreshold: number;
|
|
9
|
+
includeProjectContext: boolean;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export interface AutoLoadResult {
|
|
13
|
+
coreMemoryLoaded: boolean;
|
|
14
|
+
memoriesLoaded: number;
|
|
15
|
+
projectContextLoaded: boolean;
|
|
16
|
+
tokensUsed: number;
|
|
17
|
+
duration: number;
|
|
18
|
+
warnings: string[];
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface SessionState {
|
|
22
|
+
sessionId: string;
|
|
23
|
+
projectId: string;
|
|
24
|
+
autoLoaded: boolean;
|
|
25
|
+
startedAt: Date;
|
|
26
|
+
lastActivityAt: Date;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export const DEFAULT_AUTO_LOAD_CONFIG: AutoLoadConfig = {
|
|
30
|
+
enabled: true,
|
|
31
|
+
includeCoreMemory: true,
|
|
32
|
+
includeRecentMemories: true,
|
|
33
|
+
recentMemoryCount: 5,
|
|
34
|
+
importanceThreshold: 70,
|
|
35
|
+
includeProjectContext: true,
|
|
36
|
+
};
|