@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.
Files changed (420) hide show
  1. package/assets/examples/09-api/ai-spec.yaml +194 -0
  2. package/assets/examples/09-api/converted.yaml +95 -0
  3. package/assets/examples/09-api/diagram-architecture.mmd +10 -0
  4. package/assets/examples/09-api/diagram-er.mmd +10 -0
  5. package/assets/examples/09-api/documentation.html +104 -0
  6. package/assets/examples/09-api/documentation.md +95 -0
  7. package/assets/examples/09-api/inferred-spec.yaml +420 -0
  8. package/assets/examples/09-api/openapi.json +61 -0
  9. package/assets/examples/10-api/README.md +216 -0
  10. package/assets/examples/10-api/ai-spec.yaml +194 -0
  11. package/assets/examples/10-api/converted.yaml +96 -0
  12. package/assets/examples/10-api/diagram-architecture.mmd +10 -0
  13. package/assets/examples/10-api/diagram-er.mmd +10 -0
  14. package/assets/examples/10-api/documentation.html +104 -0
  15. package/assets/examples/10-api/documentation.md +95 -0
  16. package/assets/examples/10-api/inferred-spec.yaml +7 -0
  17. package/assets/examples/10-api/metadata.yaml +89 -0
  18. package/assets/examples/10-api/openapi.json +61 -0
  19. package/assets/examples/10-api/package-integration-test.js +177 -0
  20. package/assets/examples/10-api/usage-example.js +323 -0
  21. package/assets/examples/10-api/usage-example.ts +363 -0
  22. package/assets/examples/10-api/workflow-test.js +113 -0
  23. package/assets/examples/manifests/01-simple-default-mappings.yaml +36 -0
  24. package/assets/examples/manifests/02-capability-mappings.yaml +55 -0
  25. package/assets/examples/manifests/03-hybrid-mappings.yaml +109 -0
  26. package/assets/examples/manifests/README.md +245 -0
  27. package/assets/examples/manifests/backend-only.yaml +43 -0
  28. package/assets/examples/manifests/blog-api.md +78 -0
  29. package/assets/examples/manifests/blog-api.specly +79 -0
  30. package/assets/examples/manifests/frontend-only.yaml +27 -0
  31. package/assets/examples/manifests/fullstack-app.yaml +44 -0
  32. package/assets/examples/manifests/fullstack-monorepo.yaml +62 -0
  33. package/assets/examples/validate-examples-with-expected-failures.cjs +328 -0
  34. package/assets/examples/validate-examples.cjs +225 -0
  35. package/assets/examples-decomposed/cloud-native-manifest.example.yaml +8 -0
  36. package/assets/examples-decomposed/cloud-native-manifest.md +379 -0
  37. package/assets/examples-decomposed/cloud-native-manifest.specly +60 -0
  38. package/assets/examples-decomposed/docker-compose-manifest.example.yaml +8 -0
  39. package/assets/examples-decomposed/docker-compose-manifest.md +326 -0
  40. package/assets/examples-decomposed/docker-compose-manifest.specly +40 -0
  41. package/assets/examples-decomposed/kubernetes-deployment-manifest.example.yaml +8 -0
  42. package/assets/examples-decomposed/kubernetes-deployment-manifest.md +237 -0
  43. package/assets/examples-decomposed/kubernetes-deployment-manifest.specly +41 -0
  44. package/assets/templates/README.md +559 -0
  45. package/assets/templates/TEMPLATE-ENHANCEMENTS-V33.md +462 -0
  46. package/assets/templates/backend-only/CLAUDE.md +73 -0
  47. package/assets/templates/backend-only/README.md +197 -0
  48. package/assets/templates/backend-only/deployments/README.md +149 -0
  49. package/assets/templates/backend-only/deployments/development.specly +53 -0
  50. package/assets/templates/backend-only/deployments/production.specly +87 -0
  51. package/assets/templates/backend-only/docs/README.md +50 -0
  52. package/assets/templates/backend-only/docs/api/README.md +7 -0
  53. package/assets/templates/backend-only/docs/diagrams/README.md +85 -0
  54. package/assets/templates/backend-only/docs/example-documentation-template.md +269 -0
  55. package/assets/templates/backend-only/docs/guides/README.md +15 -0
  56. package/assets/templates/backend-only/dot.env.example +18 -0
  57. package/assets/templates/backend-only/generated/README.md +56 -0
  58. package/assets/templates/backend-only/generated/code/integration-test.template.js +320 -0
  59. package/assets/templates/backend-only/generated/code/package.json.template +34 -0
  60. package/assets/templates/backend-only/generated/docs/README.md +49 -0
  61. package/assets/templates/backend-only/gitignore +54 -0
  62. package/assets/templates/backend-only/manifests/README.md +72 -0
  63. package/assets/templates/backend-only/manifests/docker-compose.specly +91 -0
  64. package/assets/templates/backend-only/manifests/implementation.yaml +100 -0
  65. package/assets/templates/backend-only/manifests/kubernetes.specly +140 -0
  66. package/assets/templates/backend-only/package.json +59 -0
  67. package/assets/templates/backend-only/scripts/test-all.sh +160 -0
  68. package/assets/templates/backend-only/scripts/test-generated-code.sh +165 -0
  69. package/assets/templates/backend-only/specs/main.specly +67 -0
  70. package/assets/templates/default/CLAUDE.md +141 -0
  71. package/assets/templates/default/README.md +404 -0
  72. package/assets/templates/default/deployments/README.md +149 -0
  73. package/assets/templates/default/deployments/development.specly +53 -0
  74. package/assets/templates/default/deployments/production.specly +87 -0
  75. package/assets/templates/default/docs/README.md +50 -0
  76. package/assets/templates/default/docs/api/README.md +7 -0
  77. package/assets/templates/default/docs/diagrams/README.md +85 -0
  78. package/assets/templates/default/docs/example-documentation-template.md +269 -0
  79. package/assets/templates/default/docs/guides/README.md +15 -0
  80. package/assets/templates/default/dot.env.example +18 -0
  81. package/assets/templates/default/generated/README.md +56 -0
  82. package/assets/templates/default/generated/code/integration-test.template.js +320 -0
  83. package/assets/templates/default/generated/code/package.json.template +34 -0
  84. package/assets/templates/default/generated/docs/README.md +49 -0
  85. package/assets/templates/default/gitignore +54 -0
  86. package/assets/templates/default/manifests/README.md +72 -0
  87. package/assets/templates/default/manifests/docker-compose.specly +91 -0
  88. package/assets/templates/default/manifests/implementation.yaml +176 -0
  89. package/assets/templates/default/manifests/kubernetes.specly +140 -0
  90. package/assets/templates/default/package.json +61 -0
  91. package/assets/templates/default/scripts/test-all.sh +160 -0
  92. package/assets/templates/default/scripts/test-generated-code.sh +165 -0
  93. package/assets/templates/default/specs/main.specly +67 -0
  94. package/assets/templates/frontend-only/CLAUDE.md +75 -0
  95. package/assets/templates/frontend-only/README.md +231 -0
  96. package/assets/templates/frontend-only/deployments/README.md +149 -0
  97. package/assets/templates/frontend-only/deployments/development.specly +53 -0
  98. package/assets/templates/frontend-only/deployments/production.specly +87 -0
  99. package/assets/templates/frontend-only/docs/README.md +50 -0
  100. package/assets/templates/frontend-only/docs/api/README.md +7 -0
  101. package/assets/templates/frontend-only/docs/diagrams/README.md +85 -0
  102. package/assets/templates/frontend-only/docs/example-documentation-template.md +269 -0
  103. package/assets/templates/frontend-only/docs/guides/README.md +15 -0
  104. package/assets/templates/frontend-only/dot.env.example +18 -0
  105. package/assets/templates/frontend-only/generated/README.md +56 -0
  106. package/assets/templates/frontend-only/generated/code/integration-test.template.js +320 -0
  107. package/assets/templates/frontend-only/generated/code/package.json.template +34 -0
  108. package/assets/templates/frontend-only/generated/docs/README.md +49 -0
  109. package/assets/templates/frontend-only/gitignore +54 -0
  110. package/assets/templates/frontend-only/manifests/README.md +72 -0
  111. package/assets/templates/frontend-only/manifests/docker-compose.specly +91 -0
  112. package/assets/templates/frontend-only/manifests/implementation.yaml +58 -0
  113. package/assets/templates/frontend-only/manifests/kubernetes.specly +140 -0
  114. package/assets/templates/frontend-only/package.json +59 -0
  115. package/assets/templates/frontend-only/scripts/test-all.sh +160 -0
  116. package/assets/templates/frontend-only/scripts/test-generated-code.sh +165 -0
  117. package/assets/templates/frontend-only/specs/main.specly +57 -0
  118. package/assets/templates/full-stack/AI-GUIDE.md +60 -0
  119. package/assets/templates/full-stack/CLAUDE.md +141 -0
  120. package/assets/templates/full-stack/README.md +382 -0
  121. package/assets/templates/full-stack/archive/AI-GUIDE-legacy.md +392 -0
  122. package/assets/templates/full-stack/deployments/README.md +149 -0
  123. package/assets/templates/full-stack/deployments/development.specly +53 -0
  124. package/assets/templates/full-stack/deployments/production.specly +87 -0
  125. package/assets/templates/full-stack/docs/README.md +51 -0
  126. package/assets/templates/full-stack/docs/api/README.md +7 -0
  127. package/assets/templates/full-stack/docs/diagrams/README.md +85 -0
  128. package/assets/templates/full-stack/docs/example-documentation-template.md +269 -0
  129. package/assets/templates/full-stack/docs/guides/README.md +15 -0
  130. package/assets/templates/full-stack/generated/README.md +56 -0
  131. package/assets/templates/full-stack/generated/code/integration-test.template.js +320 -0
  132. package/assets/templates/full-stack/generated/code/package.json.template +34 -0
  133. package/assets/templates/full-stack/generated/docs/README.md +49 -0
  134. package/assets/templates/full-stack/gitignore +54 -0
  135. package/assets/templates/full-stack/manifests/README.md +72 -0
  136. package/assets/templates/full-stack/manifests/docker-compose.specly +91 -0
  137. package/assets/templates/full-stack/manifests/implementation.yaml +155 -0
  138. package/assets/templates/full-stack/manifests/kubernetes.specly +140 -0
  139. package/assets/templates/full-stack/package.json +45 -0
  140. package/assets/templates/full-stack/scripts/test-all.sh +160 -0
  141. package/assets/templates/full-stack/scripts/test-generated-code.sh +165 -0
  142. package/assets/templates/full-stack/specs/example-v33.specly +297 -0
  143. package/assets/templates/full-stack/specs/main-simple.specly +73 -0
  144. package/assets/templates/full-stack/specs/main.specly +408 -0
  145. package/dist/engines/code-generator.d.ts +86 -0
  146. package/dist/engines/code-generator.d.ts.map +1 -0
  147. package/dist/engines/code-generator.js +159 -0
  148. package/dist/engines/code-generator.js.map +1 -0
  149. package/dist/engines/engine-registry.d.ts +94 -0
  150. package/dist/engines/engine-registry.d.ts.map +1 -0
  151. package/dist/engines/engine-registry.js +163 -0
  152. package/dist/engines/engine-registry.js.map +1 -0
  153. package/dist/engines/index.d.ts +10 -0
  154. package/dist/engines/index.d.ts.map +1 -0
  155. package/dist/engines/index.js +12 -0
  156. package/dist/engines/index.js.map +1 -0
  157. package/dist/engines/typescript-engine.d.ts +74 -0
  158. package/dist/engines/typescript-engine.d.ts.map +1 -0
  159. package/dist/engines/typescript-engine.js +288 -0
  160. package/dist/engines/typescript-engine.js.map +1 -0
  161. package/dist/generators/index.d.ts +11 -0
  162. package/dist/generators/index.d.ts.map +1 -0
  163. package/dist/generators/index.js +11 -0
  164. package/dist/generators/index.js.map +1 -0
  165. package/dist/index.d.ts +48 -0
  166. package/dist/index.d.ts.map +1 -0
  167. package/dist/index.js +434 -0
  168. package/dist/index.js.map +1 -0
  169. package/dist/library/index.d.ts +12 -0
  170. package/dist/library/index.d.ts.map +1 -0
  171. package/dist/library/index.js +15 -0
  172. package/dist/library/index.js.map +1 -0
  173. package/dist/library/library.d.ts +132 -0
  174. package/dist/library/library.d.ts.map +1 -0
  175. package/dist/library/library.js +343 -0
  176. package/dist/library/library.js.map +1 -0
  177. package/dist/library/loader.d.ts +73 -0
  178. package/dist/library/loader.d.ts.map +1 -0
  179. package/dist/library/loader.js +150 -0
  180. package/dist/library/loader.js.map +1 -0
  181. package/dist/library/resolver.d.ts +104 -0
  182. package/dist/library/resolver.d.ts.map +1 -0
  183. package/dist/library/resolver.js +299 -0
  184. package/dist/library/resolver.js.map +1 -0
  185. package/dist/library/validator.d.ts +65 -0
  186. package/dist/library/validator.d.ts.map +1 -0
  187. package/dist/library/validator.js +203 -0
  188. package/dist/library/validator.js.map +1 -0
  189. package/dist/types/index.d.ts +7 -0
  190. package/dist/types/index.d.ts.map +1 -0
  191. package/dist/types/index.js +7 -0
  192. package/dist/types/index.js.map +1 -0
  193. package/dist/types/instance-factory.d.ts +289 -0
  194. package/dist/types/instance-factory.d.ts.map +1 -0
  195. package/dist/types/instance-factory.js +8 -0
  196. package/dist/types/instance-factory.js.map +1 -0
  197. package/dist/types/unified-mappings.d.ts +163 -0
  198. package/dist/types/unified-mappings.d.ts.map +1 -0
  199. package/dist/types/unified-mappings.js +110 -0
  200. package/dist/types/unified-mappings.js.map +1 -0
  201. package/dist/utils/ai-spec-loader.d.ts +77 -0
  202. package/dist/utils/ai-spec-loader.d.ts.map +1 -0
  203. package/dist/utils/ai-spec-loader.js +138 -0
  204. package/dist/utils/ai-spec-loader.js.map +1 -0
  205. package/dist/utils/index.d.ts +9 -0
  206. package/dist/utils/index.d.ts.map +1 -0
  207. package/dist/utils/index.js +9 -0
  208. package/dist/utils/index.js.map +1 -0
  209. package/dist/utils/manifest-loader.d.ts +107 -0
  210. package/dist/utils/manifest-loader.d.ts.map +1 -0
  211. package/dist/utils/manifest-loader.js +168 -0
  212. package/dist/utils/manifest-loader.js.map +1 -0
  213. package/dist/utils/mapping-migration.d.ts +53 -0
  214. package/dist/utils/mapping-migration.d.ts.map +1 -0
  215. package/dist/utils/mapping-migration.js +194 -0
  216. package/dist/utils/mapping-migration.js.map +1 -0
  217. package/libs/instance-factories/CURVED-INTERFACE.md +278 -0
  218. package/libs/instance-factories/README.md +433 -0
  219. package/libs/instance-factories/applications/generic-app.yaml +52 -0
  220. package/libs/instance-factories/applications/react-app.yaml +186 -0
  221. package/libs/instance-factories/applications/templates/generic/backend-env-generator.ts +31 -0
  222. package/libs/instance-factories/applications/templates/generic/backend-package-json-generator.ts +80 -0
  223. package/libs/instance-factories/applications/templates/generic/backend-tsconfig-generator.ts +69 -0
  224. package/libs/instance-factories/applications/templates/generic/main-generator.ts +308 -0
  225. package/libs/instance-factories/applications/templates/react/_view-components-source.ts +555 -0
  226. package/libs/instance-factories/applications/templates/react/api-client-generator.ts +436 -0
  227. package/libs/instance-factories/applications/templates/react/api-types-generator.ts +153 -0
  228. package/libs/instance-factories/applications/templates/react/app-tsx-generator.ts +94 -0
  229. package/libs/instance-factories/applications/templates/react/env-example-generator.ts +24 -0
  230. package/libs/instance-factories/applications/templates/react/field-helpers-generator.ts +106 -0
  231. package/libs/instance-factories/applications/templates/react/gitignore-generator.ts +38 -0
  232. package/libs/instance-factories/applications/templates/react/index-css-generator.ts +85 -0
  233. package/libs/instance-factories/applications/templates/react/index-html-generator.ts +30 -0
  234. package/libs/instance-factories/applications/templates/react/main-tsx-generator.ts +34 -0
  235. package/libs/instance-factories/applications/templates/react/package-json-generator.ts +54 -0
  236. package/libs/instance-factories/applications/templates/react/pattern-adapter-generator.ts +179 -0
  237. package/libs/instance-factories/applications/templates/react/react-pattern-adapter.tsx +1347 -0
  238. package/libs/instance-factories/applications/templates/react/relationship-field-generator.ts +150 -0
  239. package/libs/instance-factories/applications/templates/react/tailwind-adapter-generator.ts +704 -0
  240. package/libs/instance-factories/applications/templates/react/tailwind-adapter-wrapper-generator.ts +84 -0
  241. package/libs/instance-factories/applications/templates/react/tsconfig-generator.ts +35 -0
  242. package/libs/instance-factories/applications/templates/react/use-api-hooks-generator.ts +121 -0
  243. package/libs/instance-factories/applications/templates/react/view-dashboard-generator.ts +150 -0
  244. package/libs/instance-factories/applications/templates/react/view-detail-generator.ts +150 -0
  245. package/libs/instance-factories/applications/templates/react/view-form-generator.ts +362 -0
  246. package/libs/instance-factories/applications/templates/react/view-list-generator.ts +98 -0
  247. package/libs/instance-factories/applications/templates/react/view-router-generator.ts +89 -0
  248. package/libs/instance-factories/applications/templates/react/vite-config-generator.ts +49 -0
  249. package/libs/instance-factories/archived/fastify-prisma.yaml +104 -0
  250. package/libs/instance-factories/cli/commander-js.yaml +55 -0
  251. package/libs/instance-factories/cli/templates/commander/cli-entry-generator.d.ts +12 -0
  252. package/libs/instance-factories/cli/templates/commander/cli-entry-generator.d.ts.map +1 -0
  253. package/libs/instance-factories/cli/templates/commander/cli-entry-generator.js +115 -0
  254. package/libs/instance-factories/cli/templates/commander/cli-entry-generator.js.map +1 -0
  255. package/libs/instance-factories/cli/templates/commander/cli-entry-generator.ts +145 -0
  256. package/libs/instance-factories/cli/templates/commander/command-generator.d.ts +14 -0
  257. package/libs/instance-factories/cli/templates/commander/command-generator.d.ts.map +1 -0
  258. package/libs/instance-factories/cli/templates/commander/command-generator.js +182 -0
  259. package/libs/instance-factories/cli/templates/commander/command-generator.js.map +1 -0
  260. package/libs/instance-factories/cli/templates/commander/command-generator.ts +992 -0
  261. package/libs/instance-factories/communication/event-emitter.yaml +56 -0
  262. package/libs/instance-factories/communication/rabbitmq-events.yaml +87 -0
  263. package/libs/instance-factories/communication/templates/eventemitter/bus-generator.ts +93 -0
  264. package/libs/instance-factories/communication/templates/eventemitter/publisher-generator.ts +117 -0
  265. package/libs/instance-factories/communication/templates/eventemitter/subscriber-generator.ts +101 -0
  266. package/libs/instance-factories/controllers/fastify.yaml +127 -0
  267. package/libs/instance-factories/controllers/templates/fastify/meta-routes-generator.ts +103 -0
  268. package/libs/instance-factories/controllers/templates/fastify/routes-generator.ts +389 -0
  269. package/libs/instance-factories/controllers/templates/fastify/server-generator.ts +76 -0
  270. package/libs/instance-factories/infrastructure/docker-k8s.yaml +61 -0
  271. package/libs/instance-factories/infrastructure/templates/docker-k8s/infrastructure-generator.ts +46 -0
  272. package/libs/instance-factories/orms/prisma.yaml +89 -0
  273. package/libs/instance-factories/orms/templates/prisma/schema-generator.ts +563 -0
  274. package/libs/instance-factories/orms/templates/prisma/services-generator.ts +408 -0
  275. package/libs/instance-factories/scaffolding/generic-scaffold.yaml +65 -0
  276. package/libs/instance-factories/scaffolding/templates/generic/env-example-generator.ts +73 -0
  277. package/libs/instance-factories/scaffolding/templates/generic/env-generator.ts +85 -0
  278. package/libs/instance-factories/scaffolding/templates/generic/gitignore-generator.ts +69 -0
  279. package/libs/instance-factories/scaffolding/templates/generic/package-json-generator.ts +176 -0
  280. package/libs/instance-factories/scaffolding/templates/generic/readme-generator.ts +207 -0
  281. package/libs/instance-factories/scaffolding/templates/generic/tsconfig-generator.ts +78 -0
  282. package/libs/instance-factories/scaffolding/templates/generic/tsconfig-react-generator.ts +41 -0
  283. package/libs/instance-factories/sdks/python-sdk.yaml +66 -0
  284. package/libs/instance-factories/sdks/templates/python/sdk-generator.ts +50 -0
  285. package/libs/instance-factories/sdks/templates/typescript/sdk-generator.ts +49 -0
  286. package/libs/instance-factories/sdks/typescript-sdk.yaml +59 -0
  287. package/libs/instance-factories/services/prisma-services.yaml +71 -0
  288. package/libs/instance-factories/services/templates/prisma/behavior-generator.ts +303 -0
  289. package/libs/instance-factories/services/templates/prisma/controller-generator.ts +532 -0
  290. package/libs/instance-factories/services/templates/prisma/service-generator.ts +315 -0
  291. package/libs/instance-factories/shared/path-resolver.ts +111 -0
  292. package/libs/instance-factories/storage/mongodb.yaml +79 -0
  293. package/libs/instance-factories/storage/postgresql.yaml +75 -0
  294. package/libs/instance-factories/storage/redis.yaml +79 -0
  295. package/libs/instance-factories/storage/templates/mongodb/config-generator.ts +15 -0
  296. package/libs/instance-factories/storage/templates/mongodb/docker-generator.ts +18 -0
  297. package/libs/instance-factories/storage/templates/postgresql/config-generator.ts +54 -0
  298. package/libs/instance-factories/storage/templates/postgresql/docker-generator.ts +55 -0
  299. package/libs/instance-factories/storage/templates/redis/config-generator.ts +16 -0
  300. package/libs/instance-factories/storage/templates/redis/docker-generator.ts +18 -0
  301. package/libs/instance-factories/test-generation.ts +192 -0
  302. package/libs/instance-factories/testing/templates/vitest/tests-generator.ts +51 -0
  303. package/libs/instance-factories/testing/vitest-tests.yaml +63 -0
  304. package/libs/instance-factories/tools/templates/mcp/mcp-server-generator.ts +136 -0
  305. package/libs/instance-factories/tools/templates/mcp/static/docs/DEPLOYMENT_GUIDE.md +630 -0
  306. package/libs/instance-factories/tools/templates/mcp/static/docs/HYBRID_RESOURCE_SYSTEM.md +330 -0
  307. package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/EXTENSION_DEPLOYMENT.md +552 -0
  308. package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/LOCAL_DEPLOYMENT.md +164 -0
  309. package/libs/instance-factories/tools/templates/mcp/static/docs/deployments/WEB_DEPLOYMENT.md +247 -0
  310. package/libs/instance-factories/tools/templates/mcp/static/package.json +92 -0
  311. package/libs/instance-factories/tools/templates/mcp/static/scripts/build-enterprise.js +284 -0
  312. package/libs/instance-factories/tools/templates/mcp/static/scripts/build-extension.js +139 -0
  313. package/libs/instance-factories/tools/templates/mcp/static/scripts/build-local.js +74 -0
  314. package/libs/instance-factories/tools/templates/mcp/static/scripts/build-web.js +156 -0
  315. package/libs/instance-factories/tools/templates/mcp/static/scripts/copy-canonical-files.js +41 -0
  316. package/libs/instance-factories/tools/templates/mcp/static/scripts/test-deployments.js +259 -0
  317. package/libs/instance-factories/tools/templates/mcp/static/scripts/test-hybrid-resources.js +231 -0
  318. package/libs/instance-factories/tools/templates/mcp/static/scripts/test-hybrid-simple.js +196 -0
  319. package/libs/instance-factories/tools/templates/mcp/static/src/controllers/MCPServerController.ts +293 -0
  320. package/libs/instance-factories/tools/templates/mcp/static/src/events/EventEmitter.ts +90 -0
  321. package/libs/instance-factories/tools/templates/mcp/static/src/index.ts +24 -0
  322. package/libs/instance-factories/tools/templates/mcp/static/src/interfaces/ResourceProvider.ts +15 -0
  323. package/libs/instance-factories/tools/templates/mcp/static/src/models/LibrarySuggestion.ts +106 -0
  324. package/libs/instance-factories/tools/templates/mcp/static/src/models/SpecVerseResource.ts +75 -0
  325. package/libs/instance-factories/tools/templates/mcp/static/src/server/mcp-server.ts +239 -0
  326. package/libs/instance-factories/tools/templates/mcp/static/src/services/CLIProxyService.ts +1501 -0
  327. package/libs/instance-factories/tools/templates/mcp/static/src/services/EmbeddedResourcesAdapter.ts +211 -0
  328. package/libs/instance-factories/tools/templates/mcp/static/src/services/EntityModuleService.ts +308 -0
  329. package/libs/instance-factories/tools/templates/mcp/static/src/services/HybridResourcesProvider.ts +210 -0
  330. package/libs/instance-factories/tools/templates/mcp/static/src/services/LibraryToolsService.ts +356 -0
  331. package/libs/instance-factories/tools/templates/mcp/static/src/services/OrchestratorBridge.ts +524 -0
  332. package/libs/instance-factories/tools/templates/mcp/static/src/services/OrchestratorToolsService.ts +530 -0
  333. package/libs/instance-factories/tools/templates/mcp/static/src/services/PromptToolsService.ts +594 -0
  334. package/libs/instance-factories/tools/templates/mcp/static/src/services/ResourcesProviderService.ts +170 -0
  335. package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/CLIProxyService.init.test.ts +544 -0
  336. package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/CLIProxyService.test.ts +189 -0
  337. package/libs/instance-factories/tools/templates/mcp/static/src/tests/unit/ResourcesProviderService.test.ts +89 -0
  338. package/libs/instance-factories/tools/templates/mcp/static/src/types/index.ts +110 -0
  339. package/libs/instance-factories/tools/templates/mcp/static/tsconfig.json +28 -0
  340. package/libs/instance-factories/tools/templates/vscode/static/extension.ts +1195 -0
  341. package/libs/instance-factories/tools/templates/vscode/static/language-configuration.json +34 -0
  342. package/libs/instance-factories/tools/templates/vscode/static/schemas/specverse-v3-schema.json +4279 -0
  343. package/libs/instance-factories/tools/templates/vscode/static/syntaxes/specverse.tmLanguage.json +138 -0
  344. package/libs/instance-factories/tools/templates/vscode/static/themes/README.md +74 -0
  345. package/libs/instance-factories/tools/templates/vscode/static/themes/complete-specverse-colors.json +122 -0
  346. package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-basic-theme.json +65 -0
  347. package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-complete-theme.json +123 -0
  348. package/libs/instance-factories/tools/templates/vscode/static/themes/specverse-theme-colors.json +64 -0
  349. package/libs/instance-factories/tools/templates/vscode/vscode-extension-generator.ts +214 -0
  350. package/libs/instance-factories/validation/templates/zod/validation-generator.ts +46 -0
  351. package/libs/instance-factories/validation/zod.yaml +56 -0
  352. package/libs/instance-factories/views/index.d.ts +13 -0
  353. package/libs/instance-factories/views/index.d.ts.map +1 -0
  354. package/libs/instance-factories/views/index.js +18 -0
  355. package/libs/instance-factories/views/index.js.map +1 -0
  356. package/libs/instance-factories/views/index.ts +45 -0
  357. package/libs/instance-factories/views/react-components.yaml +129 -0
  358. package/libs/instance-factories/views/templates/ARCHITECTURE.md +198 -0
  359. package/libs/instance-factories/views/templates/react/adapters/antd-adapter.ts +869 -0
  360. package/libs/instance-factories/views/templates/react/adapters/mui-adapter.ts +953 -0
  361. package/libs/instance-factories/views/templates/react/adapters/shadcn-adapter.ts +806 -0
  362. package/libs/instance-factories/views/templates/react/app-generator.ts +55 -0
  363. package/libs/instance-factories/views/templates/react/components-generator.ts +391 -0
  364. package/libs/instance-factories/views/templates/react/forms-generator.ts +343 -0
  365. package/libs/instance-factories/views/templates/react/frontend-package-json-generator.ts +54 -0
  366. package/libs/instance-factories/views/templates/react/hooks-generator.ts +122 -0
  367. package/libs/instance-factories/views/templates/react/index-css-generator.ts +209 -0
  368. package/libs/instance-factories/views/templates/react/index-html-generator.ts +34 -0
  369. package/libs/instance-factories/views/templates/react/main-tsx-generator.ts +29 -0
  370. package/libs/instance-factories/views/templates/react/react-component-generator.d.ts +152 -0
  371. package/libs/instance-factories/views/templates/react/react-component-generator.d.ts.map +1 -0
  372. package/libs/instance-factories/views/templates/react/react-component-generator.js +398 -0
  373. package/libs/instance-factories/views/templates/react/react-component-generator.js.map +1 -0
  374. package/libs/instance-factories/views/templates/react/react-component-generator.ts +533 -0
  375. package/libs/instance-factories/views/templates/react/router-generator.ts +197 -0
  376. package/libs/instance-factories/views/templates/react/router-generic-generator.ts +103 -0
  377. package/libs/instance-factories/views/templates/react/spec-json-generator.ts +17 -0
  378. package/libs/instance-factories/views/templates/react/types-generator.ts +76 -0
  379. package/libs/instance-factories/views/templates/react/views-metadata-generator.ts +42 -0
  380. package/libs/instance-factories/views/templates/react/vite-config-generator.ts +38 -0
  381. package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.d.ts.map +1 -0
  382. package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.js.map +1 -0
  383. package/libs/instance-factories/views/templates/runtime/runtime-view-renderer.ts +474 -0
  384. package/libs/instance-factories/views/templates/shared/__tests__/composite-patterns.test.ts +242 -0
  385. package/libs/instance-factories/views/templates/shared/adapter-types.d.ts +77 -0
  386. package/libs/instance-factories/views/templates/shared/adapter-types.d.ts.map +1 -0
  387. package/libs/instance-factories/views/templates/shared/adapter-types.js +47 -0
  388. package/libs/instance-factories/views/templates/shared/adapter-types.js.map +1 -0
  389. package/libs/instance-factories/views/templates/shared/adapter-types.ts +142 -0
  390. package/libs/instance-factories/views/templates/shared/atomic-components-registry.d.ts +63 -0
  391. package/libs/instance-factories/views/templates/shared/atomic-components-registry.d.ts.map +1 -0
  392. package/libs/instance-factories/views/templates/shared/atomic-components-registry.js +822 -0
  393. package/libs/instance-factories/views/templates/shared/atomic-components-registry.js.map +1 -0
  394. package/libs/instance-factories/views/templates/shared/atomic-components-registry.ts +908 -0
  395. package/libs/instance-factories/views/templates/shared/base-generator.d.ts +247 -0
  396. package/libs/instance-factories/views/templates/shared/base-generator.d.ts.map +1 -0
  397. package/libs/instance-factories/views/templates/shared/base-generator.js +363 -0
  398. package/libs/instance-factories/views/templates/shared/base-generator.js.map +1 -0
  399. package/libs/instance-factories/views/templates/shared/base-generator.ts +608 -0
  400. package/libs/instance-factories/views/templates/shared/component-metadata.d.ts +254 -0
  401. package/libs/instance-factories/views/templates/shared/component-metadata.d.ts.map +1 -0
  402. package/libs/instance-factories/views/templates/shared/component-metadata.js +602 -0
  403. package/libs/instance-factories/views/templates/shared/component-metadata.js.map +1 -0
  404. package/libs/instance-factories/views/templates/shared/component-metadata.ts +803 -0
  405. package/libs/instance-factories/views/templates/shared/composite-pattern-types.ts +250 -0
  406. package/libs/instance-factories/views/templates/shared/composite-patterns.ts +535 -0
  407. package/libs/instance-factories/views/templates/shared/index.ts +68 -0
  408. package/libs/instance-factories/views/templates/shared/pattern-validator.ts +279 -0
  409. package/libs/instance-factories/views/templates/shared/property-mapper.d.ts +149 -0
  410. package/libs/instance-factories/views/templates/shared/property-mapper.d.ts.map +1 -0
  411. package/libs/instance-factories/views/templates/shared/property-mapper.js +580 -0
  412. package/libs/instance-factories/views/templates/shared/property-mapper.js.map +1 -0
  413. package/libs/instance-factories/views/templates/shared/property-mapper.ts +700 -0
  414. package/libs/instance-factories/views/templates/shared/syntax-mapper.d.ts +143 -0
  415. package/libs/instance-factories/views/templates/shared/syntax-mapper.d.ts.map +1 -0
  416. package/libs/instance-factories/views/templates/shared/syntax-mapper.js +420 -0
  417. package/libs/instance-factories/views/templates/shared/syntax-mapper.js.map +1 -0
  418. package/libs/instance-factories/views/templates/shared/syntax-mapper.ts +539 -0
  419. package/package.json +42 -0
  420. 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
+ }