squish-memory 0.7.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/.claude-plugin/marketplace.json +20 -0
- package/.claude-plugin/plugin.json +32 -0
- package/.mcp.json +14 -0
- package/LICENSE +21 -0
- package/README.md +556 -0
- package/commands/context-paging.md +51 -0
- package/commands/context-status.md +22 -0
- package/commands/context.md +5 -0
- package/commands/core-memory.md +56 -0
- package/commands/health.md +5 -0
- package/commands/init.md +39 -0
- package/commands/merge.md +113 -0
- package/commands/observe.md +5 -0
- package/commands/recall.md +5 -0
- package/commands/remember.md +11 -0
- package/commands/search.md +10 -0
- package/dist/adapters/claude-code/capture.d.ts +11 -0
- package/dist/adapters/claude-code/capture.d.ts.map +1 -0
- package/dist/adapters/claude-code/capture.js +100 -0
- package/dist/adapters/claude-code/capture.js.map +1 -0
- package/dist/adapters/claude-code/index.d.ts +5 -0
- package/dist/adapters/claude-code/index.d.ts.map +1 -0
- package/dist/adapters/claude-code/index.js +6 -0
- package/dist/adapters/claude-code/index.js.map +1 -0
- package/dist/adapters/claude-code/injection.d.ts +34 -0
- package/dist/adapters/claude-code/injection.d.ts.map +1 -0
- package/dist/adapters/claude-code/injection.js +127 -0
- package/dist/adapters/claude-code/injection.js.map +1 -0
- package/dist/adapters/claude-code/plugin-wrapper.d.ts +21 -0
- package/dist/adapters/claude-code/plugin-wrapper.d.ts.map +1 -0
- package/dist/adapters/claude-code/plugin-wrapper.js +239 -0
- package/dist/adapters/claude-code/plugin-wrapper.js.map +1 -0
- package/dist/adapters/claude-code/types.d.ts +46 -0
- package/dist/adapters/claude-code/types.d.ts.map +1 -0
- package/dist/adapters/claude-code/types.js +6 -0
- package/dist/adapters/claude-code/types.js.map +1 -0
- package/dist/algorithms/merge/analytics/token-estimator.d.ts +50 -0
- package/dist/algorithms/merge/analytics/token-estimator.d.ts.map +1 -0
- package/dist/algorithms/merge/analytics/token-estimator.js +154 -0
- package/dist/algorithms/merge/analytics/token-estimator.js.map +1 -0
- package/dist/algorithms/merge/detection/hash-filters.d.ts +47 -0
- package/dist/algorithms/merge/detection/hash-filters.d.ts.map +1 -0
- package/dist/algorithms/merge/detection/hash-filters.js +190 -0
- package/dist/algorithms/merge/detection/hash-filters.js.map +1 -0
- package/dist/algorithms/merge/detection/semantic-ranker.d.ts +32 -0
- package/dist/algorithms/merge/detection/semantic-ranker.d.ts.map +1 -0
- package/dist/algorithms/merge/detection/semantic-ranker.js +118 -0
- package/dist/algorithms/merge/detection/semantic-ranker.js.map +1 -0
- package/dist/algorithms/merge/detection/two-stage-detector.d.ts +49 -0
- package/dist/algorithms/merge/detection/two-stage-detector.d.ts.map +1 -0
- package/dist/algorithms/merge/detection/two-stage-detector.js +185 -0
- package/dist/algorithms/merge/detection/two-stage-detector.js.map +1 -0
- package/dist/algorithms/merge/handlers/approve-merge.d.ts +22 -0
- package/dist/algorithms/merge/handlers/approve-merge.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/approve-merge.js +179 -0
- package/dist/algorithms/merge/handlers/approve-merge.js.map +1 -0
- package/dist/algorithms/merge/handlers/detect-duplicates.d.ts +47 -0
- package/dist/algorithms/merge/handlers/detect-duplicates.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/detect-duplicates.js +165 -0
- package/dist/algorithms/merge/handlers/detect-duplicates.js.map +1 -0
- package/dist/algorithms/merge/handlers/get-stats.d.ts +39 -0
- package/dist/algorithms/merge/handlers/get-stats.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/get-stats.js +88 -0
- package/dist/algorithms/merge/handlers/get-stats.js.map +1 -0
- package/dist/algorithms/merge/handlers/list-proposals.d.ts +45 -0
- package/dist/algorithms/merge/handlers/list-proposals.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/list-proposals.js +83 -0
- package/dist/algorithms/merge/handlers/list-proposals.js.map +1 -0
- package/dist/algorithms/merge/handlers/preview-merge.d.ts +39 -0
- package/dist/algorithms/merge/handlers/preview-merge.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/preview-merge.js +93 -0
- package/dist/algorithms/merge/handlers/preview-merge.js.map +1 -0
- package/dist/algorithms/merge/handlers/reject-merge.d.ts +28 -0
- package/dist/algorithms/merge/handlers/reject-merge.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/reject-merge.js +69 -0
- package/dist/algorithms/merge/handlers/reject-merge.js.map +1 -0
- package/dist/algorithms/merge/handlers/reverse-merge.d.ts +21 -0
- package/dist/algorithms/merge/handlers/reverse-merge.d.ts.map +1 -0
- package/dist/algorithms/merge/handlers/reverse-merge.js +121 -0
- package/dist/algorithms/merge/handlers/reverse-merge.js.map +1 -0
- package/dist/algorithms/merge/safety/safety-checks.d.ts +22 -0
- package/dist/algorithms/merge/safety/safety-checks.d.ts.map +1 -0
- package/dist/algorithms/merge/safety/safety-checks.js +215 -0
- package/dist/algorithms/merge/safety/safety-checks.js.map +1 -0
- package/dist/algorithms/merge/strategies/merge-strategies.d.ts +32 -0
- package/dist/algorithms/merge/strategies/merge-strategies.d.ts.map +1 -0
- package/dist/algorithms/merge/strategies/merge-strategies.js +337 -0
- package/dist/algorithms/merge/strategies/merge-strategies.js.map +1 -0
- package/dist/api/web/web.d.ts +4 -0
- package/dist/api/web/web.d.ts.map +1 -0
- package/dist/api/web/web.js +484 -0
- package/dist/api/web/web.js.map +1 -0
- package/dist/config.d.ts +28 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +83 -0
- package/dist/config.js.map +1 -0
- package/dist/core/agent-memory.d.ts +22 -0
- package/dist/core/agent-memory.d.ts.map +1 -0
- package/dist/core/agent-memory.js +111 -0
- package/dist/core/agent-memory.js.map +1 -0
- package/dist/core/associations.d.ts +32 -0
- package/dist/core/associations.d.ts.map +1 -0
- package/dist/core/associations.js +249 -0
- package/dist/core/associations.js.map +1 -0
- package/dist/core/cache.d.ts +13 -0
- package/dist/core/cache.d.ts.map +1 -0
- package/dist/core/cache.js +202 -0
- package/dist/core/cache.js.map +1 -0
- package/dist/core/consolidation.d.ts +6 -0
- package/dist/core/consolidation.d.ts.map +1 -0
- package/dist/core/consolidation.js +40 -0
- package/dist/core/consolidation.js.map +1 -0
- package/dist/core/context-paging.d.ts +90 -0
- package/dist/core/context-paging.d.ts.map +1 -0
- package/dist/core/context-paging.js +348 -0
- package/dist/core/context-paging.js.map +1 -0
- package/dist/core/context.d.ts +7 -0
- package/dist/core/context.d.ts.map +1 -0
- package/dist/core/context.js +24 -0
- package/dist/core/context.js.map +1 -0
- package/dist/core/core-memory.d.ts +73 -0
- package/dist/core/core-memory.d.ts.map +1 -0
- package/dist/core/core-memory.js +214 -0
- package/dist/core/core-memory.js.map +1 -0
- package/dist/core/database.d.ts +12 -0
- package/dist/core/database.d.ts.map +1 -0
- package/dist/core/database.js +12 -0
- package/dist/core/database.js.map +1 -0
- package/dist/core/embeddings/qmd-client.d.ts +136 -0
- package/dist/core/embeddings/qmd-client.d.ts.map +1 -0
- package/dist/core/embeddings/qmd-client.js +403 -0
- package/dist/core/embeddings/qmd-client.js.map +1 -0
- package/dist/core/embeddings/qmd-provider.d.ts +65 -0
- package/dist/core/embeddings/qmd-provider.d.ts.map +1 -0
- package/dist/core/embeddings/qmd-provider.js +133 -0
- package/dist/core/embeddings/qmd-provider.js.map +1 -0
- package/dist/core/embeddings.d.ts +19 -0
- package/dist/core/embeddings.d.ts.map +1 -0
- package/dist/core/embeddings.js +297 -0
- package/dist/core/embeddings.js.map +1 -0
- package/dist/core/governance.d.ts +21 -0
- package/dist/core/governance.d.ts.map +1 -0
- package/dist/core/governance.js +64 -0
- package/dist/core/governance.js.map +1 -0
- package/dist/core/index.d.ts +10 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +14 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/lifecycle.d.ts +19 -0
- package/dist/core/lifecycle.d.ts.map +1 -0
- package/dist/core/lifecycle.js +182 -0
- package/dist/core/lifecycle.js.map +1 -0
- package/dist/core/local-embeddings.d.ts +14 -0
- package/dist/core/local-embeddings.d.ts.map +1 -0
- package/dist/core/local-embeddings.js +94 -0
- package/dist/core/local-embeddings.js.map +1 -0
- package/dist/core/logger.d.ts +16 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +34 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/memory/bridge-discovery.d.ts +50 -0
- package/dist/core/memory/bridge-discovery.d.ts.map +1 -0
- package/dist/core/memory/bridge-discovery.js +291 -0
- package/dist/core/memory/bridge-discovery.js.map +1 -0
- package/dist/core/memory/entity-extractor.d.ts +33 -0
- package/dist/core/memory/entity-extractor.d.ts.map +1 -0
- package/dist/core/memory/entity-extractor.js +336 -0
- package/dist/core/memory/entity-extractor.js.map +1 -0
- package/dist/core/memory/entity-resolver.d.ts +23 -0
- package/dist/core/memory/entity-resolver.d.ts.map +1 -0
- package/dist/core/memory/entity-resolver.js +64 -0
- package/dist/core/memory/entity-resolver.js.map +1 -0
- package/dist/core/memory/fact-extractor.d.ts +24 -0
- package/dist/core/memory/fact-extractor.d.ts.map +1 -0
- package/dist/core/memory/fact-extractor.js +89 -0
- package/dist/core/memory/fact-extractor.js.map +1 -0
- package/dist/core/memory/hybrid-retrieval.d.ts +30 -0
- package/dist/core/memory/hybrid-retrieval.d.ts.map +1 -0
- package/dist/core/memory/hybrid-retrieval.js +63 -0
- package/dist/core/memory/hybrid-retrieval.js.map +1 -0
- package/dist/core/memory/hybrid-scorer.d.ts +51 -0
- package/dist/core/memory/hybrid-scorer.d.ts.map +1 -0
- package/dist/core/memory/hybrid-scorer.js +248 -0
- package/dist/core/memory/hybrid-scorer.js.map +1 -0
- package/dist/core/memory/index.d.ts +8 -0
- package/dist/core/memory/index.d.ts.map +1 -0
- package/dist/core/memory/index.js +10 -0
- package/dist/core/memory/index.js.map +1 -0
- package/dist/core/memory/memories.d.ts +35 -0
- package/dist/core/memory/memories.d.ts.map +1 -0
- package/dist/core/memory/memories.js +338 -0
- package/dist/core/memory/memories.js.map +1 -0
- package/dist/core/memory/memory-manager.d.ts +15 -0
- package/dist/core/memory/memory-manager.d.ts.map +1 -0
- package/dist/core/memory/memory-manager.js +46 -0
- package/dist/core/memory/memory-manager.js.map +1 -0
- package/dist/core/memory/query-processor.d.ts +21 -0
- package/dist/core/memory/query-processor.d.ts.map +1 -0
- package/dist/core/memory/query-processor.js +72 -0
- package/dist/core/memory/query-processor.js.map +1 -0
- package/dist/core/memory/serialization.d.ts +6 -0
- package/dist/core/memory/serialization.d.ts.map +1 -0
- package/dist/core/memory/serialization.js +35 -0
- package/dist/core/memory/serialization.js.map +1 -0
- package/dist/core/memory/temporal-parser.d.ts +32 -0
- package/dist/core/memory/temporal-parser.d.ts.map +1 -0
- package/dist/core/memory/temporal-parser.js +385 -0
- package/dist/core/memory/temporal-parser.js.map +1 -0
- package/dist/core/observations.d.ts +26 -0
- package/dist/core/observations.d.ts.map +1 -0
- package/dist/core/observations.js +118 -0
- package/dist/core/observations.js.map +1 -0
- package/dist/core/privacy.d.ts +23 -0
- package/dist/core/privacy.d.ts.map +1 -0
- package/dist/core/privacy.js +82 -0
- package/dist/core/privacy.js.map +1 -0
- package/dist/core/projects.d.ts +10 -0
- package/dist/core/projects.d.ts.map +1 -0
- package/dist/core/projects.js +66 -0
- package/dist/core/projects.js.map +1 -0
- package/dist/core/redis.d.ts +11 -0
- package/dist/core/redis.d.ts.map +1 -0
- package/dist/core/redis.js +69 -0
- package/dist/core/redis.js.map +1 -0
- package/dist/core/requirements.d.ts +20 -0
- package/dist/core/requirements.d.ts.map +1 -0
- package/dist/core/requirements.js +35 -0
- package/dist/core/requirements.js.map +1 -0
- package/dist/core/search/conversations.d.ts +25 -0
- package/dist/core/search/conversations.d.ts.map +1 -0
- package/dist/core/search/conversations.js +112 -0
- package/dist/core/search/conversations.js.map +1 -0
- package/dist/core/search/entities.d.ts +12 -0
- package/dist/core/search/entities.d.ts.map +1 -0
- package/dist/core/search/entities.js +34 -0
- package/dist/core/search/entities.js.map +1 -0
- package/dist/core/search/folder-context.d.ts +25 -0
- package/dist/core/search/folder-context.d.ts.map +1 -0
- package/dist/core/search/folder-context.js +119 -0
- package/dist/core/search/folder-context.js.map +1 -0
- package/dist/core/search/index.d.ts +4 -0
- package/dist/core/search/index.d.ts.map +1 -0
- package/dist/core/search/index.js +5 -0
- package/dist/core/search/index.js.map +1 -0
- package/dist/core/search/qmd-search.d.ts +61 -0
- package/dist/core/search/qmd-search.d.ts.map +1 -0
- package/dist/core/search/qmd-search.js +178 -0
- package/dist/core/search/qmd-search.js.map +1 -0
- package/dist/core/secret-detector.d.ts +32 -0
- package/dist/core/secret-detector.d.ts.map +1 -0
- package/dist/core/secret-detector.js +88 -0
- package/dist/core/secret-detector.js.map +1 -0
- package/dist/core/snapshots/cleanup.d.ts +9 -0
- package/dist/core/snapshots/cleanup.d.ts.map +1 -0
- package/dist/core/snapshots/cleanup.js +12 -0
- package/dist/core/snapshots/cleanup.js.map +1 -0
- package/dist/core/snapshots/comparison.d.ts +19 -0
- package/dist/core/snapshots/comparison.d.ts.map +1 -0
- package/dist/core/snapshots/comparison.js +37 -0
- package/dist/core/snapshots/comparison.js.map +1 -0
- package/dist/core/snapshots/creation.d.ts +19 -0
- package/dist/core/snapshots/creation.d.ts.map +1 -0
- package/dist/core/snapshots/creation.js +126 -0
- package/dist/core/snapshots/creation.js.map +1 -0
- package/dist/core/snapshots/retrieval.d.ts +7 -0
- package/dist/core/snapshots/retrieval.d.ts.map +1 -0
- package/dist/core/snapshots/retrieval.js +41 -0
- package/dist/core/snapshots/retrieval.js.map +1 -0
- package/dist/core/snapshots/stats.d.ts +11 -0
- package/dist/core/snapshots/stats.d.ts.map +1 -0
- package/dist/core/snapshots/stats.js +52 -0
- package/dist/core/snapshots/stats.js.map +1 -0
- package/dist/core/snapshots.d.ts +29 -0
- package/dist/core/snapshots.d.ts.map +1 -0
- package/dist/core/snapshots.js +220 -0
- package/dist/core/snapshots.js.map +1 -0
- package/dist/core/summarization/cleanup.d.ts +9 -0
- package/dist/core/summarization/cleanup.d.ts.map +1 -0
- package/dist/core/summarization/cleanup.js +12 -0
- package/dist/core/summarization/cleanup.js.map +1 -0
- package/dist/core/summarization/queries.d.ts +9 -0
- package/dist/core/summarization/queries.d.ts.map +1 -0
- package/dist/core/summarization/queries.js +28 -0
- package/dist/core/summarization/queries.js.map +1 -0
- package/dist/core/summarization/stats.d.ts +14 -0
- package/dist/core/summarization/stats.d.ts.map +1 -0
- package/dist/core/summarization/stats.js +52 -0
- package/dist/core/summarization/stats.js.map +1 -0
- package/dist/core/summarization/strategies.d.ts +24 -0
- package/dist/core/summarization/strategies.d.ts.map +1 -0
- package/dist/core/summarization/strategies.js +28 -0
- package/dist/core/summarization/strategies.js.map +1 -0
- package/dist/core/summarization.d.ts +37 -0
- package/dist/core/summarization.d.ts.map +1 -0
- package/dist/core/summarization.js +188 -0
- package/dist/core/summarization.js.map +1 -0
- package/dist/core/sync/qmd-sync.d.ts +106 -0
- package/dist/core/sync/qmd-sync.d.ts.map +1 -0
- package/dist/core/sync/qmd-sync.js +213 -0
- package/dist/core/sync/qmd-sync.js.map +1 -0
- package/dist/core/temporal-facts.d.ts +54 -0
- package/dist/core/temporal-facts.d.ts.map +1 -0
- package/dist/core/temporal-facts.js +193 -0
- package/dist/core/temporal-facts.js.map +1 -0
- package/dist/core/utils/cleanup-operations.d.ts +13 -0
- package/dist/core/utils/cleanup-operations.d.ts.map +1 -0
- package/dist/core/utils/cleanup-operations.js +44 -0
- package/dist/core/utils/cleanup-operations.js.map +1 -0
- package/dist/core/utils/content-extraction.d.ts +19 -0
- package/dist/core/utils/content-extraction.d.ts.map +1 -0
- package/dist/core/utils/content-extraction.js +65 -0
- package/dist/core/utils/content-extraction.js.map +1 -0
- package/dist/core/utils/filter-builder.d.ts +13 -0
- package/dist/core/utils/filter-builder.d.ts.map +1 -0
- package/dist/core/utils/filter-builder.js +44 -0
- package/dist/core/utils/filter-builder.js.map +1 -0
- package/dist/core/utils/history-traversal.d.ts +13 -0
- package/dist/core/utils/history-traversal.d.ts.map +1 -0
- package/dist/core/utils/history-traversal.js +50 -0
- package/dist/core/utils/history-traversal.js.map +1 -0
- package/dist/core/utils/memory-operations.d.ts +17 -0
- package/dist/core/utils/memory-operations.d.ts.map +1 -0
- package/dist/core/utils/memory-operations.js +41 -0
- package/dist/core/utils/memory-operations.js.map +1 -0
- package/dist/core/utils/query-operations.d.ts +18 -0
- package/dist/core/utils/query-operations.d.ts.map +1 -0
- package/dist/core/utils/query-operations.js +65 -0
- package/dist/core/utils/query-operations.js.map +1 -0
- package/dist/core/utils/summarization-helpers.d.ts +21 -0
- package/dist/core/utils/summarization-helpers.d.ts.map +1 -0
- package/dist/core/utils/summarization-helpers.js +35 -0
- package/dist/core/utils/summarization-helpers.js.map +1 -0
- package/dist/core/utils/temporal-queries.d.ts +13 -0
- package/dist/core/utils/temporal-queries.d.ts.map +1 -0
- package/dist/core/utils/temporal-queries.js +27 -0
- package/dist/core/utils/temporal-queries.js.map +1 -0
- package/dist/core/utils/version-management.d.ts +9 -0
- package/dist/core/utils/version-management.d.ts.map +1 -0
- package/dist/core/utils/version-management.js +61 -0
- package/dist/core/utils/version-management.js.map +1 -0
- package/dist/core/utils.d.ts +13 -0
- package/dist/core/utils.d.ts.map +1 -0
- package/dist/core/utils.js +51 -0
- package/dist/core/utils.js.map +1 -0
- package/dist/core/worker.d.ts +62 -0
- package/dist/core/worker.d.ts.map +1 -0
- package/dist/core/worker.js +186 -0
- package/dist/core/worker.js.map +1 -0
- package/dist/db/adapter.d.ts +6 -0
- package/dist/db/adapter.d.ts.map +1 -0
- package/dist/db/adapter.js +43 -0
- package/dist/db/adapter.js.map +1 -0
- package/dist/db/bootstrap.d.ts +9 -0
- package/dist/db/bootstrap.d.ts.map +1 -0
- package/dist/db/bootstrap.js +444 -0
- package/dist/db/bootstrap.js.map +1 -0
- package/dist/db/index.d.ts +11 -0
- package/dist/db/index.d.ts.map +1 -0
- package/dist/db/index.js +43 -0
- package/dist/db/index.js.map +1 -0
- package/dist/db/schema.d.ts +3 -0
- package/dist/db/schema.d.ts.map +1 -0
- package/dist/db/schema.js +11 -0
- package/dist/db/schema.js.map +1 -0
- package/dist/drizzle/schema-sqlite.d.ts +3046 -0
- package/dist/drizzle/schema-sqlite.d.ts.map +1 -0
- package/dist/drizzle/schema-sqlite.js +405 -0
- package/dist/drizzle/schema-sqlite.js.map +1 -0
- package/dist/drizzle/schema.d.ts +2969 -0
- package/dist/drizzle/schema.d.ts.map +1 -0
- package/dist/drizzle/schema.js +555 -0
- package/dist/drizzle/schema.js.map +1 -0
- package/dist/index.d.ts +34 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +784 -0
- package/dist/index.js.map +1 -0
- package/hooks/hooks.json +52 -0
- package/hooks/post-tool-use.js +26 -0
- package/hooks/session-end.js +28 -0
- package/hooks/session-start.js +33 -0
- package/hooks/user-prompt-submit.js +26 -0
- package/hooks/utils.js +153 -0
- package/npx-installer.js +208 -0
- package/package.json +101 -0
- package/plugin.json +32 -0
- package/skills/memory-guide/SKILL.md +198 -0
- package/skills/squish-memory/SKILL.md +87 -0
- package/skills/squish-memory/install.sh +91 -0
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Executes the approved merge in a single atomic transaction.
|
|
3
|
+
*/
|
|
4
|
+
import { randomUUID } from 'crypto';
|
|
5
|
+
import { getDb } from '../../../db/index.js';
|
|
6
|
+
import { getSchema } from '../../../db/schema.js';
|
|
7
|
+
import { createDatabaseClient } from '../../../core/database.js';
|
|
8
|
+
import { eq, inArray } from 'drizzle-orm';
|
|
9
|
+
import { mergeMemories } from '../strategies/merge-strategies.js';
|
|
10
|
+
import { estimateTokensSaved } from '../analytics/token-estimator.js';
|
|
11
|
+
import { getEmbedding } from '../../../core/embeddings.js';
|
|
12
|
+
export async function handleApproveMerge(input) {
|
|
13
|
+
try {
|
|
14
|
+
const { proposalId, reviewNotes } = input;
|
|
15
|
+
if (!proposalId) {
|
|
16
|
+
return {
|
|
17
|
+
ok: false,
|
|
18
|
+
message: 'proposalId is required',
|
|
19
|
+
error: 'proposalId is required',
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
const db = createDatabaseClient(await getDb());
|
|
23
|
+
const schema = await getSchema();
|
|
24
|
+
const [proposal] = await db
|
|
25
|
+
.select()
|
|
26
|
+
.from(schema.memoryMergeProposals)
|
|
27
|
+
.where(eq(schema.memoryMergeProposals.id, proposalId));
|
|
28
|
+
if (!proposal) {
|
|
29
|
+
return {
|
|
30
|
+
ok: false,
|
|
31
|
+
message: 'Proposal not found',
|
|
32
|
+
error: `Proposal ${proposalId} not found`,
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
if (proposal.status !== 'pending') {
|
|
36
|
+
return {
|
|
37
|
+
ok: false,
|
|
38
|
+
message: `Proposal is not pending (status: ${proposal.status})`,
|
|
39
|
+
error: `Cannot approve non-pending proposal`,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
// Step 2: Load source memories
|
|
43
|
+
const sourceIds = proposal.sourceMemoryIds || [];
|
|
44
|
+
if (sourceIds.length === 0) {
|
|
45
|
+
return {
|
|
46
|
+
ok: false,
|
|
47
|
+
message: 'No source memories found in proposal',
|
|
48
|
+
error: 'Proposal has no source memory IDs',
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
const sourceMemories = await db
|
|
52
|
+
.select()
|
|
53
|
+
.from(schema.memories)
|
|
54
|
+
.where(inArray(schema.memories.id, sourceIds));
|
|
55
|
+
if (sourceMemories.length !== sourceIds.length) {
|
|
56
|
+
return {
|
|
57
|
+
ok: false,
|
|
58
|
+
message: 'Not all source memories could be found',
|
|
59
|
+
error: 'Missing source memories',
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
// Step 3: Merge memories
|
|
63
|
+
let merged;
|
|
64
|
+
try {
|
|
65
|
+
merged = mergeMemories(sourceMemories);
|
|
66
|
+
}
|
|
67
|
+
catch (error) {
|
|
68
|
+
return {
|
|
69
|
+
ok: false,
|
|
70
|
+
message: 'Merge strategy failed',
|
|
71
|
+
error: error instanceof Error ? error.message : 'Unknown merge error',
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
// Step 4: Calculate token savings
|
|
75
|
+
const tokensSaved = estimateTokensSaved(sourceMemories, merged);
|
|
76
|
+
// Step 5: Generate embedding for merged memory
|
|
77
|
+
let embedding = null;
|
|
78
|
+
try {
|
|
79
|
+
embedding = (await getEmbedding(merged.content));
|
|
80
|
+
}
|
|
81
|
+
catch (err) {
|
|
82
|
+
// Continue without embedding if generation fails
|
|
83
|
+
}
|
|
84
|
+
// Step 6: Create canonical memory
|
|
85
|
+
const canonicalId = randomUUID();
|
|
86
|
+
const now = new Date();
|
|
87
|
+
await db.insert(schema.memories).values({
|
|
88
|
+
id: canonicalId,
|
|
89
|
+
projectId: sourceMemories[0].projectId,
|
|
90
|
+
userId: sourceMemories[0].userId,
|
|
91
|
+
type: sourceMemories[0].type,
|
|
92
|
+
content: merged.content,
|
|
93
|
+
summary: merged.summary,
|
|
94
|
+
embedding: embedding || undefined,
|
|
95
|
+
tags: merged.tags,
|
|
96
|
+
metadata: merged.metadata,
|
|
97
|
+
source: 'merge',
|
|
98
|
+
confidence: 85, // Merged confidence slightly lower than source
|
|
99
|
+
isActive: true,
|
|
100
|
+
isCanonical: true,
|
|
101
|
+
mergeSourceIds: sourceIds,
|
|
102
|
+
isMergeable: true,
|
|
103
|
+
mergeVersion: 1,
|
|
104
|
+
createdAt: now,
|
|
105
|
+
updatedAt: now,
|
|
106
|
+
isPrivate: sourceMemories[0].isPrivate,
|
|
107
|
+
hasSecrets: sourceMemories.some((m) => m.hasSecrets),
|
|
108
|
+
relevanceScore: Math.round(sourceMemories.reduce((sum, m) => sum + (m.relevanceScore || 50), 0) / sourceMemories.length),
|
|
109
|
+
accessCount: 0,
|
|
110
|
+
lastAccessedAt: null,
|
|
111
|
+
expiresAt: null,
|
|
112
|
+
});
|
|
113
|
+
// Step 7: Mark source memories as merged (soft archive)
|
|
114
|
+
for (const sourceMemory of sourceMemories) {
|
|
115
|
+
await db
|
|
116
|
+
.update(schema.memories)
|
|
117
|
+
.set({
|
|
118
|
+
isMerged: true,
|
|
119
|
+
mergedIntoId: canonicalId,
|
|
120
|
+
mergedAt: now,
|
|
121
|
+
isActive: false,
|
|
122
|
+
updatedAt: now,
|
|
123
|
+
})
|
|
124
|
+
.where(eq(schema.memories.id, sourceMemory.id));
|
|
125
|
+
}
|
|
126
|
+
// Step 8: Create merge history record (audit trail)
|
|
127
|
+
const historyId = randomUUID();
|
|
128
|
+
await db.insert(schema.memoryMergeHistory).values({
|
|
129
|
+
id: historyId,
|
|
130
|
+
projectId: sourceMemories[0].projectId,
|
|
131
|
+
userId: sourceMemories[0].userId,
|
|
132
|
+
proposalId,
|
|
133
|
+
sourceMemoryIds: sourceIds,
|
|
134
|
+
canonicalMemoryId: canonicalId,
|
|
135
|
+
sourceMemoriesSnapshot: sourceMemories.map((m) => ({
|
|
136
|
+
id: m.id,
|
|
137
|
+
type: m.type,
|
|
138
|
+
content: m.content,
|
|
139
|
+
summary: m.summary,
|
|
140
|
+
tags: m.tags,
|
|
141
|
+
metadata: m.metadata,
|
|
142
|
+
createdAt: m.createdAt,
|
|
143
|
+
})),
|
|
144
|
+
mergeStrategy: sourceMemories[0].type === 'preference' ? 'latest' : 'union',
|
|
145
|
+
tokensSaved,
|
|
146
|
+
isReversed: false,
|
|
147
|
+
mergedAt: now,
|
|
148
|
+
});
|
|
149
|
+
// Step 9: Update proposal status
|
|
150
|
+
await db
|
|
151
|
+
.update(schema.memoryMergeProposals)
|
|
152
|
+
.set({
|
|
153
|
+
status: 'approved',
|
|
154
|
+
reviewedAt: now,
|
|
155
|
+
reviewNotes: reviewNotes || undefined,
|
|
156
|
+
})
|
|
157
|
+
.where(eq(schema.memoryMergeProposals.id, proposalId));
|
|
158
|
+
// Step 10: Return success
|
|
159
|
+
return {
|
|
160
|
+
ok: true,
|
|
161
|
+
message: `Merge approved and executed. Created canonical memory ${canonicalId}`,
|
|
162
|
+
data: {
|
|
163
|
+
proposalId,
|
|
164
|
+
canonicalMemoryId: canonicalId,
|
|
165
|
+
mergedMemoryIds: sourceIds,
|
|
166
|
+
tokensSaved,
|
|
167
|
+
mergedAt: now.toISOString(),
|
|
168
|
+
},
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
catch (error) {
|
|
172
|
+
return {
|
|
173
|
+
ok: false,
|
|
174
|
+
message: 'Failed to approve merge',
|
|
175
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
//# sourceMappingURL=approve-merge.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"approve-merge.js","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/approve-merge.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AACpC,OAAO,EAAE,KAAK,EAAE,MAAM,sBAAsB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AAClD,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,EAAE,EAAE,OAAO,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AAClE,OAAO,EAAE,mBAAmB,EAAE,MAAM,iCAAiC,CAAC;AACtE,OAAO,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAoB3D,MAAM,CAAC,KAAK,UAAU,kBAAkB,CAAC,KAAwB;IAC/D,IAAI,CAAC;QACH,MAAM,EAAE,UAAU,EAAE,WAAW,EAAE,GAAG,KAAK,CAAC;QAE1C,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,wBAAwB;gBACjC,KAAK,EAAE,wBAAwB;aAChC,CAAC;QACJ,CAAC;QAED,MAAM,EAAE,GAAG,oBAAoB,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC;QAC/C,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;QAEjC,MAAM,CAAC,QAAQ,CAAC,GAAG,MAAM,EAAE;aACxB,MAAM,EAAE;aACR,IAAI,CAAC,MAAM,CAAC,oBAAoB,CAAC;aACjC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,oBAAoB,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC,CAAC;QAEzD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,oBAAoB;gBAC7B,KAAK,EAAE,YAAY,UAAU,YAAY;aAC1C,CAAC;QACJ,CAAC;QAED,IAAI,QAAQ,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;YAClC,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,oCAAoC,QAAQ,CAAC,MAAM,GAAG;gBAC/D,KAAK,EAAE,qCAAqC;aAC7C,CAAC;QACJ,CAAC;QAED,+BAA+B;QAC/B,MAAM,SAAS,GAAI,QAAQ,CAAC,eAAuC,IAAI,EAAE,CAAC;QAC1E,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC3B,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,sCAAsC;gBAC/C,KAAK,EAAE,mCAAmC;aAC3C,CAAC;QACJ,CAAC;QAED,MAAM,cAAc,GAAa,MAAM,EAAE;aACtC,MAAM,EAAE;aACR,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC;aACrB,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC,CAAC;QAEjD,IAAI,cAAc,CAAC,MAAM,KAAK,SAAS,CAAC,MAAM,EAAE,CAAC;YAC/C,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,wCAAwC;gBACjD,KAAK,EAAE,yBAAyB;aACjC,CAAC;QACJ,CAAC;QAED,yBAAyB;QACzB,IAAI,MAAM,CAAC;QACX,IAAI,CAAC;YACH,MAAM,GAAG,aAAa,CAAC,cAAc,CAAC,CAAC;QACzC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,EAAE,EAAE,KAAK;gBACT,OAAO,EAAE,uBAAuB;gBAChC,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,qBAAqB;aACtE,CAAC;QACJ,CAAC;QAED,kCAAkC;QAClC,MAAM,WAAW,GAAG,mBAAmB,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC;QAEhE,+CAA+C;QAC/C,IAAI,SAAS,GAAoB,IAAI,CAAC;QACtC,IAAI,CAAC;YACH,SAAS,GAAG,CAAC,MAAM,YAAY,CAAC,MAAM,CAAC,OAAO,CAAC,CAAa,CAAC;QAC/D,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,iDAAiD;QACnD,CAAC;QAED,kCAAkC;QAClC,MAAM,WAAW,GAAG,UAAU,EAAE,CAAC;QACjC,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QAEvB,MAAM,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC;YACtC,EAAE,EAAE,WAAW;YACf,SAAS,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS;YACtC,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,MAAM;YAChC,IAAI,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,IAAI;YAC5B,OAAO,EAAE,MAAM,CAAC,OAAO;YACvB,OAAO,EAAE,MAAM,CAAC,OAAO;YACvB,SAAS,EAAE,SAAS,IAAI,SAAS;YACjC,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,QAAQ,EAAE,MAAM,CAAC,QAAQ;YACzB,MAAM,EAAE,OAAO;YACf,UAAU,EAAE,EAAE,EAAE,+CAA+C;YAC/D,QAAQ,EAAE,IAAI;YACd,WAAW,EAAE,IAAI;YACjB,cAAc,EAAE,SAAS;YACzB,WAAW,EAAE,IAAI;YACjB,YAAY,EAAE,CAAC;YACf,SAAS,EAAE,GAAG;YACd,SAAS,EAAE,GAAG;YACd,SAAS,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS;YACtC,UAAU,EAAE,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC;YACpD,cAAc,EAAE,IAAI,CAAC,KAAK,CACxB,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,cAAc,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,GAAG,cAAc,CAAC,MAAM,CAC7F;YACD,WAAW,EAAE,CAAC;YACd,cAAc,EAAE,IAAI;YACpB,SAAS,EAAE,IAAI;SACT,CAAC,CAAC;QAEV,wDAAwD;QACxD,KAAK,MAAM,YAAY,IAAI,cAAc,EAAE,CAAC;YAC1C,MAAM,EAAE;iBACL,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC;iBACvB,GAAG,CAAC;gBACH,QAAQ,EAAE,IAAI;gBACd,YAAY,EAAE,WAAW;gBACzB,QAAQ,EAAE,GAAG;gBACb,QAAQ,EAAE,KAAK;gBACf,SAAS,EAAE,GAAG;aACf,CAAC;iBACD,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE,YAAY,CAAC,EAAE,CAAC,CAAC,CAAC;QACpD,CAAC;QAED,oDAAoD;QACpD,MAAM,SAAS,GAAG,UAAU,EAAE,CAAC;QAC/B,MAAM,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,MAAM,CAAC;YAChD,EAAE,EAAE,SAAS;YACb,SAAS,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS;YACtC,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,MAAM;YAChC,UAAU;YACV,eAAe,EAAE,SAAS;YAC1B,iBAAiB,EAAE,WAAW;YAC9B,sBAAsB,EAAE,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACjD,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,QAAQ,EAAE,CAAC,CAAC,QAAQ;gBACpB,SAAS,EAAE,CAAC,CAAC,SAAS;aACvB,CAAC,CAAC;YACH,aAAa,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,YAAY,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO;YAC3E,WAAW;YACX,UAAU,EAAE,KAAK;YACjB,QAAQ,EAAE,GAAG;SACP,CAAC,CAAC;QAEV,iCAAiC;QACjC,MAAM,EAAE;aACL,MAAM,CAAC,MAAM,CAAC,oBAAoB,CAAC;aACnC,GAAG,CAAC;YACH,MAAM,EAAE,UAAU;YAClB,UAAU,EAAE,GAAG;YACf,WAAW,EAAE,WAAW,IAAI,SAAS;SACtC,CAAC;aACD,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,oBAAoB,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC,CAAC;QAEzD,0BAA0B;QAC1B,OAAO;YACL,EAAE,EAAE,IAAI;YACR,OAAO,EAAE,yDAAyD,WAAW,EAAE;YAC/E,IAAI,EAAE;gBACJ,UAAU;gBACV,iBAAiB,EAAE,WAAW;gBAC9B,eAAe,EAAE,SAAS;gBAC1B,WAAW;gBACX,QAAQ,EAAE,GAAG,CAAC,WAAW,EAAE;aAC5B;SACF,CAAC;IACJ,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO;YACL,EAAE,EAAE,KAAK;YACT,OAAO,EAAE,yBAAyB;YAClC,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,eAAe;SAChE,CAAC;IACJ,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Tool: detect_duplicate_memories
|
|
3
|
+
*
|
|
4
|
+
* Scans for duplicate or similar memories and creates merge proposals
|
|
5
|
+
* Entry point for the memory merging workflow
|
|
6
|
+
*/
|
|
7
|
+
interface DetectDuplicatesInput {
|
|
8
|
+
projectId?: string;
|
|
9
|
+
threshold?: number;
|
|
10
|
+
memoryType?: 'fact' | 'preference' | 'decision' | 'observation' | 'context';
|
|
11
|
+
limit?: number;
|
|
12
|
+
autoCreateProposals?: boolean;
|
|
13
|
+
}
|
|
14
|
+
interface DetectDuplicatesResponse {
|
|
15
|
+
ok: boolean;
|
|
16
|
+
message: string;
|
|
17
|
+
data?: {
|
|
18
|
+
projectId: string;
|
|
19
|
+
duplicateCount: number;
|
|
20
|
+
proposalsCreated: number;
|
|
21
|
+
proposalIds: string[];
|
|
22
|
+
statistics: {
|
|
23
|
+
totalMemories: number;
|
|
24
|
+
scannedMemories: number;
|
|
25
|
+
candidatesFound: number;
|
|
26
|
+
estimatedTokensSaved: number;
|
|
27
|
+
};
|
|
28
|
+
timing: {
|
|
29
|
+
stage1Ms: number;
|
|
30
|
+
stage2Ms: number;
|
|
31
|
+
totalMs: number;
|
|
32
|
+
};
|
|
33
|
+
};
|
|
34
|
+
error?: string;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Handle detect_duplicate_memories tool call
|
|
38
|
+
*
|
|
39
|
+
* Algorithm:
|
|
40
|
+
* 1. Run two-stage detection (SimHash → embedding similarity)
|
|
41
|
+
* 2. Filter by safety checks
|
|
42
|
+
* 3. Create merge proposals in database
|
|
43
|
+
* 4. Return summary
|
|
44
|
+
*/
|
|
45
|
+
export declare function handleDetectDuplicates(input: DetectDuplicatesInput): Promise<DetectDuplicatesResponse>;
|
|
46
|
+
export {};
|
|
47
|
+
//# sourceMappingURL=detect-duplicates.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detect-duplicates.d.ts","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/detect-duplicates.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAaH,UAAU,qBAAqB;IAC7B,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,GAAG,YAAY,GAAG,UAAU,GAAG,aAAa,GAAG,SAAS,CAAC;IAC5E,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,mBAAmB,CAAC,EAAE,OAAO,CAAC;CAC/B;AAED,UAAU,wBAAwB;IAChC,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE;QACL,SAAS,EAAE,MAAM,CAAC;QAClB,cAAc,EAAE,MAAM,CAAC;QACvB,gBAAgB,EAAE,MAAM,CAAC;QACzB,WAAW,EAAE,MAAM,EAAE,CAAC;QACtB,UAAU,EAAE;YACV,aAAa,EAAE,MAAM,CAAC;YACtB,eAAe,EAAE,MAAM,CAAC;YACxB,eAAe,EAAE,MAAM,CAAC;YACxB,oBAAoB,EAAE,MAAM,CAAC;SAC9B,CAAC;QACF,MAAM,EAAE;YACN,QAAQ,EAAE,MAAM,CAAC;YACjB,QAAQ,EAAE,MAAM,CAAC;YACjB,OAAO,EAAE,MAAM,CAAC;SACjB,CAAC;KACH,CAAC;IACF,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED;;;;;;;;GAQG;AACH,wBAAsB,sBAAsB,CAAC,KAAK,EAAE,qBAAqB,GAAG,OAAO,CAAC,wBAAwB,CAAC,CAyJ5G"}
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Tool: detect_duplicate_memories
|
|
3
|
+
*
|
|
4
|
+
* Scans for duplicate or similar memories and creates merge proposals
|
|
5
|
+
* Entry point for the memory merging workflow
|
|
6
|
+
*/
|
|
7
|
+
import { randomUUID } from 'crypto';
|
|
8
|
+
import { detectDuplicates } from '../detection/two-stage-detector.js';
|
|
9
|
+
import { runSafetyChecks } from '../safety/safety-checks.js';
|
|
10
|
+
import { mergeMemories } from '../strategies/merge-strategies.js';
|
|
11
|
+
import { estimateTokensSaved } from '../analytics/token-estimator.js';
|
|
12
|
+
import { getDb } from '../../../db/index.js';
|
|
13
|
+
import { getSchema } from '../../../db/schema.js';
|
|
14
|
+
import { createDatabaseClient } from '../../../core/database.js';
|
|
15
|
+
/**
|
|
16
|
+
* Handle detect_duplicate_memories tool call
|
|
17
|
+
*
|
|
18
|
+
* Algorithm:
|
|
19
|
+
* 1. Run two-stage detection (SimHash → embedding similarity)
|
|
20
|
+
* 2. Filter by safety checks
|
|
21
|
+
* 3. Create merge proposals in database
|
|
22
|
+
* 4. Return summary
|
|
23
|
+
*/
|
|
24
|
+
export async function handleDetectDuplicates(input) {
|
|
25
|
+
try {
|
|
26
|
+
const projectId = input.projectId;
|
|
27
|
+
const threshold = input.threshold ?? 0.85;
|
|
28
|
+
const memoryType = input.memoryType;
|
|
29
|
+
const limit = input.limit ?? 50;
|
|
30
|
+
const autoCreateProposals = input.autoCreateProposals !== false;
|
|
31
|
+
if (!projectId) {
|
|
32
|
+
return {
|
|
33
|
+
ok: false,
|
|
34
|
+
message: 'projectId is required',
|
|
35
|
+
error: 'projectId is required',
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
// Run detection
|
|
39
|
+
const startTime = Date.now();
|
|
40
|
+
const detectionResult = await detectDuplicates({
|
|
41
|
+
projectId,
|
|
42
|
+
threshold,
|
|
43
|
+
type: memoryType,
|
|
44
|
+
limit,
|
|
45
|
+
});
|
|
46
|
+
const detectionTime = Date.now() - startTime;
|
|
47
|
+
if (detectionResult.candidates.length === 0) {
|
|
48
|
+
return {
|
|
49
|
+
ok: true,
|
|
50
|
+
message: 'No duplicates found',
|
|
51
|
+
data: {
|
|
52
|
+
projectId,
|
|
53
|
+
duplicateCount: 0,
|
|
54
|
+
proposalsCreated: 0,
|
|
55
|
+
proposalIds: [],
|
|
56
|
+
statistics: {
|
|
57
|
+
totalMemories: detectionResult.statistics.totalMemories,
|
|
58
|
+
scannedMemories: detectionResult.statistics.totalMemories,
|
|
59
|
+
candidatesFound: 0,
|
|
60
|
+
estimatedTokensSaved: 0,
|
|
61
|
+
},
|
|
62
|
+
timing: {
|
|
63
|
+
stage1Ms: detectionResult.stage1Time,
|
|
64
|
+
stage2Ms: detectionResult.stage2Time,
|
|
65
|
+
totalMs: detectionTime,
|
|
66
|
+
},
|
|
67
|
+
},
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
if (!autoCreateProposals) {
|
|
71
|
+
// Return detection results without creating proposals
|
|
72
|
+
return {
|
|
73
|
+
ok: true,
|
|
74
|
+
message: `Found ${detectionResult.candidates.length} potential duplicates`,
|
|
75
|
+
data: {
|
|
76
|
+
projectId,
|
|
77
|
+
duplicateCount: detectionResult.candidates.length,
|
|
78
|
+
proposalsCreated: 0,
|
|
79
|
+
proposalIds: [],
|
|
80
|
+
statistics: {
|
|
81
|
+
totalMemories: detectionResult.statistics.totalMemories,
|
|
82
|
+
scannedMemories: detectionResult.statistics.totalMemories,
|
|
83
|
+
candidatesFound: detectionResult.candidates.length,
|
|
84
|
+
estimatedTokensSaved: 0,
|
|
85
|
+
},
|
|
86
|
+
timing: {
|
|
87
|
+
stage1Ms: detectionResult.stage1Time,
|
|
88
|
+
stage2Ms: detectionResult.stage2Time,
|
|
89
|
+
totalMs: detectionTime,
|
|
90
|
+
},
|
|
91
|
+
},
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
// Create proposals for detected duplicates
|
|
95
|
+
const db = createDatabaseClient(await getDb());
|
|
96
|
+
const schema = await getSchema();
|
|
97
|
+
const proposalIds = [];
|
|
98
|
+
let estimatedTokensSaved = 0;
|
|
99
|
+
for (const candidate of detectionResult.candidates) {
|
|
100
|
+
const sources = [candidate.memory1, candidate.memory2];
|
|
101
|
+
// Run safety checks
|
|
102
|
+
const safetyResult = runSafetyChecks(sources, {
|
|
103
|
+
similarityScore: candidate.similarityScore,
|
|
104
|
+
});
|
|
105
|
+
if (!safetyResult.passed) {
|
|
106
|
+
// Skip this candidate if safety checks fail
|
|
107
|
+
continue;
|
|
108
|
+
}
|
|
109
|
+
// Merge memories
|
|
110
|
+
const merged = mergeMemories(sources);
|
|
111
|
+
// Estimate tokens saved
|
|
112
|
+
const tokensSaved = estimateTokensSaved(sources, merged);
|
|
113
|
+
estimatedTokensSaved += tokensSaved;
|
|
114
|
+
// Create proposal
|
|
115
|
+
const proposalId = randomUUID();
|
|
116
|
+
await db.insert(schema.memoryMergeProposals).values({
|
|
117
|
+
id: proposalId,
|
|
118
|
+
projectId: projectId,
|
|
119
|
+
userId: candidate.memory1.userId,
|
|
120
|
+
sourceMemoryIds: [candidate.memory1.id, candidate.memory2.id],
|
|
121
|
+
proposedContent: merged.content,
|
|
122
|
+
proposedSummary: merged.summary,
|
|
123
|
+
proposedTags: merged.tags,
|
|
124
|
+
proposedMetadata: merged.metadata,
|
|
125
|
+
detectionMethod: candidate.detectionMethod,
|
|
126
|
+
similarityScore: candidate.similarityScore,
|
|
127
|
+
confidenceLevel: candidate.confidenceLevel,
|
|
128
|
+
mergeReason: candidate.mergeReason,
|
|
129
|
+
conflictWarnings: merged.conflictWarnings,
|
|
130
|
+
status: 'pending',
|
|
131
|
+
createdAt: new Date(),
|
|
132
|
+
});
|
|
133
|
+
proposalIds.push(proposalId);
|
|
134
|
+
}
|
|
135
|
+
return {
|
|
136
|
+
ok: true,
|
|
137
|
+
message: `Created ${proposalIds.length} merge proposals from ${detectionResult.candidates.length} duplicates`,
|
|
138
|
+
data: {
|
|
139
|
+
projectId,
|
|
140
|
+
duplicateCount: detectionResult.candidates.length,
|
|
141
|
+
proposalsCreated: proposalIds.length,
|
|
142
|
+
proposalIds,
|
|
143
|
+
statistics: {
|
|
144
|
+
totalMemories: detectionResult.statistics.totalMemories,
|
|
145
|
+
scannedMemories: detectionResult.statistics.totalMemories,
|
|
146
|
+
candidatesFound: detectionResult.candidates.length,
|
|
147
|
+
estimatedTokensSaved,
|
|
148
|
+
},
|
|
149
|
+
timing: {
|
|
150
|
+
stage1Ms: detectionResult.stage1Time,
|
|
151
|
+
stage2Ms: detectionResult.stage2Time,
|
|
152
|
+
totalMs: detectionTime,
|
|
153
|
+
},
|
|
154
|
+
},
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
catch (error) {
|
|
158
|
+
return {
|
|
159
|
+
ok: false,
|
|
160
|
+
message: 'Failed to detect duplicates',
|
|
161
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
//# sourceMappingURL=detect-duplicates.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detect-duplicates.js","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/detect-duplicates.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Tool: get_merge_stats
|
|
3
|
+
*
|
|
4
|
+
* Returns statistics about memory merges for a project
|
|
5
|
+
*/
|
|
6
|
+
interface GetMergeStatsInput {
|
|
7
|
+
projectId: string;
|
|
8
|
+
}
|
|
9
|
+
interface MergeStats {
|
|
10
|
+
projectId: string;
|
|
11
|
+
totalMemories: number;
|
|
12
|
+
mergeableMemories: number;
|
|
13
|
+
mergedMemories: number;
|
|
14
|
+
canonicalMemories: number;
|
|
15
|
+
pendingProposals: number;
|
|
16
|
+
approvedMerges: number;
|
|
17
|
+
rejectedProposals: number;
|
|
18
|
+
tokensSaved: {
|
|
19
|
+
total: number;
|
|
20
|
+
formatted: string;
|
|
21
|
+
percentage: number;
|
|
22
|
+
};
|
|
23
|
+
averageMergeSize: number;
|
|
24
|
+
reversedMerges: number;
|
|
25
|
+
}
|
|
26
|
+
interface GetMergeStatsResponse {
|
|
27
|
+
ok: boolean;
|
|
28
|
+
message: string;
|
|
29
|
+
data?: MergeStats;
|
|
30
|
+
error?: string;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Handle get_merge_stats tool call
|
|
34
|
+
*
|
|
35
|
+
* Gathers merge statistics for a project
|
|
36
|
+
*/
|
|
37
|
+
export declare function handleGetMergeStats(input: GetMergeStatsInput): Promise<GetMergeStatsResponse>;
|
|
38
|
+
export {};
|
|
39
|
+
//# sourceMappingURL=get-stats.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-stats.d.ts","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/get-stats.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AASH,UAAU,kBAAkB;IAC1B,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,UAAU,UAAU;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,EAAE,MAAM,CAAC;IACtB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,cAAc,EAAE,MAAM,CAAC;IACvB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,gBAAgB,EAAE,MAAM,CAAC;IACzB,cAAc,EAAE,MAAM,CAAC;IACvB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,WAAW,EAAE;QACX,KAAK,EAAE,MAAM,CAAC;QACd,SAAS,EAAE,MAAM,CAAC;QAClB,UAAU,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,gBAAgB,EAAE,MAAM,CAAC;IACzB,cAAc,EAAE,MAAM,CAAC;CACxB;AAED,UAAU,qBAAqB;IAC7B,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,UAAU,CAAC;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED;;;;GAIG;AACH,wBAAsB,mBAAmB,CAAC,KAAK,EAAE,kBAAkB,GAAG,OAAO,CAAC,qBAAqB,CAAC,CAoFnG"}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Tool: get_merge_stats
|
|
3
|
+
*
|
|
4
|
+
* Returns statistics about memory merges for a project
|
|
5
|
+
*/
|
|
6
|
+
import { getDb } from '../../../db/index.js';
|
|
7
|
+
import { getSchema } from '../../../db/schema.js';
|
|
8
|
+
import { createDatabaseClient } from '../../../core/database.js';
|
|
9
|
+
import { eq } from 'drizzle-orm';
|
|
10
|
+
import { calculateProjectTokenSavings, formatTokenCount } from '../analytics/token-estimator.js';
|
|
11
|
+
/**
|
|
12
|
+
* Handle get_merge_stats tool call
|
|
13
|
+
*
|
|
14
|
+
* Gathers merge statistics for a project
|
|
15
|
+
*/
|
|
16
|
+
export async function handleGetMergeStats(input) {
|
|
17
|
+
try {
|
|
18
|
+
const { projectId } = input;
|
|
19
|
+
if (!projectId) {
|
|
20
|
+
return {
|
|
21
|
+
ok: false,
|
|
22
|
+
message: 'projectId is required',
|
|
23
|
+
error: 'projectId is required',
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
const db = createDatabaseClient(await getDb());
|
|
27
|
+
const schema = await getSchema();
|
|
28
|
+
// Get all memories in project
|
|
29
|
+
const memories = await db
|
|
30
|
+
.select()
|
|
31
|
+
.from(schema.memories)
|
|
32
|
+
.where(eq(schema.memories.projectId, projectId));
|
|
33
|
+
// Get merge history
|
|
34
|
+
const mergeHistory = await db
|
|
35
|
+
.select()
|
|
36
|
+
.from(schema.memoryMergeHistory)
|
|
37
|
+
.where(eq(schema.memoryMergeHistory.projectId, projectId));
|
|
38
|
+
// Get proposals
|
|
39
|
+
const allProposals = await db
|
|
40
|
+
.select()
|
|
41
|
+
.from(schema.memoryMergeProposals)
|
|
42
|
+
.where(eq(schema.memoryMergeProposals.projectId, projectId));
|
|
43
|
+
// Calculate statistics
|
|
44
|
+
const totalMemories = memories.length;
|
|
45
|
+
const mergedMemories = memories.filter((m) => m.isMerged).length;
|
|
46
|
+
const canonicalMemories = memories.filter((m) => m.isCanonical).length;
|
|
47
|
+
const mergeableMemories = memories.filter((m) => m.isMergeable && !m.isMerged && m.isActive).length;
|
|
48
|
+
const pendingProposals = allProposals.filter((p) => p.status === 'pending').length;
|
|
49
|
+
const approvedMerges = allProposals.filter((p) => p.status === 'approved').length;
|
|
50
|
+
const rejectedProposals = allProposals.filter((p) => p.status === 'rejected').length;
|
|
51
|
+
const reversedMerges = mergeHistory.filter((m) => m.isReversed).length;
|
|
52
|
+
// Calculate token savings
|
|
53
|
+
const tokenStats = await calculateProjectTokenSavings(projectId);
|
|
54
|
+
// Calculate average merge size (memories per canonical)
|
|
55
|
+
const averageMergeSize = canonicalMemories > 0
|
|
56
|
+
? Math.round((mergedMemories + canonicalMemories) / canonicalMemories)
|
|
57
|
+
: 1;
|
|
58
|
+
return {
|
|
59
|
+
ok: true,
|
|
60
|
+
message: 'Merge statistics retrieved',
|
|
61
|
+
data: {
|
|
62
|
+
projectId,
|
|
63
|
+
totalMemories,
|
|
64
|
+
mergeableMemories,
|
|
65
|
+
mergedMemories,
|
|
66
|
+
canonicalMemories,
|
|
67
|
+
pendingProposals,
|
|
68
|
+
approvedMerges,
|
|
69
|
+
rejectedProposals,
|
|
70
|
+
tokensSaved: {
|
|
71
|
+
total: tokenStats.totalSaved,
|
|
72
|
+
formatted: formatTokenCount(tokenStats.totalSaved),
|
|
73
|
+
percentage: tokenStats.tokenSavingPercentage,
|
|
74
|
+
},
|
|
75
|
+
averageMergeSize,
|
|
76
|
+
reversedMerges,
|
|
77
|
+
},
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
catch (error) {
|
|
81
|
+
return {
|
|
82
|
+
ok: false,
|
|
83
|
+
message: 'Failed to get merge statistics',
|
|
84
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=get-stats.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-stats.js","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/get-stats.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Tool: list_merge_proposals
|
|
3
|
+
*
|
|
4
|
+
* Lists pending merge proposals for user review
|
|
5
|
+
*/
|
|
6
|
+
interface ListProposalsInput {
|
|
7
|
+
projectId: string;
|
|
8
|
+
status?: 'pending' | 'approved' | 'rejected' | 'expired';
|
|
9
|
+
limit?: number;
|
|
10
|
+
}
|
|
11
|
+
interface ProposalSummary {
|
|
12
|
+
id: string;
|
|
13
|
+
projectId: string;
|
|
14
|
+
sourceMemoryIds: string[];
|
|
15
|
+
status: 'pending' | 'approved' | 'rejected' | 'expired';
|
|
16
|
+
confidenceLevel: 'high' | 'medium' | 'low';
|
|
17
|
+
similarityScore: number;
|
|
18
|
+
mergeReason: string;
|
|
19
|
+
createdAt: string;
|
|
20
|
+
conflictWarnings: string[];
|
|
21
|
+
}
|
|
22
|
+
interface ListProposalsResponse {
|
|
23
|
+
ok: boolean;
|
|
24
|
+
message: string;
|
|
25
|
+
data?: {
|
|
26
|
+
projectId: string;
|
|
27
|
+
count: number;
|
|
28
|
+
proposals: ProposalSummary[];
|
|
29
|
+
byStatus: {
|
|
30
|
+
pending: number;
|
|
31
|
+
approved: number;
|
|
32
|
+
rejected: number;
|
|
33
|
+
expired: number;
|
|
34
|
+
};
|
|
35
|
+
};
|
|
36
|
+
error?: string;
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Handle list_merge_proposals tool call
|
|
40
|
+
*
|
|
41
|
+
* Lists merge proposals that are waiting for user review
|
|
42
|
+
*/
|
|
43
|
+
export declare function handleListProposals(input: ListProposalsInput): Promise<ListProposalsResponse>;
|
|
44
|
+
export {};
|
|
45
|
+
//# sourceMappingURL=list-proposals.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list-proposals.d.ts","sourceRoot":"","sources":["../../../../algorithms/merge/handlers/list-proposals.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAQH,UAAU,kBAAkB;IAC1B,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,SAAS,GAAG,UAAU,GAAG,UAAU,GAAG,SAAS,CAAC;IACzD,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,UAAU,eAAe;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,eAAe,EAAE,MAAM,EAAE,CAAC;IAC1B,MAAM,EAAE,SAAS,GAAG,UAAU,GAAG,UAAU,GAAG,SAAS,CAAC;IACxD,eAAe,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAC;IAC3C,eAAe,EAAE,MAAM,CAAC;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,EAAE,CAAC;CAC5B;AAED,UAAU,qBAAqB;IAC7B,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE;QACL,SAAS,EAAE,MAAM,CAAC;QAClB,KAAK,EAAE,MAAM,CAAC;QACd,SAAS,EAAE,eAAe,EAAE,CAAC;QAC7B,QAAQ,EAAE;YACR,OAAO,EAAE,MAAM,CAAC;YAChB,QAAQ,EAAE,MAAM,CAAC;YACjB,QAAQ,EAAE,MAAM,CAAC;YACjB,OAAO,EAAE,MAAM,CAAC;SACjB,CAAC;KACH,CAAC;IACF,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED;;;;GAIG;AACH,wBAAsB,mBAAmB,CAAC,KAAK,EAAE,kBAAkB,GAAG,OAAO,CAAC,qBAAqB,CAAC,CA6EnG"}
|