@specverse/engine-realize 3.5.3
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/assets/examples/09-api/ai-spec.yaml +194 -0
- package/assets/examples/09-api/converted.yaml +95 -0
- package/assets/examples/09-api/diagram-architecture.mmd +10 -0
- package/assets/examples/09-api/diagram-er.mmd +10 -0
- package/assets/examples/09-api/documentation.html +104 -0
- package/assets/examples/09-api/documentation.md +95 -0
- package/assets/examples/09-api/inferred-spec.yaml +420 -0
- package/assets/examples/09-api/openapi.json +61 -0
- package/assets/examples/10-api/README.md +216 -0
- package/assets/examples/10-api/ai-spec.yaml +194 -0
- package/assets/examples/10-api/converted.yaml +96 -0
- package/assets/examples/10-api/diagram-architecture.mmd +10 -0
- package/assets/examples/10-api/diagram-er.mmd +10 -0
- package/assets/examples/10-api/documentation.html +104 -0
- package/assets/examples/10-api/documentation.md +95 -0
- package/assets/examples/10-api/inferred-spec.yaml +7 -0
- package/assets/examples/10-api/metadata.yaml +89 -0
- package/assets/examples/10-api/openapi.json +61 -0
- package/assets/examples/10-api/package-integration-test.js +177 -0
- package/assets/examples/10-api/usage-example.js +323 -0
- package/assets/examples/10-api/usage-example.ts +363 -0
- package/assets/examples/10-api/workflow-test.js +113 -0
- package/assets/examples/manifests/01-simple-default-mappings.yaml +36 -0
- package/assets/examples/manifests/02-capability-mappings.yaml +55 -0
- package/assets/examples/manifests/03-hybrid-mappings.yaml +109 -0
- package/assets/examples/manifests/README.md +245 -0
- package/assets/examples/manifests/backend-only.yaml +43 -0
- package/assets/examples/manifests/blog-api.md +78 -0
- package/assets/examples/manifests/blog-api.specly +79 -0
- package/assets/examples/manifests/frontend-only.yaml +27 -0
- package/assets/examples/manifests/fullstack-app.yaml +44 -0
- package/assets/examples/manifests/fullstack-monorepo.yaml +62 -0
- package/assets/examples/validate-examples-with-expected-failures.cjs +328 -0
- package/assets/examples/validate-examples.cjs +225 -0
- package/assets/examples-decomposed/cloud-native-manifest.example.yaml +8 -0
- package/assets/examples-decomposed/cloud-native-manifest.md +379 -0
- package/assets/examples-decomposed/cloud-native-manifest.specly +60 -0
- package/assets/examples-decomposed/docker-compose-manifest.example.yaml +8 -0
- package/assets/examples-decomposed/docker-compose-manifest.md +326 -0
- package/assets/examples-decomposed/docker-compose-manifest.specly +40 -0
- package/assets/examples-decomposed/kubernetes-deployment-manifest.example.yaml +8 -0
- package/assets/examples-decomposed/kubernetes-deployment-manifest.md +237 -0
- package/assets/examples-decomposed/kubernetes-deployment-manifest.specly +41 -0
- package/assets/templates/README.md +559 -0
- package/assets/templates/TEMPLATE-ENHANCEMENTS-V33.md +462 -0
- package/assets/templates/backend-only/CLAUDE.md +73 -0
- package/assets/templates/backend-only/README.md +197 -0
- package/assets/templates/backend-only/deployments/README.md +149 -0
- package/assets/templates/backend-only/deployments/development.specly +53 -0
- package/assets/templates/backend-only/deployments/production.specly +87 -0
- package/assets/templates/backend-only/docs/README.md +50 -0
- package/assets/templates/backend-only/docs/api/README.md +7 -0
- package/assets/templates/backend-only/docs/diagrams/README.md +85 -0
- package/assets/templates/backend-only/docs/example-documentation-template.md +269 -0
- package/assets/templates/backend-only/docs/guides/README.md +15 -0
- package/assets/templates/backend-only/dot.env.example +18 -0
- package/assets/templates/backend-only/generated/README.md +56 -0
- package/assets/templates/backend-only/generated/code/integration-test.template.js +320 -0
- package/assets/templates/backend-only/generated/code/package.json.template +34 -0
- package/assets/templates/backend-only/generated/docs/README.md +49 -0
- package/assets/templates/backend-only/gitignore +54 -0
- package/assets/templates/backend-only/manifests/README.md +72 -0
- package/assets/templates/backend-only/manifests/docker-compose.specly +91 -0
- package/assets/templates/backend-only/manifests/implementation.yaml +100 -0
- package/assets/templates/backend-only/manifests/kubernetes.specly +140 -0
- package/assets/templates/backend-only/package.json +59 -0
- package/assets/templates/backend-only/scripts/test-all.sh +160 -0
- package/assets/templates/backend-only/scripts/test-generated-code.sh +165 -0
- package/assets/templates/backend-only/specs/main.specly +67 -0
- package/assets/templates/default/CLAUDE.md +141 -0
- package/assets/templates/default/README.md +404 -0
- package/assets/templates/default/deployments/README.md +149 -0
- package/assets/templates/default/deployments/development.specly +53 -0
- package/assets/templates/default/deployments/production.specly +87 -0
- package/assets/templates/default/docs/README.md +50 -0
- package/assets/templates/default/docs/api/README.md +7 -0
- package/assets/templates/default/docs/diagrams/README.md +85 -0
- package/assets/templates/default/docs/example-documentation-template.md +269 -0
- package/assets/templates/default/docs/guides/README.md +15 -0
- package/assets/templates/default/dot.env.example +18 -0
- package/assets/templates/default/generated/README.md +56 -0
- package/assets/templates/default/generated/code/integration-test.template.js +320 -0
- package/assets/templates/default/generated/code/package.json.template +34 -0
- package/assets/templates/default/generated/docs/README.md +49 -0
- package/assets/templates/default/gitignore +54 -0
- package/assets/templates/default/manifests/README.md +72 -0
- package/assets/templates/default/manifests/docker-compose.specly +91 -0
- package/assets/templates/default/manifests/implementation.yaml +176 -0
- package/assets/templates/default/manifests/kubernetes.specly +140 -0
- package/assets/templates/default/package.json +61 -0
- package/assets/templates/default/scripts/test-all.sh +160 -0
- package/assets/templates/default/scripts/test-generated-code.sh +165 -0
- package/assets/templates/default/specs/main.specly +67 -0
- package/assets/templates/frontend-only/CLAUDE.md +75 -0
- package/assets/templates/frontend-only/README.md +231 -0
- package/assets/templates/frontend-only/deployments/README.md +149 -0
- package/assets/templates/frontend-only/deployments/development.specly +53 -0
- package/assets/templates/frontend-only/deployments/production.specly +87 -0
- package/assets/templates/frontend-only/docs/README.md +50 -0
- package/assets/templates/frontend-only/docs/api/README.md +7 -0
- package/assets/templates/frontend-only/docs/diagrams/README.md +85 -0
- package/assets/templates/frontend-only/docs/example-documentation-template.md +269 -0
- package/assets/templates/frontend-only/docs/guides/README.md +15 -0
- package/assets/templates/frontend-only/dot.env.example +18 -0
- package/assets/templates/frontend-only/generated/README.md +56 -0
- package/assets/templates/frontend-only/generated/code/integration-test.template.js +320 -0
- package/assets/templates/frontend-only/generated/code/package.json.template +34 -0
- package/assets/templates/frontend-only/generated/docs/README.md +49 -0
- package/assets/templates/frontend-only/gitignore +54 -0
- package/assets/templates/frontend-only/manifests/README.md +72 -0
- package/assets/templates/frontend-only/manifests/docker-compose.specly +91 -0
- package/assets/templates/frontend-only/manifests/implementation.yaml +58 -0
- package/assets/templates/frontend-only/manifests/kubernetes.specly +140 -0
- package/assets/templates/frontend-only/package.json +59 -0
- package/assets/templates/frontend-only/scripts/test-all.sh +160 -0
- package/assets/templates/frontend-only/scripts/test-generated-code.sh +165 -0
- package/assets/templates/frontend-only/specs/main.specly +57 -0
- package/assets/templates/full-stack/AI-GUIDE.md +60 -0
- package/assets/templates/full-stack/CLAUDE.md +141 -0
- package/assets/templates/full-stack/README.md +382 -0
- package/assets/templates/full-stack/archive/AI-GUIDE-legacy.md +392 -0
- package/assets/templates/full-stack/deployments/README.md +149 -0
- package/assets/templates/full-stack/deployments/development.specly +53 -0
- package/assets/templates/full-stack/deployments/production.specly +87 -0
- package/assets/templates/full-stack/docs/README.md +51 -0
- package/assets/templates/full-stack/docs/api/README.md +7 -0
- package/assets/templates/full-stack/docs/diagrams/README.md +85 -0
- package/assets/templates/full-stack/docs/example-documentation-template.md +269 -0
- package/assets/templates/full-stack/docs/guides/README.md +15 -0
- package/assets/templates/full-stack/generated/README.md +56 -0
- package/assets/templates/full-stack/generated/code/integration-test.template.js +320 -0
- package/assets/templates/full-stack/generated/code/package.json.template +34 -0
- package/assets/templates/full-stack/generated/docs/README.md +49 -0
- package/assets/templates/full-stack/gitignore +54 -0
- package/assets/templates/full-stack/manifests/README.md +72 -0
- package/assets/templates/full-stack/manifests/docker-compose.specly +91 -0
- package/assets/templates/full-stack/manifests/implementation.yaml +155 -0
- package/assets/templates/full-stack/manifests/kubernetes.specly +140 -0
- package/assets/templates/full-stack/package.json +45 -0
- package/assets/templates/full-stack/scripts/test-all.sh +160 -0
- package/assets/templates/full-stack/scripts/test-generated-code.sh +165 -0
- package/assets/templates/full-stack/specs/example-v33.specly +297 -0
- package/assets/templates/full-stack/specs/main-simple.specly +73 -0
- package/assets/templates/full-stack/specs/main.specly +408 -0
- package/dist/engines/code-generator.d.ts +86 -0
- package/dist/engines/code-generator.d.ts.map +1 -0
- package/dist/engines/code-generator.js +159 -0
- package/dist/engines/code-generator.js.map +1 -0
- package/dist/engines/engine-registry.d.ts +94 -0
- package/dist/engines/engine-registry.d.ts.map +1 -0
- package/dist/engines/engine-registry.js +163 -0
- package/dist/engines/engine-registry.js.map +1 -0
- package/dist/engines/index.d.ts +10 -0
- package/dist/engines/index.d.ts.map +1 -0
- package/dist/engines/index.js +12 -0
- package/dist/engines/index.js.map +1 -0
- package/dist/engines/typescript-engine.d.ts +74 -0
- package/dist/engines/typescript-engine.d.ts.map +1 -0
- package/dist/engines/typescript-engine.js +288 -0
- package/dist/engines/typescript-engine.js.map +1 -0
- package/dist/generators/index.d.ts +11 -0
- package/dist/generators/index.d.ts.map +1 -0
- package/dist/generators/index.js +11 -0
- package/dist/generators/index.js.map +1 -0
- package/dist/index.d.ts +48 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +434 -0
- package/dist/index.js.map +1 -0
- package/dist/library/index.d.ts +12 -0
- package/dist/library/index.d.ts.map +1 -0
- package/dist/library/index.js +15 -0
- package/dist/library/index.js.map +1 -0
- package/dist/library/library.d.ts +132 -0
- package/dist/library/library.d.ts.map +1 -0
- package/dist/library/library.js +343 -0
- package/dist/library/library.js.map +1 -0
- package/dist/library/loader.d.ts +73 -0
- package/dist/library/loader.d.ts.map +1 -0
- package/dist/library/loader.js +150 -0
- package/dist/library/loader.js.map +1 -0
- package/dist/library/resolver.d.ts +104 -0
- package/dist/library/resolver.d.ts.map +1 -0
- package/dist/library/resolver.js +299 -0
- package/dist/library/resolver.js.map +1 -0
- package/dist/library/validator.d.ts +65 -0
- package/dist/library/validator.d.ts.map +1 -0
- package/dist/library/validator.js +203 -0
- package/dist/library/validator.js.map +1 -0
- package/dist/types/index.d.ts +7 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +7 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/instance-factory.d.ts +289 -0
- package/dist/types/instance-factory.d.ts.map +1 -0
- package/dist/types/instance-factory.js +8 -0
- package/dist/types/instance-factory.js.map +1 -0
- package/dist/types/unified-mappings.d.ts +163 -0
- package/dist/types/unified-mappings.d.ts.map +1 -0
- package/dist/types/unified-mappings.js +110 -0
- package/dist/types/unified-mappings.js.map +1 -0
- package/dist/utils/ai-spec-loader.d.ts +77 -0
- package/dist/utils/ai-spec-loader.d.ts.map +1 -0
- package/dist/utils/ai-spec-loader.js +138 -0
- package/dist/utils/ai-spec-loader.js.map +1 -0
- package/dist/utils/index.d.ts +9 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +9 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/manifest-loader.d.ts +107 -0
- package/dist/utils/manifest-loader.d.ts.map +1 -0
- package/dist/utils/manifest-loader.js +168 -0
- package/dist/utils/manifest-loader.js.map +1 -0
- package/dist/utils/mapping-migration.d.ts +53 -0
- package/dist/utils/mapping-migration.d.ts.map +1 -0
- package/dist/utils/mapping-migration.js +194 -0
- package/dist/utils/mapping-migration.js.map +1 -0
- package/libs/instance-factories/CURVED-INTERFACE.md +278 -0
- package/libs/instance-factories/README.md +433 -0
- package/libs/instance-factories/applications/generic-app.yaml +52 -0
- package/libs/instance-factories/applications/react-app.yaml +186 -0
- package/libs/instance-factories/applications/templates/generic/backend-env-generator.ts +31 -0
- package/libs/instance-factories/applications/templates/generic/backend-package-json-generator.ts +80 -0
- package/libs/instance-factories/applications/templates/generic/backend-tsconfig-generator.ts +69 -0
- package/libs/instance-factories/applications/templates/generic/main-generator.ts +308 -0
- package/libs/instance-factories/applications/templates/react/_view-components-source.ts +555 -0
- package/libs/instance-factories/applications/templates/react/api-client-generator.ts +436 -0
- package/libs/instance-factories/applications/templates/react/api-types-generator.ts +153 -0
- package/libs/instance-factories/applications/templates/react/app-tsx-generator.ts +94 -0
- package/libs/instance-factories/applications/templates/react/env-example-generator.ts +24 -0
- package/libs/instance-factories/applications/templates/react/field-helpers-generator.ts +106 -0
- package/libs/instance-factories/applications/templates/react/gitignore-generator.ts +38 -0
- package/libs/instance-factories/applications/templates/react/index-css-generator.ts +85 -0
- package/libs/instance-factories/applications/templates/react/index-html-generator.ts +30 -0
- package/libs/instance-factories/applications/templates/react/main-tsx-generator.ts +34 -0
- package/libs/instance-factories/applications/templates/react/package-json-generator.ts +54 -0
- package/libs/instance-factories/applications/templates/react/pattern-adapter-generator.ts +179 -0
- package/libs/instance-factories/applications/templates/react/react-pattern-adapter.tsx +1347 -0
- package/libs/instance-factories/applications/templates/react/relationship-field-generator.ts +150 -0
- package/libs/instance-factories/applications/templates/react/tailwind-adapter-generator.ts +704 -0
- package/libs/instance-factories/applications/templates/react/tailwind-adapter-wrapper-generator.ts +84 -0
- package/libs/instance-factories/applications/templates/react/tsconfig-generator.ts +35 -0
- package/libs/instance-factories/applications/templates/react/use-api-hooks-generator.ts +121 -0
- package/libs/instance-factories/applications/templates/react/view-dashboard-generator.ts +150 -0
- package/libs/instance-factories/applications/templates/react/view-detail-generator.ts +150 -0
- package/libs/instance-factories/applications/templates/react/view-form-generator.ts +362 -0
- package/libs/instance-factories/applications/templates/react/view-list-generator.ts +98 -0
- package/libs/instance-factories/applications/templates/react/view-router-generator.ts +89 -0
- package/libs/instance-factories/applications/templates/react/vite-config-generator.ts +49 -0
- package/libs/instance-factories/archived/fastify-prisma.yaml +104 -0
- package/libs/instance-factories/cli/commander-js.yaml +55 -0
- package/libs/instance-factories/cli/templates/commander/cli-entry-generator.d.ts +12 -0
- package/libs/instance-factories/cli/templates/commander/cli-entry-generator.d.ts.map +1 -0
- package/libs/instance-factories/cli/templates/commander/cli-entry-generator.js +115 -0
- package/libs/instance-factories/cli/templates/commander/cli-entry-generator.js.map +1 -0
- package/libs/instance-factories/cli/templates/commander/cli-entry-generator.ts +145 -0
- package/libs/instance-factories/cli/templates/commander/command-generator.d.ts +14 -0
- package/libs/instance-factories/cli/templates/commander/command-generator.d.ts.map +1 -0
- package/libs/instance-factories/cli/templates/commander/command-generator.js +182 -0
- package/libs/instance-factories/cli/templates/commander/command-generator.js.map +1 -0
- package/libs/instance-factories/cli/templates/commander/command-generator.ts +992 -0
- package/libs/instance-factories/communication/event-emitter.yaml +56 -0
- package/libs/instance-factories/communication/rabbitmq-events.yaml +87 -0
- package/libs/instance-factories/communication/templates/eventemitter/bus-generator.ts +93 -0
- package/libs/instance-factories/communication/templates/eventemitter/publisher-generator.ts +117 -0
- package/libs/instance-factories/communication/templates/eventemitter/subscriber-generator.ts +101 -0
- package/libs/instance-factories/controllers/fastify.yaml +127 -0
- package/libs/instance-factories/controllers/templates/fastify/meta-routes-generator.ts +103 -0
- package/libs/instance-factories/controllers/templates/fastify/routes-generator.ts +389 -0
- package/libs/instance-factories/controllers/templates/fastify/server-generator.ts +76 -0
- package/libs/instance-factories/infrastructure/docker-k8s.yaml +61 -0
- package/libs/instance-factories/infrastructure/templates/docker-k8s/infrastructure-generator.ts +46 -0
- package/libs/instance-factories/orms/prisma.yaml +89 -0
- package/libs/instance-factories/orms/templates/prisma/schema-generator.ts +563 -0
- package/libs/instance-factories/orms/templates/prisma/services-generator.ts +408 -0
- package/libs/instance-factories/scaffolding/generic-scaffold.yaml +65 -0
- package/libs/instance-factories/scaffolding/templates/generic/env-example-generator.ts +73 -0
- package/libs/instance-factories/scaffolding/templates/generic/env-generator.ts +85 -0
- package/libs/instance-factories/scaffolding/templates/generic/gitignore-generator.ts +69 -0
- package/libs/instance-factories/scaffolding/templates/generic/package-json-generator.ts +176 -0
- package/libs/instance-factories/scaffolding/templates/generic/readme-generator.ts +207 -0
- package/libs/instance-factories/scaffolding/templates/generic/tsconfig-generator.ts +78 -0
- package/libs/instance-factories/scaffolding/templates/generic/tsconfig-react-generator.ts +41 -0
- package/libs/instance-factories/sdks/python-sdk.yaml +66 -0
- package/libs/instance-factories/sdks/templates/python/sdk-generator.ts +50 -0
- package/libs/instance-factories/sdks/templates/typescript/sdk-generator.ts +49 -0
- package/libs/instance-factories/sdks/typescript-sdk.yaml +59 -0
- package/libs/instance-factories/services/prisma-services.yaml +71 -0
- package/libs/instance-factories/services/templates/prisma/behavior-generator.ts +303 -0
- package/libs/instance-factories/services/templates/prisma/controller-generator.ts +532 -0
- package/libs/instance-factories/services/templates/prisma/service-generator.ts +315 -0
- package/libs/instance-factories/shared/path-resolver.ts +111 -0
- package/libs/instance-factories/storage/mongodb.yaml +79 -0
- package/libs/instance-factories/storage/postgresql.yaml +75 -0
- package/libs/instance-factories/storage/redis.yaml +79 -0
- package/libs/instance-factories/storage/templates/mongodb/config-generator.ts +15 -0
- package/libs/instance-factories/storage/templates/mongodb/docker-generator.ts +18 -0
- package/libs/instance-factories/storage/templates/postgresql/config-generator.ts +54 -0
- package/libs/instance-factories/storage/templates/postgresql/docker-generator.ts +55 -0
- package/libs/instance-factories/storage/templates/redis/config-generator.ts +16 -0
- package/libs/instance-factories/storage/templates/redis/docker-generator.ts +18 -0
- package/libs/instance-factories/test-generation.ts +192 -0
- package/libs/instance-factories/testing/templates/vitest/tests-generator.ts +51 -0
- package/libs/instance-factories/testing/vitest-tests.yaml +63 -0
- package/libs/instance-factories/tools/templates/mcp/mcp-server-generator.ts +136 -0
- package/libs/instance-factories/tools/templates/mcp/static/docs/DEPLOYMENT_GUIDE.md +630 -0
- package/libs/instance-factories/tools/templates/mcp/static/docs/HYBRID_RESOURCE_SYSTEM.md +330 -0
- package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/EXTENSION_DEPLOYMENT.md +552 -0
- package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/LOCAL_DEPLOYMENT.md +164 -0
- package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/WEB_DEPLOYMENT.md +247 -0
- package/libs/instance-factories/tools/templates/mcp/static/package.json +92 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/build-enterprise.js +284 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/build-extension.js +139 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/build-local.js +74 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/build-web.js +156 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/copy-canonical-files.js +41 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/test-deployments.js +259 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/test-hybrid-resources.js +231 -0
- package/libs/instance-factories/tools/templates/mcp/static/scripts/test-hybrid-simple.js +196 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/controllers/MCPServerController.ts +293 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/events/EventEmitter.ts +90 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/index.ts +24 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/interfaces/ResourceProvider.ts +15 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/models/LibrarySuggestion.ts +106 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/models/SpecVerseResource.ts +75 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/server/mcp-server.ts +239 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/CLIProxyService.ts +1501 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/EmbeddedResourcesAdapter.ts +211 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/EntityModuleService.ts +308 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/HybridResourcesProvider.ts +210 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/LibraryToolsService.ts +356 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/OrchestratorBridge.ts +524 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/OrchestratorToolsService.ts +530 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/PromptToolsService.ts +594 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/services/ResourcesProviderService.ts +170 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/CLIProxyService.init.test.ts +544 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/CLIProxyService.test.ts +189 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/ResourcesProviderService.test.ts +89 -0
- package/libs/instance-factories/tools/templates/mcp/static/src/types/index.ts +110 -0
- package/libs/instance-factories/tools/templates/mcp/static/tsconfig.json +28 -0
- package/libs/instance-factories/tools/templates/vscode/static/extension.ts +1195 -0
- package/libs/instance-factories/tools/templates/vscode/static/language-configuration.json +34 -0
- package/libs/instance-factories/tools/templates/vscode/static/schemas/specverse-v3-schema.json +4279 -0
- package/libs/instance-factories/tools/templates/vscode/static/syntaxes/specverse.tmLanguage.json +138 -0
- package/libs/instance-factories/tools/templates/vscode/static/themes/README.md +74 -0
- package/libs/instance-factories/tools/templates/vscode/static/themes/complete-specverse-colors.json +122 -0
- package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-basic-theme.json +65 -0
- package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-complete-theme.json +123 -0
- package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-theme-colors.json +64 -0
- package/libs/instance-factories/tools/templates/vscode/vscode-extension-generator.ts +214 -0
- package/libs/instance-factories/validation/templates/zod/validation-generator.ts +46 -0
- package/libs/instance-factories/validation/zod.yaml +56 -0
- package/libs/instance-factories/views/index.d.ts +13 -0
- package/libs/instance-factories/views/index.d.ts.map +1 -0
- package/libs/instance-factories/views/index.js +18 -0
- package/libs/instance-factories/views/index.js.map +1 -0
- package/libs/instance-factories/views/index.ts +45 -0
- package/libs/instance-factories/views/react-components.yaml +129 -0
- package/libs/instance-factories/views/templates/ARCHITECTURE.md +198 -0
- package/libs/instance-factories/views/templates/react/adapters/antd-adapter.ts +869 -0
- package/libs/instance-factories/views/templates/react/adapters/mui-adapter.ts +953 -0
- package/libs/instance-factories/views/templates/react/adapters/shadcn-adapter.ts +806 -0
- package/libs/instance-factories/views/templates/react/app-generator.ts +55 -0
- package/libs/instance-factories/views/templates/react/components-generator.ts +391 -0
- package/libs/instance-factories/views/templates/react/forms-generator.ts +343 -0
- package/libs/instance-factories/views/templates/react/frontend-package-json-generator.ts +54 -0
- package/libs/instance-factories/views/templates/react/hooks-generator.ts +122 -0
- package/libs/instance-factories/views/templates/react/index-css-generator.ts +209 -0
- package/libs/instance-factories/views/templates/react/index-html-generator.ts +34 -0
- package/libs/instance-factories/views/templates/react/main-tsx-generator.ts +29 -0
- package/libs/instance-factories/views/templates/react/react-component-generator.d.ts +152 -0
- package/libs/instance-factories/views/templates/react/react-component-generator.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/react/react-component-generator.js +398 -0
- package/libs/instance-factories/views/templates/react/react-component-generator.js.map +1 -0
- package/libs/instance-factories/views/templates/react/react-component-generator.ts +533 -0
- package/libs/instance-factories/views/templates/react/router-generator.ts +197 -0
- package/libs/instance-factories/views/templates/react/router-generic-generator.ts +103 -0
- package/libs/instance-factories/views/templates/react/spec-json-generator.ts +17 -0
- package/libs/instance-factories/views/templates/react/types-generator.ts +76 -0
- package/libs/instance-factories/views/templates/react/views-metadata-generator.ts +42 -0
- package/libs/instance-factories/views/templates/react/vite-config-generator.ts +38 -0
- package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.js.map +1 -0
- package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.ts +474 -0
- package/libs/instance-factories/views/templates/shared/__tests__/composite-patterns.test.ts +242 -0
- package/libs/instance-factories/views/templates/shared/adapter-types.d.ts +77 -0
- package/libs/instance-factories/views/templates/shared/adapter-types.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/adapter-types.js +47 -0
- package/libs/instance-factories/views/templates/shared/adapter-types.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/adapter-types.ts +142 -0
- package/libs/instance-factories/views/templates/shared/atomic-components-registry.d.ts +63 -0
- package/libs/instance-factories/views/templates/shared/atomic-components-registry.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/atomic-components-registry.js +822 -0
- package/libs/instance-factories/views/templates/shared/atomic-components-registry.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/atomic-components-registry.ts +908 -0
- package/libs/instance-factories/views/templates/shared/base-generator.d.ts +247 -0
- package/libs/instance-factories/views/templates/shared/base-generator.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/base-generator.js +363 -0
- package/libs/instance-factories/views/templates/shared/base-generator.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/base-generator.ts +608 -0
- package/libs/instance-factories/views/templates/shared/component-metadata.d.ts +254 -0
- package/libs/instance-factories/views/templates/shared/component-metadata.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/component-metadata.js +602 -0
- package/libs/instance-factories/views/templates/shared/component-metadata.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/component-metadata.ts +803 -0
- package/libs/instance-factories/views/templates/shared/composite-pattern-types.ts +250 -0
- package/libs/instance-factories/views/templates/shared/composite-patterns.ts +535 -0
- package/libs/instance-factories/views/templates/shared/index.ts +68 -0
- package/libs/instance-factories/views/templates/shared/pattern-validator.ts +279 -0
- package/libs/instance-factories/views/templates/shared/property-mapper.d.ts +149 -0
- package/libs/instance-factories/views/templates/shared/property-mapper.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/property-mapper.js +580 -0
- package/libs/instance-factories/views/templates/shared/property-mapper.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/property-mapper.ts +700 -0
- package/libs/instance-factories/views/templates/shared/syntax-mapper.d.ts +143 -0
- package/libs/instance-factories/views/templates/shared/syntax-mapper.d.ts.map +1 -0
- package/libs/instance-factories/views/templates/shared/syntax-mapper.js +420 -0
- package/libs/instance-factories/views/templates/shared/syntax-mapper.js.map +1 -0
- package/libs/instance-factories/views/templates/shared/syntax-mapper.ts +539 -0
- package/package.json +42 -0
- package/schema/SPECVERSE-SCHEMA.json +4274 -0
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* React App.tsx Root Component Generator
|
|
3
|
+
*
|
|
4
|
+
* Generates the root App component with React Query and React Router setup
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import type { TemplateContext } from '@specverse/engine-realize';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Generate App.tsx root component
|
|
11
|
+
*/
|
|
12
|
+
export default function generateApp(context: TemplateContext): string {
|
|
13
|
+
const { spec } = context;
|
|
14
|
+
|
|
15
|
+
const componentName = spec?.metadata?.component || spec?.component?.name || 'App';
|
|
16
|
+
|
|
17
|
+
return `/**
|
|
18
|
+
* ${componentName} - Root Application Component
|
|
19
|
+
* Auto-generated by SpecVerse
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import { RouterProvider } from 'react-router-dom';
|
|
23
|
+
import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
|
24
|
+
import { router } from './router';
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* React Query client configuration
|
|
28
|
+
*/
|
|
29
|
+
const queryClient = new QueryClient({
|
|
30
|
+
defaultOptions: {
|
|
31
|
+
queries: {
|
|
32
|
+
staleTime: 1000 * 60 * 5, // 5 minutes
|
|
33
|
+
retry: 1,
|
|
34
|
+
refetchOnWindowFocus: false
|
|
35
|
+
},
|
|
36
|
+
mutations: {
|
|
37
|
+
retry: 1
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Root App Component
|
|
44
|
+
*/
|
|
45
|
+
export default function App() {
|
|
46
|
+
return (
|
|
47
|
+
<QueryClientProvider client={queryClient}>
|
|
48
|
+
<div className="app-container">
|
|
49
|
+
<RouterProvider router={router} />
|
|
50
|
+
</div>
|
|
51
|
+
</QueryClientProvider>
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
`;
|
|
55
|
+
}
|
|
@@ -0,0 +1,391 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* React Components Generator
|
|
3
|
+
*
|
|
4
|
+
* Generates direct React components from SpecVerse views.
|
|
5
|
+
* Components use actual model attributes from the spec — no runtime schema discovery.
|
|
6
|
+
* The pattern adapter is still shipped for optional dynamic rendering.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { TemplateContext } from '@specverse/engine-realize';
|
|
10
|
+
|
|
11
|
+
export default function generateReactComponent(context: TemplateContext): string {
|
|
12
|
+
const { view, model, spec } = context;
|
|
13
|
+
|
|
14
|
+
if (!view) {
|
|
15
|
+
throw new Error('View is required in template context');
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const componentName = view.name || `${model?.name || 'Unknown'}View`;
|
|
19
|
+
|
|
20
|
+
let modelName: string;
|
|
21
|
+
if (model?.name) { modelName = model.name; }
|
|
22
|
+
else if (Array.isArray(view.model)) { modelName = view.model[0]; }
|
|
23
|
+
else if (view.model) { modelName = view.model; }
|
|
24
|
+
else if (Array.isArray(view.modelReference)) { modelName = view.modelReference[0]; }
|
|
25
|
+
else { modelName = view.modelReference || 'Unknown'; }
|
|
26
|
+
|
|
27
|
+
const viewType = view.type || 'list';
|
|
28
|
+
const columns = getModelDisplayColumns(model);
|
|
29
|
+
const allAttrs = getModelAttributes(model);
|
|
30
|
+
const lowerModel = modelName.charAt(0).toLowerCase() + modelName.slice(1);
|
|
31
|
+
const pluralModel = `${lowerModel}s`;
|
|
32
|
+
const apiPath = `/api/${pluralModel}`;
|
|
33
|
+
|
|
34
|
+
switch (viewType) {
|
|
35
|
+
case 'list':
|
|
36
|
+
return generateListComponent(componentName, modelName, lowerModel, pluralModel, apiPath, columns, view, getBelongsToRelationships(model));
|
|
37
|
+
case 'detail':
|
|
38
|
+
return generateDetailComponent(componentName, modelName, lowerModel, pluralModel, apiPath, allAttrs, view);
|
|
39
|
+
case 'form':
|
|
40
|
+
return generateFormComponent(componentName, modelName, lowerModel, pluralModel, apiPath, allAttrs, view, getBelongsToRelationships(model));
|
|
41
|
+
case 'dashboard':
|
|
42
|
+
return generateDashboardComponent(componentName, modelName, lowerModel, pluralModel, apiPath, columns, view);
|
|
43
|
+
default:
|
|
44
|
+
return generateListComponent(componentName, modelName, lowerModel, pluralModel, apiPath, columns, view);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function getModelAttributes(model: any): Array<{ name: string; type: string; required: boolean }> {
|
|
49
|
+
if (!model?.attributes) return [];
|
|
50
|
+
const attrs = Array.isArray(model.attributes) ? model.attributes
|
|
51
|
+
: Object.entries(model.attributes).map(([name, def]: [string, any]) =>
|
|
52
|
+
typeof def === 'string' ? { name, type: def.split(' ')[0], required: def.includes('required') }
|
|
53
|
+
: { name, ...def });
|
|
54
|
+
return attrs.filter((a: any) => a.name !== 'id' && a.name !== 'createdAt' && a.name !== 'updatedAt');
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function getBelongsToRelationships(model: any): Array<{ name: string; target: string }> {
|
|
58
|
+
if (!model?.relationships) return [];
|
|
59
|
+
const rels = Array.isArray(model.relationships) ? model.relationships
|
|
60
|
+
: Object.entries(model.relationships).map(([name, def]: [string, any]) => ({ name, ...def }));
|
|
61
|
+
return rels.filter((r: any) => r.type === 'belongsTo').map((r: any) => ({
|
|
62
|
+
name: r.name,
|
|
63
|
+
target: r.target || r.targetModel || r.name,
|
|
64
|
+
}));
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function getModelDisplayColumns(model: any): string[] {
|
|
68
|
+
const attrs = getModelAttributes(model);
|
|
69
|
+
if (attrs.length === 0) return ['id'];
|
|
70
|
+
const display = attrs
|
|
71
|
+
.filter((a: any) => !['createdBy', 'updatedBy'].includes(a.name))
|
|
72
|
+
.slice(0, 6)
|
|
73
|
+
.map((a: any) => a.name);
|
|
74
|
+
return display.length > 0 ? display : ['id'];
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
function generateListComponent(name: string, model: string, lower: string, plural: string, api: string, columns: string[], view: any, belongsToRels: Array<{ name: string; target: string }> = []): string {
|
|
78
|
+
return `import { useState, useEffect } from 'react';
|
|
79
|
+
import { Link } from 'react-router-dom';
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* ${name}
|
|
83
|
+
* ${view.description || `List view for ${model}`}
|
|
84
|
+
*/
|
|
85
|
+
function formatCell(value: any): string {
|
|
86
|
+
if (value === null || value === undefined) return '—';
|
|
87
|
+
if (typeof value === 'string' && /^\\d{4}-\\d{2}-\\d{2}T/.test(value)) {
|
|
88
|
+
return new Date(value).toLocaleDateString('en-GB', { day: '2-digit', month: 'short', year: 'numeric' });
|
|
89
|
+
}
|
|
90
|
+
if (typeof value === 'boolean') return value ? 'Yes' : 'No';
|
|
91
|
+
return String(value);
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function ${name}() {
|
|
95
|
+
const [items, setItems] = useState<any[]>([]);
|
|
96
|
+
const [loading, setLoading] = useState(true);
|
|
97
|
+
|
|
98
|
+
useEffect(() => {
|
|
99
|
+
fetch('${api}')
|
|
100
|
+
.then(r => r.json())
|
|
101
|
+
.then(data => { setItems(data); setLoading(false); })
|
|
102
|
+
.catch(() => setLoading(false));
|
|
103
|
+
}, []);
|
|
104
|
+
|
|
105
|
+
if (loading) return <div className="p-6">Loading...</div>;
|
|
106
|
+
|
|
107
|
+
return (
|
|
108
|
+
<div className="p-6">
|
|
109
|
+
<div className="flex justify-between items-center mb-6">
|
|
110
|
+
<h1 className="text-2xl font-bold">${model}s</h1>
|
|
111
|
+
<Link to="/${lower}form" className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700">
|
|
112
|
+
+ New ${model}
|
|
113
|
+
</Link>
|
|
114
|
+
</div>
|
|
115
|
+
|
|
116
|
+
{items.length === 0 ? (
|
|
117
|
+
<div className="text-center py-12 text-gray-500">
|
|
118
|
+
<p className="text-lg">No ${lower}s yet</p>
|
|
119
|
+
<p className="text-sm mt-1">Create your first ${lower} to get started</p>
|
|
120
|
+
</div>
|
|
121
|
+
) : (
|
|
122
|
+
<div className="bg-white shadow rounded-lg overflow-hidden">
|
|
123
|
+
<table className="min-w-full divide-y divide-gray-200">
|
|
124
|
+
<thead className="bg-gray-50">
|
|
125
|
+
<tr>
|
|
126
|
+
${columns.map(c => ` <th className="px-4 py-3 text-left text-xs font-medium text-gray-500 uppercase whitespace-nowrap">${c}</th>`).join('\n')}
|
|
127
|
+
${belongsToRels.map(r => ` <th className="px-4 py-3 text-left text-xs font-medium text-gray-500 uppercase whitespace-nowrap">${r.target}</th>`).join('\n')}
|
|
128
|
+
<th className="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase">Actions</th>
|
|
129
|
+
</tr>
|
|
130
|
+
</thead>
|
|
131
|
+
<tbody className="divide-y divide-gray-200">
|
|
132
|
+
{items.map((item) => (
|
|
133
|
+
<tr key={item.id} className="hover:bg-gray-50">
|
|
134
|
+
${columns.map(c => ` <td className="px-4 py-3 text-sm text-gray-900 whitespace-nowrap">{formatCell(item.${c})}</td>`).join('\n')}
|
|
135
|
+
${belongsToRels.map(r => {
|
|
136
|
+
const targetLower = r.target.charAt(0).toLowerCase() + r.target.slice(1);
|
|
137
|
+
return ` <td className="px-4 py-3 text-sm whitespace-nowrap">
|
|
138
|
+
{item.${r.name} ? (
|
|
139
|
+
<Link to={\`/${targetLower}detail?id=\${item.${r.name}.id}\`} className="text-blue-600 hover:text-blue-800 hover:underline">
|
|
140
|
+
{item.${r.name}.name || item.${r.name}.title || item.${r.name}.guestName || item.${r.name}.id}
|
|
141
|
+
</Link>
|
|
142
|
+
) : '—'}
|
|
143
|
+
</td>`;
|
|
144
|
+
}).join('\n')}
|
|
145
|
+
<td className="px-6 py-4 text-right">
|
|
146
|
+
<Link to={\`/${lower}detail?id=\${item.id}\`} className="text-blue-600 hover:text-blue-800 mr-3">View</Link>
|
|
147
|
+
<Link to={\`/${lower}form?id=\${item.id}\`} className="text-green-600 hover:text-green-800">Edit</Link>
|
|
148
|
+
</td>
|
|
149
|
+
</tr>
|
|
150
|
+
))}
|
|
151
|
+
</tbody>
|
|
152
|
+
</table>
|
|
153
|
+
</div>
|
|
154
|
+
)}
|
|
155
|
+
</div>
|
|
156
|
+
);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export default ${name};
|
|
160
|
+
`;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
function generateDetailComponent(name: string, model: string, lower: string, plural: string, api: string, attrs: any[], view: any): string {
|
|
164
|
+
return `import { useState, useEffect } from 'react';
|
|
165
|
+
import { Link, useSearchParams } from 'react-router-dom';
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* ${name}
|
|
169
|
+
* ${view.description || `Detail view for ${model}`}
|
|
170
|
+
*/
|
|
171
|
+
function ${name}() {
|
|
172
|
+
const [searchParams] = useSearchParams();
|
|
173
|
+
const id = searchParams.get('id');
|
|
174
|
+
const [item, setItem] = useState<any>(null);
|
|
175
|
+
const [loading, setLoading] = useState(true);
|
|
176
|
+
|
|
177
|
+
useEffect(() => {
|
|
178
|
+
if (!id) { setLoading(false); return; }
|
|
179
|
+
fetch(\`${api}/\${id}\`)
|
|
180
|
+
.then(r => r.ok ? r.json() : null)
|
|
181
|
+
.then(data => { setItem(data); setLoading(false); })
|
|
182
|
+
.catch(() => setLoading(false));
|
|
183
|
+
}, [id]);
|
|
184
|
+
|
|
185
|
+
if (loading) return <div className="p-6">Loading...</div>;
|
|
186
|
+
if (!id) return <div className="p-6 text-gray-500">Select a ${lower} from the list to view details.</div>;
|
|
187
|
+
if (!item) return <div className="p-6 text-red-600">${model} not found.</div>;
|
|
188
|
+
|
|
189
|
+
return (
|
|
190
|
+
<div className="p-6">
|
|
191
|
+
<div className="flex justify-between items-center mb-6">
|
|
192
|
+
<h1 className="text-2xl font-bold">{item.name || item.title || '${model} ' + item.id}</h1>
|
|
193
|
+
<div className="space-x-2">
|
|
194
|
+
<Link to={\`/${lower}form?id=\${item.id}\`} className="px-4 py-2 bg-green-600 text-white rounded hover:bg-green-700">Edit</Link>
|
|
195
|
+
<Link to="/${lower}list" className="px-4 py-2 bg-gray-200 text-gray-700 rounded hover:bg-gray-300">Back</Link>
|
|
196
|
+
</div>
|
|
197
|
+
</div>
|
|
198
|
+
|
|
199
|
+
<div className="bg-white shadow rounded-lg p-6">
|
|
200
|
+
<dl className="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
201
|
+
<div>
|
|
202
|
+
<dt className="text-sm font-medium text-gray-500">ID</dt>
|
|
203
|
+
<dd className="mt-1 text-sm text-gray-900">{item.id}</dd>
|
|
204
|
+
</div>
|
|
205
|
+
${attrs.map(a => ` <div>
|
|
206
|
+
<dt className="text-sm font-medium text-gray-500">${a.name}</dt>
|
|
207
|
+
<dd className="mt-1 text-sm text-gray-900">{String(item.${a.name} ?? '—')}</dd>
|
|
208
|
+
</div>`).join('\n')}
|
|
209
|
+
</dl>
|
|
210
|
+
</div>
|
|
211
|
+
</div>
|
|
212
|
+
);
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
export default ${name};
|
|
216
|
+
`;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
function generateFormComponent(name: string, model: string, lower: string, plural: string, api: string, attrs: any[], view: any, belongsToRels: Array<{ name: string; target: string }> = []): string {
|
|
220
|
+
const editableAttrs = attrs.filter(a => !['createdAt', 'updatedAt', 'createdBy', 'updatedBy'].includes(a.name));
|
|
221
|
+
|
|
222
|
+
// Generate state and fetch for each belongsTo relationship
|
|
223
|
+
const relStateDefs = belongsToRels.map(r => {
|
|
224
|
+
const relLower = r.target.charAt(0).toLowerCase() + r.target.slice(1);
|
|
225
|
+
return `const [${relLower}Options, set${r.target}Options] = useState<any[]>([]);`;
|
|
226
|
+
}).join('\n ');
|
|
227
|
+
|
|
228
|
+
const relFetches = belongsToRels.map(r => {
|
|
229
|
+
const relLower = r.target.charAt(0).toLowerCase() + r.target.slice(1);
|
|
230
|
+
return `fetch('/api/${relLower}s').then(r => r.json()).then(d => set${r.target}Options(d)).catch(() => {});`;
|
|
231
|
+
}).join('\n ');
|
|
232
|
+
|
|
233
|
+
const relFields = belongsToRels.map(r => {
|
|
234
|
+
const fkField = `${r.name}Id`;
|
|
235
|
+
const relLower = r.target.charAt(0).toLowerCase() + r.target.slice(1);
|
|
236
|
+
return ` <div>
|
|
237
|
+
<label className="block text-sm font-medium text-gray-700 mb-1">${r.target}</label>
|
|
238
|
+
<select
|
|
239
|
+
value={form.${fkField} ?? ''}
|
|
240
|
+
onChange={e => setForm({...form, ${fkField}: e.target.value})}
|
|
241
|
+
className="w-full px-3 py-2 border rounded-md focus:ring-blue-500 focus:border-blue-500"
|
|
242
|
+
required
|
|
243
|
+
>
|
|
244
|
+
<option value="">Select ${r.target}...</option>
|
|
245
|
+
{${relLower}Options.map((opt: any) => (
|
|
246
|
+
<option key={opt.id} value={opt.id}>{opt.name || opt.title || opt.guestName || opt.id}</option>
|
|
247
|
+
))}
|
|
248
|
+
</select>
|
|
249
|
+
</div>`;
|
|
250
|
+
}).join('\n');
|
|
251
|
+
|
|
252
|
+
return `import { useState, useEffect } from 'react';
|
|
253
|
+
import { useNavigate, useSearchParams } from 'react-router-dom';
|
|
254
|
+
|
|
255
|
+
/**
|
|
256
|
+
* ${name}
|
|
257
|
+
* ${view.description || `Form for creating and editing ${model}`}
|
|
258
|
+
*/
|
|
259
|
+
function ${name}() {
|
|
260
|
+
const navigate = useNavigate();
|
|
261
|
+
const [searchParams] = useSearchParams();
|
|
262
|
+
const id = searchParams.get('id');
|
|
263
|
+
const [form, setForm] = useState<any>({});
|
|
264
|
+
const [loading, setLoading] = useState(!!id);
|
|
265
|
+
const [saving, setSaving] = useState(false);
|
|
266
|
+
const [error, setError] = useState<string | null>(null);
|
|
267
|
+
${relStateDefs}
|
|
268
|
+
|
|
269
|
+
useEffect(() => {
|
|
270
|
+
if (!id) return;
|
|
271
|
+
fetch(\`${api}/\${id}\`)
|
|
272
|
+
.then(r => r.json())
|
|
273
|
+
.then(data => { setForm(data); setLoading(false); })
|
|
274
|
+
.catch(() => setLoading(false));
|
|
275
|
+
}, [id]);
|
|
276
|
+
|
|
277
|
+
useEffect(() => {
|
|
278
|
+
${relFetches}
|
|
279
|
+
}, []);
|
|
280
|
+
|
|
281
|
+
const handleSubmit = async (e: React.FormEvent) => {
|
|
282
|
+
e.preventDefault();
|
|
283
|
+
setSaving(true);
|
|
284
|
+
setError(null);
|
|
285
|
+
try {
|
|
286
|
+
// Convert datetime-local values to full ISO-8601 for Prisma
|
|
287
|
+
const submitData = { ...form };
|
|
288
|
+
for (const [key, val] of Object.entries(submitData)) {
|
|
289
|
+
if (typeof val === 'string' && /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}$/.test(val)) {
|
|
290
|
+
submitData[key] = new Date(val).toISOString();
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
const method = id ? 'PUT' : 'POST';
|
|
294
|
+
const url = id ? \`${api}/\${id}\` : '${api}';
|
|
295
|
+
const res = await fetch(url, { method, headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(submitData) });
|
|
296
|
+
if (!res.ok) { const err = await res.json(); throw new Error(err.message || 'Save failed'); }
|
|
297
|
+
navigate('/${lower}list');
|
|
298
|
+
} catch (e: any) { setError(e.message); }
|
|
299
|
+
finally { setSaving(false); }
|
|
300
|
+
};
|
|
301
|
+
|
|
302
|
+
const handleDelete = async () => {
|
|
303
|
+
if (!id || !confirm('Delete this ${lower}?')) return;
|
|
304
|
+
await fetch(\`${api}/\${id}\`, { method: 'DELETE' });
|
|
305
|
+
navigate('/${lower}list');
|
|
306
|
+
};
|
|
307
|
+
|
|
308
|
+
if (loading) return <div className="p-6">Loading...</div>;
|
|
309
|
+
|
|
310
|
+
return (
|
|
311
|
+
<div className="p-6 max-w-2xl">
|
|
312
|
+
<h1 className="text-2xl font-bold mb-6">{id ? 'Edit' : 'New'} ${model}</h1>
|
|
313
|
+
|
|
314
|
+
{error && <div className="mb-4 p-3 bg-red-100 text-red-700 rounded">{error}</div>}
|
|
315
|
+
|
|
316
|
+
<form onSubmit={handleSubmit} className="bg-white shadow rounded-lg p-6 space-y-4">
|
|
317
|
+
${relFields}
|
|
318
|
+
${editableAttrs.map(a => ` <div>
|
|
319
|
+
<label className="block text-sm font-medium text-gray-700 mb-1">${a.name}</label>
|
|
320
|
+
<input
|
|
321
|
+
type="${a.type === 'Integer' || a.type === 'Number' || a.type === 'Decimal' ? 'number' : a.type === 'Date' || a.type === 'DateTime' ? 'datetime-local' : a.type === 'Email' ? 'email' : a.type === 'Boolean' ? 'checkbox' : 'text'}"
|
|
322
|
+
value={form.${a.name} ?? ''}
|
|
323
|
+
onChange={e => setForm({...form, ${a.name}: ${a.type === 'Integer' ? 'parseInt(e.target.value)' : a.type === 'Number' || a.type === 'Decimal' ? 'parseFloat(e.target.value)' : a.type === 'Boolean' ? 'e.target.checked' : 'e.target.value'}})}
|
|
324
|
+
className="w-full px-3 py-2 border rounded-md focus:ring-blue-500 focus:border-blue-500"
|
|
325
|
+
${a.required ? 'required' : ''}
|
|
326
|
+
/>
|
|
327
|
+
</div>`).join('\n')}
|
|
328
|
+
|
|
329
|
+
<div className="flex justify-between pt-4">
|
|
330
|
+
<div>
|
|
331
|
+
{id && <button type="button" onClick={handleDelete} className="px-4 py-2 bg-red-600 text-white rounded hover:bg-red-700">Delete</button>}
|
|
332
|
+
</div>
|
|
333
|
+
<div className="space-x-2">
|
|
334
|
+
<button type="button" onClick={() => navigate('/${lower}list')} className="px-4 py-2 bg-gray-200 text-gray-700 rounded hover:bg-gray-300">Cancel</button>
|
|
335
|
+
<button type="submit" disabled={saving} className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700 disabled:opacity-50">
|
|
336
|
+
{saving ? 'Saving...' : id ? 'Update' : 'Create'}
|
|
337
|
+
</button>
|
|
338
|
+
</div>
|
|
339
|
+
</div>
|
|
340
|
+
</form>
|
|
341
|
+
</div>
|
|
342
|
+
);
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
export default ${name};
|
|
346
|
+
`;
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
function generateDashboardComponent(name: string, model: string, lower: string, plural: string, api: string, columns: string[], view: any): string {
|
|
350
|
+
return `import { useState, useEffect } from 'react';
|
|
351
|
+
|
|
352
|
+
/**
|
|
353
|
+
* ${name}
|
|
354
|
+
* ${view.description || 'System dashboard'}
|
|
355
|
+
*/
|
|
356
|
+
function ${name}() {
|
|
357
|
+
const [stats, setStats] = useState<Record<string, number>>({});
|
|
358
|
+
const [loading, setLoading] = useState(true);
|
|
359
|
+
|
|
360
|
+
useEffect(() => {
|
|
361
|
+
// Fetch counts for all models
|
|
362
|
+
Promise.all([
|
|
363
|
+
${['guesthouses', 'rooms', 'guests', 'bookings', 'houses'].map(p => ` fetch('/api/${p}').then(r => r.ok ? r.json() : []).then(d => ['${p}', Array.isArray(d) ? d.length : 0]).catch(() => ['${p}', 0])`).join(',\n')}
|
|
364
|
+
]).then(results => {
|
|
365
|
+
const s: Record<string, number> = {};
|
|
366
|
+
results.forEach(([name, count]) => { if (count > 0 || name === '${plural}') s[name as string] = count as number; });
|
|
367
|
+
setStats(s);
|
|
368
|
+
setLoading(false);
|
|
369
|
+
});
|
|
370
|
+
}, []);
|
|
371
|
+
|
|
372
|
+
if (loading) return <div className="p-6">Loading dashboard...</div>;
|
|
373
|
+
|
|
374
|
+
return (
|
|
375
|
+
<div className="p-6">
|
|
376
|
+
<h1 className="text-2xl font-bold mb-6">Dashboard</h1>
|
|
377
|
+
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
|
|
378
|
+
{Object.entries(stats).map(([name, count]) => (
|
|
379
|
+
<div key={name} className="bg-white shadow rounded-lg p-6">
|
|
380
|
+
<h3 className="text-sm font-medium text-gray-500 uppercase">{name}</h3>
|
|
381
|
+
<p className="text-3xl font-bold mt-2">{count}</p>
|
|
382
|
+
</div>
|
|
383
|
+
))}
|
|
384
|
+
</div>
|
|
385
|
+
</div>
|
|
386
|
+
);
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
export default ${name};
|
|
390
|
+
`;
|
|
391
|
+
}
|