@soleri/core 2.9.0 → 2.11.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/dist/agency/agency-manager.d.ts +47 -0
- package/dist/agency/agency-manager.d.ts.map +1 -0
- package/dist/agency/agency-manager.js +281 -0
- package/dist/agency/agency-manager.js.map +1 -0
- package/dist/agency/index.d.ts +3 -0
- package/dist/agency/index.d.ts.map +1 -0
- package/dist/agency/index.js +2 -0
- package/dist/agency/index.js.map +1 -0
- package/dist/agency/types.d.ts +69 -0
- package/dist/agency/types.d.ts.map +1 -0
- package/dist/agency/types.js +5 -0
- package/dist/agency/types.js.map +1 -0
- package/dist/brain/brain.d.ts +0 -1
- package/dist/brain/brain.d.ts.map +1 -1
- package/dist/brain/brain.js +14 -0
- package/dist/brain/brain.js.map +1 -1
- package/dist/brain/intelligence.d.ts +5 -1
- package/dist/brain/intelligence.d.ts.map +1 -1
- package/dist/brain/intelligence.js +83 -0
- package/dist/brain/intelligence.js.map +1 -1
- package/dist/brain/types.d.ts +21 -0
- package/dist/brain/types.d.ts.map +1 -1
- package/dist/chat/agent-loop-types.d.ts +82 -0
- package/dist/chat/agent-loop-types.d.ts.map +1 -0
- package/dist/chat/agent-loop-types.js +8 -0
- package/dist/chat/agent-loop-types.js.map +1 -0
- package/dist/chat/agent-loop.d.ts +19 -0
- package/dist/chat/agent-loop.d.ts.map +1 -0
- package/dist/chat/agent-loop.js +261 -0
- package/dist/chat/agent-loop.js.map +1 -0
- package/dist/chat/auth-manager.d.ts +49 -0
- package/dist/chat/auth-manager.d.ts.map +1 -0
- package/dist/chat/auth-manager.js +152 -0
- package/dist/chat/auth-manager.js.map +1 -0
- package/dist/chat/browser-session.d.ts +86 -0
- package/dist/chat/browser-session.d.ts.map +1 -0
- package/dist/chat/browser-session.js +143 -0
- package/dist/chat/browser-session.js.map +1 -0
- package/dist/chat/cancellation.d.ts +54 -0
- package/dist/chat/cancellation.d.ts.map +1 -0
- package/dist/chat/cancellation.js +80 -0
- package/dist/chat/cancellation.js.map +1 -0
- package/dist/chat/chat-session.d.ts +86 -0
- package/dist/chat/chat-session.d.ts.map +1 -0
- package/dist/chat/chat-session.js +252 -0
- package/dist/chat/chat-session.js.map +1 -0
- package/dist/chat/file-handler.d.ts +63 -0
- package/dist/chat/file-handler.d.ts.map +1 -0
- package/dist/chat/file-handler.js +182 -0
- package/dist/chat/file-handler.js.map +1 -0
- package/dist/chat/fragment-buffer.d.ts +49 -0
- package/dist/chat/fragment-buffer.d.ts.map +1 -0
- package/dist/chat/fragment-buffer.js +130 -0
- package/dist/chat/fragment-buffer.js.map +1 -0
- package/dist/chat/index.d.ts +24 -0
- package/dist/chat/index.d.ts.map +1 -0
- package/dist/chat/index.js +15 -0
- package/dist/chat/index.js.map +1 -0
- package/dist/chat/mcp-bridge.d.ts +60 -0
- package/dist/chat/mcp-bridge.d.ts.map +1 -0
- package/dist/chat/mcp-bridge.js +111 -0
- package/dist/chat/mcp-bridge.js.map +1 -0
- package/dist/chat/notifications.d.ts +82 -0
- package/dist/chat/notifications.d.ts.map +1 -0
- package/dist/chat/notifications.js +119 -0
- package/dist/chat/notifications.js.map +1 -0
- package/dist/chat/output-compressor.d.ts +30 -0
- package/dist/chat/output-compressor.d.ts.map +1 -0
- package/dist/chat/output-compressor.js +95 -0
- package/dist/chat/output-compressor.js.map +1 -0
- package/dist/chat/queue.d.ts +91 -0
- package/dist/chat/queue.d.ts.map +1 -0
- package/dist/chat/queue.js +146 -0
- package/dist/chat/queue.js.map +1 -0
- package/dist/chat/response-chunker.d.ts +29 -0
- package/dist/chat/response-chunker.d.ts.map +1 -0
- package/dist/chat/response-chunker.js +163 -0
- package/dist/chat/response-chunker.js.map +1 -0
- package/dist/chat/self-update.d.ts +62 -0
- package/dist/chat/self-update.d.ts.map +1 -0
- package/dist/chat/self-update.js +90 -0
- package/dist/chat/self-update.js.map +1 -0
- package/dist/chat/types.d.ts +105 -0
- package/dist/chat/types.d.ts.map +1 -0
- package/dist/chat/types.js +8 -0
- package/dist/chat/types.js.map +1 -0
- package/dist/chat/voice.d.ts +39 -0
- package/dist/chat/voice.d.ts.map +1 -0
- package/dist/chat/voice.js +80 -0
- package/dist/chat/voice.js.map +1 -0
- package/dist/claudemd/compose.d.ts +31 -0
- package/dist/claudemd/compose.d.ts.map +1 -0
- package/dist/claudemd/compose.js +105 -0
- package/dist/claudemd/compose.js.map +1 -0
- package/dist/claudemd/index.d.ts +5 -0
- package/dist/claudemd/index.d.ts.map +1 -0
- package/dist/claudemd/index.js +3 -0
- package/dist/claudemd/index.js.map +1 -0
- package/dist/claudemd/inject.d.ts +31 -0
- package/dist/claudemd/inject.d.ts.map +1 -0
- package/dist/claudemd/inject.js +157 -0
- package/dist/claudemd/inject.js.map +1 -0
- package/dist/claudemd/types.d.ts +41 -0
- package/dist/claudemd/types.d.ts.map +1 -0
- package/dist/claudemd/types.js +5 -0
- package/dist/claudemd/types.js.map +1 -0
- package/dist/context/context-engine.d.ts +31 -0
- package/dist/context/context-engine.d.ts.map +1 -0
- package/dist/context/context-engine.js +245 -0
- package/dist/context/context-engine.js.map +1 -0
- package/dist/context/index.d.ts +3 -0
- package/dist/context/index.d.ts.map +1 -0
- package/dist/context/index.js +2 -0
- package/dist/context/index.js.map +1 -0
- package/dist/context/types.d.ts +54 -0
- package/dist/context/types.d.ts.map +1 -0
- package/dist/context/types.js +5 -0
- package/dist/context/types.js.map +1 -0
- package/dist/enforcement/adapters/claude-code.d.ts +18 -0
- package/dist/enforcement/adapters/claude-code.d.ts.map +1 -0
- package/dist/enforcement/adapters/claude-code.js +106 -0
- package/dist/enforcement/adapters/claude-code.js.map +1 -0
- package/dist/enforcement/adapters/index.d.ts +2 -0
- package/dist/enforcement/adapters/index.d.ts.map +1 -0
- package/dist/enforcement/adapters/index.js +2 -0
- package/dist/enforcement/adapters/index.js.map +1 -0
- package/dist/enforcement/index.d.ts +4 -0
- package/dist/enforcement/index.d.ts.map +1 -0
- package/dist/enforcement/index.js +3 -0
- package/dist/enforcement/index.js.map +1 -0
- package/dist/enforcement/registry.d.ts +23 -0
- package/dist/enforcement/registry.d.ts.map +1 -0
- package/dist/enforcement/registry.js +63 -0
- package/dist/enforcement/registry.js.map +1 -0
- package/dist/enforcement/types.d.ts +51 -0
- package/dist/enforcement/types.d.ts.map +1 -0
- package/dist/enforcement/types.js +8 -0
- package/dist/enforcement/types.js.map +1 -0
- package/dist/facades/facade-factory.d.ts +10 -3
- package/dist/facades/facade-factory.d.ts.map +1 -1
- package/dist/facades/facade-factory.js +94 -5
- package/dist/facades/facade-factory.js.map +1 -1
- package/dist/facades/types.d.ts +15 -1
- package/dist/facades/types.d.ts.map +1 -1
- package/dist/facades/types.js +6 -0
- package/dist/facades/types.js.map +1 -1
- package/dist/health/health-registry.d.ts +40 -0
- package/dist/health/health-registry.d.ts.map +1 -0
- package/dist/health/health-registry.js +134 -0
- package/dist/health/health-registry.js.map +1 -0
- package/dist/health/index.d.ts +5 -0
- package/dist/health/index.d.ts.map +1 -0
- package/dist/health/index.js +3 -0
- package/dist/health/index.js.map +1 -0
- package/dist/health/vault-integrity.d.ts +13 -0
- package/dist/health/vault-integrity.d.ts.map +1 -0
- package/dist/health/vault-integrity.js +49 -0
- package/dist/health/vault-integrity.js.map +1 -0
- package/dist/index.d.ts +67 -6
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +51 -3
- package/dist/index.js.map +1 -1
- package/dist/intake/intake-pipeline.d.ts +0 -7
- package/dist/intake/intake-pipeline.d.ts.map +1 -1
- package/dist/intake/intake-pipeline.js +1 -1
- package/dist/intake/intake-pipeline.js.map +1 -1
- package/dist/intelligence/types.d.ts +1 -0
- package/dist/intelligence/types.d.ts.map +1 -1
- package/dist/migrations/index.d.ts +6 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/index.js +5 -0
- package/dist/migrations/index.js.map +1 -0
- package/dist/migrations/migration-runner.d.ts +51 -0
- package/dist/migrations/migration-runner.d.ts.map +1 -0
- package/dist/migrations/migration-runner.js +141 -0
- package/dist/migrations/migration-runner.js.map +1 -0
- package/dist/packs/index.d.ts +10 -0
- package/dist/packs/index.d.ts.map +1 -0
- package/dist/packs/index.js +8 -0
- package/dist/packs/index.js.map +1 -0
- package/dist/packs/lockfile.d.ts +97 -0
- package/dist/packs/lockfile.d.ts.map +1 -0
- package/dist/packs/lockfile.js +129 -0
- package/dist/packs/lockfile.js.map +1 -0
- package/dist/packs/pack-installer.d.ts +41 -0
- package/dist/packs/pack-installer.d.ts.map +1 -0
- package/dist/packs/pack-installer.js +253 -0
- package/dist/packs/pack-installer.js.map +1 -0
- package/dist/packs/resolver.d.ts +51 -0
- package/dist/packs/resolver.d.ts.map +1 -0
- package/dist/packs/resolver.js +195 -0
- package/dist/packs/resolver.js.map +1 -0
- package/dist/packs/types.d.ts +186 -0
- package/dist/packs/types.d.ts.map +1 -0
- package/dist/packs/types.js +69 -0
- package/dist/packs/types.js.map +1 -0
- package/dist/persistence/postgres-provider.d.ts +42 -7
- package/dist/persistence/postgres-provider.d.ts.map +1 -1
- package/dist/persistence/postgres-provider.js +187 -46
- package/dist/persistence/postgres-provider.js.map +1 -1
- package/dist/planning/gap-analysis.d.ts.map +1 -1
- package/dist/planning/gap-analysis.js +3 -1
- package/dist/planning/gap-analysis.js.map +1 -1
- package/dist/playbooks/index.d.ts +2 -0
- package/dist/playbooks/index.d.ts.map +1 -1
- package/dist/playbooks/index.js +2 -0
- package/dist/playbooks/index.js.map +1 -1
- package/dist/playbooks/playbook-executor.d.ts +100 -0
- package/dist/playbooks/playbook-executor.d.ts.map +1 -0
- package/dist/playbooks/playbook-executor.js +207 -0
- package/dist/playbooks/playbook-executor.js.map +1 -0
- package/dist/plugins/index.d.ts +7 -0
- package/dist/plugins/index.d.ts.map +1 -0
- package/dist/plugins/index.js +7 -0
- package/dist/plugins/index.js.map +1 -0
- package/dist/plugins/plugin-loader.d.ts +28 -0
- package/dist/plugins/plugin-loader.d.ts.map +1 -0
- package/dist/plugins/plugin-loader.js +150 -0
- package/dist/plugins/plugin-loader.js.map +1 -0
- package/dist/plugins/plugin-registry.d.ts +58 -0
- package/dist/plugins/plugin-registry.d.ts.map +1 -0
- package/dist/plugins/plugin-registry.js +157 -0
- package/dist/plugins/plugin-registry.js.map +1 -0
- package/dist/plugins/types.d.ts +180 -0
- package/dist/plugins/types.d.ts.map +1 -0
- package/dist/plugins/types.js +48 -0
- package/dist/plugins/types.js.map +1 -0
- package/dist/runtime/admin-extra-ops.d.ts.map +1 -1
- package/dist/runtime/admin-extra-ops.js +181 -8
- package/dist/runtime/admin-extra-ops.js.map +1 -1
- package/dist/runtime/capture-ops.d.ts.map +1 -1
- package/dist/runtime/capture-ops.js +106 -7
- package/dist/runtime/capture-ops.js.map +1 -1
- package/dist/runtime/deprecation.d.ts +33 -0
- package/dist/runtime/deprecation.d.ts.map +1 -0
- package/dist/runtime/deprecation.js +41 -0
- package/dist/runtime/deprecation.js.map +1 -0
- package/dist/runtime/facades/admin-facade.d.ts.map +1 -1
- package/dist/runtime/facades/admin-facade.js +12 -1
- package/dist/runtime/facades/admin-facade.js.map +1 -1
- package/dist/runtime/facades/agency-facade.d.ts +7 -0
- package/dist/runtime/facades/agency-facade.d.ts.map +1 -0
- package/dist/runtime/facades/agency-facade.js +103 -0
- package/dist/runtime/facades/agency-facade.js.map +1 -0
- package/dist/runtime/facades/brain-facade.d.ts.map +1 -1
- package/dist/runtime/facades/brain-facade.js +58 -0
- package/dist/runtime/facades/brain-facade.js.map +1 -1
- package/dist/runtime/facades/chat-facade.d.ts +7 -0
- package/dist/runtime/facades/chat-facade.d.ts.map +1 -0
- package/dist/runtime/facades/chat-facade.js +808 -0
- package/dist/runtime/facades/chat-facade.js.map +1 -0
- package/dist/runtime/facades/context-facade.d.ts +7 -0
- package/dist/runtime/facades/context-facade.d.ts.map +1 -0
- package/dist/runtime/facades/context-facade.js +45 -0
- package/dist/runtime/facades/context-facade.js.map +1 -0
- package/dist/runtime/facades/index.d.ts.map +1 -1
- package/dist/runtime/facades/index.js +18 -0
- package/dist/runtime/facades/index.js.map +1 -1
- package/dist/runtime/facades/vault-facade.d.ts.map +1 -1
- package/dist/runtime/facades/vault-facade.js +247 -1
- package/dist/runtime/facades/vault-facade.js.map +1 -1
- package/dist/runtime/feature-flags.d.ts +18 -0
- package/dist/runtime/feature-flags.d.ts.map +1 -0
- package/dist/runtime/feature-flags.js +90 -0
- package/dist/runtime/feature-flags.js.map +1 -0
- package/dist/runtime/pack-ops.d.ts +9 -0
- package/dist/runtime/pack-ops.d.ts.map +1 -0
- package/dist/runtime/pack-ops.js +76 -0
- package/dist/runtime/pack-ops.js.map +1 -0
- package/dist/runtime/playbook-ops.d.ts +3 -7
- package/dist/runtime/playbook-ops.d.ts.map +1 -1
- package/dist/runtime/playbook-ops.js +101 -10
- package/dist/runtime/playbook-ops.js.map +1 -1
- package/dist/runtime/plugin-ops.d.ts +9 -0
- package/dist/runtime/plugin-ops.d.ts.map +1 -0
- package/dist/runtime/plugin-ops.js +235 -0
- package/dist/runtime/plugin-ops.js.map +1 -0
- package/dist/runtime/runtime.d.ts.map +1 -1
- package/dist/runtime/runtime.js +72 -5
- package/dist/runtime/runtime.js.map +1 -1
- package/dist/runtime/telemetry-ops.d.ts +10 -0
- package/dist/runtime/telemetry-ops.d.ts.map +1 -0
- package/dist/runtime/telemetry-ops.js +53 -0
- package/dist/runtime/telemetry-ops.js.map +1 -0
- package/dist/runtime/types.d.ts +35 -0
- package/dist/runtime/types.d.ts.map +1 -1
- package/dist/runtime/vault-sharing-ops.d.ts +13 -0
- package/dist/runtime/vault-sharing-ops.d.ts.map +1 -0
- package/dist/runtime/vault-sharing-ops.js +345 -0
- package/dist/runtime/vault-sharing-ops.js.map +1 -0
- package/dist/streams/index.d.ts +1 -1
- package/dist/streams/index.d.ts.map +1 -1
- package/dist/streams/index.js +1 -1
- package/dist/streams/index.js.map +1 -1
- package/dist/streams/replayable-stream.d.ts +13 -1
- package/dist/streams/replayable-stream.d.ts.map +1 -1
- package/dist/streams/replayable-stream.js +27 -3
- package/dist/streams/replayable-stream.js.map +1 -1
- package/dist/text/similarity.d.ts +0 -1
- package/dist/text/similarity.d.ts.map +1 -1
- package/dist/text/similarity.js +1 -1
- package/dist/text/similarity.js.map +1 -1
- package/dist/transport/http-server.d.ts +56 -0
- package/dist/transport/http-server.d.ts.map +1 -0
- package/dist/transport/http-server.js +210 -0
- package/dist/transport/http-server.js.map +1 -0
- package/dist/transport/index.d.ts +11 -0
- package/dist/transport/index.d.ts.map +1 -0
- package/dist/transport/index.js +10 -0
- package/dist/transport/index.js.map +1 -0
- package/dist/transport/lsp-server.d.ts +140 -0
- package/dist/transport/lsp-server.d.ts.map +1 -0
- package/dist/transport/lsp-server.js +239 -0
- package/dist/transport/lsp-server.js.map +1 -0
- package/dist/transport/rate-limiter.d.ts +35 -0
- package/dist/transport/rate-limiter.d.ts.map +1 -0
- package/dist/transport/rate-limiter.js +72 -0
- package/dist/transport/rate-limiter.js.map +1 -0
- package/dist/transport/session-manager.d.ts +49 -0
- package/dist/transport/session-manager.d.ts.map +1 -0
- package/dist/transport/session-manager.js +83 -0
- package/dist/transport/session-manager.js.map +1 -0
- package/dist/transport/token-auth.d.ts +29 -0
- package/dist/transport/token-auth.d.ts.map +1 -0
- package/dist/transport/token-auth.js +84 -0
- package/dist/transport/token-auth.js.map +1 -0
- package/dist/transport/types.d.ts +61 -0
- package/dist/transport/types.d.ts.map +1 -0
- package/dist/transport/types.js +5 -0
- package/dist/transport/types.js.map +1 -0
- package/dist/transport/ws-server.d.ts +78 -0
- package/dist/transport/ws-server.d.ts.map +1 -0
- package/dist/transport/ws-server.js +342 -0
- package/dist/transport/ws-server.js.map +1 -0
- package/dist/vault/git-vault-sync.d.ts +107 -0
- package/dist/vault/git-vault-sync.d.ts.map +1 -0
- package/dist/vault/git-vault-sync.js +251 -0
- package/dist/vault/git-vault-sync.js.map +1 -0
- package/dist/vault/knowledge-review.d.ts +67 -0
- package/dist/vault/knowledge-review.d.ts.map +1 -0
- package/dist/vault/knowledge-review.js +133 -0
- package/dist/vault/knowledge-review.js.map +1 -0
- package/dist/vault/obsidian-sync.d.ts +94 -0
- package/dist/vault/obsidian-sync.d.ts.map +1 -0
- package/dist/vault/obsidian-sync.js +247 -0
- package/dist/vault/obsidian-sync.js.map +1 -0
- package/dist/vault/scope-detector.d.ts +31 -0
- package/dist/vault/scope-detector.d.ts.map +1 -0
- package/dist/vault/scope-detector.js +182 -0
- package/dist/vault/scope-detector.js.map +1 -0
- package/dist/vault/vault-branching.d.ts +71 -0
- package/dist/vault/vault-branching.d.ts.map +1 -0
- package/dist/vault/vault-branching.js +180 -0
- package/dist/vault/vault-branching.js.map +1 -0
- package/dist/vault/vault-manager.d.ts +89 -0
- package/dist/vault/vault-manager.d.ts.map +1 -0
- package/dist/vault/vault-manager.js +199 -0
- package/dist/vault/vault-manager.js.map +1 -0
- package/dist/vault/vault-types.d.ts +30 -0
- package/dist/vault/vault-types.d.ts.map +1 -0
- package/dist/vault/vault-types.js +10 -0
- package/dist/vault/vault-types.js.map +1 -0
- package/dist/vault/vault.d.ts +10 -0
- package/dist/vault/vault.d.ts.map +1 -1
- package/dist/vault/vault.js +36 -3
- package/dist/vault/vault.js.map +1 -1
- package/package.json +1 -1
- package/src/__tests__/admin-extra-ops.test.ts +31 -11
- package/src/__tests__/agency-manager.test.ts +374 -0
- package/src/__tests__/agent-loop.test.ts +256 -0
- package/src/__tests__/capture-ops.test.ts +275 -0
- package/src/__tests__/chat-differentiators.test.ts +251 -0
- package/src/__tests__/chat-enhanced.test.ts +390 -0
- package/src/__tests__/chat-transport.test.ts +665 -0
- package/src/__tests__/claudemd.test.ts +282 -0
- package/src/__tests__/context-engine.test.ts +256 -0
- package/src/__tests__/core-ops.test.ts +97 -5
- package/src/__tests__/deprecation.test.ts +78 -0
- package/src/__tests__/enforcement.test.ts +153 -0
- package/src/__tests__/facade-factory.test.ts +271 -0
- package/src/__tests__/feature-flags.test.ts +138 -0
- package/src/__tests__/git-vault-sync.test.ts +230 -0
- package/src/__tests__/health-registry.test.ts +173 -0
- package/src/__tests__/knowledge-review.test.ts +104 -0
- package/src/__tests__/lsp-transport.test.ts +442 -0
- package/src/__tests__/migration-runner.test.ts +170 -0
- package/src/__tests__/normalize.test.ts +10 -0
- package/src/__tests__/obsidian-sync.test.ts +354 -0
- package/src/__tests__/pack-lockfile.test.ts +261 -0
- package/src/__tests__/pack-ops.test.ts +146 -0
- package/src/__tests__/pack-system.test.ts +423 -0
- package/src/__tests__/playbook-executor.test.ts +249 -0
- package/src/__tests__/playbook-ops-execution.test.ts +189 -0
- package/src/__tests__/plugin-ops.test.ts +411 -0
- package/src/__tests__/plugin-system.test.ts +509 -0
- package/src/__tests__/postgres-provider.test.ts +64 -6
- package/src/__tests__/replayable-stream.test.ts +112 -1
- package/src/__tests__/scope-detector.test.ts +121 -0
- package/src/__tests__/session-lifecycle.test.ts +259 -0
- package/src/__tests__/transport.test.ts +758 -0
- package/src/__tests__/vault-branching.test.ts +274 -0
- package/src/__tests__/vault-connect.test.ts +179 -0
- package/src/__tests__/vault-integrity.test.ts +71 -0
- package/src/__tests__/vault-manager.test.ts +238 -0
- package/src/__tests__/vault-scaling.test.ts +281 -0
- package/src/__tests__/vault-sharing.test.ts +270 -0
- package/src/__tests__/ws-transport.test.ts +479 -0
- package/src/agency/agency-manager.ts +326 -0
- package/src/agency/index.ts +13 -0
- package/src/agency/types.ts +88 -0
- package/src/brain/brain.ts +15 -11
- package/src/brain/intelligence.ts +103 -0
- package/src/brain/types.ts +26 -0
- package/src/chat/agent-loop-types.ts +99 -0
- package/src/chat/agent-loop.ts +357 -0
- package/src/chat/auth-manager.ts +171 -0
- package/src/chat/browser-session.ts +188 -0
- package/src/chat/cancellation.ts +99 -0
- package/src/chat/chat-session.ts +283 -0
- package/src/chat/file-handler.ts +230 -0
- package/src/chat/fragment-buffer.ts +160 -0
- package/src/chat/index.ts +72 -0
- package/src/chat/mcp-bridge.ts +135 -0
- package/src/chat/notifications.ts +164 -0
- package/src/chat/output-compressor.ts +116 -0
- package/src/chat/queue.ts +208 -0
- package/src/chat/response-chunker.ts +200 -0
- package/src/chat/self-update.ts +117 -0
- package/src/chat/types.ts +126 -0
- package/src/chat/voice.ts +134 -0
- package/src/claudemd/compose.ts +142 -0
- package/src/claudemd/index.ts +17 -0
- package/src/claudemd/inject.ts +170 -0
- package/src/claudemd/types.ts +45 -0
- package/src/context/context-engine.ts +302 -0
- package/src/context/index.ts +11 -0
- package/src/context/types.ts +69 -0
- package/src/enforcement/adapters/claude-code.ts +135 -0
- package/src/enforcement/adapters/index.ts +1 -0
- package/src/enforcement/index.ts +10 -0
- package/src/enforcement/registry.ts +82 -0
- package/src/enforcement/types.ts +56 -0
- package/src/facades/facade-factory.ts +138 -5
- package/src/facades/types.ts +21 -0
- package/src/health/health-registry.ts +165 -0
- package/src/health/index.ts +11 -0
- package/src/health/vault-integrity.ts +66 -0
- package/src/index.ts +294 -2
- package/src/intake/intake-pipeline.ts +1 -1
- package/src/intelligence/types.ts +1 -0
- package/src/migrations/index.ts +6 -0
- package/src/migrations/migration-runner.ts +185 -0
- package/src/packs/index.ts +20 -0
- package/src/packs/lockfile.ts +180 -0
- package/src/packs/pack-installer.ts +289 -0
- package/src/packs/resolver.ts +237 -0
- package/src/packs/types.ts +125 -0
- package/src/persistence/postgres-provider.ts +211 -58
- package/src/planning/gap-analysis.ts +52 -7
- package/src/playbooks/index.ts +11 -0
- package/src/playbooks/playbook-executor.ts +301 -0
- package/src/plugins/index.ts +19 -0
- package/src/plugins/plugin-loader.ts +183 -0
- package/src/plugins/plugin-registry.ts +187 -0
- package/src/plugins/types.ts +119 -0
- package/src/runtime/admin-extra-ops.ts +193 -8
- package/src/runtime/capture-ops.ts +113 -8
- package/src/runtime/deprecation.ts +58 -0
- package/src/runtime/facades/admin-facade.ts +16 -1
- package/src/runtime/facades/agency-facade.ts +111 -0
- package/src/runtime/facades/brain-facade.ts +60 -0
- package/src/runtime/facades/chat-facade.ts +918 -0
- package/src/runtime/facades/context-facade.ts +55 -0
- package/src/runtime/facades/index.ts +22 -1
- package/src/runtime/facades/vault-facade.ts +261 -1
- package/src/runtime/feature-flags.ts +101 -0
- package/src/runtime/pack-ops.ts +85 -0
- package/src/runtime/playbook-ops.ts +113 -9
- package/src/runtime/plugin-ops.ts +258 -0
- package/src/runtime/runtime.ts +84 -5
- package/src/runtime/telemetry-ops.ts +57 -0
- package/src/runtime/types.ts +35 -0
- package/src/runtime/vault-sharing-ops.ts +372 -0
- package/src/streams/index.ts +1 -1
- package/src/streams/replayable-stream.ts +34 -3
- package/src/text/similarity.ts +1 -1
- package/src/transport/http-server.ts +269 -0
- package/src/transport/index.ts +48 -0
- package/src/transport/lsp-server.ts +401 -0
- package/src/transport/rate-limiter.ts +97 -0
- package/src/transport/session-manager.ts +120 -0
- package/src/transport/token-auth.ts +96 -0
- package/src/transport/types.ts +66 -0
- package/src/transport/ws-server.ts +415 -0
- package/src/vault/git-vault-sync.ts +318 -0
- package/src/vault/knowledge-review.ts +221 -0
- package/src/vault/obsidian-sync.ts +346 -0
- package/src/vault/scope-detector.ts +219 -0
- package/src/vault/vault-branching.ts +264 -0
- package/src/vault/vault-manager.ts +237 -0
- package/src/vault/vault-types.ts +50 -0
- package/src/vault/vault.ts +41 -3
- package/src/governance/index.ts +0 -18
|
@@ -0,0 +1,346 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Obsidian Bidirectional Sync — export/import vault entries as Obsidian markdown.
|
|
3
|
+
*
|
|
4
|
+
* Supports three modes:
|
|
5
|
+
* - Push (vault → Obsidian)
|
|
6
|
+
* - Pull (Obsidian → vault)
|
|
7
|
+
* - Bidirectional (timestamp-based merge with conflict detection)
|
|
8
|
+
*
|
|
9
|
+
* Format: YAML frontmatter + body content + wikilinks for related entries.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { mkdirSync, writeFileSync, readFileSync, readdirSync } from 'node:fs';
|
|
13
|
+
import { join, extname, relative, dirname } from 'node:path';
|
|
14
|
+
import type { Vault } from './vault.js';
|
|
15
|
+
import type { IntelligenceEntry } from '../intelligence/types.js';
|
|
16
|
+
|
|
17
|
+
// ─── Types ────────────────────────────────────────────────────────────
|
|
18
|
+
|
|
19
|
+
export interface ObsidianSyncConfig {
|
|
20
|
+
vault: Vault;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export interface ExportOptions {
|
|
24
|
+
types?: string[];
|
|
25
|
+
domains?: string[];
|
|
26
|
+
dryRun?: boolean;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface ImportOptions {
|
|
30
|
+
defaultType?: string;
|
|
31
|
+
defaultDomain?: string;
|
|
32
|
+
dryRun?: boolean;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export type SyncMode = 'push' | 'pull' | 'bidirectional';
|
|
36
|
+
|
|
37
|
+
export interface SyncOptions {
|
|
38
|
+
mode?: SyncMode;
|
|
39
|
+
dryRun?: boolean;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export interface ExportResult {
|
|
43
|
+
exported: number;
|
|
44
|
+
files: string[];
|
|
45
|
+
skipped: number;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export interface ImportResult {
|
|
49
|
+
imported: number;
|
|
50
|
+
updated: number;
|
|
51
|
+
skipped: number;
|
|
52
|
+
conflicts: ConflictInfo[];
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export interface SyncResult {
|
|
56
|
+
pushed: number;
|
|
57
|
+
pulled: number;
|
|
58
|
+
conflicts: ConflictInfo[];
|
|
59
|
+
mode: SyncMode;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
export interface ConflictInfo {
|
|
63
|
+
title: string;
|
|
64
|
+
id: string;
|
|
65
|
+
vaultUpdated: number;
|
|
66
|
+
obsidianUpdated: number;
|
|
67
|
+
vaultSnippet: string;
|
|
68
|
+
obsidianSnippet: string;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// ─── Format Helpers ──────────────────────────────────────────────────
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Convert a vault entry to Obsidian-compatible markdown with YAML frontmatter.
|
|
75
|
+
*/
|
|
76
|
+
export function toObsidianMarkdown(entry: IntelligenceEntry): string {
|
|
77
|
+
const lines: string[] = ['---'];
|
|
78
|
+
|
|
79
|
+
lines.push(`id: "${entry.id}"`);
|
|
80
|
+
lines.push(`type: "${entry.type}"`);
|
|
81
|
+
if (entry.domain) lines.push(`domain: "${entry.domain}"`);
|
|
82
|
+
if (entry.severity) lines.push(`severity: "${entry.severity}"`);
|
|
83
|
+
if (entry.tags && entry.tags.length > 0) {
|
|
84
|
+
lines.push(`tags: [${entry.tags.map((t) => `"${t}"`).join(', ')}]`);
|
|
85
|
+
}
|
|
86
|
+
lines.push(`updated: ${Date.now()}`);
|
|
87
|
+
lines.push('---');
|
|
88
|
+
lines.push('');
|
|
89
|
+
lines.push(`# ${entry.title}`);
|
|
90
|
+
lines.push('');
|
|
91
|
+
lines.push(entry.description);
|
|
92
|
+
|
|
93
|
+
return lines.join('\n');
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
/**
|
|
97
|
+
* Parse Obsidian markdown with YAML frontmatter back to vault entry fields.
|
|
98
|
+
*/
|
|
99
|
+
export function fromObsidianMarkdown(content: string): {
|
|
100
|
+
id?: string;
|
|
101
|
+
type?: string;
|
|
102
|
+
domain?: string;
|
|
103
|
+
severity?: string;
|
|
104
|
+
tags?: string[];
|
|
105
|
+
title?: string;
|
|
106
|
+
description?: string;
|
|
107
|
+
updated?: number;
|
|
108
|
+
} {
|
|
109
|
+
const result: Record<string, unknown> = {};
|
|
110
|
+
|
|
111
|
+
// Parse YAML frontmatter
|
|
112
|
+
const fmMatch = content.match(/^---\n([\s\S]*?)\n---/);
|
|
113
|
+
if (fmMatch) {
|
|
114
|
+
const yaml = fmMatch[1];
|
|
115
|
+
for (const line of yaml.split('\n')) {
|
|
116
|
+
const kv = line.match(/^(\w+):\s*(.+)$/);
|
|
117
|
+
if (!kv) continue;
|
|
118
|
+
const [, key, value] = kv;
|
|
119
|
+
if (key === 'tags') {
|
|
120
|
+
const tagMatch = value.match(/\[([^\]]*)\]/);
|
|
121
|
+
if (tagMatch) {
|
|
122
|
+
result.tags = tagMatch[1]
|
|
123
|
+
.split(',')
|
|
124
|
+
.map((t) => t.trim().replace(/^"|"$/g, ''))
|
|
125
|
+
.filter(Boolean);
|
|
126
|
+
}
|
|
127
|
+
} else if (key === 'updated') {
|
|
128
|
+
result.updated = parseInt(value, 10);
|
|
129
|
+
} else {
|
|
130
|
+
result[key] = value.replace(/^"|"$/g, '');
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Parse body
|
|
136
|
+
const body = content.replace(/^---\n[\s\S]*?\n---\n*/, '');
|
|
137
|
+
const titleMatch = body.match(/^# (.+)$/m);
|
|
138
|
+
if (titleMatch) {
|
|
139
|
+
result.title = titleMatch[1];
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
const description = body.replace(/^# .+\n*/, '').trim();
|
|
143
|
+
if (description) {
|
|
144
|
+
result.description = description;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// Type inference when missing
|
|
148
|
+
if (!result.type && description) {
|
|
149
|
+
const lower = description.toLowerCase();
|
|
150
|
+
if (/\b(don't|avoid|never|anti-pattern)\b/.test(lower)) {
|
|
151
|
+
result.type = 'anti-pattern';
|
|
152
|
+
} else if (/\b(always|prefer|use|pattern)\b/.test(lower)) {
|
|
153
|
+
result.type = 'pattern';
|
|
154
|
+
} else if (/^rule:/i.test(description)) {
|
|
155
|
+
result.type = 'rule';
|
|
156
|
+
} else {
|
|
157
|
+
result.type = 'concept';
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
return result as {
|
|
162
|
+
id?: string;
|
|
163
|
+
type?: string;
|
|
164
|
+
domain?: string;
|
|
165
|
+
severity?: string;
|
|
166
|
+
tags?: string[];
|
|
167
|
+
title?: string;
|
|
168
|
+
description?: string;
|
|
169
|
+
updated?: number;
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
* Slugify a title for use as a filename.
|
|
175
|
+
*/
|
|
176
|
+
export function titleToSlug(title: string): string {
|
|
177
|
+
return title
|
|
178
|
+
.toLowerCase()
|
|
179
|
+
.replace(/[^\w\s-]/g, '')
|
|
180
|
+
.replace(/\s+/g, '-')
|
|
181
|
+
.replace(/-+/g, '-')
|
|
182
|
+
.replace(/^-|-$/g, '')
|
|
183
|
+
.slice(0, 80);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
// ─── Sync Engine ─────────────────────────────────────────────────────
|
|
187
|
+
|
|
188
|
+
export class ObsidianSync {
|
|
189
|
+
private vault: Vault;
|
|
190
|
+
|
|
191
|
+
constructor(config: ObsidianSyncConfig) {
|
|
192
|
+
this.vault = config.vault;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
/**
|
|
196
|
+
* Export vault entries to Obsidian markdown files.
|
|
197
|
+
*/
|
|
198
|
+
export(obsidianDir: string, opts: ExportOptions = {}): ExportResult {
|
|
199
|
+
let entries = this.vault.list({});
|
|
200
|
+
|
|
201
|
+
// Filter by types and domains (vault.list takes singular, we support arrays)
|
|
202
|
+
if (opts.types && opts.types.length > 0) {
|
|
203
|
+
entries = entries.filter((e) => opts.types!.includes(e.type));
|
|
204
|
+
}
|
|
205
|
+
if (opts.domains && opts.domains.length > 0) {
|
|
206
|
+
entries = entries.filter((e) => opts.domains!.includes(e.domain || ''));
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
const files: string[] = [];
|
|
210
|
+
let skipped = 0;
|
|
211
|
+
|
|
212
|
+
for (const entry of entries) {
|
|
213
|
+
const domain = entry.domain || 'general';
|
|
214
|
+
const slug = titleToSlug(entry.title);
|
|
215
|
+
if (!slug) {
|
|
216
|
+
skipped++;
|
|
217
|
+
continue;
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
const dir = join(obsidianDir, domain);
|
|
221
|
+
const filePath = join(dir, `${slug}.md`);
|
|
222
|
+
const relPath = relative(obsidianDir, filePath);
|
|
223
|
+
|
|
224
|
+
if (!opts.dryRun) {
|
|
225
|
+
mkdirSync(dir, { recursive: true });
|
|
226
|
+
writeFileSync(filePath, toObsidianMarkdown(entry), 'utf-8');
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
files.push(relPath);
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
return { exported: files.length, files, skipped };
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
/**
|
|
236
|
+
* Import Obsidian markdown files into the vault.
|
|
237
|
+
*/
|
|
238
|
+
import(obsidianDir: string, opts: ImportOptions = {}): ImportResult {
|
|
239
|
+
const mdFiles = this.findMarkdownFiles(obsidianDir);
|
|
240
|
+
let imported = 0;
|
|
241
|
+
let updated = 0;
|
|
242
|
+
let skipped = 0;
|
|
243
|
+
const conflicts: ConflictInfo[] = [];
|
|
244
|
+
|
|
245
|
+
for (const file of mdFiles) {
|
|
246
|
+
const content = readFileSync(file, 'utf-8');
|
|
247
|
+
const parsed = fromObsidianMarkdown(content);
|
|
248
|
+
|
|
249
|
+
if (!parsed.title || !parsed.description) {
|
|
250
|
+
skipped++;
|
|
251
|
+
continue;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Use directory name as domain fallback
|
|
255
|
+
const relDir = dirname(relative(obsidianDir, file));
|
|
256
|
+
const domain = parsed.domain || (relDir !== '.' ? relDir : opts.defaultDomain || 'general');
|
|
257
|
+
const type = parsed.type || opts.defaultType || 'concept';
|
|
258
|
+
|
|
259
|
+
// Check if exists by ID
|
|
260
|
+
if (parsed.id) {
|
|
261
|
+
const existing = this.vault.get(parsed.id);
|
|
262
|
+
if (existing) {
|
|
263
|
+
if (!opts.dryRun) {
|
|
264
|
+
this.vault.update(parsed.id, {
|
|
265
|
+
title: parsed.title,
|
|
266
|
+
description: parsed.description,
|
|
267
|
+
domain,
|
|
268
|
+
tags: parsed.tags || [],
|
|
269
|
+
});
|
|
270
|
+
}
|
|
271
|
+
updated++;
|
|
272
|
+
continue;
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// New entry
|
|
277
|
+
if (!opts.dryRun) {
|
|
278
|
+
this.vault.seed([
|
|
279
|
+
{
|
|
280
|
+
id: parsed.id || `obsidian-${titleToSlug(parsed.title)}-${Date.now()}`,
|
|
281
|
+
type: type as 'pattern' | 'anti-pattern' | 'rule',
|
|
282
|
+
domain,
|
|
283
|
+
title: parsed.title,
|
|
284
|
+
description: parsed.description,
|
|
285
|
+
severity: (parsed.severity as 'critical' | 'warning' | 'suggestion') || 'suggestion',
|
|
286
|
+
tags: parsed.tags || [],
|
|
287
|
+
},
|
|
288
|
+
]);
|
|
289
|
+
}
|
|
290
|
+
imported++;
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
return { imported, updated, skipped, conflicts };
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
/**
|
|
297
|
+
* Bidirectional sync between vault and Obsidian directory.
|
|
298
|
+
*/
|
|
299
|
+
sync(obsidianDir: string, opts: SyncOptions = {}): SyncResult {
|
|
300
|
+
const mode = opts.mode || 'bidirectional';
|
|
301
|
+
|
|
302
|
+
if (mode === 'push') {
|
|
303
|
+
const result = this.export(obsidianDir, { dryRun: opts.dryRun });
|
|
304
|
+
return { pushed: result.exported, pulled: 0, conflicts: [], mode };
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
if (mode === 'pull') {
|
|
308
|
+
const result = this.import(obsidianDir, { dryRun: opts.dryRun });
|
|
309
|
+
return {
|
|
310
|
+
pushed: 0,
|
|
311
|
+
pulled: result.imported + result.updated,
|
|
312
|
+
conflicts: result.conflicts,
|
|
313
|
+
mode,
|
|
314
|
+
};
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
// Bidirectional: export first, then import new entries
|
|
318
|
+
const exportResult = this.export(obsidianDir, { dryRun: opts.dryRun });
|
|
319
|
+
const importResult = this.import(obsidianDir, { dryRun: opts.dryRun });
|
|
320
|
+
|
|
321
|
+
return {
|
|
322
|
+
pushed: exportResult.exported,
|
|
323
|
+
pulled: importResult.imported + importResult.updated,
|
|
324
|
+
conflicts: importResult.conflicts,
|
|
325
|
+
mode,
|
|
326
|
+
};
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
private findMarkdownFiles(dir: string): string[] {
|
|
330
|
+
const results: string[] = [];
|
|
331
|
+
try {
|
|
332
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
333
|
+
for (const entry of entries) {
|
|
334
|
+
const fullPath = join(dir, entry.name);
|
|
335
|
+
if (entry.isDirectory()) {
|
|
336
|
+
results.push(...this.findMarkdownFiles(fullPath));
|
|
337
|
+
} else if (extname(entry.name) === '.md') {
|
|
338
|
+
results.push(fullPath);
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
} catch {
|
|
342
|
+
// Directory doesn't exist or can't be read
|
|
343
|
+
}
|
|
344
|
+
return results;
|
|
345
|
+
}
|
|
346
|
+
}
|
|
@@ -0,0 +1,219 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Scope detector — classifies knowledge entries into agent/project/team tiers.
|
|
3
|
+
*
|
|
4
|
+
* Uses weighted signal heuristics from content, category, tags, and title
|
|
5
|
+
* to determine where an entry should live:
|
|
6
|
+
* - agent: personal preferences, local config, workflow habits
|
|
7
|
+
* - project: repo-specific patterns, architecture decisions, internal APIs
|
|
8
|
+
* - team: universal patterns, language best practices, design principles
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
export type ScopeTier = 'agent' | 'project' | 'team';
|
|
12
|
+
export type ConfidenceLevel = 'HIGH' | 'MEDIUM' | 'LOW';
|
|
13
|
+
|
|
14
|
+
export interface ScopeSignal {
|
|
15
|
+
tier: ScopeTier;
|
|
16
|
+
source: 'content' | 'category' | 'tags' | 'title';
|
|
17
|
+
indicator: string;
|
|
18
|
+
weight: number;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface ScopeDetectionResult {
|
|
22
|
+
tier: ScopeTier;
|
|
23
|
+
confidence: ConfidenceLevel;
|
|
24
|
+
reason: string;
|
|
25
|
+
signals: ScopeSignal[];
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface ScopeInput {
|
|
29
|
+
title: string;
|
|
30
|
+
description: string;
|
|
31
|
+
category?: string;
|
|
32
|
+
tags?: string[];
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// ─── Signal patterns ─────────────────────────────────────────────────
|
|
36
|
+
|
|
37
|
+
const TEAM_CONTENT_PATTERNS: Array<{ pattern: RegExp; weight: number; desc: string }> = [
|
|
38
|
+
{ pattern: /accessibility|a11y|aria-|screen\s+reader/i, weight: 0.85, desc: 'accessibility' },
|
|
39
|
+
{ pattern: /design\s+system|semantic\s+token/i, weight: 0.8, desc: 'design system' },
|
|
40
|
+
{ pattern: /contrast\s+ratio|wcag/i, weight: 0.85, desc: 'WCAG guidelines' },
|
|
41
|
+
{
|
|
42
|
+
pattern: /clean\s+code|solid\s+principle|dry|kiss/i,
|
|
43
|
+
weight: 0.7,
|
|
44
|
+
desc: 'clean code principles',
|
|
45
|
+
},
|
|
46
|
+
{ pattern: /security|xss|csrf|injection|sanitiz/i, weight: 0.75, desc: 'security patterns' },
|
|
47
|
+
{ pattern: /performance|lazy\s+load|memoiz/i, weight: 0.6, desc: 'performance patterns' },
|
|
48
|
+
{ pattern: /type\s+safety|type\s+guard/i, weight: 0.55, desc: 'type safety' },
|
|
49
|
+
{ pattern: /error\s+handling|error\s+boundary/i, weight: 0.6, desc: 'error handling' },
|
|
50
|
+
{ pattern: /touch\s+target|tap\s+target|fitts/i, weight: 0.8, desc: 'UX touch targets' },
|
|
51
|
+
{ pattern: /focus\s+(ring|state|indicator)/i, weight: 0.8, desc: 'focus states' },
|
|
52
|
+
{ pattern: /best\s+practice|anti.?pattern/i, weight: 0.65, desc: 'best practice' },
|
|
53
|
+
];
|
|
54
|
+
|
|
55
|
+
const PROJECT_CONTENT_PATTERNS: Array<{ pattern: RegExp; weight: number; desc: string }> = [
|
|
56
|
+
{ pattern: /@[\w-]+\/[\w-]+/, weight: 0.9, desc: 'scoped package (@org/pkg)' },
|
|
57
|
+
{ pattern: /packages\/[\w/-]+/, weight: 0.8, desc: 'monorepo path' },
|
|
58
|
+
{ pattern: /src\/[\w/-]+\.(ts|js|tsx|jsx)/, weight: 0.7, desc: 'project file path' },
|
|
59
|
+
{
|
|
60
|
+
pattern: /our\s+(project|codebase|team|repo)/i,
|
|
61
|
+
weight: 0.75,
|
|
62
|
+
desc: 'project-specific language',
|
|
63
|
+
},
|
|
64
|
+
{ pattern: /this\s+(project|codebase|repository)/i, weight: 0.8, desc: 'explicit project ref' },
|
|
65
|
+
{ pattern: /internal\s+(api|service)/i, weight: 0.7, desc: 'internal service' },
|
|
66
|
+
{ pattern: /\.config\.(ts|js|mjs)/i, weight: 0.5, desc: 'config file reference' },
|
|
67
|
+
{ pattern: /localhost|127\.0\.0\.1/i, weight: 0.6, desc: 'local dev reference' },
|
|
68
|
+
];
|
|
69
|
+
|
|
70
|
+
const AGENT_CONTENT_PATTERNS: Array<{ pattern: RegExp; weight: number; desc: string }> = [
|
|
71
|
+
{ pattern: /I\s+prefer|my\s+workflow|my\s+setup/i, weight: 0.8, desc: 'personal preference' },
|
|
72
|
+
{ pattern: /~\/|\/Users\/|\/home\//i, weight: 0.7, desc: 'home directory path' },
|
|
73
|
+
{ pattern: /my\s+(editor|IDE|terminal|shell)/i, weight: 0.75, desc: 'personal tooling' },
|
|
74
|
+
{ pattern: /alias(es)?|shortcut/i, weight: 0.6, desc: 'personal aliases' },
|
|
75
|
+
{ pattern: /habit|routine|always\s+do/i, weight: 0.55, desc: 'personal habit' },
|
|
76
|
+
];
|
|
77
|
+
|
|
78
|
+
const TEAM_CATEGORIES = new Set([
|
|
79
|
+
'styling',
|
|
80
|
+
'accessibility',
|
|
81
|
+
'performance',
|
|
82
|
+
'security',
|
|
83
|
+
'design',
|
|
84
|
+
'methodology',
|
|
85
|
+
]);
|
|
86
|
+
const PROJECT_CATEGORIES = new Set([
|
|
87
|
+
'monorepo',
|
|
88
|
+
'prisma',
|
|
89
|
+
'infrastructure',
|
|
90
|
+
'deployment',
|
|
91
|
+
'config',
|
|
92
|
+
]);
|
|
93
|
+
|
|
94
|
+
const TEAM_TAGS = new Set([
|
|
95
|
+
'universal',
|
|
96
|
+
'best-practice',
|
|
97
|
+
'a11y',
|
|
98
|
+
'accessibility',
|
|
99
|
+
'wcag',
|
|
100
|
+
'security',
|
|
101
|
+
'performance',
|
|
102
|
+
'clean-code',
|
|
103
|
+
'design-system',
|
|
104
|
+
'pattern',
|
|
105
|
+
'principle',
|
|
106
|
+
]);
|
|
107
|
+
const PROJECT_TAGS = new Set([
|
|
108
|
+
'project-specific',
|
|
109
|
+
'internal',
|
|
110
|
+
'monorepo',
|
|
111
|
+
'infrastructure',
|
|
112
|
+
'deployment',
|
|
113
|
+
]);
|
|
114
|
+
const AGENT_TAGS = new Set(['personal', 'preference', 'workflow', 'habit', 'local']);
|
|
115
|
+
|
|
116
|
+
// ─── Analysis functions ──────────────────────────────────────────────
|
|
117
|
+
|
|
118
|
+
function analyzeContent(text: string): ScopeSignal[] {
|
|
119
|
+
const signals: ScopeSignal[] = [];
|
|
120
|
+
|
|
121
|
+
for (const { pattern, weight, desc } of TEAM_CONTENT_PATTERNS) {
|
|
122
|
+
if (pattern.test(text)) {
|
|
123
|
+
signals.push({ tier: 'team', source: 'content', indicator: desc, weight });
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
for (const { pattern, weight, desc } of PROJECT_CONTENT_PATTERNS) {
|
|
127
|
+
if (pattern.test(text)) {
|
|
128
|
+
signals.push({ tier: 'project', source: 'content', indicator: desc, weight });
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
for (const { pattern, weight, desc } of AGENT_CONTENT_PATTERNS) {
|
|
132
|
+
if (pattern.test(text)) {
|
|
133
|
+
signals.push({ tier: 'agent', source: 'content', indicator: desc, weight });
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
return signals;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
function analyzeCategory(category: string): ScopeSignal[] {
|
|
141
|
+
const norm = category.toLowerCase();
|
|
142
|
+
if (TEAM_CATEGORIES.has(norm)) {
|
|
143
|
+
return [{ tier: 'team', source: 'category', indicator: `category "${category}"`, weight: 0.7 }];
|
|
144
|
+
}
|
|
145
|
+
if (PROJECT_CATEGORIES.has(norm)) {
|
|
146
|
+
return [
|
|
147
|
+
{ tier: 'project', source: 'category', indicator: `category "${category}"`, weight: 0.6 },
|
|
148
|
+
];
|
|
149
|
+
}
|
|
150
|
+
return [];
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
function analyzeTags(tags: string[]): ScopeSignal[] {
|
|
154
|
+
const signals: ScopeSignal[] = [];
|
|
155
|
+
for (const tag of tags) {
|
|
156
|
+
const norm = tag.toLowerCase();
|
|
157
|
+
if (TEAM_TAGS.has(norm)) {
|
|
158
|
+
signals.push({ tier: 'team', source: 'tags', indicator: `tag "${tag}"`, weight: 0.8 });
|
|
159
|
+
} else if (PROJECT_TAGS.has(norm)) {
|
|
160
|
+
signals.push({ tier: 'project', source: 'tags', indicator: `tag "${tag}"`, weight: 0.8 });
|
|
161
|
+
} else if (AGENT_TAGS.has(norm)) {
|
|
162
|
+
signals.push({ tier: 'agent', source: 'tags', indicator: `tag "${tag}"`, weight: 0.8 });
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
return signals;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function computeConfidence(scores: Record<ScopeTier, number>, winner: ScopeTier): ConfidenceLevel {
|
|
169
|
+
const winScore = scores[winner];
|
|
170
|
+
const others = Object.entries(scores)
|
|
171
|
+
.filter(([t]) => t !== winner)
|
|
172
|
+
.map(([, v]) => v);
|
|
173
|
+
const runnerUp = Math.max(...others, 0);
|
|
174
|
+
|
|
175
|
+
if (winScore === 0) return 'LOW';
|
|
176
|
+
if (runnerUp === 0 && winScore >= 0.5) return 'HIGH';
|
|
177
|
+
const ratio = winScore / (winScore + runnerUp);
|
|
178
|
+
if (ratio >= 0.7 && winScore >= 1.0) return 'HIGH';
|
|
179
|
+
if (ratio >= 0.55) return 'MEDIUM';
|
|
180
|
+
return 'LOW';
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// ─── Public API ──────────────────────────────────────────────────────
|
|
184
|
+
|
|
185
|
+
export function detectScope(input: ScopeInput): ScopeDetectionResult {
|
|
186
|
+
const signals: ScopeSignal[] = [];
|
|
187
|
+
const fullText = `${input.title} ${input.description}`;
|
|
188
|
+
|
|
189
|
+
signals.push(...analyzeContent(fullText));
|
|
190
|
+
if (input.category) signals.push(...analyzeCategory(input.category));
|
|
191
|
+
if (input.tags?.length) signals.push(...analyzeTags(input.tags));
|
|
192
|
+
|
|
193
|
+
const scores: Record<ScopeTier, number> = { agent: 0, project: 0, team: 0 };
|
|
194
|
+
for (const s of signals) {
|
|
195
|
+
scores[s.tier] += s.weight;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
// Default to agent when no signals
|
|
199
|
+
let winner: ScopeTier = 'agent';
|
|
200
|
+
let maxScore = 0;
|
|
201
|
+
for (const [tier, score] of Object.entries(scores) as Array<[ScopeTier, number]>) {
|
|
202
|
+
if (score > maxScore) {
|
|
203
|
+
maxScore = score;
|
|
204
|
+
winner = tier;
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const confidence = computeConfidence(scores, winner);
|
|
209
|
+
const topSignals = signals
|
|
210
|
+
.filter((s) => s.tier === winner)
|
|
211
|
+
.sort((a, b) => b.weight - a.weight)
|
|
212
|
+
.slice(0, 3);
|
|
213
|
+
const reason =
|
|
214
|
+
topSignals.length > 0
|
|
215
|
+
? topSignals.map((s) => s.indicator).join('; ')
|
|
216
|
+
: 'No clear signals — defaulting to agent tier';
|
|
217
|
+
|
|
218
|
+
return { tier: winner, confidence, reason, signals };
|
|
219
|
+
}
|