@happyvertical/smrt-core 0.30.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/AGENTS.md +124 -0
- package/CLAUDE.md +1 -0
- package/LICENSE +7 -0
- package/README.md +265 -0
- package/bin/smrt-prebuild.js +26 -0
- package/dist/__tests__/fixtures/advisor-test-classes.d.ts +28 -0
- package/dist/__tests__/fixtures/advisor-test-classes.d.ts.map +1 -0
- package/dist/__tests__/fixtures/collection-coverage-fixtures.d.ts +12 -0
- package/dist/__tests__/fixtures/collection-coverage-fixtures.d.ts.map +1 -0
- package/dist/__tests__/fixtures/inheritance-resolver-fixtures.d.ts +28 -0
- package/dist/__tests__/fixtures/inheritance-resolver-fixtures.d.ts.map +1 -0
- package/dist/__tests__/fixtures/inheritance-test-classes.d.ts +43 -0
- package/dist/__tests__/fixtures/inheritance-test-classes.d.ts.map +1 -0
- package/dist/__tests__/fixtures/mcp-integration-test-classes.d.ts +18 -0
- package/dist/__tests__/fixtures/mcp-integration-test-classes.d.ts.map +1 -0
- package/dist/__tests__/fixtures/object-ai-memory-fixtures.d.ts +15 -0
- package/dist/__tests__/fixtures/object-ai-memory-fixtures.d.ts.map +1 -0
- package/dist/__tests__/fixtures/object-spec-test-classes.d.ts +13 -0
- package/dist/__tests__/fixtures/object-spec-test-classes.d.ts.map +1 -0
- package/dist/__tests__/fixtures/registry-test-classes.d.ts +23 -0
- package/dist/__tests__/fixtures/registry-test-classes.d.ts.map +1 -0
- package/dist/adapters/ai-usage.d.ts +23 -0
- package/dist/adapters/ai-usage.d.ts.map +1 -0
- package/dist/adapters/ai-usage.js +105 -0
- package/dist/adapters/ai-usage.js.map +1 -0
- package/dist/adapters/cost-rates.d.ts +20 -0
- package/dist/adapters/cost-rates.d.ts.map +1 -0
- package/dist/adapters/cost-rates.js +40 -0
- package/dist/adapters/cost-rates.js.map +1 -0
- package/dist/adapters/index.d.ts +19 -0
- package/dist/adapters/index.d.ts.map +1 -0
- package/dist/adapters/metrics.d.ts +111 -0
- package/dist/adapters/metrics.d.ts.map +1 -0
- package/dist/adapters/metrics.js +169 -0
- package/dist/adapters/metrics.js.map +1 -0
- package/dist/adapters/pubsub.d.ts +124 -0
- package/dist/adapters/pubsub.d.ts.map +1 -0
- package/dist/adapters/pubsub.js +121 -0
- package/dist/adapters/pubsub.js.map +1 -0
- package/dist/browser.d.ts +32 -0
- package/dist/browser.d.ts.map +1 -0
- package/dist/browser.js +68 -0
- package/dist/browser.js.map +1 -0
- package/dist/child-accessors.d.ts +27 -0
- package/dist/child-accessors.d.ts.map +1 -0
- package/dist/child-accessors.js +35 -0
- package/dist/child-accessors.js.map +1 -0
- package/dist/class.d.ts +313 -0
- package/dist/class.d.ts.map +1 -0
- package/dist/class.js +896 -0
- package/dist/class.js.map +1 -0
- package/dist/collection-cache.d.ts +110 -0
- package/dist/collection-cache.d.ts.map +1 -0
- package/dist/collection-cache.js +187 -0
- package/dist/collection-cache.js.map +1 -0
- package/dist/collection.d.ts +894 -0
- package/dist/collection.d.ts.map +1 -0
- package/dist/collection.js +1987 -0
- package/dist/collection.js.map +1 -0
- package/dist/config/global-config.d.ts +3 -0
- package/dist/config/global-config.d.ts.map +1 -0
- package/dist/config/global-config.js +19 -0
- package/dist/config/global-config.js.map +1 -0
- package/dist/config.d.ts +145 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +57 -0
- package/dist/config.js.map +1 -0
- package/dist/consumer-plugin/index.d.ts +22 -0
- package/dist/consumer-plugin/index.d.ts.map +1 -0
- package/dist/consumer-plugin/index.js +452 -0
- package/dist/consumer-plugin/index.js.map +1 -0
- package/dist/consumer-plugin.d.ts +8 -0
- package/dist/consumer-plugin.d.ts.map +1 -0
- package/dist/consumer-plugin.js +5 -0
- package/dist/consumer-plugin.js.map +1 -0
- package/dist/database.d.ts +95 -0
- package/dist/database.d.ts.map +1 -0
- package/dist/database.js +32 -0
- package/dist/database.js.map +1 -0
- package/dist/decorators/compatibility.d.ts +14 -0
- package/dist/decorators/compatibility.d.ts.map +1 -0
- package/dist/decorators/compatibility.js +111 -0
- package/dist/decorators/compatibility.js.map +1 -0
- package/dist/decorators/index.d.ts +381 -0
- package/dist/decorators/index.d.ts.map +1 -0
- package/dist/decorators/index.js +104 -0
- package/dist/decorators/index.js.map +1 -0
- package/dist/dispatch/bus.d.ts +306 -0
- package/dist/dispatch/bus.d.ts.map +1 -0
- package/dist/dispatch/bus.js +583 -0
- package/dist/dispatch/bus.js.map +1 -0
- package/dist/dispatch/collections/DispatchSubscriptions.d.ts +79 -0
- package/dist/dispatch/collections/DispatchSubscriptions.d.ts.map +1 -0
- package/dist/dispatch/collections/DispatchSubscriptions.js +243 -0
- package/dist/dispatch/collections/DispatchSubscriptions.js.map +1 -0
- package/dist/dispatch/collections/Dispatches.d.ts +98 -0
- package/dist/dispatch/collections/Dispatches.d.ts.map +1 -0
- package/dist/dispatch/collections/Dispatches.js +358 -0
- package/dist/dispatch/collections/Dispatches.js.map +1 -0
- package/dist/dispatch/index.d.ts +47 -0
- package/dist/dispatch/index.d.ts.map +1 -0
- package/dist/dispatch/models/Dispatch.d.ts +101 -0
- package/dist/dispatch/models/Dispatch.d.ts.map +1 -0
- package/dist/dispatch/models/Dispatch.js +162 -0
- package/dist/dispatch/models/Dispatch.js.map +1 -0
- package/dist/dispatch/models/DispatchSubscription.d.ts +83 -0
- package/dist/dispatch/models/DispatchSubscription.d.ts.map +1 -0
- package/dist/dispatch/models/DispatchSubscription.js +112 -0
- package/dist/dispatch/models/DispatchSubscription.js.map +1 -0
- package/dist/dispatch/tenant-resolver.d.ts +98 -0
- package/dist/dispatch/tenant-resolver.d.ts.map +1 -0
- package/dist/dispatch/tenant-resolver.js +32 -0
- package/dist/dispatch/tenant-resolver.js.map +1 -0
- package/dist/dispatch/types.d.ts +149 -0
- package/dist/dispatch/types.d.ts.map +1 -0
- package/dist/embeddings/hash.d.ts +33 -0
- package/dist/embeddings/hash.d.ts.map +1 -0
- package/dist/embeddings/hash.js +37 -0
- package/dist/embeddings/hash.js.map +1 -0
- package/dist/embeddings/index.d.ts +36 -0
- package/dist/embeddings/index.d.ts.map +1 -0
- package/dist/embeddings/provider.d.ts +75 -0
- package/dist/embeddings/provider.d.ts.map +1 -0
- package/dist/embeddings/provider.js +170 -0
- package/dist/embeddings/provider.js.map +1 -0
- package/dist/embeddings/similarity.d.ts +47 -0
- package/dist/embeddings/similarity.d.ts.map +1 -0
- package/dist/embeddings/similarity.js +64 -0
- package/dist/embeddings/similarity.js.map +1 -0
- package/dist/embeddings/storage.d.ts +125 -0
- package/dist/embeddings/storage.d.ts.map +1 -0
- package/dist/embeddings/storage.js +283 -0
- package/dist/embeddings/storage.js.map +1 -0
- package/dist/embeddings/types.d.ts +250 -0
- package/dist/embeddings/types.d.ts.map +1 -0
- package/dist/errors.d.ts +363 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +669 -0
- package/dist/errors.js.map +1 -0
- package/dist/generators/cli.d.ts +162 -0
- package/dist/generators/cli.d.ts.map +1 -0
- package/dist/generators/cli.js +462 -0
- package/dist/generators/cli.js.map +1 -0
- package/dist/generators/index.d.ts +13 -0
- package/dist/generators/index.d.ts.map +1 -0
- package/dist/generators/mcp-runtime-template.d.ts +60 -0
- package/dist/generators/mcp-runtime-template.d.ts.map +1 -0
- package/dist/generators/mcp-runtime-template.js +509 -0
- package/dist/generators/mcp-runtime-template.js.map +1 -0
- package/dist/generators/mcp.d.ts +231 -0
- package/dist/generators/mcp.d.ts.map +1 -0
- package/dist/generators/mcp.js +1220 -0
- package/dist/generators/mcp.js.map +1 -0
- package/dist/generators/rest.d.ts +171 -0
- package/dist/generators/rest.d.ts.map +1 -0
- package/dist/generators/rest.js +591 -0
- package/dist/generators/rest.js.map +1 -0
- package/dist/generators/swagger.d.ts +21 -0
- package/dist/generators/swagger.d.ts.map +1 -0
- package/dist/generators/swagger.js +307 -0
- package/dist/generators/swagger.js.map +1 -0
- package/dist/generators/tenant-gate.d.ts +74 -0
- package/dist/generators/tenant-gate.d.ts.map +1 -0
- package/dist/generators/tenant-gate.js +15 -0
- package/dist/generators/tenant-gate.js.map +1 -0
- package/dist/generators.d.ts +8 -0
- package/dist/generators.d.ts.map +1 -0
- package/dist/generators.js +19 -0
- package/dist/generators.js.map +1 -0
- package/dist/hierarchical.d.ts +103 -0
- package/dist/hierarchical.d.ts.map +1 -0
- package/dist/hierarchical.js +184 -0
- package/dist/hierarchical.js.map +1 -0
- package/dist/index.d.ts +57 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +202 -0
- package/dist/index.js.map +1 -0
- package/dist/interceptors.d.ts +251 -0
- package/dist/interceptors.d.ts.map +1 -0
- package/dist/interceptors.js +259 -0
- package/dist/interceptors.js.map +1 -0
- package/dist/junction.d.ts +99 -0
- package/dist/junction.d.ts.map +1 -0
- package/dist/junction.js +136 -0
- package/dist/junction.js.map +1 -0
- package/dist/knowledge.d.ts +11 -0
- package/dist/knowledge.d.ts.map +1 -0
- package/dist/knowledge.js +310 -0
- package/dist/knowledge.js.map +1 -0
- package/dist/lazy-config.d.ts +160 -0
- package/dist/lazy-config.d.ts.map +1 -0
- package/dist/lazy-config.js +146 -0
- package/dist/lazy-config.js.map +1 -0
- package/dist/manifest/discover-base-classes.d.ts +78 -0
- package/dist/manifest/discover-base-classes.d.ts.map +1 -0
- package/dist/manifest/discover-base-classes.js +85 -0
- package/dist/manifest/discover-base-classes.js.map +1 -0
- package/dist/manifest/discover-smrt-packages.d.ts +48 -0
- package/dist/manifest/discover-smrt-packages.d.ts.map +1 -0
- package/dist/manifest/discover-smrt-packages.js +361 -0
- package/dist/manifest/discover-smrt-packages.js.map +1 -0
- package/dist/manifest/generator.d.ts +93 -0
- package/dist/manifest/generator.d.ts.map +1 -0
- package/dist/manifest/generator.js +380 -0
- package/dist/manifest/generator.js.map +1 -0
- package/dist/manifest/index.d.ts +16 -0
- package/dist/manifest/index.d.ts.map +1 -0
- package/dist/manifest/index.js +51 -0
- package/dist/manifest/index.js.map +1 -0
- package/dist/manifest/manager.d.ts +51 -0
- package/dist/manifest/manager.d.ts.map +1 -0
- package/dist/manifest/manager.js +89 -0
- package/dist/manifest/manager.js.map +1 -0
- package/dist/manifest/manifest-loader.d.ts +187 -0
- package/dist/manifest/manifest-loader.d.ts.map +1 -0
- package/dist/manifest/manifest-loader.js +847 -0
- package/dist/manifest/manifest-loader.js.map +1 -0
- package/dist/manifest/sources/composite.d.ts +22 -0
- package/dist/manifest/sources/composite.d.ts.map +1 -0
- package/dist/manifest/sources/composite.js +60 -0
- package/dist/manifest/sources/composite.js.map +1 -0
- package/dist/manifest/sources/embedded.d.ts +7 -0
- package/dist/manifest/sources/embedded.d.ts.map +1 -0
- package/dist/manifest/sources/embedded.js +30 -0
- package/dist/manifest/sources/embedded.js.map +1 -0
- package/dist/manifest/sources/explicit-paths.d.ts +17 -0
- package/dist/manifest/sources/explicit-paths.d.ts.map +1 -0
- package/dist/manifest/sources/explicit-paths.js +35 -0
- package/dist/manifest/sources/explicit-paths.js.map +1 -0
- package/dist/manifest/sources/fallback.d.ts +25 -0
- package/dist/manifest/sources/fallback.d.ts.map +1 -0
- package/dist/manifest/sources/fallback.js +63 -0
- package/dist/manifest/sources/fallback.js.map +1 -0
- package/dist/manifest/sources/index.d.ts +17 -0
- package/dist/manifest/sources/index.d.ts.map +1 -0
- package/dist/manifest/sources/local-test.d.ts +7 -0
- package/dist/manifest/sources/local-test.d.ts.map +1 -0
- package/dist/manifest/sources/local-test.js +21 -0
- package/dist/manifest/sources/local-test.js.map +1 -0
- package/dist/manifest/sources/static.d.ts +7 -0
- package/dist/manifest/sources/static.d.ts.map +1 -0
- package/dist/manifest/sources/static.js +19 -0
- package/dist/manifest/sources/static.js.map +1 -0
- package/dist/manifest/sources/test.d.ts +7 -0
- package/dist/manifest/sources/test.d.ts.map +1 -0
- package/dist/manifest/sources/test.js +21 -0
- package/dist/manifest/sources/test.js.map +1 -0
- package/dist/manifest/sources/types.d.ts +79 -0
- package/dist/manifest/sources/types.d.ts.map +1 -0
- package/dist/manifest/sources/types.js +61 -0
- package/dist/manifest/sources/types.js.map +1 -0
- package/dist/manifest/static-manifest.d.ts +4 -0
- package/dist/manifest/static-manifest.d.ts.map +1 -0
- package/dist/manifest/static-manifest.js +1535 -0
- package/dist/manifest/static-manifest.js.map +1 -0
- package/dist/manifest/store.d.ts +111 -0
- package/dist/manifest/store.d.ts.map +1 -0
- package/dist/manifest/store.js +431 -0
- package/dist/manifest/store.js.map +1 -0
- package/dist/manifest/test-manifest-loader.d.ts +3 -0
- package/dist/manifest/test-manifest-loader.d.ts.map +1 -0
- package/dist/manifest/test-manifest-stub.d.ts +4 -0
- package/dist/manifest/test-manifest-stub.d.ts.map +1 -0
- package/dist/manifest/test-manifest-stub.js +80013 -0
- package/dist/manifest/test-manifest-stub.js.map +1 -0
- package/dist/manifest.d.ts +8 -0
- package/dist/manifest.d.ts.map +1 -0
- package/dist/manifest.js +20 -0
- package/dist/manifest.js.map +1 -0
- package/dist/manifest.json +1489 -0
- package/dist/mcp-advisor/index.d.ts +499 -0
- package/dist/mcp-advisor/index.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/add-ai-methods.d.ts +6 -0
- package/dist/mcp-advisor/tools/add-ai-methods.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/configure-decorators.d.ts +6 -0
- package/dist/mcp-advisor/tools/configure-decorators.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/generate-collection.d.ts +6 -0
- package/dist/mcp-advisor/tools/generate-collection.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/generate-field-definitions.d.ts +6 -0
- package/dist/mcp-advisor/tools/generate-field-definitions.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/generate-smrt-class.d.ts +6 -0
- package/dist/mcp-advisor/tools/generate-smrt-class.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/get-object-config.d.ts +6 -0
- package/dist/mcp-advisor/tools/get-object-config.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/get-object-schema.d.ts +10 -0
- package/dist/mcp-advisor/tools/get-object-schema.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/list-registered-objects.d.ts +9 -0
- package/dist/mcp-advisor/tools/list-registered-objects.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/preview-api-endpoints.d.ts +9 -0
- package/dist/mcp-advisor/tools/preview-api-endpoints.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/preview-mcp-tools.d.ts +9 -0
- package/dist/mcp-advisor/tools/preview-mcp-tools.d.ts.map +1 -0
- package/dist/mcp-advisor/tools/validate-smrt-object.d.ts +6 -0
- package/dist/mcp-advisor/tools/validate-smrt-object.d.ts.map +1 -0
- package/dist/mcp-advisor/types.d.ts +209 -0
- package/dist/mcp-advisor/types.d.ts.map +1 -0
- package/dist/migrations/backfill-tracker.d.ts +84 -0
- package/dist/migrations/backfill-tracker.d.ts.map +1 -0
- package/dist/migrations/backfill-tracker.js +118 -0
- package/dist/migrations/backfill-tracker.js.map +1 -0
- package/dist/migrations/checksum.d.ts +43 -0
- package/dist/migrations/checksum.d.ts.map +1 -0
- package/dist/migrations/checksum.js +32 -0
- package/dist/migrations/checksum.js.map +1 -0
- package/dist/migrations/differ.d.ts +186 -0
- package/dist/migrations/differ.d.ts.map +1 -0
- package/dist/migrations/differ.js +601 -0
- package/dist/migrations/differ.js.map +1 -0
- package/dist/migrations/generator.d.ts +133 -0
- package/dist/migrations/generator.d.ts.map +1 -0
- package/dist/migrations/generator.js +328 -0
- package/dist/migrations/generator.js.map +1 -0
- package/dist/migrations/index.d.ts +20 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/orchestrate.d.ts +148 -0
- package/dist/migrations/orchestrate.d.ts.map +1 -0
- package/dist/migrations/orchestrate.js +118 -0
- package/dist/migrations/orchestrate.js.map +1 -0
- package/dist/migrations/tracker.d.ts +134 -0
- package/dist/migrations/tracker.d.ts.map +1 -0
- package/dist/migrations/tracker.js +624 -0
- package/dist/migrations/tracker.js.map +1 -0
- package/dist/migrations/types.d.ts +221 -0
- package/dist/migrations/types.d.ts.map +1 -0
- package/dist/migrations.d.ts +7 -0
- package/dist/migrations.d.ts.map +1 -0
- package/dist/migrations.js +26 -0
- package/dist/migrations.js.map +1 -0
- package/dist/node_modules/.pnpm/balanced-match@4.0.4/node_modules/balanced-match/dist/esm/index.js +56 -0
- package/dist/node_modules/.pnpm/balanced-match@4.0.4/node_modules/balanced-match/dist/esm/index.js.map +1 -0
- package/dist/node_modules/.pnpm/brace-expansion@5.0.5/node_modules/brace-expansion/dist/esm/index.js +163 -0
- package/dist/node_modules/.pnpm/brace-expansion@5.0.5/node_modules/brace-expansion/dist/esm/index.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/assert-valid-pattern.js +13 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/assert-valid-pattern.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/ast.js +654 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/ast.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/brace-expressions.js +111 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/brace-expressions.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/escape.js +10 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/escape.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/index.js +824 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/index.js.map +1 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/unescape.js +10 -0
- package/dist/node_modules/.pnpm/minimatch@10.2.3/node_modules/minimatch/dist/esm/unescape.js.map +1 -0
- package/dist/object.d.ts +1202 -0
- package/dist/object.d.ts.map +1 -0
- package/dist/object.js +2731 -0
- package/dist/object.js.map +1 -0
- package/dist/polymorphic-association.d.ts +69 -0
- package/dist/polymorphic-association.d.ts.map +1 -0
- package/dist/polymorphic-association.js +96 -0
- package/dist/polymorphic-association.js.map +1 -0
- package/dist/prebuild/cli.d.ts +7 -0
- package/dist/prebuild/cli.d.ts.map +1 -0
- package/dist/prebuild/cli.js +29 -0
- package/dist/prebuild/cli.js.map +1 -0
- package/dist/prebuild/index.d.ts +22 -0
- package/dist/prebuild/index.d.ts.map +1 -0
- package/dist/prebuild/index.js +239 -0
- package/dist/prebuild/index.js.map +1 -0
- package/dist/prebuild.d.ts +8 -0
- package/dist/prebuild.d.ts.map +1 -0
- package/dist/prebuild.js +6 -0
- package/dist/prebuild.js.map +1 -0
- package/dist/registry/cache-config.d.ts +13 -0
- package/dist/registry/cache-config.d.ts.map +1 -0
- package/dist/registry/cache-config.js +17 -0
- package/dist/registry/cache-config.js.map +1 -0
- package/dist/registry/class-registration.d.ts +31 -0
- package/dist/registry/class-registration.d.ts.map +1 -0
- package/dist/registry/class-registration.js +1074 -0
- package/dist/registry/class-registration.js.map +1 -0
- package/dist/registry/collection-resolution.d.ts +45 -0
- package/dist/registry/collection-resolution.d.ts.map +1 -0
- package/dist/registry/collection-resolution.js +121 -0
- package/dist/registry/collection-resolution.js.map +1 -0
- package/dist/registry/collision-policy.d.ts +179 -0
- package/dist/registry/collision-policy.d.ts.map +1 -0
- package/dist/registry/collision-policy.js +153 -0
- package/dist/registry/collision-policy.js.map +1 -0
- package/dist/registry/diagnostics.d.ts +58 -0
- package/dist/registry/diagnostics.d.ts.map +1 -0
- package/dist/registry/diagnostics.js +54 -0
- package/dist/registry/diagnostics.js.map +1 -0
- package/dist/registry/embedding-manager.d.ts +23 -0
- package/dist/registry/embedding-manager.d.ts.map +1 -0
- package/dist/registry/embedding-manager.js +62 -0
- package/dist/registry/embedding-manager.js.map +1 -0
- package/dist/registry/index.d.ts +13 -0
- package/dist/registry/index.d.ts.map +1 -0
- package/dist/registry/inheritance-resolver.d.ts +13 -0
- package/dist/registry/inheritance-resolver.d.ts.map +1 -0
- package/dist/registry/inheritance-resolver.js +244 -0
- package/dist/registry/inheritance-resolver.js.map +1 -0
- package/dist/registry/manifest-field-merge.d.ts +4 -0
- package/dist/registry/manifest-field-merge.d.ts.map +1 -0
- package/dist/registry/manifest-field-merge.js +82 -0
- package/dist/registry/manifest-field-merge.js.map +1 -0
- package/dist/registry/name-resolver.d.ts +102 -0
- package/dist/registry/name-resolver.d.ts.map +1 -0
- package/dist/registry/name-resolver.js +241 -0
- package/dist/registry/name-resolver.js.map +1 -0
- package/dist/registry/relationship-graph.d.ts +16 -0
- package/dist/registry/relationship-graph.d.ts.map +1 -0
- package/dist/registry/relationship-graph.js +79 -0
- package/dist/registry/relationship-graph.js.map +1 -0
- package/dist/registry/schema-builder.d.ts +113 -0
- package/dist/registry/schema-builder.d.ts.map +1 -0
- package/dist/registry/schema-builder.js +474 -0
- package/dist/registry/schema-builder.js.map +1 -0
- package/dist/registry/shared-state.d.ts +62 -0
- package/dist/registry/shared-state.d.ts.map +1 -0
- package/dist/registry/shared-state.js +135 -0
- package/dist/registry/shared-state.js.map +1 -0
- package/dist/registry/types.d.ts +667 -0
- package/dist/registry/types.d.ts.map +1 -0
- package/dist/registry/validator.d.ts +13 -0
- package/dist/registry/validator.d.ts.map +1 -0
- package/dist/registry/validator.js +138 -0
- package/dist/registry/validator.js.map +1 -0
- package/dist/registry.d.ts +1358 -0
- package/dist/registry.d.ts.map +1 -0
- package/dist/registry.js +2301 -0
- package/dist/registry.js.map +1 -0
- package/dist/runtime/client.d.ts +34 -0
- package/dist/runtime/client.d.ts.map +1 -0
- package/dist/runtime/client.js +104 -0
- package/dist/runtime/client.js.map +1 -0
- package/dist/runtime/index.d.ts +10 -0
- package/dist/runtime/index.d.ts.map +1 -0
- package/dist/runtime/mcp.d.ts +47 -0
- package/dist/runtime/mcp.d.ts.map +1 -0
- package/dist/runtime/mcp.js +72 -0
- package/dist/runtime/mcp.js.map +1 -0
- package/dist/runtime/server.d.ts +92 -0
- package/dist/runtime/server.d.ts.map +1 -0
- package/dist/runtime/server.js +390 -0
- package/dist/runtime/server.js.map +1 -0
- package/dist/runtime/types.d.ts +58 -0
- package/dist/runtime/types.d.ts.map +1 -0
- package/dist/runtime.d.ts +8 -0
- package/dist/runtime.d.ts.map +1 -0
- package/dist/runtime.js +10 -0
- package/dist/runtime.js.map +1 -0
- package/dist/scanner/import-scanner.d.ts +7 -0
- package/dist/scanner/import-scanner.d.ts.map +1 -0
- package/dist/scanner/index.d.ts +12 -0
- package/dist/scanner/index.d.ts.map +1 -0
- package/dist/scanner/manifest-generator.d.ts +304 -0
- package/dist/scanner/manifest-generator.d.ts.map +1 -0
- package/dist/scanner/manifest-generator.js +1707 -0
- package/dist/scanner/manifest-generator.js.map +1 -0
- package/dist/scanner/test-file-patterns.d.ts +18 -0
- package/dist/scanner/test-file-patterns.d.ts.map +1 -0
- package/dist/scanner/test-file-patterns.js +16 -0
- package/dist/scanner/test-file-patterns.js.map +1 -0
- package/dist/scanner/types.d.ts +313 -0
- package/dist/scanner/types.d.ts.map +1 -0
- package/dist/scanner/types.js +2 -0
- package/dist/scanner/types.js.map +1 -0
- package/dist/scanner.d.ts +6 -0
- package/dist/scanner.d.ts.map +1 -0
- package/dist/scanner.js +6 -0
- package/dist/scanner.js.map +1 -0
- package/dist/schema/code-generator.d.ts +53 -0
- package/dist/schema/code-generator.d.ts.map +1 -0
- package/dist/schema/ddl/base-strategy.d.ts +80 -0
- package/dist/schema/ddl/base-strategy.d.ts.map +1 -0
- package/dist/schema/ddl/base-strategy.js +240 -0
- package/dist/schema/ddl/base-strategy.js.map +1 -0
- package/dist/schema/ddl/duckdb-strategy.d.ts +33 -0
- package/dist/schema/ddl/duckdb-strategy.d.ts.map +1 -0
- package/dist/schema/ddl/duckdb-strategy.js +74 -0
- package/dist/schema/ddl/duckdb-strategy.js.map +1 -0
- package/dist/schema/ddl/index.d.ts +53 -0
- package/dist/schema/ddl/index.d.ts.map +1 -0
- package/dist/schema/ddl/index.js +80 -0
- package/dist/schema/ddl/index.js.map +1 -0
- package/dist/schema/ddl/json-duckdb-strategy.d.ts +8 -0
- package/dist/schema/ddl/json-duckdb-strategy.d.ts.map +1 -0
- package/dist/schema/ddl/json-duckdb-strategy.js +14 -0
- package/dist/schema/ddl/json-duckdb-strategy.js.map +1 -0
- package/dist/schema/ddl/postgres-strategy.d.ts +29 -0
- package/dist/schema/ddl/postgres-strategy.d.ts.map +1 -0
- package/dist/schema/ddl/postgres-strategy.js +102 -0
- package/dist/schema/ddl/postgres-strategy.js.map +1 -0
- package/dist/schema/ddl/sqlite-strategy.d.ts +38 -0
- package/dist/schema/ddl/sqlite-strategy.d.ts.map +1 -0
- package/dist/schema/ddl/sqlite-strategy.js +74 -0
- package/dist/schema/ddl/sqlite-strategy.js.map +1 -0
- package/dist/schema/ddl/types.d.ts +114 -0
- package/dist/schema/ddl/types.d.ts.map +1 -0
- package/dist/schema/generator.d.ts +176 -0
- package/dist/schema/generator.d.ts.map +1 -0
- package/dist/schema/generator.js +1076 -0
- package/dist/schema/generator.js.map +1 -0
- package/dist/schema/index-utils.d.ts +19 -0
- package/dist/schema/index-utils.d.ts.map +1 -0
- package/dist/schema/index-utils.js +32 -0
- package/dist/schema/index-utils.js.map +1 -0
- package/dist/schema/index.d.ts +13 -0
- package/dist/schema/index.d.ts.map +1 -0
- package/dist/schema/override-system.d.ts +43 -0
- package/dist/schema/override-system.d.ts.map +1 -0
- package/dist/schema/schema-aggregator.d.ts +112 -0
- package/dist/schema/schema-aggregator.d.ts.map +1 -0
- package/dist/schema/schema-manager.d.ts +95 -0
- package/dist/schema/schema-manager.d.ts.map +1 -0
- package/dist/schema/schema-manager.js +283 -0
- package/dist/schema/schema-manager.js.map +1 -0
- package/dist/schema/sql-identifiers.d.ts +107 -0
- package/dist/schema/sql-identifiers.d.ts.map +1 -0
- package/dist/schema/sql-identifiers.js +190 -0
- package/dist/schema/sql-identifiers.js.map +1 -0
- package/dist/schema/table-verifier.d.ts +10 -0
- package/dist/schema/table-verifier.d.ts.map +1 -0
- package/dist/schema/table-verifier.js +37 -0
- package/dist/schema/table-verifier.js.map +1 -0
- package/dist/schema/types.d.ts +241 -0
- package/dist/schema/types.d.ts.map +1 -0
- package/dist/schema/utils.d.ts +32 -0
- package/dist/schema/utils.d.ts.map +1 -0
- package/dist/schema/utils.js +134 -0
- package/dist/schema/utils.js.map +1 -0
- package/dist/scripts/create-wrappers.js +89 -0
- package/dist/scripts/generate-manifest.js +155 -0
- package/dist/scripts/generate-test-manifest.js +77 -0
- package/dist/scripts/migrate-datetime-to-timestamp.ts +310 -0
- package/dist/scripts/prepack.js +49 -0
- package/dist/signals/bus.d.ts +64 -0
- package/dist/signals/bus.d.ts.map +1 -0
- package/dist/signals/bus.js +102 -0
- package/dist/signals/bus.js.map +1 -0
- package/dist/signals/index.d.ts +11 -0
- package/dist/signals/index.d.ts.map +1 -0
- package/dist/signals/sanitizer.d.ts +54 -0
- package/dist/signals/sanitizer.d.ts.map +1 -0
- package/dist/signals/sanitizer.js +111 -0
- package/dist/signals/sanitizer.js.map +1 -0
- package/dist/smrt-knowledge.json +335 -0
- package/dist/system/compatibility.d.ts +8 -0
- package/dist/system/compatibility.d.ts.map +1 -0
- package/dist/system/compatibility.js +409 -0
- package/dist/system/compatibility.js.map +1 -0
- package/dist/system/index.d.ts +9 -0
- package/dist/system/index.d.ts.map +1 -0
- package/dist/system/schema.d.ts +69 -0
- package/dist/system/schema.d.ts.map +1 -0
- package/dist/system/schema.js +271 -0
- package/dist/system/schema.js.map +1 -0
- package/dist/system/types.d.ts +135 -0
- package/dist/system/types.d.ts.map +1 -0
- package/dist/system-fields.d.ts +44 -0
- package/dist/system-fields.d.ts.map +1 -0
- package/dist/system-fields.js +55 -0
- package/dist/system-fields.js.map +1 -0
- package/dist/table-cache.d.ts +28 -0
- package/dist/table-cache.d.ts.map +1 -0
- package/dist/table-cache.js +21 -0
- package/dist/table-cache.js.map +1 -0
- package/dist/test-utils.d.ts +140 -0
- package/dist/test-utils.d.ts.map +1 -0
- package/dist/testing/database.d.ts +73 -0
- package/dist/testing/database.d.ts.map +1 -0
- package/dist/testing/database.js +204 -0
- package/dist/testing/database.js.map +1 -0
- package/dist/testing/index.d.ts +21 -0
- package/dist/testing/index.d.ts.map +1 -0
- package/dist/testing.d.ts +6 -0
- package/dist/testing.d.ts.map +1 -0
- package/dist/testing.js +5 -0
- package/dist/testing.js.map +1 -0
- package/dist/tools/index.d.ts +8 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/tool-executor.d.ts +101 -0
- package/dist/tools/tool-executor.d.ts.map +1 -0
- package/dist/tools/tool-executor.js +142 -0
- package/dist/tools/tool-executor.js.map +1 -0
- package/dist/tools/tool-generator.d.ts +54 -0
- package/dist/tools/tool-generator.d.ts.map +1 -0
- package/dist/tools/tool-generator.js +121 -0
- package/dist/tools/tool-generator.js.map +1 -0
- package/dist/utils/chunk.d.ts +32 -0
- package/dist/utils/chunk.d.ts.map +1 -0
- package/dist/utils/chunk.js +14 -0
- package/dist/utils/chunk.js.map +1 -0
- package/dist/utils/import-workspace-module.d.ts +8 -0
- package/dist/utils/import-workspace-module.d.ts.map +1 -0
- package/dist/utils/import-workspace-module.js +81 -0
- package/dist/utils/import-workspace-module.js.map +1 -0
- package/dist/utils/json.d.ts +102 -0
- package/dist/utils/json.d.ts.map +1 -0
- package/dist/utils/json.js +43 -0
- package/dist/utils/json.js.map +1 -0
- package/dist/utils/lru-cache.d.ts +69 -0
- package/dist/utils/lru-cache.d.ts.map +1 -0
- package/dist/utils/lru-cache.js +100 -0
- package/dist/utils/lru-cache.js.map +1 -0
- package/dist/utils/naming.d.ts +16 -0
- package/dist/utils/naming.d.ts.map +1 -0
- package/dist/utils/naming.js +23 -0
- package/dist/utils/naming.js.map +1 -0
- package/dist/utils/qualified-names.d.ts +122 -0
- package/dist/utils/qualified-names.d.ts.map +1 -0
- package/dist/utils/qualified-names.js +82 -0
- package/dist/utils/qualified-names.js.map +1 -0
- package/dist/utils/scanner-module.d.ts +37 -0
- package/dist/utils/scanner-module.d.ts.map +1 -0
- package/dist/utils.d.ts +177 -0
- package/dist/utils.d.ts.map +1 -0
- package/dist/utils.js +185 -0
- package/dist/utils.js.map +1 -0
- package/dist/vite-plugin/import-build-aware.d.ts +68 -0
- package/dist/vite-plugin/import-build-aware.d.ts.map +1 -0
- package/dist/vite-plugin/import-build-aware.js +72 -0
- package/dist/vite-plugin/import-build-aware.js.map +1 -0
- package/dist/vite-plugin/index.d.ts +59 -0
- package/dist/vite-plugin/index.d.ts.map +1 -0
- package/dist/vite-plugin/index.js +1400 -0
- package/dist/vite-plugin/index.js.map +1 -0
- package/dist/vite-plugin/sveltekit-generator.d.ts +66 -0
- package/dist/vite-plugin/sveltekit-generator.d.ts.map +1 -0
- package/dist/vite-plugin/sveltekit-generator.js +1375 -0
- package/dist/vite-plugin/sveltekit-generator.js.map +1 -0
- package/dist/vite-plugin/templates/default-ui.ts +432 -0
- package/dist/vite-plugin/templates/default.html +206 -0
- package/dist/vite-plugin.d.ts +8 -0
- package/dist/vite-plugin.d.ts.map +1 -0
- package/dist/vite-plugin.js +11 -0
- package/dist/vite-plugin.js.map +1 -0
- package/package.json +208 -0
|
@@ -0,0 +1,583 @@
|
|
|
1
|
+
import { createLogger } from "@happyvertical/logger";
|
|
2
|
+
import { getDatabase } from "@happyvertical/sql";
|
|
3
|
+
import { ensureDispatchSystemTableCompatibility, ensureDispatchSubscriptionsSystemTableCompatibility } from "../system/compatibility.js";
|
|
4
|
+
import { CREATE_SMRT_DISPATCH_TABLE, CREATE_SMRT_DISPATCH_SUBSCRIPTIONS_TABLE } from "../system/schema.js";
|
|
5
|
+
import { DispatchCollection } from "./collections/Dispatches.js";
|
|
6
|
+
import { DispatchSubscriptionCollection } from "./collections/DispatchSubscriptions.js";
|
|
7
|
+
import { Dispatch } from "./models/Dispatch.js";
|
|
8
|
+
import { DispatchSubscription } from "./models/DispatchSubscription.js";
|
|
9
|
+
import { resolveDispatchTenantId, resolveDispatchTenantScope } from "./tenant-resolver.js";
|
|
10
|
+
const logger = createLogger({ level: "info" });
|
|
11
|
+
class DispatchBus {
|
|
12
|
+
db;
|
|
13
|
+
handlers = /* @__PURE__ */ new Map();
|
|
14
|
+
initialized = false;
|
|
15
|
+
/**
|
|
16
|
+
* Reserved subscriber name used internally for in-memory `on()` handlers.
|
|
17
|
+
* Callers may not register persistent subscriptions under this name, nor
|
|
18
|
+
* assert it as an emit `source`, so they cannot impersonate the in-memory
|
|
19
|
+
* pseudo-subscriber (S5 #1398).
|
|
20
|
+
*/
|
|
21
|
+
static RESERVED_SUBSCRIBER = "_in_memory_";
|
|
22
|
+
/** Maximum stored length of a caller-asserted dispatch source label. */
|
|
23
|
+
static MAX_SOURCE_LENGTH = 256;
|
|
24
|
+
/**
|
|
25
|
+
* Normalize a caller-asserted `source` into untrusted metadata.
|
|
26
|
+
*
|
|
27
|
+
* `source` is a declared label, not an authenticated identity, so it must not
|
|
28
|
+
* be trusted for authorization. This caps its length and **rejects** the
|
|
29
|
+
* reserved internal sentinel (`_in_memory_`) by throwing — the contract is
|
|
30
|
+
* that the sentinel is not an accepted source, so a caller cannot quietly
|
|
31
|
+
* impersonate the in-memory subscriber pseudo-source (S5 #1398). Empty/missing
|
|
32
|
+
* values default to `'unknown'` (unchanged behavior).
|
|
33
|
+
*
|
|
34
|
+
* @throws Error if `source` is the reserved in-memory sentinel.
|
|
35
|
+
*/
|
|
36
|
+
static sanitizeSource(source) {
|
|
37
|
+
const raw = (source ?? "").trim();
|
|
38
|
+
if (!raw) {
|
|
39
|
+
return "unknown";
|
|
40
|
+
}
|
|
41
|
+
if (raw === DispatchBus.RESERVED_SUBSCRIBER) {
|
|
42
|
+
throw new Error(
|
|
43
|
+
`DispatchBus.emit: "${DispatchBus.RESERVED_SUBSCRIBER}" is a reserved source and cannot be asserted`
|
|
44
|
+
);
|
|
45
|
+
}
|
|
46
|
+
return raw.slice(0, DispatchBus.MAX_SOURCE_LENGTH);
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Create a new DispatchBus (use createDispatchBus factory instead)
|
|
50
|
+
*/
|
|
51
|
+
constructor(db) {
|
|
52
|
+
this.db = db;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Initialize the dispatch tables
|
|
56
|
+
*/
|
|
57
|
+
async initialize() {
|
|
58
|
+
if (this.initialized) return;
|
|
59
|
+
const dispatchExists = await DispatchCollection.tableExists(this.db);
|
|
60
|
+
if (!dispatchExists) {
|
|
61
|
+
const statements = CREATE_SMRT_DISPATCH_TABLE.split(";").filter(
|
|
62
|
+
(s) => s.trim()
|
|
63
|
+
);
|
|
64
|
+
for (const stmt of statements) {
|
|
65
|
+
await this.db.query(stmt);
|
|
66
|
+
}
|
|
67
|
+
} else {
|
|
68
|
+
await ensureDispatchSystemTableCompatibility(this.db);
|
|
69
|
+
}
|
|
70
|
+
const subsExists = await DispatchSubscriptionCollection.tableExists(
|
|
71
|
+
this.db
|
|
72
|
+
);
|
|
73
|
+
if (!subsExists) {
|
|
74
|
+
const statements = CREATE_SMRT_DISPATCH_SUBSCRIPTIONS_TABLE.split(
|
|
75
|
+
";"
|
|
76
|
+
).filter((s) => s.trim());
|
|
77
|
+
for (const stmt of statements) {
|
|
78
|
+
await this.db.query(stmt);
|
|
79
|
+
}
|
|
80
|
+
} else {
|
|
81
|
+
await ensureDispatchSubscriptionsSystemTableCompatibility(this.db);
|
|
82
|
+
}
|
|
83
|
+
this.initialized = true;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Emits a dispatch message.
|
|
87
|
+
*
|
|
88
|
+
* Two things happen atomically (from the caller's perspective):
|
|
89
|
+
* 1. One or more `Dispatch` rows are inserted with `status: 'pending'` into `_smrt_dispatch`.
|
|
90
|
+
* 2. All matching in-memory handlers (registered via `on()`) are invoked
|
|
91
|
+
* fire-and-forget (errors are caught and logged, not thrown back to the caller).
|
|
92
|
+
*
|
|
93
|
+
* Persistent subscriptions registered via `subscribe()` will see this dispatch
|
|
94
|
+
* the next time `process(subscriber, handler)` is called for their subscriber name.
|
|
95
|
+
*
|
|
96
|
+
* **Delivery modes** affect how dispatches are created:
|
|
97
|
+
* - `compete` subscribers share a single dispatch (`target_subscriber = NULL`).
|
|
98
|
+
* First subscriber to `process()` claims it (at-most-once delivery).
|
|
99
|
+
* - `fanout` subscribers each get their own dispatch copy (`target_subscriber` set
|
|
100
|
+
* to the subscriber name), so each processes independently.
|
|
101
|
+
*
|
|
102
|
+
* If no subscriptions exist, one dispatch is created for future processing.
|
|
103
|
+
*
|
|
104
|
+
* **Security (S5 #1398):**
|
|
105
|
+
* - `tenant_id` is derived **server-side** from the active tenant context and
|
|
106
|
+
* is never read from caller options, so it cannot be spoofed. Subscribers in
|
|
107
|
+
* a different tenant cannot see or claim this dispatch (see {@link process}).
|
|
108
|
+
* When there is no active tenant context (system/global), `tenant_id` is
|
|
109
|
+
* `NULL` and the dispatch is visible to all scopes — preserving pre-tenancy
|
|
110
|
+
* behavior.
|
|
111
|
+
* - `options.source` is **caller-asserted, untrusted metadata** — it is a
|
|
112
|
+
* declared label only and must not be relied on as an authenticated emitter
|
|
113
|
+
* identity. It is length-capped and the reserved internal sentinel
|
|
114
|
+
* (`_in_memory_`) is rejected so a caller cannot impersonate the in-memory
|
|
115
|
+
* subscriber pseudo-source.
|
|
116
|
+
*
|
|
117
|
+
* @param type - Signal type string, e.g. `'campaign.completed'` or `'invoice.paid'`
|
|
118
|
+
* @param payload - Any JSON-serializable data to attach to the dispatch
|
|
119
|
+
* @param options.source - Declared (untrusted) name of the emitting agent/component (default `'unknown'`)
|
|
120
|
+
* @param options.sourceId - Optional ID of the specific emitting entity
|
|
121
|
+
* @param options.metadata - Optional additional JSON metadata for the dispatch record
|
|
122
|
+
* @returns A persisted `Dispatch` instance (the compete dispatch, or the first fanout copy if fanout-only)
|
|
123
|
+
*
|
|
124
|
+
* @example
|
|
125
|
+
* ```typescript
|
|
126
|
+
* const dispatch = await bus.emit(
|
|
127
|
+
* 'campaign.completed',
|
|
128
|
+
* { campaignId: 'cmp-001', impressions: 10_000 },
|
|
129
|
+
* { source: 'suasor', sourceId: agentId },
|
|
130
|
+
* );
|
|
131
|
+
* console.log(dispatch.id); // UUID of the created dispatch record
|
|
132
|
+
* ```
|
|
133
|
+
*
|
|
134
|
+
* @see {@link on} for in-memory handlers
|
|
135
|
+
* @see {@link subscribe} for persistent subscriptions
|
|
136
|
+
* @see {@link process} to consume pending dispatches
|
|
137
|
+
*/
|
|
138
|
+
async emit(type, payload, options = {}) {
|
|
139
|
+
await this.initialize();
|
|
140
|
+
const tenantId = resolveDispatchTenantId() ?? null;
|
|
141
|
+
const source = DispatchBus.sanitizeSource(options.source);
|
|
142
|
+
const matchingSubs = await DispatchSubscriptionCollection.findBySignalType(
|
|
143
|
+
this.db,
|
|
144
|
+
type,
|
|
145
|
+
resolveDispatchTenantScope()
|
|
146
|
+
);
|
|
147
|
+
const fanoutSubs = matchingSubs.filter((s) => s.delivery === "fanout");
|
|
148
|
+
const competeSubs = matchingSubs.filter((s) => s.delivery !== "fanout");
|
|
149
|
+
const dispatch = new Dispatch({
|
|
150
|
+
type,
|
|
151
|
+
source,
|
|
152
|
+
source_id: options.sourceId || null,
|
|
153
|
+
payload: JSON.stringify(payload || {}),
|
|
154
|
+
metadata: JSON.stringify(options.metadata || {}),
|
|
155
|
+
status: "pending",
|
|
156
|
+
correlation_id: options.correlationId || null,
|
|
157
|
+
tenant_id: tenantId
|
|
158
|
+
});
|
|
159
|
+
let returnDispatch = dispatch;
|
|
160
|
+
for (const sub of fanoutSubs) {
|
|
161
|
+
const fanoutDispatch = new Dispatch({
|
|
162
|
+
type,
|
|
163
|
+
source,
|
|
164
|
+
source_id: options.sourceId || null,
|
|
165
|
+
payload: JSON.stringify(payload || {}),
|
|
166
|
+
metadata: JSON.stringify(options.metadata || {}),
|
|
167
|
+
status: "pending",
|
|
168
|
+
target_subscriber: sub.subscriber,
|
|
169
|
+
correlation_id: options.correlationId || null,
|
|
170
|
+
tenant_id: tenantId
|
|
171
|
+
});
|
|
172
|
+
await DispatchCollection.insert(this.db, fanoutDispatch);
|
|
173
|
+
if (competeSubs.length === 0 && matchingSubs.length > 0) {
|
|
174
|
+
returnDispatch = fanoutDispatch;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
if (competeSubs.length > 0 || matchingSubs.length === 0) {
|
|
178
|
+
await DispatchCollection.insert(this.db, dispatch);
|
|
179
|
+
}
|
|
180
|
+
this.notifyHandlers(type, payload, dispatch.getMetadata());
|
|
181
|
+
return returnDispatch;
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Registers an in-memory handler for a signal type pattern.
|
|
185
|
+
*
|
|
186
|
+
* In-memory handlers are called immediately (fire-and-forget) during `emit()`.
|
|
187
|
+
* They are stored only in memory and lost on process restart — use `subscribe()`
|
|
188
|
+
* for durable, restart-safe subscriptions.
|
|
189
|
+
*
|
|
190
|
+
* Pattern matching supports a single-segment wildcard `*`:
|
|
191
|
+
* - `'campaign.*'` matches `'campaign.completed'` and `'campaign.failed'`
|
|
192
|
+
* - `'campaign.*'` does **not** match `'campaign.phase.two'`
|
|
193
|
+
*
|
|
194
|
+
* Multiple handlers may be registered for the same pattern — all will be called.
|
|
195
|
+
*
|
|
196
|
+
* @param pattern - Signal type pattern, optionally with a trailing `.*` wildcard
|
|
197
|
+
* @param handler - Async or sync function to call with `(payload, metadata)`
|
|
198
|
+
*
|
|
199
|
+
* @example
|
|
200
|
+
* ```typescript
|
|
201
|
+
* bus.on('invoice.*', async (payload, metadata) => {
|
|
202
|
+
* console.log(`Invoice event from ${metadata.source}:`, payload);
|
|
203
|
+
* });
|
|
204
|
+
* ```
|
|
205
|
+
*
|
|
206
|
+
* @see {@link off} to remove a handler
|
|
207
|
+
* @see {@link subscribe} for persistent (restart-safe) subscriptions
|
|
208
|
+
*/
|
|
209
|
+
on(pattern, handler) {
|
|
210
|
+
const subscription = new DispatchSubscription({
|
|
211
|
+
signal_type: pattern,
|
|
212
|
+
subscriber: "_in_memory_",
|
|
213
|
+
handler: "callback"
|
|
214
|
+
});
|
|
215
|
+
const registered = { pattern, handler, subscription };
|
|
216
|
+
if (!this.handlers.has(pattern)) {
|
|
217
|
+
this.handlers.set(pattern, []);
|
|
218
|
+
}
|
|
219
|
+
this.handlers.get(pattern)?.push(registered);
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Remove an in-memory handler
|
|
223
|
+
*
|
|
224
|
+
* @param pattern - Signal type pattern
|
|
225
|
+
* @param handler - Handler function to remove
|
|
226
|
+
* @returns True if handler was found and removed
|
|
227
|
+
*/
|
|
228
|
+
off(pattern, handler) {
|
|
229
|
+
const handlers = this.handlers.get(pattern);
|
|
230
|
+
if (!handlers) return false;
|
|
231
|
+
const index = handlers.findIndex((h) => h.handler === handler);
|
|
232
|
+
if (index === -1) return false;
|
|
233
|
+
handlers.splice(index, 1);
|
|
234
|
+
if (handlers.length === 0) {
|
|
235
|
+
this.handlers.delete(pattern);
|
|
236
|
+
}
|
|
237
|
+
return true;
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
* Creates or updates a persistent subscription in the database.
|
|
241
|
+
*
|
|
242
|
+
* Persistent subscriptions survive process restarts. When `process(subscriber, handler)`
|
|
243
|
+
* is called later, all pending dispatches matching this subscriber's signal patterns
|
|
244
|
+
* will be delivered.
|
|
245
|
+
*
|
|
246
|
+
* Calling `subscribe()` with the same `signalType`/`subscriber` pair is idempotent
|
|
247
|
+
* (upsert) — it is safe to call on every agent startup.
|
|
248
|
+
*
|
|
249
|
+
* Set `delivery: 'fanout'` to give each subscriber their own dispatch copy.
|
|
250
|
+
* Default is `'compete'` (at-most-once, first subscriber to process claims it).
|
|
251
|
+
*
|
|
252
|
+
* @param options.signalType - Signal type pattern to subscribe to (wildcards supported)
|
|
253
|
+
* @param options.subscriber - Name that identifies this subscriber (e.g. agent class name)
|
|
254
|
+
* @param options.handler - Optional method name on the subscriber to call (default `'handleDispatch'`)
|
|
255
|
+
* @param options.delivery - `'compete'` (default) for at-most-once, `'fanout'` for per-subscriber copies
|
|
256
|
+
* @param options.enabled - If `false`, subscription is created but disabled (default `true`)
|
|
257
|
+
*
|
|
258
|
+
* @example
|
|
259
|
+
* ```typescript
|
|
260
|
+
* // Subscribe on agent startup (idempotent)
|
|
261
|
+
* await bus.subscribe({ signalType: 'campaign.*', subscriber: 'FiscusAgent' });
|
|
262
|
+
*
|
|
263
|
+
* // Fan-out: each subscriber gets their own dispatch copy
|
|
264
|
+
* await bus.subscribe({
|
|
265
|
+
* signalType: 'campaign.*',
|
|
266
|
+
* subscriber: 'AuditorAgent',
|
|
267
|
+
* delivery: 'fanout',
|
|
268
|
+
* });
|
|
269
|
+
*
|
|
270
|
+
* // Later, process matching pending dispatches
|
|
271
|
+
* await bus.process('FiscusAgent', async (payload, metadata) => { ... });
|
|
272
|
+
* ```
|
|
273
|
+
*
|
|
274
|
+
* @see {@link process} to consume pending dispatches for this subscriber
|
|
275
|
+
* @see {@link unsubscribe} to remove the subscription
|
|
276
|
+
* @see {@link on} for non-persistent in-memory handling
|
|
277
|
+
*/
|
|
278
|
+
async subscribe(options) {
|
|
279
|
+
await this.initialize();
|
|
280
|
+
const subscriber = (options.subscriber ?? "").trim();
|
|
281
|
+
if (!subscriber) {
|
|
282
|
+
throw new Error("DispatchBus.subscribe requires a non-empty subscriber");
|
|
283
|
+
}
|
|
284
|
+
if (subscriber === DispatchBus.RESERVED_SUBSCRIBER) {
|
|
285
|
+
throw new Error(
|
|
286
|
+
`DispatchBus.subscribe: "${DispatchBus.RESERVED_SUBSCRIBER}" is a reserved subscriber name`
|
|
287
|
+
);
|
|
288
|
+
}
|
|
289
|
+
const signalType = (options.signalType ?? "").trim();
|
|
290
|
+
if (!signalType) {
|
|
291
|
+
throw new Error("DispatchBus.subscribe requires a non-empty signalType");
|
|
292
|
+
}
|
|
293
|
+
const subscription = new DispatchSubscription({
|
|
294
|
+
signal_type: signalType,
|
|
295
|
+
subscriber,
|
|
296
|
+
handler: options.handler || "handleDispatch",
|
|
297
|
+
delivery: options.delivery || "compete",
|
|
298
|
+
enabled: options.enabled !== false ? 1 : 0,
|
|
299
|
+
// Server-derived tenant scope (never from caller options). NULL when there
|
|
300
|
+
// is no active tenant context (global subscription).
|
|
301
|
+
tenant_id: resolveDispatchTenantId() ?? null
|
|
302
|
+
});
|
|
303
|
+
await DispatchSubscriptionCollection.upsert(this.db, subscription);
|
|
304
|
+
}
|
|
305
|
+
/**
|
|
306
|
+
* Remove a persistent subscription
|
|
307
|
+
*
|
|
308
|
+
* @param signalType - Signal type pattern
|
|
309
|
+
* @param subscriber - Subscriber name
|
|
310
|
+
*/
|
|
311
|
+
async unsubscribe(signalType, subscriber) {
|
|
312
|
+
await this.initialize();
|
|
313
|
+
await DispatchSubscriptionCollection.deleteByKey(
|
|
314
|
+
this.db,
|
|
315
|
+
signalType,
|
|
316
|
+
subscriber,
|
|
317
|
+
resolveDispatchTenantScope()
|
|
318
|
+
);
|
|
319
|
+
}
|
|
320
|
+
/**
|
|
321
|
+
* Processes pending dispatches for a named subscriber.
|
|
322
|
+
*
|
|
323
|
+
* For each matching `pending` dispatch:
|
|
324
|
+
* 1. Sets status to `processing`
|
|
325
|
+
* 2. Calls `handler(payload, metadata)`
|
|
326
|
+
* 3. On success: sets status to `completed`
|
|
327
|
+
* 4. On error: sets status to `failed` (stores the error message)
|
|
328
|
+
*
|
|
329
|
+
* Uses a wildcard-aware query strategy: subscriptions with `*` patterns fetch
|
|
330
|
+
* all pending dispatches and filter in memory; exact-match subscriptions use
|
|
331
|
+
* a direct SQL `IN` query for efficiency.
|
|
332
|
+
*
|
|
333
|
+
* @param subscriber - The subscriber name (must match a `subscribe()` call)
|
|
334
|
+
* @param handler - Function to call for each pending dispatch
|
|
335
|
+
* @param options.limit - Maximum dispatches to process in one call (default 100)
|
|
336
|
+
* @param options.signalTypes - Optional filter to process only specific signal types
|
|
337
|
+
* @returns Number of dispatches successfully processed (excludes failed)
|
|
338
|
+
*
|
|
339
|
+
* @example
|
|
340
|
+
* ```typescript
|
|
341
|
+
* const count = await bus.process('FiscusAgent', async (payload, metadata) => {
|
|
342
|
+
* if (metadata.signalType === 'campaign.completed') {
|
|
343
|
+
* await generateInvoice(payload.campaignId);
|
|
344
|
+
* }
|
|
345
|
+
* });
|
|
346
|
+
* console.log(`Processed ${count} dispatches`);
|
|
347
|
+
* ```
|
|
348
|
+
*
|
|
349
|
+
* @see {@link subscribe} to register the persistent subscription first
|
|
350
|
+
* @see {@link retry} to reset failed dispatches back to pending
|
|
351
|
+
*/
|
|
352
|
+
async process(subscriber, handler, options = {}) {
|
|
353
|
+
await this.initialize();
|
|
354
|
+
const tenantScope = resolveDispatchTenantScope();
|
|
355
|
+
const subscriptions = await DispatchSubscriptionCollection.findBySubscriber(
|
|
356
|
+
this.db,
|
|
357
|
+
subscriber,
|
|
358
|
+
true,
|
|
359
|
+
tenantScope
|
|
360
|
+
);
|
|
361
|
+
if (subscriptions.length === 0) {
|
|
362
|
+
return 0;
|
|
363
|
+
}
|
|
364
|
+
const signalTypes = subscriptions.map((s) => s.signalType);
|
|
365
|
+
const hasWildcards = signalTypes.some((t) => t.includes("*"));
|
|
366
|
+
const isVisibleToSubscriber = (dispatch) => {
|
|
367
|
+
if (tenantScope.enforced) {
|
|
368
|
+
if (tenantScope.tenantId === null) {
|
|
369
|
+
if (dispatch.tenantId !== null) {
|
|
370
|
+
return false;
|
|
371
|
+
}
|
|
372
|
+
} else if (dispatch.tenantId !== null && dispatch.tenantId !== tenantScope.tenantId) {
|
|
373
|
+
return false;
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
if (dispatch.targetSubscriber === subscriber) {
|
|
377
|
+
return true;
|
|
378
|
+
}
|
|
379
|
+
if (dispatch.targetSubscriber !== null) {
|
|
380
|
+
return false;
|
|
381
|
+
}
|
|
382
|
+
const matchingSub = subscriptions.find(
|
|
383
|
+
(sub) => sub.matches(dispatch.type)
|
|
384
|
+
);
|
|
385
|
+
return matchingSub?.delivery === "compete";
|
|
386
|
+
};
|
|
387
|
+
let pendingDispatches;
|
|
388
|
+
if (hasWildcards) {
|
|
389
|
+
const allPending = await DispatchCollection.list(this.db, {
|
|
390
|
+
status: "pending",
|
|
391
|
+
limit: options.limit || 100,
|
|
392
|
+
tenantScope
|
|
393
|
+
});
|
|
394
|
+
pendingDispatches = allPending.filter(
|
|
395
|
+
(dispatch) => subscriptions.some((sub) => sub.matches(dispatch.type)) && isVisibleToSubscriber(dispatch)
|
|
396
|
+
);
|
|
397
|
+
} else {
|
|
398
|
+
const raw = await DispatchCollection.findPending(
|
|
399
|
+
this.db,
|
|
400
|
+
signalTypes,
|
|
401
|
+
options.limit || 100,
|
|
402
|
+
subscriber,
|
|
403
|
+
tenantScope
|
|
404
|
+
);
|
|
405
|
+
pendingDispatches = raw.filter(isVisibleToSubscriber);
|
|
406
|
+
}
|
|
407
|
+
if (options.signalTypes && options.signalTypes.length > 0) {
|
|
408
|
+
pendingDispatches = pendingDispatches.filter(
|
|
409
|
+
(d) => options.signalTypes?.includes(d.type)
|
|
410
|
+
);
|
|
411
|
+
}
|
|
412
|
+
let processed = 0;
|
|
413
|
+
for (const dispatch of pendingDispatches) {
|
|
414
|
+
const claimed = await DispatchCollection.claim(
|
|
415
|
+
this.db,
|
|
416
|
+
dispatch,
|
|
417
|
+
tenantScope
|
|
418
|
+
);
|
|
419
|
+
if (!claimed) {
|
|
420
|
+
continue;
|
|
421
|
+
}
|
|
422
|
+
try {
|
|
423
|
+
await handler(dispatch.payload, dispatch.getMetadata());
|
|
424
|
+
dispatch.markCompleted(subscriber);
|
|
425
|
+
await DispatchCollection.update(this.db, dispatch);
|
|
426
|
+
processed++;
|
|
427
|
+
} catch (error) {
|
|
428
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
429
|
+
dispatch.markFailed(errorMessage);
|
|
430
|
+
await DispatchCollection.update(this.db, dispatch);
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
return processed;
|
|
434
|
+
}
|
|
435
|
+
/**
|
|
436
|
+
* Retry failed dispatches
|
|
437
|
+
*
|
|
438
|
+
* Resets failed dispatches to pending status so they can be processed again.
|
|
439
|
+
*
|
|
440
|
+
* @param options - Retry options
|
|
441
|
+
* @returns Number of dispatches reset
|
|
442
|
+
*/
|
|
443
|
+
async retry(options = {}) {
|
|
444
|
+
await this.initialize();
|
|
445
|
+
const retryable = await DispatchCollection.findRetryable(
|
|
446
|
+
this.db,
|
|
447
|
+
options,
|
|
448
|
+
resolveDispatchTenantScope()
|
|
449
|
+
);
|
|
450
|
+
for (const dispatch of retryable) {
|
|
451
|
+
dispatch.resetForRetry();
|
|
452
|
+
await DispatchCollection.update(this.db, dispatch);
|
|
453
|
+
}
|
|
454
|
+
return retryable.length;
|
|
455
|
+
}
|
|
456
|
+
/**
|
|
457
|
+
* Clean up old dispatches
|
|
458
|
+
*
|
|
459
|
+
* @param options - Cleanup options
|
|
460
|
+
* @returns Number of dispatches deleted
|
|
461
|
+
*/
|
|
462
|
+
async cleanup(options = {}) {
|
|
463
|
+
await this.initialize();
|
|
464
|
+
return DispatchCollection.cleanup(
|
|
465
|
+
this.db,
|
|
466
|
+
options,
|
|
467
|
+
resolveDispatchTenantScope()
|
|
468
|
+
);
|
|
469
|
+
}
|
|
470
|
+
/**
|
|
471
|
+
* Lists dispatches, scoped to the active tenant context (S5 #1398).
|
|
472
|
+
*
|
|
473
|
+
* The tenant scope is derived **server-side** from the active context and
|
|
474
|
+
* **overrides any caller-supplied scope** — callers cannot select another
|
|
475
|
+
* tenant or widen visibility to all tenants. Scoping rules:
|
|
476
|
+
*
|
|
477
|
+
* - tenancy off → no tenant filter (pre-tenancy behavior).
|
|
478
|
+
* - active tenant T → that tenant's dispatches plus global (NULL) dispatches.
|
|
479
|
+
* - tenancy on but no active tenant → global (NULL) dispatches only
|
|
480
|
+
* (fail-closed; never all tenants).
|
|
481
|
+
*
|
|
482
|
+
* @param options - Non-tenant list filters (status/type/source/etc.). Any
|
|
483
|
+
* `tenantScope` field is ignored and replaced with the server-derived one.
|
|
484
|
+
*/
|
|
485
|
+
async list(options = {}) {
|
|
486
|
+
await this.initialize();
|
|
487
|
+
const { tenantScope: _ignoredCallerScope, ...safeOptions } = options;
|
|
488
|
+
return DispatchCollection.list(this.db, {
|
|
489
|
+
...safeOptions,
|
|
490
|
+
tenantScope: resolveDispatchTenantScope()
|
|
491
|
+
});
|
|
492
|
+
}
|
|
493
|
+
/**
|
|
494
|
+
* Gets a dispatch by ID, enforcing the active tenant scope (S5 #1398).
|
|
495
|
+
*
|
|
496
|
+
* Applies the same server-derived tenant predicate as {@link list}: a
|
|
497
|
+
* subscriber in tenant A cannot fetch tenant B's dispatch by id, and when
|
|
498
|
+
* tenancy is on with no active tenant only global (NULL) dispatches are
|
|
499
|
+
* returned. Returns `null` when the dispatch exists but is out of scope.
|
|
500
|
+
*/
|
|
501
|
+
async get(id) {
|
|
502
|
+
await this.initialize();
|
|
503
|
+
return DispatchCollection.getScoped(
|
|
504
|
+
this.db,
|
|
505
|
+
id,
|
|
506
|
+
resolveDispatchTenantScope()
|
|
507
|
+
);
|
|
508
|
+
}
|
|
509
|
+
/**
|
|
510
|
+
* List all subscriptions
|
|
511
|
+
*/
|
|
512
|
+
async listSubscriptions(subscriber) {
|
|
513
|
+
await this.initialize();
|
|
514
|
+
const tenantScope = resolveDispatchTenantScope();
|
|
515
|
+
if (subscriber) {
|
|
516
|
+
return DispatchSubscriptionCollection.findBySubscriber(
|
|
517
|
+
this.db,
|
|
518
|
+
subscriber,
|
|
519
|
+
false,
|
|
520
|
+
tenantScope
|
|
521
|
+
);
|
|
522
|
+
}
|
|
523
|
+
return DispatchSubscriptionCollection.list(this.db, false, tenantScope);
|
|
524
|
+
}
|
|
525
|
+
/**
|
|
526
|
+
* Notify in-memory handlers (fire-and-forget)
|
|
527
|
+
*/
|
|
528
|
+
notifyHandlers(type, payload, metadata) {
|
|
529
|
+
const matchingHandlers = [];
|
|
530
|
+
for (const [, handlers] of this.handlers) {
|
|
531
|
+
for (const registered of handlers) {
|
|
532
|
+
if (registered.subscription.matches(type)) {
|
|
533
|
+
matchingHandlers.push(registered);
|
|
534
|
+
}
|
|
535
|
+
}
|
|
536
|
+
}
|
|
537
|
+
for (const registered of matchingHandlers) {
|
|
538
|
+
try {
|
|
539
|
+
const result = registered.handler(payload, metadata);
|
|
540
|
+
if (result && typeof result.catch === "function") {
|
|
541
|
+
void result.catch((error) => {
|
|
542
|
+
logger.error(
|
|
543
|
+
`DispatchBus: Handler for "${registered.pattern}" failed`,
|
|
544
|
+
{ error }
|
|
545
|
+
);
|
|
546
|
+
});
|
|
547
|
+
}
|
|
548
|
+
} catch (error) {
|
|
549
|
+
logger.error(
|
|
550
|
+
`DispatchBus: Handler for "${registered.pattern}" failed`,
|
|
551
|
+
{ error }
|
|
552
|
+
);
|
|
553
|
+
}
|
|
554
|
+
}
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
async function createDispatchBus(options = {}) {
|
|
558
|
+
const dbConfig = options.db || options.persistence;
|
|
559
|
+
if (!dbConfig) {
|
|
560
|
+
throw new Error("DispatchBus requires a database configuration");
|
|
561
|
+
}
|
|
562
|
+
let db;
|
|
563
|
+
if (typeof dbConfig === "string") {
|
|
564
|
+
db = await getDatabase(dbConfig);
|
|
565
|
+
} else if ("query" in dbConfig) {
|
|
566
|
+
db = dbConfig;
|
|
567
|
+
} else if ("type" in dbConfig && "url" in dbConfig) {
|
|
568
|
+
db = await getDatabase({
|
|
569
|
+
type: dbConfig.type,
|
|
570
|
+
url: dbConfig.url
|
|
571
|
+
});
|
|
572
|
+
} else {
|
|
573
|
+
throw new Error("Invalid database configuration for DispatchBus");
|
|
574
|
+
}
|
|
575
|
+
const bus = new DispatchBus(db);
|
|
576
|
+
await bus.initialize();
|
|
577
|
+
return bus;
|
|
578
|
+
}
|
|
579
|
+
export {
|
|
580
|
+
DispatchBus,
|
|
581
|
+
createDispatchBus
|
|
582
|
+
};
|
|
583
|
+
//# sourceMappingURL=bus.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bus.js","sources":["../../src/dispatch/bus.ts"],"sourcesContent":["/**\n * DispatchBus - Central hub for inter-agent communication\n *\n * The DispatchBus provides both in-memory handlers and persistent subscriptions\n * for asynchronous agent-to-agent messaging.\n *\n * @example\n * ```typescript\n * const bus = await createDispatchBus({ db: { type: 'sqlite', url: 'app.db' } });\n *\n * // In-memory handler (immediate)\n * bus.on('campaign.completed', async (payload, metadata) => {\n * console.log(`Campaign ${payload.campaignId} completed`);\n * });\n *\n * // Persistent subscription (processed later)\n * await bus.subscribe({ signalType: 'campaign.*', subscriber: 'fiscus' });\n *\n * // Emit a dispatch\n * await bus.emit('campaign.completed', { campaignId: '123' }, { source: 'suasor' });\n *\n * // Process pending dispatches\n * await bus.process('fiscus');\n * ```\n */\n\nimport { createLogger } from '@happyvertical/logger';\nimport type { DatabaseInterface } from '@happyvertical/sql';\nimport { getDatabase } from '@happyvertical/sql';\nimport {\n ensureDispatchSubscriptionsSystemTableCompatibility,\n ensureDispatchSystemTableCompatibility,\n} from '../system/compatibility.js';\nimport {\n CREATE_SMRT_DISPATCH_SUBSCRIPTIONS_TABLE,\n CREATE_SMRT_DISPATCH_TABLE,\n} from '../system/schema.js';\nimport { DispatchCollection } from './collections/Dispatches.js';\nimport { DispatchSubscriptionCollection } from './collections/DispatchSubscriptions.js';\nimport { Dispatch } from './models/Dispatch.js';\nimport { DispatchSubscription } from './models/DispatchSubscription.js';\nimport {\n resolveDispatchTenantId,\n resolveDispatchTenantScope,\n} from './tenant-resolver.js';\nimport type {\n DispatchBusOptions,\n DispatchCleanupOptions,\n DispatchCleanupResult,\n DispatchEmitOptions,\n DispatchHandler,\n DispatchListOptions,\n DispatchMetadata,\n DispatchProcessOptions,\n DispatchRetryOptions,\n DispatchSubscribeOptions,\n} from './types.js';\n\nconst logger = createLogger({ level: 'info' });\n\n/**\n * Registered in-memory handler\n */\ninterface RegisteredHandler {\n pattern: string;\n handler: DispatchHandler;\n subscription: DispatchSubscription;\n}\n\n/**\n * Central hub for inter-agent messaging with both in-memory and persistent delivery.\n *\n * The `DispatchBus` combines two complementary delivery models:\n *\n * - **In-memory handlers** (`on(pattern, handler)`) — called synchronously (fire-and-forget)\n * when a dispatch is emitted. Fast, but lost on process restart.\n * - **Persistent subscriptions** (`subscribe({ signalType, subscriber })`) — stored in\n * `_smrt_dispatch_subscriptions` and processed later by calling `process(subscriber, handler)`.\n * Survive restarts; suitable for background workers and scheduled agents.\n *\n * Signal types support single-segment wildcards: `'campaign.*'` matches `'campaign.completed'`\n * and `'campaign.failed'`, but not `'campaign.phase.two'`.\n *\n * Dispatch lifecycle: `pending → processing → completed` (or `failed` on handler error).\n * Use `retry()` to reset failed dispatches back to `pending`.\n *\n * Create instances via `createDispatchBus()` — do not use `new DispatchBus()` directly\n * in application code.\n *\n * @example\n * ```typescript\n * const bus = await createDispatchBus({ db: myDb });\n *\n * // In-memory: immediate, fire-and-forget\n * bus.on('invoice.paid', async (payload) => {\n * console.log('Invoice paid:', payload.invoiceId);\n * });\n *\n * // Persistent: processed by 'notifications' agent on next run\n * await bus.subscribe({ signalType: 'invoice.*', subscriber: 'notifications' });\n *\n * // Emit (notifies in-memory handlers immediately, stores persistent dispatch)\n * await bus.emit('invoice.paid', { invoiceId: 'inv-001' }, { source: 'billing' });\n *\n * // Later, in the notifications agent:\n * await bus.process('notifications', async (payload) => {\n * await sendEmail(payload.invoiceId);\n * });\n * ```\n */\nexport class DispatchBus {\n private db: DatabaseInterface;\n private handlers: Map<string, RegisteredHandler[]> = new Map();\n private initialized: boolean = false;\n\n /**\n * Reserved subscriber name used internally for in-memory `on()` handlers.\n * Callers may not register persistent subscriptions under this name, nor\n * assert it as an emit `source`, so they cannot impersonate the in-memory\n * pseudo-subscriber (S5 #1398).\n */\n private static readonly RESERVED_SUBSCRIBER = '_in_memory_';\n\n /** Maximum stored length of a caller-asserted dispatch source label. */\n private static readonly MAX_SOURCE_LENGTH = 256;\n\n /**\n * Normalize a caller-asserted `source` into untrusted metadata.\n *\n * `source` is a declared label, not an authenticated identity, so it must not\n * be trusted for authorization. This caps its length and **rejects** the\n * reserved internal sentinel (`_in_memory_`) by throwing — the contract is\n * that the sentinel is not an accepted source, so a caller cannot quietly\n * impersonate the in-memory subscriber pseudo-source (S5 #1398). Empty/missing\n * values default to `'unknown'` (unchanged behavior).\n *\n * @throws Error if `source` is the reserved in-memory sentinel.\n */\n private static sanitizeSource(source: string | undefined): string {\n const raw = (source ?? '').trim();\n if (!raw) {\n return 'unknown';\n }\n if (raw === DispatchBus.RESERVED_SUBSCRIBER) {\n throw new Error(\n `DispatchBus.emit: \"${DispatchBus.RESERVED_SUBSCRIBER}\" is a reserved source and cannot be asserted`,\n );\n }\n return raw.slice(0, DispatchBus.MAX_SOURCE_LENGTH);\n }\n\n /**\n * Create a new DispatchBus (use createDispatchBus factory instead)\n */\n constructor(db: DatabaseInterface) {\n this.db = db;\n }\n\n /**\n * Initialize the dispatch tables\n */\n async initialize(): Promise<void> {\n if (this.initialized) return;\n\n // Create dispatch tables if they don't exist\n const dispatchExists = await DispatchCollection.tableExists(this.db);\n if (!dispatchExists) {\n // Split the DDL into separate statements and execute each\n const statements = CREATE_SMRT_DISPATCH_TABLE.split(';').filter((s) =>\n s.trim(),\n );\n for (const stmt of statements) {\n await this.db.query(stmt);\n }\n } else {\n await ensureDispatchSystemTableCompatibility(this.db);\n }\n\n const subsExists = await DispatchSubscriptionCollection.tableExists(\n this.db,\n );\n if (!subsExists) {\n const statements = CREATE_SMRT_DISPATCH_SUBSCRIPTIONS_TABLE.split(\n ';',\n ).filter((s) => s.trim());\n for (const stmt of statements) {\n await this.db.query(stmt);\n }\n } else {\n await ensureDispatchSubscriptionsSystemTableCompatibility(this.db);\n }\n\n this.initialized = true;\n }\n\n /**\n * Emits a dispatch message.\n *\n * Two things happen atomically (from the caller's perspective):\n * 1. One or more `Dispatch` rows are inserted with `status: 'pending'` into `_smrt_dispatch`.\n * 2. All matching in-memory handlers (registered via `on()`) are invoked\n * fire-and-forget (errors are caught and logged, not thrown back to the caller).\n *\n * Persistent subscriptions registered via `subscribe()` will see this dispatch\n * the next time `process(subscriber, handler)` is called for their subscriber name.\n *\n * **Delivery modes** affect how dispatches are created:\n * - `compete` subscribers share a single dispatch (`target_subscriber = NULL`).\n * First subscriber to `process()` claims it (at-most-once delivery).\n * - `fanout` subscribers each get their own dispatch copy (`target_subscriber` set\n * to the subscriber name), so each processes independently.\n *\n * If no subscriptions exist, one dispatch is created for future processing.\n *\n * **Security (S5 #1398):**\n * - `tenant_id` is derived **server-side** from the active tenant context and\n * is never read from caller options, so it cannot be spoofed. Subscribers in\n * a different tenant cannot see or claim this dispatch (see {@link process}).\n * When there is no active tenant context (system/global), `tenant_id` is\n * `NULL` and the dispatch is visible to all scopes — preserving pre-tenancy\n * behavior.\n * - `options.source` is **caller-asserted, untrusted metadata** — it is a\n * declared label only and must not be relied on as an authenticated emitter\n * identity. It is length-capped and the reserved internal sentinel\n * (`_in_memory_`) is rejected so a caller cannot impersonate the in-memory\n * subscriber pseudo-source.\n *\n * @param type - Signal type string, e.g. `'campaign.completed'` or `'invoice.paid'`\n * @param payload - Any JSON-serializable data to attach to the dispatch\n * @param options.source - Declared (untrusted) name of the emitting agent/component (default `'unknown'`)\n * @param options.sourceId - Optional ID of the specific emitting entity\n * @param options.metadata - Optional additional JSON metadata for the dispatch record\n * @returns A persisted `Dispatch` instance (the compete dispatch, or the first fanout copy if fanout-only)\n *\n * @example\n * ```typescript\n * const dispatch = await bus.emit(\n * 'campaign.completed',\n * { campaignId: 'cmp-001', impressions: 10_000 },\n * { source: 'suasor', sourceId: agentId },\n * );\n * console.log(dispatch.id); // UUID of the created dispatch record\n * ```\n *\n * @see {@link on} for in-memory handlers\n * @see {@link subscribe} for persistent subscriptions\n * @see {@link process} to consume pending dispatches\n */\n async emit(\n type: string,\n payload: unknown,\n options: DispatchEmitOptions = {},\n ): Promise<Dispatch> {\n await this.initialize();\n\n // Derive the tenant scope server-side from the active context. This is the\n // trust anchor for cross-tenant isolation (S5 #1398) — it is never taken\n // from caller options, so it cannot be spoofed. `undefined`/`null` means\n // \"no tenant context\" → tenant_id stays NULL (global dispatch).\n const tenantId = resolveDispatchTenantId() ?? null;\n\n // Treat the caller-asserted source as untrusted metadata.\n const source = DispatchBus.sanitizeSource(options.source);\n\n // Query subscriptions to determine if fan-out copies are needed.\n // This runs on every emit() — uses SQL-level exact matching with\n // in-memory fallback only for wildcard subscriptions to minimize I/O.\n // Scoped to the active tenant so emit only fans out to this tenant's\n // (and, for global emits, global) subscriptions (S5 #1398).\n const matchingSubs = await DispatchSubscriptionCollection.findBySignalType(\n this.db,\n type,\n resolveDispatchTenantScope(),\n );\n\n // Separate into compete and fanout subscribers\n const fanoutSubs = matchingSubs.filter((s) => s.delivery === 'fanout');\n const competeSubs = matchingSubs.filter((s) => s.delivery !== 'fanout');\n\n // Create the base dispatch record\n const dispatch = new Dispatch({\n type,\n source,\n source_id: options.sourceId || null,\n payload: JSON.stringify(payload || {}),\n metadata: JSON.stringify(options.metadata || {}),\n status: 'pending',\n correlation_id: options.correlationId || null,\n tenant_id: tenantId,\n });\n\n // Track first persisted dispatch to return to caller\n let returnDispatch = dispatch;\n\n // For fanout subscribers: create a per-subscriber dispatch copy\n for (const sub of fanoutSubs) {\n const fanoutDispatch = new Dispatch({\n type,\n source,\n source_id: options.sourceId || null,\n payload: JSON.stringify(payload || {}),\n metadata: JSON.stringify(options.metadata || {}),\n status: 'pending',\n target_subscriber: sub.subscriber,\n correlation_id: options.correlationId || null,\n tenant_id: tenantId,\n });\n await DispatchCollection.insert(this.db, fanoutDispatch);\n // If no compete dispatch will be persisted, return the first fanout copy\n if (competeSubs.length === 0 && matchingSubs.length > 0) {\n returnDispatch = fanoutDispatch;\n }\n }\n\n // For compete subscribers (or if no subscriptions found): insert the original\n // dispatch with target_subscriber = NULL so any compete subscriber can claim it\n if (competeSubs.length > 0 || matchingSubs.length === 0) {\n await DispatchCollection.insert(this.db, dispatch);\n }\n\n // Notify in-memory handlers immediately (fire-and-forget)\n this.notifyHandlers(type, payload, dispatch.getMetadata());\n\n return returnDispatch;\n }\n\n /**\n * Registers an in-memory handler for a signal type pattern.\n *\n * In-memory handlers are called immediately (fire-and-forget) during `emit()`.\n * They are stored only in memory and lost on process restart — use `subscribe()`\n * for durable, restart-safe subscriptions.\n *\n * Pattern matching supports a single-segment wildcard `*`:\n * - `'campaign.*'` matches `'campaign.completed'` and `'campaign.failed'`\n * - `'campaign.*'` does **not** match `'campaign.phase.two'`\n *\n * Multiple handlers may be registered for the same pattern — all will be called.\n *\n * @param pattern - Signal type pattern, optionally with a trailing `.*` wildcard\n * @param handler - Async or sync function to call with `(payload, metadata)`\n *\n * @example\n * ```typescript\n * bus.on('invoice.*', async (payload, metadata) => {\n * console.log(`Invoice event from ${metadata.source}:`, payload);\n * });\n * ```\n *\n * @see {@link off} to remove a handler\n * @see {@link subscribe} for persistent (restart-safe) subscriptions\n */\n on(pattern: string, handler: DispatchHandler): void {\n const subscription = new DispatchSubscription({\n signal_type: pattern,\n subscriber: '_in_memory_',\n handler: 'callback',\n });\n\n const registered: RegisteredHandler = { pattern, handler, subscription };\n\n if (!this.handlers.has(pattern)) {\n this.handlers.set(pattern, []);\n }\n this.handlers.get(pattern)?.push(registered);\n }\n\n /**\n * Remove an in-memory handler\n *\n * @param pattern - Signal type pattern\n * @param handler - Handler function to remove\n * @returns True if handler was found and removed\n */\n off(pattern: string, handler: DispatchHandler): boolean {\n const handlers = this.handlers.get(pattern);\n if (!handlers) return false;\n\n const index = handlers.findIndex((h) => h.handler === handler);\n if (index === -1) return false;\n\n handlers.splice(index, 1);\n if (handlers.length === 0) {\n this.handlers.delete(pattern);\n }\n return true;\n }\n\n /**\n * Creates or updates a persistent subscription in the database.\n *\n * Persistent subscriptions survive process restarts. When `process(subscriber, handler)`\n * is called later, all pending dispatches matching this subscriber's signal patterns\n * will be delivered.\n *\n * Calling `subscribe()` with the same `signalType`/`subscriber` pair is idempotent\n * (upsert) — it is safe to call on every agent startup.\n *\n * Set `delivery: 'fanout'` to give each subscriber their own dispatch copy.\n * Default is `'compete'` (at-most-once, first subscriber to process claims it).\n *\n * @param options.signalType - Signal type pattern to subscribe to (wildcards supported)\n * @param options.subscriber - Name that identifies this subscriber (e.g. agent class name)\n * @param options.handler - Optional method name on the subscriber to call (default `'handleDispatch'`)\n * @param options.delivery - `'compete'` (default) for at-most-once, `'fanout'` for per-subscriber copies\n * @param options.enabled - If `false`, subscription is created but disabled (default `true`)\n *\n * @example\n * ```typescript\n * // Subscribe on agent startup (idempotent)\n * await bus.subscribe({ signalType: 'campaign.*', subscriber: 'FiscusAgent' });\n *\n * // Fan-out: each subscriber gets their own dispatch copy\n * await bus.subscribe({\n * signalType: 'campaign.*',\n * subscriber: 'AuditorAgent',\n * delivery: 'fanout',\n * });\n *\n * // Later, process matching pending dispatches\n * await bus.process('FiscusAgent', async (payload, metadata) => { ... });\n * ```\n *\n * @see {@link process} to consume pending dispatches for this subscriber\n * @see {@link unsubscribe} to remove the subscription\n * @see {@link on} for non-persistent in-memory handling\n */\n async subscribe(options: DispatchSubscribeOptions): Promise<void> {\n await this.initialize();\n\n // Gate the subscriber namespace (S5 #1398): require a concrete subscriber\n // name and reject the reserved internal sentinel so callers cannot\n // register a persistent subscription that masquerades as the in-memory\n // pseudo-subscriber.\n const subscriber = (options.subscriber ?? '').trim();\n if (!subscriber) {\n throw new Error('DispatchBus.subscribe requires a non-empty subscriber');\n }\n if (subscriber === DispatchBus.RESERVED_SUBSCRIBER) {\n throw new Error(\n `DispatchBus.subscribe: \"${DispatchBus.RESERVED_SUBSCRIBER}\" is a reserved subscriber name`,\n );\n }\n const signalType = (options.signalType ?? '').trim();\n if (!signalType) {\n throw new Error('DispatchBus.subscribe requires a non-empty signalType');\n }\n\n const subscription = new DispatchSubscription({\n signal_type: signalType,\n subscriber,\n handler: options.handler || 'handleDispatch',\n delivery: options.delivery || 'compete',\n enabled: options.enabled !== false ? 1 : 0,\n // Server-derived tenant scope (never from caller options). NULL when there\n // is no active tenant context (global subscription).\n tenant_id: resolveDispatchTenantId() ?? null,\n });\n\n await DispatchSubscriptionCollection.upsert(this.db, subscription);\n }\n\n /**\n * Remove a persistent subscription\n *\n * @param signalType - Signal type pattern\n * @param subscriber - Subscriber name\n */\n async unsubscribe(signalType: string, subscriber: string): Promise<void> {\n await this.initialize();\n // Scoped to the active tenant so one tenant cannot remove another's\n // subscription (S5 #1398).\n await DispatchSubscriptionCollection.deleteByKey(\n this.db,\n signalType,\n subscriber,\n resolveDispatchTenantScope(),\n );\n }\n\n /**\n * Processes pending dispatches for a named subscriber.\n *\n * For each matching `pending` dispatch:\n * 1. Sets status to `processing`\n * 2. Calls `handler(payload, metadata)`\n * 3. On success: sets status to `completed`\n * 4. On error: sets status to `failed` (stores the error message)\n *\n * Uses a wildcard-aware query strategy: subscriptions with `*` patterns fetch\n * all pending dispatches and filter in memory; exact-match subscriptions use\n * a direct SQL `IN` query for efficiency.\n *\n * @param subscriber - The subscriber name (must match a `subscribe()` call)\n * @param handler - Function to call for each pending dispatch\n * @param options.limit - Maximum dispatches to process in one call (default 100)\n * @param options.signalTypes - Optional filter to process only specific signal types\n * @returns Number of dispatches successfully processed (excludes failed)\n *\n * @example\n * ```typescript\n * const count = await bus.process('FiscusAgent', async (payload, metadata) => {\n * if (metadata.signalType === 'campaign.completed') {\n * await generateInvoice(payload.campaignId);\n * }\n * });\n * console.log(`Processed ${count} dispatches`);\n * ```\n *\n * @see {@link subscribe} to register the persistent subscription first\n * @see {@link retry} to reset failed dispatches back to pending\n */\n async process(\n subscriber: string,\n handler: DispatchHandler,\n options: DispatchProcessOptions = {},\n ): Promise<number> {\n await this.initialize();\n\n // Resolve the active tenant scope server-side (S5 #1398). This distinguishes\n // three states (see resolveDispatchTenantScope):\n // - tenancy off → no filter; system/global + non-tenant deployments behave\n // exactly as before.\n // - active tenant T → only T's dispatches plus global (NULL) dispatches are\n // claimable, so a subscriber in tenant A cannot see/claim tenant B's.\n // - tenancy on but no active tenant → fail-closed to global (NULL) rows\n // only; a missing context never leaks other tenants' dispatches.\n const tenantScope = resolveDispatchTenantScope();\n\n // Get subscriber's subscriptions, scoped to the active tenant so a tenant-A\n // processor only matches tenant-A (and, when global, global) subscriptions.\n const subscriptions = await DispatchSubscriptionCollection.findBySubscriber(\n this.db,\n subscriber,\n true,\n tenantScope,\n );\n\n if (subscriptions.length === 0) {\n return 0;\n }\n\n // Get signal types (including wildcards)\n const signalTypes = subscriptions.map((s) => s.signalType);\n\n // For non-wildcard subscriptions, we can query directly\n // For wildcards, we need to get all pending and filter\n const hasWildcards = signalTypes.some((t) => t.includes('*'));\n\n // Helper: check if a dispatch should be visible to this subscriber\n const isVisibleToSubscriber = (dispatch: Dispatch): boolean => {\n // Tenant isolation (defense in depth — the SQL queries below also filter).\n if (tenantScope.enforced) {\n if (tenantScope.tenantId === null) {\n // Tenancy on, no active tenant → only global (NULL) rows are visible.\n if (dispatch.tenantId !== null) {\n return false;\n }\n } else if (\n dispatch.tenantId !== null &&\n dispatch.tenantId !== tenantScope.tenantId\n ) {\n // Active tenant → that tenant's rows plus global (NULL) rows only.\n return false;\n }\n }\n if (dispatch.targetSubscriber === subscriber) {\n // Targeted at this subscriber (fanout copy) — always visible\n return true;\n }\n if (dispatch.targetSubscriber !== null) {\n // Targeted at a different subscriber — not visible\n return false;\n }\n // Null target (compete dispatch): only visible if this subscriber\n // has a compete subscription matching this dispatch type\n const matchingSub = subscriptions.find((sub) =>\n sub.matches(dispatch.type),\n );\n return matchingSub?.delivery === 'compete';\n };\n\n let pendingDispatches: Dispatch[];\n if (hasWildcards) {\n // Get all pending dispatches and filter by subscription patterns + target\n const allPending = await DispatchCollection.list(this.db, {\n status: 'pending',\n limit: options.limit || 100,\n tenantScope,\n });\n\n pendingDispatches = allPending.filter(\n (dispatch) =>\n subscriptions.some((sub) => sub.matches(dispatch.type)) &&\n isVisibleToSubscriber(dispatch),\n );\n } else {\n // Direct query for exact signal types (with target_subscriber filter)\n const raw = await DispatchCollection.findPending(\n this.db,\n signalTypes,\n options.limit || 100,\n subscriber,\n tenantScope,\n );\n // Post-filter for delivery mode correctness\n pendingDispatches = raw.filter(isVisibleToSubscriber);\n }\n\n // Filter by specific signal types if provided\n if (options.signalTypes && options.signalTypes.length > 0) {\n pendingDispatches = pendingDispatches.filter((d) =>\n options.signalTypes?.includes(d.type),\n );\n }\n\n let processed = 0;\n\n for (const dispatch of pendingDispatches) {\n // Atomically claim the dispatch (S5 #1398). The conditional UPDATE\n // (WHERE id = ? AND status = 'pending' [AND tenant predicate]) closes the\n // TOCTOU window: if a competing worker — or a system-scoped processor —\n // already moved this row out of 'pending', our claim affects no rows and\n // we skip it, so a compete/global dispatch is processed at most once.\n const claimed = await DispatchCollection.claim(\n this.db,\n dispatch,\n tenantScope,\n );\n if (!claimed) {\n continue;\n }\n\n try {\n // Invoke handler\n await handler(dispatch.payload, dispatch.getMetadata());\n\n // Mark as completed\n dispatch.markCompleted(subscriber);\n await DispatchCollection.update(this.db, dispatch);\n processed++;\n } catch (error) {\n // Mark as failed\n const errorMessage =\n error instanceof Error ? error.message : String(error);\n dispatch.markFailed(errorMessage);\n await DispatchCollection.update(this.db, dispatch);\n }\n }\n\n return processed;\n }\n\n /**\n * Retry failed dispatches\n *\n * Resets failed dispatches to pending status so they can be processed again.\n *\n * @param options - Retry options\n * @returns Number of dispatches reset\n */\n async retry(options: DispatchRetryOptions = {}): Promise<number> {\n await this.initialize();\n\n // Tenant isolation (S5 #1398): only the active tenant's failed dispatches\n // (plus global ones) are eligible — a tenant's retry() must not reset\n // another tenant's rows. Scope is derived server-side; callers cannot widen\n // it.\n const retryable = await DispatchCollection.findRetryable(\n this.db,\n options,\n resolveDispatchTenantScope(),\n );\n\n for (const dispatch of retryable) {\n dispatch.resetForRetry();\n await DispatchCollection.update(this.db, dispatch);\n }\n\n return retryable.length;\n }\n\n /**\n * Clean up old dispatches\n *\n * @param options - Cleanup options\n * @returns Number of dispatches deleted\n */\n async cleanup(\n options: DispatchCleanupOptions = {},\n ): Promise<DispatchCleanupResult> {\n await this.initialize();\n // Tenant isolation (S5 #1398): only the active tenant's dispatches (plus\n // global ones) are deleted — a tenant's cleanup() must not delete another\n // tenant's rows. Scope is derived server-side; callers cannot widen it.\n return DispatchCollection.cleanup(\n this.db,\n options,\n resolveDispatchTenantScope(),\n );\n }\n\n /**\n * Lists dispatches, scoped to the active tenant context (S5 #1398).\n *\n * The tenant scope is derived **server-side** from the active context and\n * **overrides any caller-supplied scope** — callers cannot select another\n * tenant or widen visibility to all tenants. Scoping rules:\n *\n * - tenancy off → no tenant filter (pre-tenancy behavior).\n * - active tenant T → that tenant's dispatches plus global (NULL) dispatches.\n * - tenancy on but no active tenant → global (NULL) dispatches only\n * (fail-closed; never all tenants).\n *\n * @param options - Non-tenant list filters (status/type/source/etc.). Any\n * `tenantScope` field is ignored and replaced with the server-derived one.\n */\n async list(options: DispatchListOptions = {}): Promise<Dispatch[]> {\n await this.initialize();\n // Strip any caller-supplied scope and inject the server-derived one.\n const { tenantScope: _ignoredCallerScope, ...safeOptions } = options;\n return DispatchCollection.list(this.db, {\n ...safeOptions,\n tenantScope: resolveDispatchTenantScope(),\n });\n }\n\n /**\n * Gets a dispatch by ID, enforcing the active tenant scope (S5 #1398).\n *\n * Applies the same server-derived tenant predicate as {@link list}: a\n * subscriber in tenant A cannot fetch tenant B's dispatch by id, and when\n * tenancy is on with no active tenant only global (NULL) dispatches are\n * returned. Returns `null` when the dispatch exists but is out of scope.\n */\n async get(id: string): Promise<Dispatch | null> {\n await this.initialize();\n return DispatchCollection.getScoped(\n this.db,\n id,\n resolveDispatchTenantScope(),\n );\n }\n\n /**\n * List all subscriptions\n */\n async listSubscriptions(\n subscriber?: string,\n ): Promise<DispatchSubscription[]> {\n await this.initialize();\n\n // Scoped to the active tenant so listing never leaks another tenant's\n // subscriptions (S5 #1398).\n const tenantScope = resolveDispatchTenantScope();\n\n if (subscriber) {\n return DispatchSubscriptionCollection.findBySubscriber(\n this.db,\n subscriber,\n false,\n tenantScope,\n );\n }\n return DispatchSubscriptionCollection.list(this.db, false, tenantScope);\n }\n\n /**\n * Notify in-memory handlers (fire-and-forget)\n */\n private notifyHandlers(\n type: string,\n payload: unknown,\n metadata: DispatchMetadata,\n ): void {\n // Collect all matching handlers\n const matchingHandlers: RegisteredHandler[] = [];\n\n for (const [, handlers] of this.handlers) {\n for (const registered of handlers) {\n if (registered.subscription.matches(type)) {\n matchingHandlers.push(registered);\n }\n }\n }\n\n // Fire-and-forget - don't await\n for (const registered of matchingHandlers) {\n try {\n const result = registered.handler(payload, metadata);\n // Handle both sync and async handlers\n if (result && typeof result.catch === 'function') {\n void result.catch((error: unknown) => {\n logger.error(\n `DispatchBus: Handler for \"${registered.pattern}\" failed`,\n { error },\n );\n });\n }\n } catch (error) {\n logger.error(\n `DispatchBus: Handler for \"${registered.pattern}\" failed`,\n { error },\n );\n }\n }\n }\n}\n\n/**\n * Create a DispatchBus instance\n *\n * @param options - Bus configuration\n * @returns Initialized DispatchBus\n */\nexport async function createDispatchBus(\n options: DispatchBusOptions = {},\n): Promise<DispatchBus> {\n // Resolve database configuration\n const dbConfig = options.db || options.persistence;\n\n if (!dbConfig) {\n throw new Error('DispatchBus requires a database configuration');\n }\n\n // Get or create database interface\n let db: DatabaseInterface;\n if (typeof dbConfig === 'string') {\n db = await getDatabase(dbConfig);\n } else if ('query' in dbConfig) {\n // Already a DatabaseInterface (has query method)\n db = dbConfig as DatabaseInterface;\n } else if ('type' in dbConfig && 'url' in dbConfig) {\n // Database config object - use getDatabase with type and url\n db = await getDatabase({\n type: dbConfig.type as 'sqlite' | 'postgres' | 'duckdb',\n url: dbConfig.url,\n });\n } else {\n throw new Error('Invalid database configuration for DispatchBus');\n }\n\n const bus = new DispatchBus(db);\n await bus.initialize();\n\n return bus;\n}\n"],"names":[],"mappings":"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"}
|