@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,509 @@
|
|
|
1
|
+
function capitalize(str) {
|
|
2
|
+
return str.charAt(0).toUpperCase() + str.slice(1);
|
|
3
|
+
}
|
|
4
|
+
function generateRuntimeBootstrap(options = {}) {
|
|
5
|
+
const {
|
|
6
|
+
name = "smrt-mcp-server",
|
|
7
|
+
version = "1.0.0",
|
|
8
|
+
description = "Auto-generated MCP server from SMRT objects",
|
|
9
|
+
debug = false,
|
|
10
|
+
tools = [],
|
|
11
|
+
tenantScopedObjects = []
|
|
12
|
+
} = options;
|
|
13
|
+
const toolsCode = tools.length > 0 ? JSON.stringify(tools, null, 2) : "[]";
|
|
14
|
+
const tenantScopedSet = Array.from(
|
|
15
|
+
new Set(tenantScopedObjects.map((n) => n.toLowerCase()))
|
|
16
|
+
);
|
|
17
|
+
const hasTenantScoped = tenantScopedSet.length > 0;
|
|
18
|
+
const generateSwitchCases = (indent) => {
|
|
19
|
+
return tools.map((tool) => {
|
|
20
|
+
const [objectName, action] = tool.name.split("_");
|
|
21
|
+
switch (action) {
|
|
22
|
+
case "list":
|
|
23
|
+
return `${indent}case '${tool.name}': {
|
|
24
|
+
${indent} const limit = args.limit ?? 50;
|
|
25
|
+
${indent} const offset = args.offset ?? 0;
|
|
26
|
+
${indent} const where = args.where ?? {};
|
|
27
|
+
|
|
28
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
29
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
30
|
+
${indent} ai: aiConfig
|
|
31
|
+
${indent} });
|
|
32
|
+
|
|
33
|
+
${indent} const items = await collection.list({ where, limit, offset });
|
|
34
|
+
${indent} const itemsPublic = items.map((item) => item.toPublicJSON());
|
|
35
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify(itemsPublic) }] };
|
|
36
|
+
${indent}}`;
|
|
37
|
+
case "get":
|
|
38
|
+
return `${indent}case '${tool.name}': {
|
|
39
|
+
${indent} if (!args.id && !args.slug) {
|
|
40
|
+
${indent} throw new Error('Either id or slug is required');
|
|
41
|
+
${indent} }
|
|
42
|
+
|
|
43
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
44
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
45
|
+
${indent} ai: aiConfig
|
|
46
|
+
${indent} });
|
|
47
|
+
|
|
48
|
+
${indent} const filter = args.id || args.slug;
|
|
49
|
+
${indent} const item = await collection.get(filter);
|
|
50
|
+
|
|
51
|
+
${indent} if (!item) {
|
|
52
|
+
${indent} throw new Error('Object not found');
|
|
53
|
+
${indent} }
|
|
54
|
+
|
|
55
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify(item.toPublicJSON()) }] };
|
|
56
|
+
${indent}}`;
|
|
57
|
+
case "create":
|
|
58
|
+
return `${indent}case '${tool.name}': {
|
|
59
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
60
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
61
|
+
${indent} ai: aiConfig
|
|
62
|
+
${indent} });
|
|
63
|
+
|
|
64
|
+
${indent} const newItem = await collection.create(applyWritablePolicy('${capitalize(objectName)}', args));
|
|
65
|
+
${indent} await newItem.save();
|
|
66
|
+
|
|
67
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify(newItem.toPublicJSON()) }] };
|
|
68
|
+
${indent}}`;
|
|
69
|
+
case "update":
|
|
70
|
+
return `${indent}case '${tool.name}': {
|
|
71
|
+
${indent} const { id, ...updateData } = args;
|
|
72
|
+
${indent} if (!id) {
|
|
73
|
+
${indent} throw new Error('ID is required for update');
|
|
74
|
+
${indent} }
|
|
75
|
+
|
|
76
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
77
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
78
|
+
${indent} ai: aiConfig
|
|
79
|
+
${indent} });
|
|
80
|
+
|
|
81
|
+
${indent} const existing = await collection.get(id);
|
|
82
|
+
${indent} if (!existing) {
|
|
83
|
+
${indent} throw new Error('Object not found');
|
|
84
|
+
${indent} }
|
|
85
|
+
|
|
86
|
+
${indent} Object.assign(existing, applyWritablePolicy('${capitalize(objectName)}', updateData));
|
|
87
|
+
${indent} await existing.save();
|
|
88
|
+
|
|
89
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify(existing.toPublicJSON()) }] };
|
|
90
|
+
${indent}}`;
|
|
91
|
+
case "delete":
|
|
92
|
+
return `${indent}case '${tool.name}': {
|
|
93
|
+
${indent} if (!args.id) {
|
|
94
|
+
${indent} throw new Error('ID is required for delete');
|
|
95
|
+
${indent} }
|
|
96
|
+
|
|
97
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
98
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
99
|
+
${indent} ai: aiConfig
|
|
100
|
+
${indent} });
|
|
101
|
+
|
|
102
|
+
${indent} const toDelete = await collection.get(args.id);
|
|
103
|
+
${indent} if (!toDelete) {
|
|
104
|
+
${indent} throw new Error('Object not found');
|
|
105
|
+
${indent} }
|
|
106
|
+
|
|
107
|
+
${indent} await toDelete.delete();
|
|
108
|
+
|
|
109
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify({ success: true, message: 'Object deleted successfully' }) }] };
|
|
110
|
+
${indent}}`;
|
|
111
|
+
default:
|
|
112
|
+
return `${indent}case '${tool.name}': {
|
|
113
|
+
${indent} const { id, options = {}, ...directArgs } = args;
|
|
114
|
+
|
|
115
|
+
${indent} if (!id) {
|
|
116
|
+
${indent} throw new Error('ID is required for custom action ${action}');
|
|
117
|
+
${indent} }
|
|
118
|
+
|
|
119
|
+
${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {
|
|
120
|
+
${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },
|
|
121
|
+
${indent} ai: aiConfig
|
|
122
|
+
${indent} });
|
|
123
|
+
|
|
124
|
+
${indent} const object = await collection.get(id);
|
|
125
|
+
${indent} if (!object) {
|
|
126
|
+
${indent} throw new Error('Object not found');
|
|
127
|
+
${indent} }
|
|
128
|
+
|
|
129
|
+
${indent} if (typeof object['${action}'] !== 'function') {
|
|
130
|
+
${indent} throw new Error('Method ${action} not found on object');
|
|
131
|
+
${indent} }
|
|
132
|
+
|
|
133
|
+
${indent} const methodArgs = Object.keys(options).length > 0 ? options : directArgs;
|
|
134
|
+
${indent} const result = await object['${action}'](methodArgs);
|
|
135
|
+
|
|
136
|
+
${indent} return { content: [{ type: 'text', text: JSON.stringify(toPublicResult(result)) }] };
|
|
137
|
+
${indent}}`;
|
|
138
|
+
}
|
|
139
|
+
}).join("\n\n");
|
|
140
|
+
};
|
|
141
|
+
const switchCases = generateSwitchCases(" ");
|
|
142
|
+
return `#!/usr/bin/env node
|
|
143
|
+
/**
|
|
144
|
+
* Auto-generated MCP Server
|
|
145
|
+
* Generated by @smrt/core MCPGenerator
|
|
146
|
+
*
|
|
147
|
+
* This server exposes SMRT objects as MCP tools for AI integration.
|
|
148
|
+
*
|
|
149
|
+
* SECURITY (#1540): tool responses exclude @field({ sensitive }) fields and
|
|
150
|
+
* create/update bodies are mass-assignment guarded. This stdio server has NO
|
|
151
|
+
* per-call authentication principal — its trust boundary is the host process /
|
|
152
|
+
* MCP client that launches it. Run it only in a trusted context, or front it
|
|
153
|
+
* with an authenticated gateway. Do not expose it directly to untrusted callers.
|
|
154
|
+
*/
|
|
155
|
+
|
|
156
|
+
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
|
157
|
+
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
|
158
|
+
import {
|
|
159
|
+
CallToolRequestSchema,
|
|
160
|
+
ListToolsRequestSchema,
|
|
161
|
+
type CallToolRequest,
|
|
162
|
+
type ListToolsRequest,
|
|
163
|
+
} from '@modelcontextprotocol/sdk/types.js';
|
|
164
|
+
import { ObjectRegistry } from '@happyvertical/smrt-core';
|
|
165
|
+
import { config } from '@happyvertical/smrt-config';
|
|
166
|
+
${hasTenantScoped ? "import { enableTenancy, runTenantScopedEntryPoint } from '@happyvertical/smrt-tenancy';\n" : ""}
|
|
167
|
+
// Server configuration
|
|
168
|
+
const SERVER_NAME = ${JSON.stringify(name)};
|
|
169
|
+
const SERVER_VERSION = ${JSON.stringify(version)};
|
|
170
|
+
const SERVER_DESCRIPTION = ${JSON.stringify(description)};
|
|
171
|
+
const DEBUG = ${debug};
|
|
172
|
+
|
|
173
|
+
// Static tool definitions (generated at build time)
|
|
174
|
+
const TOOLS = ${toolsCode};
|
|
175
|
+
${hasTenantScoped ? `
|
|
176
|
+
// Fail-closed tenant context (#1554): tenant-scoped objects must run inside a
|
|
177
|
+
// tenant. This stdio server has no auth principal, so the tenant is taken from
|
|
178
|
+
// the environment; without it (and with tenancy enabled) tenant-scoped tools
|
|
179
|
+
// throw rather than reading across all tenants.
|
|
180
|
+
const TENANT_SCOPED = new Set(${JSON.stringify(tenantScopedSet)});
|
|
181
|
+
const MCP_TENANT_ID = process.env.SMRT_MCP_TENANT_ID || undefined;
|
|
182
|
+
const MCP_ALLOW_CROSS_TENANT = process.env.SMRT_MCP_ALLOW_CROSS_TENANT === 'true';
|
|
183
|
+
` : ""}
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Mass-assignment guard (#1540): strip framework/server-managed and
|
|
187
|
+
* \`@field({ readonly: true })\` fields from create/update bodies, intersecting
|
|
188
|
+
* with the optional \`@smrt({ api: { writable: [...] } })\` allowlist.
|
|
189
|
+
*/
|
|
190
|
+
function applyWritablePolicy(objectName: string, data: any): Record<string, any> {
|
|
191
|
+
if (!data || typeof data !== 'object') return {};
|
|
192
|
+
const serverManaged = new Set([
|
|
193
|
+
'id', 'tenantId', 'tenant_id',
|
|
194
|
+
'createdAt', 'created_at', 'updatedAt', 'updated_at',
|
|
195
|
+
]);
|
|
196
|
+
const readonly = new Set<string>();
|
|
197
|
+
let writable: string[] | null = null;
|
|
198
|
+
const apiConfig = ObjectRegistry.getConfig(objectName)?.api as any;
|
|
199
|
+
if (apiConfig && typeof apiConfig === 'object' && Array.isArray(apiConfig.writable)) {
|
|
200
|
+
writable = apiConfig.writable;
|
|
201
|
+
}
|
|
202
|
+
for (const [name, def] of ObjectRegistry.getFields(objectName)) {
|
|
203
|
+
if (def && ((def as any).readonly === true || (def as any)._meta?.readonly === true)) {
|
|
204
|
+
readonly.add(name);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
const result: Record<string, any> = {};
|
|
208
|
+
for (const [key, value] of Object.entries(data)) {
|
|
209
|
+
if (key.startsWith('_')) continue;
|
|
210
|
+
if (serverManaged.has(key)) continue;
|
|
211
|
+
if (readonly.has(key)) continue;
|
|
212
|
+
if (writable && !writable.includes(key)) continue;
|
|
213
|
+
result[key] = value;
|
|
214
|
+
}
|
|
215
|
+
return result;
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
/**
|
|
219
|
+
* Sensitive-field-safe serialization for custom-action results (#1540).
|
|
220
|
+
* Recurses through arrays and plain objects so nested SmrtObjects are stripped
|
|
221
|
+
* too; non-plain instances (Date, etc.) and primitives pass through. Cycle-safe.
|
|
222
|
+
*/
|
|
223
|
+
function toPublicResult(value: any, seen: WeakSet<object> = new WeakSet()): any {
|
|
224
|
+
if (value === null || typeof value !== 'object') return value;
|
|
225
|
+
if (typeof value.toPublicJSON === 'function') return value.toPublicJSON();
|
|
226
|
+
if (Array.isArray(value)) {
|
|
227
|
+
if (seen.has(value)) return value;
|
|
228
|
+
seen.add(value);
|
|
229
|
+
return value.map((entry: any) => toPublicResult(entry, seen));
|
|
230
|
+
}
|
|
231
|
+
const proto = Object.getPrototypeOf(value);
|
|
232
|
+
if (proto !== Object.prototype && proto !== null) return value;
|
|
233
|
+
if (seen.has(value)) return value;
|
|
234
|
+
seen.add(value);
|
|
235
|
+
const out: Record<string, any> = {};
|
|
236
|
+
for (const [key, entry] of Object.entries(value)) {
|
|
237
|
+
out[key] = toPublicResult(entry, seen);
|
|
238
|
+
}
|
|
239
|
+
return out;
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
/**
|
|
243
|
+
* Main server startup function
|
|
244
|
+
*/
|
|
245
|
+
async function main() {
|
|
246
|
+
try {
|
|
247
|
+
if (DEBUG) {
|
|
248
|
+
console.error(\`[MCP] Starting server: \${SERVER_NAME} v\${SERVER_VERSION}\`);
|
|
249
|
+
}
|
|
250
|
+
${hasTenantScoped ? `
|
|
251
|
+
// Fail-closed tenant context (#1554): install the tenancy interceptor so
|
|
252
|
+
// tenant-scoped tools are actually filtered, and so the entry-point gate
|
|
253
|
+
// throws (rather than passing through) when no tenant is supplied. Without
|
|
254
|
+
// this, a tenant set via SMRT_MCP_TENANT_ID would only set async context
|
|
255
|
+
// with no interceptor to enforce it.
|
|
256
|
+
enableTenancy();
|
|
257
|
+
` : ""}
|
|
258
|
+
// Load configuration from environment and .smrt.config files
|
|
259
|
+
const appConfig = await config.load();
|
|
260
|
+
const aiConfig = appConfig?.ai || {};
|
|
261
|
+
|
|
262
|
+
if (DEBUG) {
|
|
263
|
+
console.error(\`[MCP] Loaded \${TOOLS.length} static tools\`);
|
|
264
|
+
console.error(\`[MCP] Available tools:\`, TOOLS.map(t => t.name).join(', '));
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
// Create MCP server
|
|
268
|
+
const server = new Server(
|
|
269
|
+
{
|
|
270
|
+
name: SERVER_NAME,
|
|
271
|
+
version: SERVER_VERSION,
|
|
272
|
+
},
|
|
273
|
+
{
|
|
274
|
+
capabilities: {
|
|
275
|
+
tools: {},
|
|
276
|
+
},
|
|
277
|
+
}
|
|
278
|
+
);
|
|
279
|
+
|
|
280
|
+
// Register ListTools handler
|
|
281
|
+
server.setRequestHandler(ListToolsRequestSchema, async (_request: ListToolsRequest) => {
|
|
282
|
+
if (DEBUG) {
|
|
283
|
+
console.error(\`[MCP] ListTools request received\`);
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
return {
|
|
287
|
+
tools: TOOLS,
|
|
288
|
+
};
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
// Register CallTool handler
|
|
292
|
+
server.setRequestHandler(CallToolRequestSchema, async (request: CallToolRequest) => {
|
|
293
|
+
const { name: toolName, arguments: args = {} } = request.params;
|
|
294
|
+
|
|
295
|
+
if (DEBUG) {
|
|
296
|
+
console.error(\`[MCP] CallTool request: \${toolName}\`);
|
|
297
|
+
console.error(\`[MCP] Arguments:\`, JSON.stringify(args, null, 2));
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
try {
|
|
301
|
+
// Static switch statement for tool execution
|
|
302
|
+
const runToolBody = async () => {
|
|
303
|
+
switch (toolName) {
|
|
304
|
+
${switchCases}
|
|
305
|
+
|
|
306
|
+
default:
|
|
307
|
+
throw new Error(\`Unknown tool: \${toolName}\`);
|
|
308
|
+
}
|
|
309
|
+
};
|
|
310
|
+
${hasTenantScoped ? `
|
|
311
|
+
// Fail-closed tenant context for tenant-scoped tools (#1554).
|
|
312
|
+
const [toolObject] = toolName.split('_');
|
|
313
|
+
const result =
|
|
314
|
+
toolObject && TENANT_SCOPED.has(toolObject.toLowerCase())
|
|
315
|
+
? await runTenantScopedEntryPoint(
|
|
316
|
+
{ tenantScoped: true, tenantId: MCP_TENANT_ID, allowCrossTenant: MCP_ALLOW_CROSS_TENANT, surface: 'MCP' },
|
|
317
|
+
runToolBody,
|
|
318
|
+
)
|
|
319
|
+
: await runToolBody();` : `
|
|
320
|
+
const result = await runToolBody();`}
|
|
321
|
+
|
|
322
|
+
if (DEBUG) {
|
|
323
|
+
console.error(\`[MCP] Tool executed successfully: \${toolName}\`);
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
return result;
|
|
327
|
+
} catch (error) {
|
|
328
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
329
|
+
console.error(\`[MCP] Tool execution failed: \${toolName}\`, error);
|
|
330
|
+
|
|
331
|
+
return {
|
|
332
|
+
content: [
|
|
333
|
+
{
|
|
334
|
+
type: 'text',
|
|
335
|
+
text: \`Error executing tool \${toolName}: \${errorMessage}\`,
|
|
336
|
+
},
|
|
337
|
+
],
|
|
338
|
+
isError: true,
|
|
339
|
+
};
|
|
340
|
+
}
|
|
341
|
+
});
|
|
342
|
+
|
|
343
|
+
// Setup stdio transport
|
|
344
|
+
const transport = new StdioServerTransport();
|
|
345
|
+
|
|
346
|
+
// Connect server to transport
|
|
347
|
+
await server.connect(transport);
|
|
348
|
+
|
|
349
|
+
if (DEBUG) {
|
|
350
|
+
console.error(\`[MCP] Server connected via stdio transport\`);
|
|
351
|
+
console.error(\`[MCP] Ready to receive requests\`);
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
// Handle graceful shutdown
|
|
355
|
+
process.on('SIGINT', async () => {
|
|
356
|
+
if (DEBUG) {
|
|
357
|
+
console.error(\`[MCP] Received SIGINT, shutting down gracefully\`);
|
|
358
|
+
}
|
|
359
|
+
await server.close();
|
|
360
|
+
process.exit(0);
|
|
361
|
+
});
|
|
362
|
+
|
|
363
|
+
process.on('SIGTERM', async () => {
|
|
364
|
+
if (DEBUG) {
|
|
365
|
+
console.error(\`[MCP] Received SIGTERM, shutting down gracefully\`);
|
|
366
|
+
}
|
|
367
|
+
await server.close();
|
|
368
|
+
process.exit(0);
|
|
369
|
+
});
|
|
370
|
+
} catch (error) {
|
|
371
|
+
console.error('[MCP] Fatal error during server startup:', error);
|
|
372
|
+
process.exit(1);
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
// Start the server
|
|
377
|
+
main().catch((error) => {
|
|
378
|
+
console.error('[MCP] Unhandled error:', error);
|
|
379
|
+
process.exit(1);
|
|
380
|
+
});
|
|
381
|
+
`;
|
|
382
|
+
}
|
|
383
|
+
function generateMCPScript(serverPath = "dist/mcp-server.js") {
|
|
384
|
+
return `node ${serverPath}`;
|
|
385
|
+
}
|
|
386
|
+
function generateClaudeConfig(serverName, serverPath) {
|
|
387
|
+
return {
|
|
388
|
+
mcpServers: {
|
|
389
|
+
[serverName]: {
|
|
390
|
+
command: "node",
|
|
391
|
+
args: [serverPath]
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
function generateMCPDocumentation(serverName, serverPath) {
|
|
397
|
+
return `# MCP Server Setup
|
|
398
|
+
|
|
399
|
+
This project includes an auto-generated MCP (Model Context Protocol) server that exposes SMRT objects as tools for AI integration.
|
|
400
|
+
|
|
401
|
+
## Quick Start
|
|
402
|
+
|
|
403
|
+
### 1. Build the MCP Server
|
|
404
|
+
|
|
405
|
+
\`\`\`bash
|
|
406
|
+
npm run build
|
|
407
|
+
\`\`\`
|
|
408
|
+
|
|
409
|
+
This generates the MCP server at: \`${serverPath}\`
|
|
410
|
+
|
|
411
|
+
### 2. Configure Claude Desktop
|
|
412
|
+
|
|
413
|
+
Add the following to your Claude Desktop configuration file:
|
|
414
|
+
|
|
415
|
+
**macOS**: \`~/.config/Claude/claude_desktop_config.json\`
|
|
416
|
+
**Windows**: \`%APPDATA%\\Claude\\claude_desktop_config.json\`
|
|
417
|
+
|
|
418
|
+
\`\`\`json
|
|
419
|
+
{
|
|
420
|
+
"mcpServers": {
|
|
421
|
+
"${serverName}": {
|
|
422
|
+
"command": "node",
|
|
423
|
+
"args": ["/absolute/path/to/${serverPath}"]
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
\`\`\`
|
|
428
|
+
|
|
429
|
+
Replace \`/absolute/path/to/\` with the actual absolute path to your project directory.
|
|
430
|
+
|
|
431
|
+
### 3. Restart Claude Desktop
|
|
432
|
+
|
|
433
|
+
Close and reopen Claude Desktop to load the new MCP server.
|
|
434
|
+
|
|
435
|
+
### 4. Test the Integration
|
|
436
|
+
|
|
437
|
+
In Claude Code, you can now use the auto-generated tools. For example:
|
|
438
|
+
|
|
439
|
+
- \`list_products\` - List all products
|
|
440
|
+
- \`get_product\` - Get a specific product by ID
|
|
441
|
+
- \`create_product\` - Create a new product
|
|
442
|
+
- And more...
|
|
443
|
+
|
|
444
|
+
## Environment Variables
|
|
445
|
+
|
|
446
|
+
The MCP server supports optional environment variables:
|
|
447
|
+
|
|
448
|
+
- \`DATABASE_URL\` - Database connection string
|
|
449
|
+
|
|
450
|
+
**AI Provider Configuration (in priority order):**
|
|
451
|
+
1. **Generic configuration** (supports any provider):
|
|
452
|
+
- \`SMRT_AI_PROVIDER\` - Provider name (e.g., 'openai', 'anthropic', 'claude-cli', 'gemini')
|
|
453
|
+
- \`SMRT_AI_API_KEY\` - API key for the provider
|
|
454
|
+
- \`SMRT_AI_MODEL\` - Model to use (optional)
|
|
455
|
+
|
|
456
|
+
2. **Provider-specific fallbacks**:
|
|
457
|
+
- \`OPENAI_API_KEY\` - OpenAI API key (auto-detects provider as 'openai')
|
|
458
|
+
- \`ANTHROPIC_API_KEY\` - Anthropic API key (auto-detects provider as 'anthropic')
|
|
459
|
+
- \`CLAUDE_API_KEY\` + \`CLAUDE_MODEL\` - Claude CLI provider (defaults to 'sonnet')
|
|
460
|
+
|
|
461
|
+
**Examples:**
|
|
462
|
+
\`\`\`bash
|
|
463
|
+
# Using generic configuration (recommended)
|
|
464
|
+
export SMRT_AI_PROVIDER=claude-cli
|
|
465
|
+
export SMRT_AI_MODEL=sonnet
|
|
466
|
+
|
|
467
|
+
# Using provider-specific configuration
|
|
468
|
+
export CLAUDE_API_KEY=your-key
|
|
469
|
+
export CLAUDE_MODEL=sonnet
|
|
470
|
+
|
|
471
|
+
# Using OpenAI
|
|
472
|
+
export OPENAI_API_KEY=your-openai-key
|
|
473
|
+
\`\`\`
|
|
474
|
+
|
|
475
|
+
## Troubleshooting
|
|
476
|
+
|
|
477
|
+
### Server Not Appearing in Claude
|
|
478
|
+
|
|
479
|
+
1. Check that the path in \`claude_desktop_config.json\` is absolute
|
|
480
|
+
2. Verify the server file exists at the specified path
|
|
481
|
+
3. Check Claude Desktop logs for errors
|
|
482
|
+
|
|
483
|
+
### Tools Not Working
|
|
484
|
+
|
|
485
|
+
1. Ensure your database is accessible (if using one)
|
|
486
|
+
2. Check that SMRT objects are properly decorated with \`@smrt()\`
|
|
487
|
+
3. Look for errors in the MCP server output
|
|
488
|
+
|
|
489
|
+
### Debug Mode
|
|
490
|
+
|
|
491
|
+
To enable debug logging, set the \`DEBUG\` constant to \`true\` in the generated server file.
|
|
492
|
+
|
|
493
|
+
## Generated Tools
|
|
494
|
+
|
|
495
|
+
The following tools are automatically generated from your SMRT objects:
|
|
496
|
+
|
|
497
|
+
- **CRUD Operations**: \`list_\`, \`get_\`, \`create_\`, \`update_\`, \`delete_\` for each object type
|
|
498
|
+
- **Custom Actions**: Any custom methods included in the \`@smrt()\` decorator configuration
|
|
499
|
+
|
|
500
|
+
See the SMRT object definitions for the complete list of available tools and their parameters.
|
|
501
|
+
`;
|
|
502
|
+
}
|
|
503
|
+
export {
|
|
504
|
+
generateClaudeConfig,
|
|
505
|
+
generateMCPDocumentation,
|
|
506
|
+
generateMCPScript,
|
|
507
|
+
generateRuntimeBootstrap
|
|
508
|
+
};
|
|
509
|
+
//# sourceMappingURL=mcp-runtime-template.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mcp-runtime-template.js","sources":["../../src/generators/mcp-runtime-template.ts"],"sourcesContent":["/**\n * Runtime bootstrap template for generated MCP servers\n *\n * This template provides stdio transport integration for SMRT-generated MCP servers.\n * It handles:\n * - Server initialization with @modelcontextprotocol/sdk\n * - Tool registration from MCPGenerator\n * - Stdio transport connection\n * - Error handling and logging\n * - Graceful shutdown\n */\n\nimport type { MCPConfig, MCPContext } from './mcp.js';\n\n/**\n * Helper function to capitalize first letter\n */\nfunction capitalize(str: string): string {\n return str.charAt(0).toUpperCase() + str.slice(1);\n}\n\nexport interface RuntimeOptions {\n /** Server name (defaults to package name) */\n name?: string;\n\n /** Server version (defaults to package version) */\n version?: string;\n\n /** Server description */\n description?: string;\n\n /** MCP generator configuration */\n config?: MCPConfig;\n\n /** MCP context (database, AI client, etc.) */\n context?: MCPContext;\n\n /** Enable debug logging */\n debug?: boolean;\n\n /** Static tool definitions (generated at build time) */\n tools?: Array<{\n name: string;\n description: string;\n inputSchema: any;\n }>;\n\n /**\n * Lowercased simple names of objects that are `@TenantScoped` (#1554). When\n * non-empty, the generated server imports the tenancy fail-closed gate and\n * wraps tenant-scoped tool calls so a stdio invocation cannot read across all\n * tenants. The tenant is sourced from `SMRT_MCP_TENANT_ID` /\n * `SMRT_MCP_ALLOW_CROSS_TENANT` env vars (this server has no auth principal).\n */\n tenantScopedObjects?: string[];\n}\n\n/**\n * Generate runtime bootstrap code for MCP server\n *\n * @param options - Runtime configuration options\n * @returns TypeScript code for server entry point\n */\nexport function generateRuntimeBootstrap(options: RuntimeOptions = {}): string {\n const {\n name = 'smrt-mcp-server',\n version = '1.0.0',\n description = 'Auto-generated MCP server from SMRT objects',\n debug = false,\n tools = [],\n tenantScopedObjects = [],\n } = options;\n\n // Generate static tool array as TypeScript code\n const toolsCode = tools.length > 0 ? JSON.stringify(tools, null, 2) : '[]';\n\n // Fail-closed tenant context (#1554): only wire the tenancy gate when at\n // least one exposed object is tenant-scoped, so apps without tenancy never\n // get a dangling import.\n const tenantScopedSet = Array.from(\n new Set(tenantScopedObjects.map((n) => n.toLowerCase())),\n );\n const hasTenantScoped = tenantScopedSet.length > 0;\n\n // Generate static switch cases using shared helper\n const generateSwitchCases = (indent: string) => {\n return tools\n .map((tool) => {\n const [objectName, action] = tool.name.split('_');\n\n switch (action) {\n case 'list':\n return `${indent}case '${tool.name}': {\n${indent} const limit = args.limit ?? 50;\n${indent} const offset = args.offset ?? 0;\n${indent} const where = args.where ?? {};\n\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const items = await collection.list({ where, limit, offset });\n${indent} const itemsPublic = items.map((item) => item.toPublicJSON());\n${indent} return { content: [{ type: 'text', text: JSON.stringify(itemsPublic) }] };\n${indent}}`;\n\n case 'get':\n return `${indent}case '${tool.name}': {\n${indent} if (!args.id && !args.slug) {\n${indent} throw new Error('Either id or slug is required');\n${indent} }\n\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const filter = args.id || args.slug;\n${indent} const item = await collection.get(filter);\n\n${indent} if (!item) {\n${indent} throw new Error('Object not found');\n${indent} }\n\n${indent} return { content: [{ type: 'text', text: JSON.stringify(item.toPublicJSON()) }] };\n${indent}}`;\n\n case 'create':\n return `${indent}case '${tool.name}': {\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const newItem = await collection.create(applyWritablePolicy('${capitalize(objectName)}', args));\n${indent} await newItem.save();\n\n${indent} return { content: [{ type: 'text', text: JSON.stringify(newItem.toPublicJSON()) }] };\n${indent}}`;\n\n case 'update':\n return `${indent}case '${tool.name}': {\n${indent} const { id, ...updateData } = args;\n${indent} if (!id) {\n${indent} throw new Error('ID is required for update');\n${indent} }\n\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const existing = await collection.get(id);\n${indent} if (!existing) {\n${indent} throw new Error('Object not found');\n${indent} }\n\n${indent} Object.assign(existing, applyWritablePolicy('${capitalize(objectName)}', updateData));\n${indent} await existing.save();\n\n${indent} return { content: [{ type: 'text', text: JSON.stringify(existing.toPublicJSON()) }] };\n${indent}}`;\n\n case 'delete':\n return `${indent}case '${tool.name}': {\n${indent} if (!args.id) {\n${indent} throw new Error('ID is required for delete');\n${indent} }\n\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const toDelete = await collection.get(args.id);\n${indent} if (!toDelete) {\n${indent} throw new Error('Object not found');\n${indent} }\n\n${indent} await toDelete.delete();\n\n${indent} return { content: [{ type: 'text', text: JSON.stringify({ success: true, message: 'Object deleted successfully' }) }] };\n${indent}}`;\n\n default:\n // Custom action\n return `${indent}case '${tool.name}': {\n${indent} const { id, options = {}, ...directArgs } = args;\n\n${indent} if (!id) {\n${indent} throw new Error('ID is required for custom action ${action}');\n${indent} }\n\n${indent} const collection = await ObjectRegistry.getCollection('${capitalize(objectName)}', {\n${indent} persistence: { type: 'sql', url: process.env.DATABASE_URL || ':memory:' },\n${indent} ai: aiConfig\n${indent} });\n\n${indent} const object = await collection.get(id);\n${indent} if (!object) {\n${indent} throw new Error('Object not found');\n${indent} }\n\n${indent} if (typeof object['${action}'] !== 'function') {\n${indent} throw new Error('Method ${action} not found on object');\n${indent} }\n\n${indent} const methodArgs = Object.keys(options).length > 0 ? options : directArgs;\n${indent} const result = await object['${action}'](methodArgs);\n\n${indent} return { content: [{ type: 'text', text: JSON.stringify(toPublicResult(result)) }] };\n${indent}}`;\n }\n })\n .join('\\n\\n');\n };\n\n const switchCases = generateSwitchCases(' ');\n\n return `#!/usr/bin/env node\n/**\n * Auto-generated MCP Server\n * Generated by @smrt/core MCPGenerator\n *\n * This server exposes SMRT objects as MCP tools for AI integration.\n *\n * SECURITY (#1540): tool responses exclude @field({ sensitive }) fields and\n * create/update bodies are mass-assignment guarded. This stdio server has NO\n * per-call authentication principal — its trust boundary is the host process /\n * MCP client that launches it. Run it only in a trusted context, or front it\n * with an authenticated gateway. Do not expose it directly to untrusted callers.\n */\n\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n CallToolRequestSchema,\n ListToolsRequestSchema,\n type CallToolRequest,\n type ListToolsRequest,\n} from '@modelcontextprotocol/sdk/types.js';\nimport { ObjectRegistry } from '@happyvertical/smrt-core';\nimport { config } from '@happyvertical/smrt-config';\n${hasTenantScoped ? \"import { enableTenancy, runTenantScopedEntryPoint } from '@happyvertical/smrt-tenancy';\\n\" : ''}\n// Server configuration\nconst SERVER_NAME = ${JSON.stringify(name)};\nconst SERVER_VERSION = ${JSON.stringify(version)};\nconst SERVER_DESCRIPTION = ${JSON.stringify(description)};\nconst DEBUG = ${debug};\n\n// Static tool definitions (generated at build time)\nconst TOOLS = ${toolsCode};\n${\n hasTenantScoped\n ? `\n// Fail-closed tenant context (#1554): tenant-scoped objects must run inside a\n// tenant. This stdio server has no auth principal, so the tenant is taken from\n// the environment; without it (and with tenancy enabled) tenant-scoped tools\n// throw rather than reading across all tenants.\nconst TENANT_SCOPED = new Set(${JSON.stringify(tenantScopedSet)});\nconst MCP_TENANT_ID = process.env.SMRT_MCP_TENANT_ID || undefined;\nconst MCP_ALLOW_CROSS_TENANT = process.env.SMRT_MCP_ALLOW_CROSS_TENANT === 'true';\n`\n : ''\n}\n\n/**\n * Mass-assignment guard (#1540): strip framework/server-managed and\n * \\`@field({ readonly: true })\\` fields from create/update bodies, intersecting\n * with the optional \\`@smrt({ api: { writable: [...] } })\\` allowlist.\n */\nfunction applyWritablePolicy(objectName: string, data: any): Record<string, any> {\n if (!data || typeof data !== 'object') return {};\n const serverManaged = new Set([\n 'id', 'tenantId', 'tenant_id',\n 'createdAt', 'created_at', 'updatedAt', 'updated_at',\n ]);\n const readonly = new Set<string>();\n let writable: string[] | null = null;\n const apiConfig = ObjectRegistry.getConfig(objectName)?.api as any;\n if (apiConfig && typeof apiConfig === 'object' && Array.isArray(apiConfig.writable)) {\n writable = apiConfig.writable;\n }\n for (const [name, def] of ObjectRegistry.getFields(objectName)) {\n if (def && ((def as any).readonly === true || (def as any)._meta?.readonly === true)) {\n readonly.add(name);\n }\n }\n const result: Record<string, any> = {};\n for (const [key, value] of Object.entries(data)) {\n if (key.startsWith('_')) continue;\n if (serverManaged.has(key)) continue;\n if (readonly.has(key)) continue;\n if (writable && !writable.includes(key)) continue;\n result[key] = value;\n }\n return result;\n}\n\n/**\n * Sensitive-field-safe serialization for custom-action results (#1540).\n * Recurses through arrays and plain objects so nested SmrtObjects are stripped\n * too; non-plain instances (Date, etc.) and primitives pass through. Cycle-safe.\n */\nfunction toPublicResult(value: any, seen: WeakSet<object> = new WeakSet()): any {\n if (value === null || typeof value !== 'object') return value;\n if (typeof value.toPublicJSON === 'function') return value.toPublicJSON();\n if (Array.isArray(value)) {\n if (seen.has(value)) return value;\n seen.add(value);\n return value.map((entry: any) => toPublicResult(entry, seen));\n }\n const proto = Object.getPrototypeOf(value);\n if (proto !== Object.prototype && proto !== null) return value;\n if (seen.has(value)) return value;\n seen.add(value);\n const out: Record<string, any> = {};\n for (const [key, entry] of Object.entries(value)) {\n out[key] = toPublicResult(entry, seen);\n }\n return out;\n}\n\n/**\n * Main server startup function\n */\nasync function main() {\n try {\n if (DEBUG) {\n console.error(\\`[MCP] Starting server: \\${SERVER_NAME} v\\${SERVER_VERSION}\\`);\n }\n${\n hasTenantScoped\n ? `\n // Fail-closed tenant context (#1554): install the tenancy interceptor so\n // tenant-scoped tools are actually filtered, and so the entry-point gate\n // throws (rather than passing through) when no tenant is supplied. Without\n // this, a tenant set via SMRT_MCP_TENANT_ID would only set async context\n // with no interceptor to enforce it.\n enableTenancy();\n`\n : ''\n}\n // Load configuration from environment and .smrt.config files\n const appConfig = await config.load();\n const aiConfig = appConfig?.ai || {};\n\n if (DEBUG) {\n console.error(\\`[MCP] Loaded \\${TOOLS.length} static tools\\`);\n console.error(\\`[MCP] Available tools:\\`, TOOLS.map(t => t.name).join(', '));\n }\n\n // Create MCP server\n const server = new Server(\n {\n name: SERVER_NAME,\n version: SERVER_VERSION,\n },\n {\n capabilities: {\n tools: {},\n },\n }\n );\n\n // Register ListTools handler\n server.setRequestHandler(ListToolsRequestSchema, async (_request: ListToolsRequest) => {\n if (DEBUG) {\n console.error(\\`[MCP] ListTools request received\\`);\n }\n\n return {\n tools: TOOLS,\n };\n });\n\n // Register CallTool handler\n server.setRequestHandler(CallToolRequestSchema, async (request: CallToolRequest) => {\n const { name: toolName, arguments: args = {} } = request.params;\n\n if (DEBUG) {\n console.error(\\`[MCP] CallTool request: \\${toolName}\\`);\n console.error(\\`[MCP] Arguments:\\`, JSON.stringify(args, null, 2));\n }\n\n try {\n // Static switch statement for tool execution\n const runToolBody = async () => {\n switch (toolName) {\n${switchCases}\n\n default:\n throw new Error(\\`Unknown tool: \\${toolName}\\`);\n }\n };\n${\n hasTenantScoped\n ? `\n // Fail-closed tenant context for tenant-scoped tools (#1554).\n const [toolObject] = toolName.split('_');\n const result =\n toolObject && TENANT_SCOPED.has(toolObject.toLowerCase())\n ? await runTenantScopedEntryPoint(\n { tenantScoped: true, tenantId: MCP_TENANT_ID, allowCrossTenant: MCP_ALLOW_CROSS_TENANT, surface: 'MCP' },\n runToolBody,\n )\n : await runToolBody();`\n : `\n const result = await runToolBody();`\n}\n\n if (DEBUG) {\n console.error(\\`[MCP] Tool executed successfully: \\${toolName}\\`);\n }\n\n return result;\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n console.error(\\`[MCP] Tool execution failed: \\${toolName}\\`, error);\n\n return {\n content: [\n {\n type: 'text',\n text: \\`Error executing tool \\${toolName}: \\${errorMessage}\\`,\n },\n ],\n isError: true,\n };\n }\n });\n\n // Setup stdio transport\n const transport = new StdioServerTransport();\n\n // Connect server to transport\n await server.connect(transport);\n\n if (DEBUG) {\n console.error(\\`[MCP] Server connected via stdio transport\\`);\n console.error(\\`[MCP] Ready to receive requests\\`);\n }\n\n // Handle graceful shutdown\n process.on('SIGINT', async () => {\n if (DEBUG) {\n console.error(\\`[MCP] Received SIGINT, shutting down gracefully\\`);\n }\n await server.close();\n process.exit(0);\n });\n\n process.on('SIGTERM', async () => {\n if (DEBUG) {\n console.error(\\`[MCP] Received SIGTERM, shutting down gracefully\\`);\n }\n await server.close();\n process.exit(0);\n });\n } catch (error) {\n console.error('[MCP] Fatal error during server startup:', error);\n process.exit(1);\n }\n}\n\n// Start the server\nmain().catch((error) => {\n console.error('[MCP] Unhandled error:', error);\n process.exit(1);\n});\n`;\n}\n\n/**\n * Generate package.json script for running MCP server\n *\n * @param serverPath - Path to generated server file (relative to package root)\n * @returns Script command for package.json\n */\nexport function generateMCPScript(\n serverPath: string = 'dist/mcp-server.js',\n): string {\n return `node ${serverPath}`;\n}\n\n/**\n * Generate Claude Desktop configuration example\n *\n * @param serverName - Name for the MCP server\n * @param serverPath - Absolute path to server file\n * @returns Configuration object for claude_desktop_config.json\n */\nexport function generateClaudeConfig(\n serverName: string,\n serverPath: string,\n): object {\n return {\n mcpServers: {\n [serverName]: {\n command: 'node',\n args: [serverPath],\n },\n },\n };\n}\n\n/**\n * Generate README documentation for MCP server setup\n *\n * @param serverName - Name of the MCP server\n * @param serverPath - Path to the server file\n * @returns Markdown documentation\n */\nexport function generateMCPDocumentation(\n serverName: string,\n serverPath: string,\n): string {\n return `# MCP Server Setup\n\nThis project includes an auto-generated MCP (Model Context Protocol) server that exposes SMRT objects as tools for AI integration.\n\n## Quick Start\n\n### 1. Build the MCP Server\n\n\\`\\`\\`bash\nnpm run build\n\\`\\`\\`\n\nThis generates the MCP server at: \\`${serverPath}\\`\n\n### 2. Configure Claude Desktop\n\nAdd the following to your Claude Desktop configuration file:\n\n**macOS**: \\`~/.config/Claude/claude_desktop_config.json\\`\n**Windows**: \\`%APPDATA%\\\\Claude\\\\claude_desktop_config.json\\`\n\n\\`\\`\\`json\n{\n \"mcpServers\": {\n \"${serverName}\": {\n \"command\": \"node\",\n \"args\": [\"/absolute/path/to/${serverPath}\"]\n }\n }\n}\n\\`\\`\\`\n\nReplace \\`/absolute/path/to/\\` with the actual absolute path to your project directory.\n\n### 3. Restart Claude Desktop\n\nClose and reopen Claude Desktop to load the new MCP server.\n\n### 4. Test the Integration\n\nIn Claude Code, you can now use the auto-generated tools. For example:\n\n- \\`list_products\\` - List all products\n- \\`get_product\\` - Get a specific product by ID\n- \\`create_product\\` - Create a new product\n- And more...\n\n## Environment Variables\n\nThe MCP server supports optional environment variables:\n\n- \\`DATABASE_URL\\` - Database connection string\n\n**AI Provider Configuration (in priority order):**\n1. **Generic configuration** (supports any provider):\n - \\`SMRT_AI_PROVIDER\\` - Provider name (e.g., 'openai', 'anthropic', 'claude-cli', 'gemini')\n - \\`SMRT_AI_API_KEY\\` - API key for the provider\n - \\`SMRT_AI_MODEL\\` - Model to use (optional)\n\n2. **Provider-specific fallbacks**:\n - \\`OPENAI_API_KEY\\` - OpenAI API key (auto-detects provider as 'openai')\n - \\`ANTHROPIC_API_KEY\\` - Anthropic API key (auto-detects provider as 'anthropic')\n - \\`CLAUDE_API_KEY\\` + \\`CLAUDE_MODEL\\` - Claude CLI provider (defaults to 'sonnet')\n\n**Examples:**\n\\`\\`\\`bash\n# Using generic configuration (recommended)\nexport SMRT_AI_PROVIDER=claude-cli\nexport SMRT_AI_MODEL=sonnet\n\n# Using provider-specific configuration\nexport CLAUDE_API_KEY=your-key\nexport CLAUDE_MODEL=sonnet\n\n# Using OpenAI\nexport OPENAI_API_KEY=your-openai-key\n\\`\\`\\`\n\n## Troubleshooting\n\n### Server Not Appearing in Claude\n\n1. Check that the path in \\`claude_desktop_config.json\\` is absolute\n2. Verify the server file exists at the specified path\n3. Check Claude Desktop logs for errors\n\n### Tools Not Working\n\n1. Ensure your database is accessible (if using one)\n2. Check that SMRT objects are properly decorated with \\`@smrt()\\`\n3. Look for errors in the MCP server output\n\n### Debug Mode\n\nTo enable debug logging, set the \\`DEBUG\\` constant to \\`true\\` in the generated server file.\n\n## Generated Tools\n\nThe following tools are automatically generated from your SMRT objects:\n\n- **CRUD Operations**: \\`list_\\`, \\`get_\\`, \\`create_\\`, \\`update_\\`, \\`delete_\\` for each object type\n- **Custom Actions**: Any custom methods included in the \\`@smrt()\\` decorator configuration\n\nSee the SMRT object definitions for the complete list of available tools and their parameters.\n`;\n}\n"],"names":[],"mappings":"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"}
|