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,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Privacy System
|
|
3
|
+
* Handles private tag filtering, secret detection, and path-based exclusions
|
|
4
|
+
*/
|
|
5
|
+
import { detectSecrets } from './secret-detector.js';
|
|
6
|
+
const DENY_PATTERNS = [
|
|
7
|
+
'.env', '.env.local', '.env.*.local',
|
|
8
|
+
'*.key', '*.pem',
|
|
9
|
+
'secrets.json', 'credentials.json',
|
|
10
|
+
'.aws', '.ssh', 'node_modules/.env'
|
|
11
|
+
];
|
|
12
|
+
const PRIVATE_TAG_REGEX = /<(private|secret)>[\s\S]*?<\/\1>/gi;
|
|
13
|
+
export function hasPrivateTags(content) {
|
|
14
|
+
return PRIVATE_TAG_REGEX.test(content);
|
|
15
|
+
}
|
|
16
|
+
export function stripPrivateTags(content) {
|
|
17
|
+
return content.replace(/<private>[\s\S]*?<\/private>/gi, '[PRIVATE]')
|
|
18
|
+
.replace(/<secret>[\s\S]*?<\/secret>/gi, '[SECRET]');
|
|
19
|
+
}
|
|
20
|
+
function isInDenyList(filePath) {
|
|
21
|
+
const normalized = filePath.toLowerCase().replace(/\\/g, '/');
|
|
22
|
+
for (const pattern of DENY_PATTERNS) {
|
|
23
|
+
if (normalized === pattern)
|
|
24
|
+
return true;
|
|
25
|
+
if (pattern.startsWith('*.') && normalized.endsWith(pattern.substring(1)))
|
|
26
|
+
return true;
|
|
27
|
+
if (pattern.endsWith('/*')) {
|
|
28
|
+
const dir = pattern.slice(0, -2);
|
|
29
|
+
if (normalized.includes(`/${dir}/`) || normalized.startsWith(`${dir}/`))
|
|
30
|
+
return true;
|
|
31
|
+
}
|
|
32
|
+
if (normalized.includes(pattern))
|
|
33
|
+
return true;
|
|
34
|
+
}
|
|
35
|
+
return false;
|
|
36
|
+
}
|
|
37
|
+
export async function shouldStore(content, filePath, privacyMode = 'moderate') {
|
|
38
|
+
if (privacyMode === 'off')
|
|
39
|
+
return true;
|
|
40
|
+
if (hasPrivateTags(content))
|
|
41
|
+
return false;
|
|
42
|
+
if (filePath && isInDenyList(filePath))
|
|
43
|
+
return false;
|
|
44
|
+
if (detectSecrets(content).length > 0)
|
|
45
|
+
return false;
|
|
46
|
+
return true;
|
|
47
|
+
}
|
|
48
|
+
export async function applyPrivacyFilters(items) {
|
|
49
|
+
const results = [];
|
|
50
|
+
for (const item of items) {
|
|
51
|
+
const content = item.content || JSON.stringify(item.details || '');
|
|
52
|
+
if (await shouldStore(content)) {
|
|
53
|
+
results.push(item);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return results;
|
|
57
|
+
}
|
|
58
|
+
export function sanitizeForStorage(content, privacyMode) {
|
|
59
|
+
let sanitized = stripPrivateTags(content);
|
|
60
|
+
if (privacyMode !== 'off') {
|
|
61
|
+
const secrets = detectSecrets(sanitized);
|
|
62
|
+
for (let i = secrets.length - 1; i >= 0; i--) {
|
|
63
|
+
const secret = secrets[i];
|
|
64
|
+
sanitized = sanitized.substring(0, secret.start) + '[REDACTED]' + sanitized.substring(secret.end);
|
|
65
|
+
}
|
|
66
|
+
sanitized = sanitized
|
|
67
|
+
.replace(/([a-zA-Z0-9_-]*[Aa]pi[_-]?[Kk]ey|Bearer\s+[^\s]+)/g, '[API_KEY]')
|
|
68
|
+
.replace(/(token|jwt|bearer)[\s=:]+[^\s,]+/gi, '[TOKEN]');
|
|
69
|
+
}
|
|
70
|
+
return sanitized;
|
|
71
|
+
}
|
|
72
|
+
export async function analyzePrivacy(content, filePath) {
|
|
73
|
+
const secrets = detectSecrets(content);
|
|
74
|
+
return {
|
|
75
|
+
isPrivate: hasPrivateTags(content),
|
|
76
|
+
hasSecrets: secrets.length > 0,
|
|
77
|
+
secretCount: secrets.length,
|
|
78
|
+
secretTypes: [...new Set(secrets.map(s => s.type))],
|
|
79
|
+
shouldStore: await shouldStore(content, filePath)
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
//# sourceMappingURL=privacy.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"privacy.js","sourceRoot":"","sources":["../../core/privacy.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export interface ProjectRecord {
|
|
2
|
+
id: string;
|
|
3
|
+
name: string;
|
|
4
|
+
path: string;
|
|
5
|
+
description?: string | null;
|
|
6
|
+
metadata?: Record<string, unknown> | null;
|
|
7
|
+
}
|
|
8
|
+
export declare function getProjectByPath(path: string): Promise<ProjectRecord | null>;
|
|
9
|
+
export declare function ensureProject(path?: string): Promise<ProjectRecord | null>;
|
|
10
|
+
//# sourceMappingURL=projects.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projects.d.ts","sourceRoot":"","sources":["../../core/projects.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,aAAa;IAC5B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC5B,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;CAC3C;AAED,wBAAsB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,CAelF;AAED,wBAAsB,aAAa,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,CA4BhF"}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { basename } from 'path';
|
|
2
|
+
import { randomUUID } from 'crypto';
|
|
3
|
+
import { eq } from 'drizzle-orm';
|
|
4
|
+
import { getDb } from '../db/index.js';
|
|
5
|
+
import { getSchema } from '../db/schema.js';
|
|
6
|
+
import { fromSqliteJson, toSqliteJson } from './memory/serialization.js';
|
|
7
|
+
import { config } from '../config.js';
|
|
8
|
+
import { createDatabaseClient } from './database.js';
|
|
9
|
+
export async function getProjectByPath(path) {
|
|
10
|
+
try {
|
|
11
|
+
const db = createDatabaseClient(await getDb());
|
|
12
|
+
const schema = await getSchema();
|
|
13
|
+
const rows = await db.select().from(schema.projects).where(eq(schema.projects.path, path)).limit(1);
|
|
14
|
+
const row = rows[0];
|
|
15
|
+
if (!row)
|
|
16
|
+
return null;
|
|
17
|
+
return normalizeProject(row);
|
|
18
|
+
}
|
|
19
|
+
catch (error) {
|
|
20
|
+
if (error.message?.includes('Database unavailable') ||
|
|
21
|
+
error.message?.includes('not a valid Win32 application')) {
|
|
22
|
+
return null; // Graceful degradation - database unavailable
|
|
23
|
+
}
|
|
24
|
+
throw error;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
export async function ensureProject(path) {
|
|
28
|
+
if (!path)
|
|
29
|
+
return null;
|
|
30
|
+
const existing = await getProjectByPath(path);
|
|
31
|
+
if (existing)
|
|
32
|
+
return existing;
|
|
33
|
+
const db = createDatabaseClient(await getDb());
|
|
34
|
+
const schema = await getSchema();
|
|
35
|
+
const id = randomUUID();
|
|
36
|
+
const name = basename(path) || path;
|
|
37
|
+
const metadata = { source: 'mcp' };
|
|
38
|
+
if (config.isTeamMode) {
|
|
39
|
+
await db.insert(schema.projects).values({
|
|
40
|
+
id,
|
|
41
|
+
name,
|
|
42
|
+
path,
|
|
43
|
+
metadata,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
else {
|
|
47
|
+
await db.insert(schema.projects).values({
|
|
48
|
+
id,
|
|
49
|
+
name,
|
|
50
|
+
path,
|
|
51
|
+
metadata: toSqliteJson(metadata),
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
return { id, name, path, metadata };
|
|
55
|
+
}
|
|
56
|
+
function normalizeProject(row) {
|
|
57
|
+
const metadata = config.isTeamMode ? row.metadata : fromSqliteJson(row.metadata);
|
|
58
|
+
return {
|
|
59
|
+
id: row.id,
|
|
60
|
+
name: row.name,
|
|
61
|
+
path: row.path,
|
|
62
|
+
description: row.description ?? null,
|
|
63
|
+
metadata,
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
//# sourceMappingURL=projects.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projects.js","sourceRoot":"","sources":["../../core/projects.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,MAAM,CAAC;AAChC,OAAO,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AACpC,OAAO,EAAE,EAAE,EAAE,MAAM,aAAa,CAAC;AACjC,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACvC,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzE,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,EAAE,oBAAoB,EAAE,MAAM,eAAe,CAAC;AAUrD,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,IAAY;IACjD,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,oBAAoB,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC;QAC/C,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;QACjC,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACpG,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QACpB,IAAI,CAAC,GAAG;YAAE,OAAO,IAAI,CAAC;QACtB,OAAO,gBAAgB,CAAC,GAAG,CAAC,CAAC;IAC/B,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,IAAI,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,sBAAsB,CAAC;YAC/C,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,+BAA+B,CAAC,EAAE,CAAC;YAC7D,OAAO,IAAI,CAAC,CAAC,8CAA8C;QAC7D,CAAC;QACD,MAAM,KAAK,CAAC;IACd,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,IAAa;IAC/C,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IACvB,MAAM,QAAQ,GAAG,MAAM,gBAAgB,CAAC,IAAI,CAAC,CAAC;IAC9C,IAAI,QAAQ;QAAE,OAAO,QAAQ,CAAC;IAE9B,MAAM,EAAE,GAAG,oBAAoB,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,EAAE,GAAG,UAAU,EAAE,CAAC;IACxB,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;IACpC,MAAM,QAAQ,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC;IAEnC,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;QACtB,MAAM,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC;YACtC,EAAE;YACF,IAAI;YACJ,IAAI;YACJ,QAAQ;SACT,CAAC,CAAC;IACL,CAAC;SAAM,CAAC;QACN,MAAM,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC;YACtC,EAAE;YACF,IAAI;YACJ,IAAI;YACJ,QAAQ,EAAE,YAAY,CAAC,QAAQ,CAAC;SACjC,CAAC,CAAC;IACL,CAAC;IAED,OAAO,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC;AACtC,CAAC;AAED,SAAS,gBAAgB,CAAC,GAAQ;IAChC,MAAM,QAAQ,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,cAAc,CAA0B,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC1G,OAAO;QACL,EAAE,EAAE,GAAG,CAAC,EAAE;QACV,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,WAAW,EAAE,GAAG,CAAC,WAAW,IAAI,IAAI;QACpC,QAAQ;KACT,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { RedisClientType } from 'redis';
|
|
2
|
+
export declare function getRedisClient(): Promise<RedisClientType>;
|
|
3
|
+
export declare function closeRedisConnection(): Promise<void>;
|
|
4
|
+
export declare function cacheGet<T>(key: string): Promise<T | null>;
|
|
5
|
+
export declare function cacheSet<T>(key: string, value: T, ttlSeconds?: number): Promise<void>;
|
|
6
|
+
export declare function cacheDelete(key: string): Promise<void>;
|
|
7
|
+
export declare function cacheClear(pattern: string): Promise<void>;
|
|
8
|
+
export declare function publish(channel: string, message: unknown): Promise<void>;
|
|
9
|
+
export declare function subscribe(channel: string, callback: (message: unknown) => void): Promise<void>;
|
|
10
|
+
export declare function checkRedisHealth(): Promise<boolean>;
|
|
11
|
+
//# sourceMappingURL=redis.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"redis.d.ts","sourceRoot":"","sources":["../../core/redis.ts"],"names":[],"mappings":"AAAA,OAAO,EAAgB,eAAe,EAAE,MAAM,OAAO,CAAC;AAQtD,wBAAsB,cAAc,IAAI,OAAO,CAAC,eAAe,CAAC,CAiB/D;AAED,wBAAsB,oBAAoB,IAAI,OAAO,CAAC,IAAI,CAAC,CAK1D;AAGD,wBAAsB,QAAQ,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,CAAC,GAAG,IAAI,CAAC,CAIhE;AAED,wBAAsB,QAAQ,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,GAAE,MAAY,GAAG,OAAO,CAAC,IAAI,CAAC,CAGhG;AAED,wBAAsB,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAG5D;AAED,wBAAsB,UAAU,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAM/D;AAGD,wBAAsB,OAAO,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAE9E;AAED,wBAAsB,SAAS,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAMpG;AAGD,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,OAAO,CAAC,CAQzD"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { createClient } from 'redis';
|
|
2
|
+
import { performRedisPublish } from './utils/memory-operations.js';
|
|
3
|
+
import { logger } from './logger.js';
|
|
4
|
+
const REDIS_URL = process.env.REDIS_URL || 'redis://localhost:6379';
|
|
5
|
+
let client = null;
|
|
6
|
+
export async function getRedisClient() {
|
|
7
|
+
if (client && client.isOpen) {
|
|
8
|
+
return client;
|
|
9
|
+
}
|
|
10
|
+
client = createClient({ url: REDIS_URL });
|
|
11
|
+
client.on('error', (err) => {
|
|
12
|
+
logger.error('Redis client error', err);
|
|
13
|
+
});
|
|
14
|
+
client.on('connect', () => {
|
|
15
|
+
logger.info('Redis connected');
|
|
16
|
+
});
|
|
17
|
+
await client.connect();
|
|
18
|
+
return client;
|
|
19
|
+
}
|
|
20
|
+
export async function closeRedisConnection() {
|
|
21
|
+
if (client && client.isOpen) {
|
|
22
|
+
await client.quit();
|
|
23
|
+
client = null;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
// Cache helpers with TTL
|
|
27
|
+
export async function cacheGet(key) {
|
|
28
|
+
const redis = await getRedisClient();
|
|
29
|
+
const value = await redis.get(key);
|
|
30
|
+
return value ? JSON.parse(value) : null;
|
|
31
|
+
}
|
|
32
|
+
export async function cacheSet(key, value, ttlSeconds = 300) {
|
|
33
|
+
const redis = await getRedisClient();
|
|
34
|
+
await redis.setEx(key, ttlSeconds, JSON.stringify(value));
|
|
35
|
+
}
|
|
36
|
+
export async function cacheDelete(key) {
|
|
37
|
+
const redis = await getRedisClient();
|
|
38
|
+
await redis.del(key);
|
|
39
|
+
}
|
|
40
|
+
export async function cacheClear(pattern) {
|
|
41
|
+
const redis = await getRedisClient();
|
|
42
|
+
const keys = await redis.keys(pattern);
|
|
43
|
+
if (keys.length > 0) {
|
|
44
|
+
await redis.del(keys);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
// Pub/Sub for real-time sync
|
|
48
|
+
export async function publish(channel, message) {
|
|
49
|
+
await performRedisPublish(getRedisClient, channel, message);
|
|
50
|
+
}
|
|
51
|
+
export async function subscribe(channel, callback) {
|
|
52
|
+
const subscriber = (await getRedisClient()).duplicate();
|
|
53
|
+
await subscriber.connect();
|
|
54
|
+
await subscriber.subscribe(channel, (message) => {
|
|
55
|
+
callback(JSON.parse(message));
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
// Check Redis health
|
|
59
|
+
export async function checkRedisHealth() {
|
|
60
|
+
try {
|
|
61
|
+
const redis = await getRedisClient();
|
|
62
|
+
const pong = await redis.ping();
|
|
63
|
+
return pong === 'PONG';
|
|
64
|
+
}
|
|
65
|
+
catch {
|
|
66
|
+
return false;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=redis.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"redis.js","sourceRoot":"","sources":["../../core/redis.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAmB,MAAM,OAAO,CAAC;AACtD,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAErC,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,wBAAwB,CAAC;AAEpE,IAAI,MAAM,GAA2B,IAAI,CAAC;AAE1C,MAAM,CAAC,KAAK,UAAU,cAAc;IAClC,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAC5B,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,MAAM,GAAG,YAAY,CAAC,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;IAE1C,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;QACzB,MAAM,CAAC,KAAK,CAAC,oBAAoB,EAAE,GAAG,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;QACxB,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,MAAM,MAAM,CAAC,OAAO,EAAE,CAAC;IACvB,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,oBAAoB;IACxC,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAC5B,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;QACpB,MAAM,GAAG,IAAI,CAAC;IAChB,CAAC;AACH,CAAC;AAED,yBAAyB;AACzB,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAI,GAAW;IAC3C,MAAM,KAAK,GAAG,MAAM,cAAc,EAAE,CAAC;IACrC,MAAM,KAAK,GAAG,MAAM,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IACnC,OAAO,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;AAC1C,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAI,GAAW,EAAE,KAAQ,EAAE,aAAqB,GAAG;IAC/E,MAAM,KAAK,GAAG,MAAM,cAAc,EAAE,CAAC;IACrC,MAAM,KAAK,CAAC,KAAK,CAAC,GAAG,EAAE,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;AAC5D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,GAAW;IAC3C,MAAM,KAAK,GAAG,MAAM,cAAc,EAAE,CAAC;IACrC,MAAM,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AACvB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,OAAe;IAC9C,MAAM,KAAK,GAAG,MAAM,cAAc,EAAE,CAAC;IACrC,MAAM,IAAI,GAAG,MAAM,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACvC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACpB,MAAM,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;AACH,CAAC;AAED,6BAA6B;AAC7B,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,OAAe,EAAE,OAAgB;IAC7D,MAAM,mBAAmB,CAAC,cAAc,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AAC9D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,OAAe,EAAE,QAAoC;IACnF,MAAM,UAAU,GAAG,CAAC,MAAM,cAAc,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACxD,MAAM,UAAU,CAAC,OAAO,EAAE,CAAC;IAC3B,MAAM,UAAU,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,EAAE;QAC9C,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC;IAChC,CAAC,CAAC,CAAC;AACL,CAAC;AAED,qBAAqB;AACrB,MAAM,CAAC,KAAK,UAAU,gBAAgB;IACpC,IAAI,CAAC;QACH,MAAM,KAAK,GAAG,MAAM,cAAc,EAAE,CAAC;QACrC,MAAM,IAAI,GAAG,MAAM,KAAK,CAAC,IAAI,EAAE,CAAC;QAChC,OAAO,IAAI,KAAK,MAAM,CAAC;IACzB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
export interface MemoryCriteria {
|
|
2
|
+
tag?: string;
|
|
3
|
+
type?: string;
|
|
4
|
+
types?: string[];
|
|
5
|
+
sector?: string;
|
|
6
|
+
minAge?: number;
|
|
7
|
+
maxAge?: number;
|
|
8
|
+
minConfidence?: number;
|
|
9
|
+
minRelevance?: number;
|
|
10
|
+
projectId?: string;
|
|
11
|
+
agentId?: string;
|
|
12
|
+
visibilityScope?: string;
|
|
13
|
+
}
|
|
14
|
+
export declare function requireMemory(criteria: MemoryCriteria): Promise<any>;
|
|
15
|
+
export declare function assertMemoryPresent(memoryId: string): Promise<void>;
|
|
16
|
+
export declare function assertMemoryNotPresent(criteria: MemoryCriteria): Promise<void>;
|
|
17
|
+
export declare function requireMemories(criteria: MemoryCriteria, minCount?: number): Promise<any[]>;
|
|
18
|
+
export declare function requireHighConfidenceMemory(criteria: MemoryCriteria, minConfidence?: number): Promise<any>;
|
|
19
|
+
export declare function requireRecentMemory(criteria: MemoryCriteria, maxAgeDays?: number): Promise<any>;
|
|
20
|
+
//# sourceMappingURL=requirements.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requirements.d.ts","sourceRoot":"","sources":["../../core/requirements.ts"],"names":[],"mappings":"AAMA,MAAM,WAAW,cAAc;IAC7B,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,wBAAsB,aAAa,CAAC,QAAQ,EAAE,cAAc,GAAG,OAAO,CAAC,GAAG,CAAC,CAa1E;AAED,wBAAsB,mBAAmB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAYzE;AAED,wBAAsB,sBAAsB,CAAC,QAAQ,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC,CAWpF;AAED,wBAAsB,eAAe,CAAC,QAAQ,EAAE,cAAc,EAAE,QAAQ,GAAE,MAAU,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,CAWpG;AAED,wBAAsB,2BAA2B,CAC/C,QAAQ,EAAE,cAAc,EACxB,aAAa,GAAE,MAAW,GACzB,OAAO,CAAC,GAAG,CAAC,CAEd;AAED,wBAAsB,mBAAmB,CACvC,QAAQ,EAAE,cAAc,EACxB,UAAU,GAAE,MAAU,GACrB,OAAO,CAAC,GAAG,CAAC,CAGd"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
// Deterministic Memory Requirements
|
|
2
|
+
import { eq } from 'drizzle-orm';
|
|
3
|
+
import { getSchema } from '../db/schema.js';
|
|
4
|
+
import { buildMemoryFilters, buildMemoryFiltersPartial } from './utils/filter-builder.js';
|
|
5
|
+
import { executeMemoryAssertion } from './utils/query-operations.js';
|
|
6
|
+
export async function requireMemory(criteria) {
|
|
7
|
+
const schema = await getSchema();
|
|
8
|
+
const filters = buildMemoryFilters(criteria, schema);
|
|
9
|
+
const memories = await executeMemoryAssertion(criteria, filters, 1, 'memory requirement', 'Required memory not found');
|
|
10
|
+
return memories[0];
|
|
11
|
+
}
|
|
12
|
+
export async function assertMemoryPresent(memoryId) {
|
|
13
|
+
const criteria = { memoryId };
|
|
14
|
+
const schema = await getSchema();
|
|
15
|
+
const filters = [eq(schema.memories.id, memoryId)];
|
|
16
|
+
await executeMemoryAssertion(criteria, filters, 1, 'asserting memory presence', 'Required memory is not present: ' + memoryId);
|
|
17
|
+
}
|
|
18
|
+
export async function assertMemoryNotPresent(criteria) {
|
|
19
|
+
const schema = await getSchema();
|
|
20
|
+
const filters = buildMemoryFiltersPartial(criteria, schema);
|
|
21
|
+
await executeMemoryAssertion(criteria, filters, 0, 'checking memory non-presence', 'Memory should not exist but was found');
|
|
22
|
+
}
|
|
23
|
+
export async function requireMemories(criteria, minCount = 1) {
|
|
24
|
+
const schema = await getSchema();
|
|
25
|
+
const filters = buildMemoryFilters(criteria, schema);
|
|
26
|
+
return executeMemoryAssertion(criteria, filters, minCount, 'checking memory requirements', `Required ${minCount} memories but found fewer`);
|
|
27
|
+
}
|
|
28
|
+
export async function requireHighConfidenceMemory(criteria, minConfidence = 80) {
|
|
29
|
+
return requireMemory({ ...criteria, minConfidence });
|
|
30
|
+
}
|
|
31
|
+
export async function requireRecentMemory(criteria, maxAgeDays = 7) {
|
|
32
|
+
const maxAge = maxAgeDays * 24 * 60 * 60 * 1000;
|
|
33
|
+
return requireMemory({ ...criteria, maxAge });
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=requirements.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requirements.js","sourceRoot":"","sources":["../../core/requirements.ts"],"names":[],"mappings":"AAAA,oCAAoC;AACpC,OAAO,EAAE,EAAE,EAAE,MAAM,aAAa,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,MAAM,2BAA2B,CAAC;AAC1F,OAAO,EAAsB,sBAAsB,EAA0B,MAAM,6BAA6B,CAAC;AAgBjH,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,QAAwB;IAC1D,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,OAAO,GAAG,kBAAkB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAErD,MAAM,QAAQ,GAAG,MAAM,sBAAsB,CAC3C,QAAQ,EACR,OAAO,EACP,CAAC,EACD,oBAAoB,EACpB,2BAA2B,CAC5B,CAAC;IAEF,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC;AACrB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,mBAAmB,CAAC,QAAgB;IACxD,MAAM,QAAQ,GAAG,EAAE,QAAQ,EAAE,CAAC;IAC9B,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,OAAO,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC,CAAC;IAEnD,MAAM,sBAAsB,CAC1B,QAAQ,EACR,OAAO,EACP,CAAC,EACD,2BAA2B,EAC3B,kCAAkC,GAAG,QAAQ,CAC9C,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAAC,QAAwB;IACnE,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,OAAO,GAAG,yBAAyB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAE5D,MAAM,sBAAsB,CAC1B,QAAQ,EACR,OAAO,EACP,CAAC,EACD,8BAA8B,EAC9B,uCAAuC,CACxC,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,QAAwB,EAAE,WAAmB,CAAC;IAClF,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,OAAO,GAAG,kBAAkB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAErD,OAAO,sBAAsB,CAC3B,QAAQ,EACR,OAAO,EACP,QAAQ,EACR,8BAA8B,EAC9B,YAAY,QAAQ,2BAA2B,CAChD,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,2BAA2B,CAC/C,QAAwB,EACxB,gBAAwB,EAAE;IAE1B,OAAO,aAAa,CAAC,EAAE,GAAG,QAAQ,EAAE,aAAa,EAAE,CAAC,CAAC;AACvD,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,QAAwB,EACxB,aAAqB,CAAC;IAEtB,MAAM,MAAM,GAAG,UAAU,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;IAChD,OAAO,aAAa,CAAC,EAAE,GAAG,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;AAChD,CAAC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export interface ConversationRecord {
|
|
2
|
+
id: string;
|
|
3
|
+
projectId?: string | null;
|
|
4
|
+
sessionId: string;
|
|
5
|
+
title?: string | null;
|
|
6
|
+
summary?: string | null;
|
|
7
|
+
messageCount?: number | null;
|
|
8
|
+
tokenCount?: number | null;
|
|
9
|
+
startedAt?: string | null;
|
|
10
|
+
endedAt?: string | null;
|
|
11
|
+
}
|
|
12
|
+
export interface ConversationSearchInput {
|
|
13
|
+
query: string;
|
|
14
|
+
limit?: number;
|
|
15
|
+
role?: 'user' | 'assistant';
|
|
16
|
+
}
|
|
17
|
+
export interface RecentConversationsInput {
|
|
18
|
+
n?: number;
|
|
19
|
+
before?: string;
|
|
20
|
+
after?: string;
|
|
21
|
+
project?: string;
|
|
22
|
+
}
|
|
23
|
+
export declare function searchConversations(input: ConversationSearchInput): Promise<ConversationRecord[]>;
|
|
24
|
+
export declare function getRecentConversations(input: RecentConversationsInput): Promise<ConversationRecord[]>;
|
|
25
|
+
//# sourceMappingURL=conversations.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conversations.d.ts","sourceRoot":"","sources":["../../../core/search/conversations.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,kBAAkB;IACjC,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CACzB;AAED,MAAM,WAAW,uBAAuB;IACtC,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,GAAG,WAAW,CAAC;CAC7B;AAED,MAAM,WAAW,wBAAwB;IACvC,CAAC,CAAC,EAAE,MAAM,CAAC;IACX,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,wBAAsB,mBAAmB,CAAC,KAAK,EAAE,uBAAuB,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC,CAMvG;AAED,wBAAsB,sBAAsB,CAAC,KAAK,EAAE,wBAAwB,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC,CA2B3G"}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { and, desc, eq, gte, lte } from 'drizzle-orm';
|
|
2
|
+
import { getDb } from '../../db/index.js';
|
|
3
|
+
import { getSchema } from '../../db/schema.js';
|
|
4
|
+
import { config } from '../../config.js';
|
|
5
|
+
import { getProjectByPath } from '../../core/projects.js';
|
|
6
|
+
import { fromSqliteJson } from '../../core/memory/serialization.js';
|
|
7
|
+
import { createDatabaseClient } from '../../core/database.js';
|
|
8
|
+
import { normalizeTimestamp, clampLimit } from '../../core/utils.js';
|
|
9
|
+
export async function searchConversations(input) {
|
|
10
|
+
const limit = clampLimit(input.limit, 5, 1, 50);
|
|
11
|
+
if (config.isTeamMode) {
|
|
12
|
+
return await searchConversationsPostgres(input, limit);
|
|
13
|
+
}
|
|
14
|
+
return await searchConversationsSqlite(input, limit);
|
|
15
|
+
}
|
|
16
|
+
export async function getRecentConversations(input) {
|
|
17
|
+
const db = createDatabaseClient(await getDb());
|
|
18
|
+
const schema = await getSchema();
|
|
19
|
+
const limit = clampLimit(input.n, 3, 1, 50);
|
|
20
|
+
const whereParts = [];
|
|
21
|
+
if (input.project) {
|
|
22
|
+
const project = await getProjectByPath(input.project);
|
|
23
|
+
if (project) {
|
|
24
|
+
whereParts.push(eq(schema.conversations.projectId, project.id));
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
if (input.before) {
|
|
28
|
+
whereParts.push(lte(schema.conversations.startedAt, new Date(input.before)));
|
|
29
|
+
}
|
|
30
|
+
if (input.after) {
|
|
31
|
+
whereParts.push(gte(schema.conversations.startedAt, new Date(input.after)));
|
|
32
|
+
}
|
|
33
|
+
const query = whereParts.length
|
|
34
|
+
? db.select().from(schema.conversations).where(and(...whereParts)).orderBy(desc(schema.conversations.startedAt)).limit(limit)
|
|
35
|
+
: db.select().from(schema.conversations).orderBy(desc(schema.conversations.startedAt)).limit(limit);
|
|
36
|
+
const rows = await query;
|
|
37
|
+
return rows.map((row) => normalizeConversation(row));
|
|
38
|
+
}
|
|
39
|
+
async function searchConversationsSqlite(input, limit) {
|
|
40
|
+
const db = createDatabaseClient(await getDb());
|
|
41
|
+
const sqlite = db.$client;
|
|
42
|
+
const params = [input.query];
|
|
43
|
+
let where = 'messages_fts MATCH ?';
|
|
44
|
+
if (input.role) {
|
|
45
|
+
where += ' AND m.role = ?';
|
|
46
|
+
params.push(input.role);
|
|
47
|
+
}
|
|
48
|
+
params.push(limit);
|
|
49
|
+
const statement = sqlite.prepare(`
|
|
50
|
+
SELECT DISTINCT
|
|
51
|
+
c.id as id,
|
|
52
|
+
c.project_id as projectId,
|
|
53
|
+
c.session_id as sessionId,
|
|
54
|
+
c.title as title,
|
|
55
|
+
c.summary as summary,
|
|
56
|
+
c.message_count as messageCount,
|
|
57
|
+
c.token_count as tokenCount,
|
|
58
|
+
c.started_at as startedAt,
|
|
59
|
+
c.ended_at as endedAt,
|
|
60
|
+
c.metadata as metadata
|
|
61
|
+
FROM conversations c
|
|
62
|
+
JOIN messages m ON m.conversation_id = c.id
|
|
63
|
+
JOIN messages_fts ON messages_fts.rowid = m.rowid
|
|
64
|
+
WHERE ${where}
|
|
65
|
+
ORDER BY c.started_at DESC
|
|
66
|
+
LIMIT ?
|
|
67
|
+
`);
|
|
68
|
+
const rows = statement.all(...params);
|
|
69
|
+
return rows.map((row) => normalizeConversation(row));
|
|
70
|
+
}
|
|
71
|
+
async function searchConversationsPostgres(input, limit) {
|
|
72
|
+
const db = createDatabaseClient(await getDb());
|
|
73
|
+
const values = [`%${input.query}%`];
|
|
74
|
+
const whereParts = [`m.content ILIKE $1`];
|
|
75
|
+
if (input.role) {
|
|
76
|
+
values.push(input.role);
|
|
77
|
+
whereParts.push(`m.role = $${values.length}`);
|
|
78
|
+
}
|
|
79
|
+
values.push(limit);
|
|
80
|
+
const rows = await db.$client.query(`SELECT DISTINCT ON (c.id)
|
|
81
|
+
c.id as "id",
|
|
82
|
+
c.project_id as "projectId",
|
|
83
|
+
c.session_id as "sessionId",
|
|
84
|
+
c.title as "title",
|
|
85
|
+
c.summary as "summary",
|
|
86
|
+
c.message_count as "messageCount",
|
|
87
|
+
c.token_count as "tokenCount",
|
|
88
|
+
c.started_at as "startedAt",
|
|
89
|
+
c.ended_at as "endedAt",
|
|
90
|
+
c.metadata as "metadata"
|
|
91
|
+
FROM conversations c
|
|
92
|
+
JOIN messages m ON m.conversation_id = c.id
|
|
93
|
+
WHERE ${whereParts.join(' AND ')}
|
|
94
|
+
ORDER BY c.id, c.started_at DESC
|
|
95
|
+
LIMIT $${values.length}`, values);
|
|
96
|
+
return rows.rows.map((row) => normalizeConversation(row));
|
|
97
|
+
}
|
|
98
|
+
function normalizeConversation(row) {
|
|
99
|
+
const metadata = config.isTeamMode ? row.metadata : fromSqliteJson(row.metadata ?? null);
|
|
100
|
+
return {
|
|
101
|
+
id: row.id,
|
|
102
|
+
projectId: row.projectId ?? row.project_id ?? null,
|
|
103
|
+
sessionId: row.sessionId ?? row.session_id,
|
|
104
|
+
title: row.title ?? null,
|
|
105
|
+
summary: row.summary ?? null,
|
|
106
|
+
messageCount: row.messageCount ?? row.message_count ?? null,
|
|
107
|
+
tokenCount: row.tokenCount ?? row.token_count ?? null,
|
|
108
|
+
startedAt: normalizeTimestamp(row.startedAt ?? row.started_at),
|
|
109
|
+
endedAt: normalizeTimestamp(row.endedAt ?? row.ended_at),
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=conversations.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conversations.js","sourceRoot":"","sources":["../../../core/search/conversations.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export interface EntityRecord {
|
|
2
|
+
id: string;
|
|
3
|
+
projectId?: string | null;
|
|
4
|
+
name: string;
|
|
5
|
+
type: string;
|
|
6
|
+
description?: string | null;
|
|
7
|
+
properties?: Record<string, unknown> | null;
|
|
8
|
+
createdAt?: string | null;
|
|
9
|
+
updatedAt?: string | null;
|
|
10
|
+
}
|
|
11
|
+
export declare function getEntitiesForProject(projectPath: string, limit: number): Promise<EntityRecord[]>;
|
|
12
|
+
//# sourceMappingURL=entities.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"entities.d.ts","sourceRoot":"","sources":["../../../core/search/entities.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,YAAY;IAC3B,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC5B,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;IAC5C,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CAC3B;AAED,wBAAsB,qBAAqB,CAAC,WAAW,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC,CAYvG"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { desc, eq } from 'drizzle-orm';
|
|
2
|
+
import { getDb } from '../../db/index.js';
|
|
3
|
+
import { getSchema } from '../../db/schema.js';
|
|
4
|
+
import { config } from '../../config.js';
|
|
5
|
+
import { getProjectByPath } from '../../core/projects.js';
|
|
6
|
+
import { fromSqliteJson } from '../../core/memory/serialization.js';
|
|
7
|
+
import { createDatabaseClient } from '../../core/database.js';
|
|
8
|
+
import { normalizeTimestamp } from '../../core/utils.js';
|
|
9
|
+
export async function getEntitiesForProject(projectPath, limit) {
|
|
10
|
+
const db = createDatabaseClient(await getDb());
|
|
11
|
+
const schema = await getSchema();
|
|
12
|
+
const project = await getProjectByPath(projectPath);
|
|
13
|
+
if (!project)
|
|
14
|
+
return [];
|
|
15
|
+
const rows = await db.select().from(schema.entities)
|
|
16
|
+
.where(eq(schema.entities.projectId, project.id))
|
|
17
|
+
.orderBy(desc(schema.entities.createdAt))
|
|
18
|
+
.limit(limit);
|
|
19
|
+
return rows.map((row) => normalizeEntity(row));
|
|
20
|
+
}
|
|
21
|
+
function normalizeEntity(row) {
|
|
22
|
+
const properties = config.isTeamMode ? row.properties : fromSqliteJson(row.properties ?? null);
|
|
23
|
+
return {
|
|
24
|
+
id: row.id,
|
|
25
|
+
projectId: row.projectId ?? row.project_id ?? null,
|
|
26
|
+
name: row.name,
|
|
27
|
+
type: row.type,
|
|
28
|
+
description: row.description ?? null,
|
|
29
|
+
properties,
|
|
30
|
+
createdAt: normalizeTimestamp(row.createdAt ?? row.created_at),
|
|
31
|
+
updatedAt: normalizeTimestamp(row.updatedAt ?? row.updated_at),
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
//# sourceMappingURL=entities.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"entities.js","sourceRoot":"","sources":["../../../core/search/entities.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,MAAM,aAAa,CAAC;AACvC,OAAO,EAAE,KAAK,EAAE,MAAM,mBAAmB,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAC/C,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,cAAc,EAAE,MAAM,oCAAoC,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AAazD,MAAM,CAAC,KAAK,UAAU,qBAAqB,CAAC,WAAmB,EAAE,KAAa;IAC5E,MAAM,EAAE,GAAG,oBAAoB,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,MAAM,SAAS,EAAE,CAAC;IACjC,MAAM,OAAO,GAAG,MAAM,gBAAgB,CAAC,WAAW,CAAC,CAAC;IACpD,IAAI,CAAC,OAAO;QAAE,OAAO,EAAE,CAAC;IAExB,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC;SACjD,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;SAChD,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;SACxC,KAAK,CAAC,KAAK,CAAC,CAAC;IAEhB,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,GAAQ,EAAE,EAAE,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC;AACtD,CAAC;AAED,SAAS,eAAe,CAAC,GAAQ;IAC/B,MAAM,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,cAAc,CAA0B,GAAG,CAAC,UAAU,IAAI,IAAI,CAAC,CAAC;IACxH,OAAO;QACL,EAAE,EAAE,GAAG,CAAC,EAAE;QACV,SAAS,EAAE,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,UAAU,IAAI,IAAI;QAClD,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,WAAW,EAAE,GAAG,CAAC,WAAW,IAAI,IAAI;QACpC,UAAU;QACV,SAAS,EAAE,kBAAkB,CAAC,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,UAAU,CAAC;QAC9D,SAAS,EAAE,kBAAkB,CAAC,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,UAAU,CAAC;KAC/D,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Folder Context Generation
|
|
3
|
+
* Generates and maintains per-folder CLAUDE.md files with session summaries
|
|
4
|
+
*/
|
|
5
|
+
interface FolderContextData {
|
|
6
|
+
projectPath: string;
|
|
7
|
+
projectName: string;
|
|
8
|
+
gitRemote?: string;
|
|
9
|
+
gitBranch?: string;
|
|
10
|
+
recentConversations: Array<{
|
|
11
|
+
content: string;
|
|
12
|
+
timestamp: string;
|
|
13
|
+
}>;
|
|
14
|
+
keyObservations: Array<{
|
|
15
|
+
summary: string;
|
|
16
|
+
timestamp: string;
|
|
17
|
+
type: string;
|
|
18
|
+
}>;
|
|
19
|
+
lastUpdated: string;
|
|
20
|
+
}
|
|
21
|
+
export declare function generateAndInjectFolderContext(projectPath: string): Promise<void>;
|
|
22
|
+
export declare function readFolderContext(projectPath: string): Promise<FolderContextData | null>;
|
|
23
|
+
export declare function injectFolderContextIntoSession(projectPath: string): Promise<string>;
|
|
24
|
+
export {};
|
|
25
|
+
//# sourceMappingURL=folder-context.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"folder-context.d.ts","sourceRoot":"","sources":["../../../core/search/folder-context.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAOH,UAAU,iBAAiB;IACzB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,mBAAmB,EAAE,KAAK,CAAC;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACnE,eAAe,EAAE,KAAK,CAAC;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAC7E,WAAW,EAAE,MAAM,CAAC;CACrB;AAmFD,wBAAsB,8BAA8B,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAQvF;AAED,wBAAsB,iBAAiB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CA0B9F;AAED,wBAAsB,8BAA8B,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAczF"}
|