swallowkit 1.0.0-beta.3 → 1.0.0-beta.31

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 (201) hide show
  1. package/LICENSE +21 -21
  2. package/README.ja.md +353 -215
  3. package/README.md +406 -216
  4. package/dist/__tests__/fixtures.d.ts +22 -0
  5. package/dist/__tests__/fixtures.d.ts.map +1 -0
  6. package/dist/__tests__/fixtures.js +146 -0
  7. package/dist/__tests__/fixtures.js.map +1 -0
  8. package/dist/cli/commands/add-auth.d.ts +10 -0
  9. package/dist/cli/commands/add-auth.d.ts.map +1 -0
  10. package/dist/cli/commands/add-auth.js +444 -0
  11. package/dist/cli/commands/add-auth.js.map +1 -0
  12. package/dist/cli/commands/add-connector.d.ts +20 -0
  13. package/dist/cli/commands/add-connector.d.ts.map +1 -0
  14. package/dist/cli/commands/add-connector.js +163 -0
  15. package/dist/cli/commands/add-connector.js.map +1 -0
  16. package/dist/cli/commands/create-model.d.ts +1 -4
  17. package/dist/cli/commands/create-model.d.ts.map +1 -1
  18. package/dist/cli/commands/create-model.js +21 -82
  19. package/dist/cli/commands/create-model.js.map +1 -1
  20. package/dist/cli/commands/dev-seeds.d.ts +57 -0
  21. package/dist/cli/commands/dev-seeds.d.ts.map +1 -0
  22. package/dist/cli/commands/dev-seeds.js +470 -0
  23. package/dist/cli/commands/dev-seeds.js.map +1 -0
  24. package/dist/cli/commands/dev.d.ts +33 -0
  25. package/dist/cli/commands/dev.d.ts.map +1 -1
  26. package/dist/cli/commands/dev.js +628 -146
  27. package/dist/cli/commands/dev.js.map +1 -1
  28. package/dist/cli/commands/index.d.ts +1 -0
  29. package/dist/cli/commands/index.d.ts.map +1 -1
  30. package/dist/cli/commands/index.js +3 -1
  31. package/dist/cli/commands/index.js.map +1 -1
  32. package/dist/cli/commands/init.d.ts +15 -0
  33. package/dist/cli/commands/init.d.ts.map +1 -1
  34. package/dist/cli/commands/init.js +2696 -1706
  35. package/dist/cli/commands/init.js.map +1 -1
  36. package/dist/cli/commands/scaffold.d.ts.map +1 -1
  37. package/dist/cli/commands/scaffold.js +448 -129
  38. package/dist/cli/commands/scaffold.js.map +1 -1
  39. package/dist/cli/index.d.ts +5 -1
  40. package/dist/cli/index.d.ts.map +1 -1
  41. package/dist/cli/index.js +200 -42
  42. package/dist/cli/index.js.map +1 -1
  43. package/dist/core/config.d.ts +8 -2
  44. package/dist/core/config.d.ts.map +1 -1
  45. package/dist/core/config.js +94 -5
  46. package/dist/core/config.js.map +1 -1
  47. package/dist/core/mock/connector-mock-server.d.ts +101 -0
  48. package/dist/core/mock/connector-mock-server.d.ts.map +1 -0
  49. package/dist/core/mock/connector-mock-server.js +480 -0
  50. package/dist/core/mock/connector-mock-server.js.map +1 -0
  51. package/dist/core/mock/zod-mock-generator.d.ts +14 -0
  52. package/dist/core/mock/zod-mock-generator.d.ts.map +1 -0
  53. package/dist/core/mock/zod-mock-generator.js +163 -0
  54. package/dist/core/mock/zod-mock-generator.js.map +1 -0
  55. package/dist/core/operations/create-model.d.ts +15 -0
  56. package/dist/core/operations/create-model.d.ts.map +1 -0
  57. package/dist/core/operations/create-model.js +171 -0
  58. package/dist/core/operations/create-model.js.map +1 -0
  59. package/dist/core/operations/runtime.d.ts +32 -0
  60. package/dist/core/operations/runtime.d.ts.map +1 -0
  61. package/dist/core/operations/runtime.js +225 -0
  62. package/dist/core/operations/runtime.js.map +1 -0
  63. package/dist/core/operations/scaffold-machine.d.ts +16 -0
  64. package/dist/core/operations/scaffold-machine.d.ts.map +1 -0
  65. package/dist/core/operations/scaffold-machine.js +63 -0
  66. package/dist/core/operations/scaffold-machine.js.map +1 -0
  67. package/dist/core/project/manifest.d.ts +92 -0
  68. package/dist/core/project/manifest.d.ts.map +1 -0
  69. package/dist/core/project/manifest.js +321 -0
  70. package/dist/core/project/manifest.js.map +1 -0
  71. package/dist/core/project/validation.d.ts +20 -0
  72. package/dist/core/project/validation.d.ts.map +1 -0
  73. package/dist/core/project/validation.js +209 -0
  74. package/dist/core/project/validation.js.map +1 -0
  75. package/dist/core/scaffold/auth-generator.d.ts +38 -0
  76. package/dist/core/scaffold/auth-generator.d.ts.map +1 -0
  77. package/dist/core/scaffold/auth-generator.js +1244 -0
  78. package/dist/core/scaffold/auth-generator.js.map +1 -0
  79. package/dist/core/scaffold/connector-functions-generator.d.ts +41 -0
  80. package/dist/core/scaffold/connector-functions-generator.d.ts.map +1 -0
  81. package/dist/core/scaffold/connector-functions-generator.js +1027 -0
  82. package/dist/core/scaffold/connector-functions-generator.js.map +1 -0
  83. package/dist/core/scaffold/functions-generator.d.ts +7 -1
  84. package/dist/core/scaffold/functions-generator.d.ts.map +1 -1
  85. package/dist/core/scaffold/functions-generator.js +920 -213
  86. package/dist/core/scaffold/functions-generator.js.map +1 -1
  87. package/dist/core/scaffold/model-parser.d.ts +20 -1
  88. package/dist/core/scaffold/model-parser.d.ts.map +1 -1
  89. package/dist/core/scaffold/model-parser.js +328 -135
  90. package/dist/core/scaffold/model-parser.js.map +1 -1
  91. package/dist/core/scaffold/native-schema-generator.d.ts +13 -0
  92. package/dist/core/scaffold/native-schema-generator.d.ts.map +1 -0
  93. package/dist/core/scaffold/native-schema-generator.js +677 -0
  94. package/dist/core/scaffold/native-schema-generator.js.map +1 -0
  95. package/dist/core/scaffold/nextjs-generator.d.ts +8 -0
  96. package/dist/core/scaffold/nextjs-generator.d.ts.map +1 -1
  97. package/dist/core/scaffold/nextjs-generator.js +314 -182
  98. package/dist/core/scaffold/nextjs-generator.js.map +1 -1
  99. package/dist/core/scaffold/openapi-generator.d.ts +3 -0
  100. package/dist/core/scaffold/openapi-generator.d.ts.map +1 -0
  101. package/dist/core/scaffold/openapi-generator.js +190 -0
  102. package/dist/core/scaffold/openapi-generator.js.map +1 -0
  103. package/dist/core/scaffold/ui-generator.d.ts +10 -4
  104. package/dist/core/scaffold/ui-generator.d.ts.map +1 -1
  105. package/dist/core/scaffold/ui-generator.js +768 -663
  106. package/dist/core/scaffold/ui-generator.js.map +1 -1
  107. package/dist/database/base-model.d.ts +3 -3
  108. package/dist/database/base-model.js +3 -3
  109. package/dist/index.d.ts +2 -2
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +2 -1
  112. package/dist/index.js.map +1 -1
  113. package/dist/machine/contracts.d.ts +16 -0
  114. package/dist/machine/contracts.d.ts.map +1 -0
  115. package/dist/machine/contracts.js +3 -0
  116. package/dist/machine/contracts.js.map +1 -0
  117. package/dist/machine/errors.d.ts +11 -0
  118. package/dist/machine/errors.d.ts.map +1 -0
  119. package/dist/machine/errors.js +34 -0
  120. package/dist/machine/errors.js.map +1 -0
  121. package/dist/machine/index.d.ts +3 -0
  122. package/dist/machine/index.d.ts.map +1 -0
  123. package/dist/machine/index.js +156 -0
  124. package/dist/machine/index.js.map +1 -0
  125. package/dist/mcp/index.d.ts +25 -0
  126. package/dist/mcp/index.d.ts.map +1 -0
  127. package/dist/mcp/index.js +184 -0
  128. package/dist/mcp/index.js.map +1 -0
  129. package/dist/types/index.d.ts +65 -0
  130. package/dist/types/index.d.ts.map +1 -1
  131. package/dist/utils/package-manager.d.ts +109 -0
  132. package/dist/utils/package-manager.d.ts.map +1 -0
  133. package/dist/utils/package-manager.js +215 -0
  134. package/dist/utils/package-manager.js.map +1 -0
  135. package/dist/utils/python-uv.d.ts +21 -0
  136. package/dist/utils/python-uv.d.ts.map +1 -0
  137. package/dist/utils/python-uv.js +111 -0
  138. package/dist/utils/python-uv.js.map +1 -0
  139. package/package.json +85 -73
  140. package/src/__tests__/__snapshots__/functions-generator.test.ts.snap +1139 -0
  141. package/src/__tests__/__snapshots__/nextjs-generator.test.ts.snap +194 -0
  142. package/src/__tests__/__snapshots__/ui-generator.test.ts.snap +532 -0
  143. package/src/__tests__/auth.test.ts +654 -0
  144. package/src/__tests__/config.test.ts +274 -0
  145. package/src/__tests__/connector-functions-generator.test.ts +288 -0
  146. package/src/__tests__/connector-mock-server.test.ts +439 -0
  147. package/src/__tests__/connector-model-bff.test.ts +162 -0
  148. package/src/__tests__/dev-seeds.test.ts +173 -0
  149. package/src/__tests__/dev.test.ts +252 -0
  150. package/src/__tests__/fixtures.ts +144 -0
  151. package/src/__tests__/functions-generator.test.ts +237 -0
  152. package/src/__tests__/init.test.ts +115 -0
  153. package/src/__tests__/machine.test.ts +251 -0
  154. package/src/__tests__/mcp.test.ts +117 -0
  155. package/src/__tests__/model-parser.test.ts +52 -0
  156. package/src/__tests__/nextjs-generator.test.ts +97 -0
  157. package/src/__tests__/openapi-generator.test.ts +43 -0
  158. package/src/__tests__/package-manager.test.ts +189 -0
  159. package/src/__tests__/python-uv.test.ts +48 -0
  160. package/src/__tests__/scaffold.test.ts +67 -0
  161. package/src/__tests__/string-utils.test.ts +75 -0
  162. package/src/__tests__/ui-generator.test.ts +144 -0
  163. package/src/__tests__/zod-mock-generator.test.ts +132 -0
  164. package/src/cli/commands/add-auth.ts +500 -0
  165. package/src/cli/commands/add-connector.ts +158 -0
  166. package/src/cli/commands/create-model.ts +62 -0
  167. package/src/cli/commands/dev-seeds.ts +614 -0
  168. package/src/cli/commands/dev.ts +1134 -0
  169. package/src/cli/commands/index.ts +9 -0
  170. package/src/cli/commands/init.ts +3480 -0
  171. package/src/cli/commands/provision.ts +193 -0
  172. package/src/cli/commands/scaffold.ts +1001 -0
  173. package/src/cli/index.ts +196 -0
  174. package/src/core/config.ts +312 -0
  175. package/src/core/mock/connector-mock-server.ts +555 -0
  176. package/src/core/mock/zod-mock-generator.ts +205 -0
  177. package/src/core/operations/create-model.ts +174 -0
  178. package/src/core/operations/runtime.ts +235 -0
  179. package/src/core/operations/scaffold-machine.ts +91 -0
  180. package/src/core/project/manifest.ts +402 -0
  181. package/src/core/project/validation.ts +229 -0
  182. package/src/core/scaffold/auth-generator.ts +1284 -0
  183. package/src/core/scaffold/connector-functions-generator.ts +1128 -0
  184. package/src/core/scaffold/functions-generator.ts +970 -0
  185. package/src/core/scaffold/model-parser.ts +841 -0
  186. package/src/core/scaffold/native-schema-generator.ts +798 -0
  187. package/src/core/scaffold/nextjs-generator.ts +370 -0
  188. package/src/core/scaffold/openapi-generator.ts +212 -0
  189. package/src/core/scaffold/ui-generator.ts +1061 -0
  190. package/src/database/base-model.ts +184 -0
  191. package/src/database/client.ts +140 -0
  192. package/src/database/repository.ts +104 -0
  193. package/src/database/runtime-check.ts +25 -0
  194. package/src/index.ts +27 -0
  195. package/src/machine/contracts.ts +17 -0
  196. package/src/machine/errors.ts +34 -0
  197. package/src/machine/index.ts +173 -0
  198. package/src/mcp/index.ts +185 -0
  199. package/src/types/index.ts +134 -0
  200. package/src/utils/package-manager.ts +229 -0
  201. package/src/utils/python-uv.ts +96 -0
@@ -40,11 +40,32 @@ Object.defineProperty(exports, "__esModule", { value: true });
40
40
  exports.scaffoldCommand = scaffoldCommand;
41
41
  const fs = __importStar(require("fs"));
42
42
  const path = __importStar(require("path"));
43
+ const child_process_1 = require("child_process");
44
+ const config_1 = require("../../core/config");
43
45
  const model_parser_1 = require("../../core/scaffold/model-parser");
44
46
  const functions_generator_1 = require("../../core/scaffold/functions-generator");
47
+ const connector_functions_generator_1 = require("../../core/scaffold/connector-functions-generator");
45
48
  const nextjs_generator_1 = require("../../core/scaffold/nextjs-generator");
46
49
  const ui_generator_1 = require("../../core/scaffold/ui-generator");
47
- const config_1 = require("../../core/config");
50
+ const native_schema_generator_1 = require("../../core/scaffold/native-schema-generator");
51
+ const package_manager_1 = require("../../utils/package-manager");
52
+ const manifest_1 = require("../../core/project/manifest");
53
+ function getMachineAwareStdio() {
54
+ return process.env.SWALLOWKIT_MACHINE_OUTPUT === "1" ? "pipe" : "inherit";
55
+ }
56
+ function runSpawnSyncCommand(command, args, cwd) {
57
+ const result = (0, child_process_1.spawnSync)(command, args, {
58
+ cwd,
59
+ stdio: getMachineAwareStdio(),
60
+ shell: true,
61
+ });
62
+ if (typeof result.status === "number" && result.status !== 0) {
63
+ throw new Error(`${command} ${args.join(" ")} exited with code ${result.status}`);
64
+ }
65
+ if (result.error) {
66
+ throw result.error;
67
+ }
68
+ }
48
69
  async function scaffoldCommand(options) {
49
70
  // SwallowKit プロジェクトディレクトリかどうかを検証
50
71
  (0, config_1.ensureSwallowKitProject)("scaffold");
@@ -57,6 +78,14 @@ async function scaffoldCommand(options) {
57
78
  console.log("🔍 Parsing model file...");
58
79
  const modelInfo = await (0, model_parser_1.parseModelFile)(modelPath);
59
80
  console.log(`✅ Model parsed: ${modelInfo.name} (${modelInfo.schemaName})`);
81
+ const backendLanguage = (0, config_1.getBackendLanguage)();
82
+ console.log(`🧠 Backend language: ${backendLanguage}`);
83
+ // コネクタモデルかどうかを判定
84
+ const isConnectorModel = !!modelInfo.connectorConfig;
85
+ if (isConnectorModel) {
86
+ console.log(`🔌 Connector model detected: ${modelInfo.connectorConfig.connector}`);
87
+ console.log(` Operations: ${modelInfo.connectorConfig.operations.join(", ")}`);
88
+ }
60
89
  // ネストスキーマ参照があれば表示
61
90
  if (modelInfo.nestedSchemaRefs.length > 0) {
62
91
  console.log(`🔗 Nested schema references detected:`);
@@ -73,36 +102,88 @@ async function scaffoldCommand(options) {
73
102
  // 4. Read shared package name
74
103
  const functionsDir = options.functionsDir || "functions";
75
104
  const sharedPackageName = readSharedPackageName();
105
+ const relatedModels = backendLanguage === "typescript"
106
+ ? [modelInfo]
107
+ : await collectModelGraph(modelPath);
76
108
  // 5. Generate BFF callFunction helper
77
109
  await generateCallFunctionHelper();
78
110
  // 6. Generate Azure Functions code
79
- await generateFunctionsCode(modelInfo, functionsDir, sharedPackageName);
111
+ if (isConnectorModel) {
112
+ await generateConnectorFunctionsCode(modelInfo, functionsDir, sharedPackageName, backendLanguage);
113
+ // 6b. Install connector driver dependencies
114
+ await installConnectorDriverDependencies(modelInfo, functionsDir, backendLanguage);
115
+ }
116
+ else {
117
+ await generateFunctionsCode(modelInfo, functionsDir, sharedPackageName, backendLanguage);
118
+ if (backendLanguage !== "typescript") {
119
+ await (0, native_schema_generator_1.generateLanguageSchemaArtifacts)(relatedModels, modelInfo, functionsDir, backendLanguage);
120
+ }
121
+ }
80
122
  // 7. Generate Next.js BFF API Routes
81
123
  const apiDir = options.apiDir || "app/api";
82
- await generateBFFRoutes(modelInfo, apiDir, sharedPackageName);
83
- // 8. Generate Cosmos DB container Bicep file
84
- await generateCosmosContainer(modelInfo);
124
+ if (isConnectorModel) {
125
+ await generateConnectorBFFRoutesFiles(modelInfo, apiDir, sharedPackageName);
126
+ }
127
+ else {
128
+ await generateBFFRoutes(modelInfo, apiDir, sharedPackageName);
129
+ }
130
+ // 8. Generate Cosmos DB container Bicep file (skip for connector models)
131
+ if (!isConnectorModel) {
132
+ await generateCosmosContainer(modelInfo);
133
+ }
85
134
  // 9. Generate UI components (unless --api-only)
86
135
  if (!options.apiOnly) {
87
- await generateUIComponents(modelInfo, sharedPackageName);
136
+ const uiAuthConfig = (0, config_1.getAuthConfig)();
137
+ const uiAuthPolicy = resolveAuthPolicy(modelInfo, uiAuthConfig);
138
+ const uiAuthOptions = uiAuthPolicy && uiAuthConfig && uiAuthConfig.provider !== 'none'
139
+ ? { authPolicy: uiAuthPolicy }
140
+ : undefined;
141
+ await generateUIComponents(modelInfo, sharedPackageName, uiAuthOptions);
88
142
  await updateNavigationMenu(modelInfo);
89
143
  }
144
+ await (0, manifest_1.syncProjectManifest)();
90
145
  console.log("\n✅ Scaffold completed successfully!");
91
146
  console.log("\n📝 Next steps:");
92
- console.log(` 1. Review generated files in ${functionsDir}/src/ and ${apiDir}/`);
147
+ console.log(` 1. Review generated files in ${describeFunctionsOutputPath(functionsDir, backendLanguage)} and ${apiDir}/`);
93
148
  if (!options.apiOnly) {
94
149
  console.log(` 2. Check the generated UI pages in app/${(0, model_parser_1.toKebabCase)(modelInfo.name)}/`);
95
150
  console.log(" 3. Navigate to the model from the homepage menu");
96
151
  }
97
- console.log(` ${options.apiOnly ? '2' : '4'}. Ensure FUNCTIONS_BASE_URL is set in your .env.local file`);
98
- console.log(` ${options.apiOnly ? '3' : '5'}. Configure CosmosDBConnection in functions/local.settings.json`);
99
- console.log(` ${options.apiOnly ? '4' : '6'}. Run 'npx swallowkit dev' to test the generated code`);
152
+ if (backendLanguage !== "typescript") {
153
+ console.log(` ${options.apiOnly ? "2" : "4"}. Review generated OpenAPI export and native schema assets in ${functionsDir}/openapi/ and ${functionsDir}/generated/`);
154
+ }
155
+ console.log(` ${options.apiOnly ? (backendLanguage === "typescript" ? "2" : "3") : (backendLanguage === "typescript" ? "4" : "5")}. Ensure BACKEND_FUNCTIONS_BASE_URL is set in your .env.local file`);
156
+ console.log(` ${options.apiOnly ? (backendLanguage === "typescript" ? "3" : "4") : (backendLanguage === "typescript" ? "5" : "6")}. Configure CosmosDBConnection in functions/local.settings.json`);
157
+ console.log(` ${options.apiOnly ? (backendLanguage === "typescript" ? "4" : "5") : (backendLanguage === "typescript" ? "6" : "7")}. Run '${(0, package_manager_1.getCommands)((0, package_manager_1.detectFromProject)()).dlx} swallowkit dev' to test the generated code`);
100
158
  }
101
159
  catch (error) {
102
160
  console.error("\n❌ Scaffold failed:", error.message);
103
161
  process.exit(1);
104
162
  }
105
163
  }
164
+ /**
165
+ * モデルの authPolicy と auth config から実効的な認可ポリシーを解決
166
+ * - モデルに authPolicy があればそれを使用
167
+ * - なければ auth.authorization.defaultPolicy に従う
168
+ * - auth 設定がなければ undefined(ガードなし)
169
+ */
170
+ function resolveAuthPolicy(modelInfo, authConfig) {
171
+ // モデルに明示的な authPolicy がある場合はそれを使用
172
+ if (modelInfo.authPolicy) {
173
+ return modelInfo.authPolicy;
174
+ }
175
+ // auth 設定がない場合はガードなし
176
+ if (!authConfig || authConfig.provider === 'none') {
177
+ return undefined;
178
+ }
179
+ // defaultPolicy が 'authenticated' なら認証のみ(ロール指定なし)のポリシーを返す
180
+ const defaultPolicy = authConfig.authorization?.defaultPolicy ?? 'authenticated';
181
+ if (defaultPolicy === 'authenticated') {
182
+ return {}; // 空のポリシー = 認証のみ、ロール制限なし
183
+ }
184
+ // defaultPolicy が 'anonymous' なら認証不要
185
+ return undefined;
186
+ }
106
187
  /**
107
188
  * モデルファイルのパスを解決
108
189
  */
@@ -152,7 +233,7 @@ function readSharedPackageName() {
152
233
  if (!fs.existsSync(sharedPkgPath)) {
153
234
  throw new Error("shared/package.json not found.\n" +
154
235
  "The shared package is required for model imports.\n" +
155
- 'Run "npx swallowkit init" to set up your project.');
236
+ `Run "${(0, package_manager_1.getCommands)((0, package_manager_1.detectFromProject)()).dlx} swallowkit init" to set up your project.`);
156
237
  }
157
238
  const pkg = JSON.parse(fs.readFileSync(sharedPkgPath, "utf-8"));
158
239
  return pkg.name;
@@ -168,34 +249,272 @@ async function generateCallFunctionHelper() {
168
249
  if (!fs.existsSync(helperDir)) {
169
250
  fs.mkdirSync(helperDir, { recursive: true });
170
251
  }
171
- const helperCode = (0, nextjs_generator_1.generateBFFCallFunction)();
252
+ // auth 設定がある場合は Authorization ヘッダー転送版を生成
253
+ const authConfig = (0, config_1.getAuthConfig)();
254
+ const hasAuth = authConfig && authConfig.provider !== 'none';
255
+ let helperCode;
256
+ if (hasAuth) {
257
+ const { generateBFFCallFunctionWithAuth } = await Promise.resolve().then(() => __importStar(require("../../core/scaffold/auth-generator")));
258
+ helperCode = generateBFFCallFunctionWithAuth();
259
+ }
260
+ else {
261
+ helperCode = (0, nextjs_generator_1.generateBFFCallFunction)();
262
+ }
172
263
  fs.writeFileSync(helperPath, helperCode, "utf-8");
173
264
  console.log(`✅ Created: ${helperPath}`);
174
265
  }
175
266
  /**
176
267
  * Generate Azure Functions CRUD code
177
268
  */
178
- async function generateFunctionsCode(modelInfo, functionsDir, sharedPackageName) {
269
+ async function generateFunctionsCode(modelInfo, functionsDir, sharedPackageName, backendLanguage) {
179
270
  console.log("\n🔨 Generating Azure Functions CRUD code...");
271
+ // Resolve auth policy: model-level authPolicy or global defaultPolicy
272
+ const authConfig = (0, config_1.getAuthConfig)();
273
+ const authPolicy = resolveAuthPolicy(modelInfo, authConfig);
274
+ if (authPolicy) {
275
+ console.log(`🔐 Auth policy detected: ${JSON.stringify(authPolicy)}`);
276
+ }
180
277
  const modelKebab = (0, model_parser_1.toKebabCase)(modelInfo.name);
181
- const functionFilePath = path.join(process.cwd(), functionsDir, "src", `${modelKebab}.ts`);
182
- // ディレクトリを作成
183
- const functionDir = path.dirname(functionFilePath);
184
- if (!fs.existsSync(functionDir)) {
185
- fs.mkdirSync(functionDir, { recursive: true });
278
+ if (backendLanguage === "typescript") {
279
+ const functionFilePath = path.join(process.cwd(), functionsDir, "src", `${modelKebab}.ts`);
280
+ const functionDir = path.dirname(functionFilePath);
281
+ if (!fs.existsSync(functionDir)) {
282
+ fs.mkdirSync(functionDir, { recursive: true });
283
+ }
284
+ const code = (0, functions_generator_1.generateCompactAzureFunctionsCRUD)(modelInfo, sharedPackageName, authPolicy);
285
+ fs.writeFileSync(functionFilePath, code, "utf-8");
286
+ console.log(`✅ Created: ${functionFilePath}`);
287
+ return;
186
288
  }
187
- // コードを生成
188
- const code = (0, functions_generator_1.generateCompactAzureFunctionsCRUD)(modelInfo, sharedPackageName);
189
- // ファイルに書き込み
190
- fs.writeFileSync(functionFilePath, code, "utf-8");
191
- console.log(`✅ Created: ${functionFilePath}`);
289
+ if (backendLanguage === "csharp") {
290
+ const crudDir = path.join(process.cwd(), functionsDir, "Crud");
291
+ const functionFilePath = path.join(crudDir, `${modelInfo.name}Functions.cs`);
292
+ fs.mkdirSync(crudDir, { recursive: true });
293
+ // Remove init-generated template (singular) to avoid route conflicts
294
+ const templatePath = path.join(crudDir, `${modelInfo.name}Function.cs`);
295
+ if (fs.existsSync(templatePath)) {
296
+ fs.unlinkSync(templatePath);
297
+ console.log(`🗑️ Removed template: ${templatePath}`);
298
+ }
299
+ fs.writeFileSync(functionFilePath, (0, functions_generator_1.generateCSharpAzureFunctionsCRUD)(modelInfo, authPolicy), "utf-8");
300
+ console.log(`✅ Created: ${functionFilePath}`);
301
+ return;
302
+ }
303
+ const blueprintsDir = path.join(process.cwd(), functionsDir, "blueprints");
304
+ const blueprintPath = path.join(blueprintsDir, `${modelKebab.replace(/-/g, "_")}.py`);
305
+ fs.mkdirSync(blueprintsDir, { recursive: true });
306
+ const { blueprint, registration } = (0, functions_generator_1.generatePythonAzureFunctionsCRUD)(modelInfo, authPolicy);
307
+ fs.writeFileSync(blueprintPath, blueprint, "utf-8");
308
+ updatePythonFunctionRegistrations(path.join(process.cwd(), functionsDir, "function_app.py"), registration);
309
+ console.log(`✅ Created: ${blueprintPath}`);
310
+ }
311
+ /**
312
+ * コネクタモデル用 Azure Functions コード生成
313
+ */
314
+ async function generateConnectorFunctionsCode(modelInfo, functionsDir, sharedPackageName, backendLanguage) {
315
+ console.log("\n🔌 Generating Connector Azure Functions code...");
316
+ const connectorConfig = modelInfo.connectorConfig;
317
+ const connectorDef = (0, config_1.getConnectorDefinition)(connectorConfig.connector);
318
+ if (!connectorDef) {
319
+ throw new Error(`Connector '${connectorConfig.connector}' not found in swallowkit.config.js.\n` +
320
+ ` Please add it to the 'connectors' section of your configuration.`);
321
+ }
322
+ // Resolve auth policy (same logic as Cosmos model scaffolding)
323
+ const authConfig = (0, config_1.getAuthConfig)();
324
+ const authPolicy = resolveAuthPolicy(modelInfo, authConfig);
325
+ if (authPolicy) {
326
+ console.log(`🔐 Auth policy detected: ${JSON.stringify(authPolicy)}`);
327
+ }
328
+ const modelKebab = (0, model_parser_1.toKebabCase)(modelInfo.name);
329
+ if (backendLanguage === "typescript") {
330
+ const functionFilePath = path.join(process.cwd(), functionsDir, "src", `${modelKebab}.ts`);
331
+ fs.mkdirSync(path.dirname(functionFilePath), { recursive: true });
332
+ let code;
333
+ if (connectorDef.type === "rdb") {
334
+ code = (0, connector_functions_generator_1.generateRdbConnectorFunctionTS)(modelInfo, sharedPackageName, connectorDef, connectorConfig, authPolicy);
335
+ }
336
+ else {
337
+ code = (0, connector_functions_generator_1.generateApiConnectorFunctionTS)(modelInfo, sharedPackageName, connectorDef, connectorConfig, authPolicy);
338
+ }
339
+ fs.writeFileSync(functionFilePath, code, "utf-8");
340
+ console.log(`✅ Created: ${functionFilePath}`);
341
+ return;
342
+ }
343
+ if (backendLanguage === "csharp") {
344
+ const functionFilePath = path.join(process.cwd(), functionsDir, "Connectors", `${modelInfo.name}ConnectorFunctions.cs`);
345
+ fs.mkdirSync(path.dirname(functionFilePath), { recursive: true });
346
+ let code;
347
+ if (connectorDef.type === "rdb") {
348
+ code = (0, connector_functions_generator_1.generateRdbConnectorFunctionCSharp)(modelInfo, connectorDef, connectorConfig);
349
+ }
350
+ else {
351
+ code = (0, connector_functions_generator_1.generateApiConnectorFunctionCSharp)(modelInfo, connectorDef, connectorConfig);
352
+ }
353
+ fs.writeFileSync(functionFilePath, code, "utf-8");
354
+ console.log(`✅ Created: ${functionFilePath}`);
355
+ return;
356
+ }
357
+ // Python
358
+ const blueprintsDir = path.join(process.cwd(), functionsDir, "blueprints");
359
+ const blueprintPath = path.join(blueprintsDir, `${modelKebab.replace(/-/g, "_")}.py`);
360
+ fs.mkdirSync(blueprintsDir, { recursive: true });
361
+ let result;
362
+ if (connectorDef.type === "rdb") {
363
+ result = (0, connector_functions_generator_1.generateRdbConnectorFunctionPython)(modelInfo, connectorDef, connectorConfig);
364
+ }
365
+ else {
366
+ result = (0, connector_functions_generator_1.generateApiConnectorFunctionPython)(modelInfo, connectorDef, connectorConfig);
367
+ }
368
+ fs.writeFileSync(blueprintPath, result.blueprint, "utf-8");
369
+ updatePythonFunctionRegistrations(path.join(process.cwd(), functionsDir, "function_app.py"), result.registration);
370
+ console.log(`✅ Created: ${blueprintPath}`);
371
+ }
372
+ /**
373
+ * コネクタモデルの scaffold 後に、生成コードが必要とする RDB/API ドライバを
374
+ * functions ディレクトリにインストールする。
375
+ */
376
+ async function installConnectorDriverDependencies(modelInfo, functionsDir, backendLanguage) {
377
+ const connectorConfig = modelInfo.connectorConfig;
378
+ const connectorDef = (0, config_1.getConnectorDefinition)(connectorConfig.connector);
379
+ if (!connectorDef || connectorDef.type !== "rdb")
380
+ return;
381
+ const rdbDef = connectorDef;
382
+ const functionsPath = path.join(process.cwd(), functionsDir);
383
+ if (backendLanguage === "typescript") {
384
+ const driverMap = {
385
+ mysql: { deps: ["mysql2"], devDeps: [] },
386
+ postgres: { deps: ["pg"], devDeps: ["@types/pg"] },
387
+ sqlserver: { deps: ["mssql"], devDeps: [] },
388
+ };
389
+ const entry = driverMap[rdbDef.provider];
390
+ if (!entry)
391
+ return;
392
+ // package.json を読んで、既にインストール済みならスキップ
393
+ const pkgJsonPath = path.join(functionsPath, "package.json");
394
+ if (fs.existsSync(pkgJsonPath)) {
395
+ const pkg = JSON.parse(fs.readFileSync(pkgJsonPath, "utf-8"));
396
+ const allDeps = { ...pkg.dependencies, ...pkg.devDependencies };
397
+ const missingDeps = entry.deps.filter(d => !allDeps[d]);
398
+ const missingDevDeps = entry.devDeps.filter(d => !allDeps[d]);
399
+ if (missingDeps.length === 0 && missingDevDeps.length === 0)
400
+ return;
401
+ }
402
+ console.log(`\n📦 Installing ${rdbDef.provider} driver dependencies...`);
403
+ const pm = (0, package_manager_1.detectFromProject)(functionsPath);
404
+ const cmds = (0, package_manager_1.getCommands)(pm);
405
+ if (entry.deps.length > 0) {
406
+ runSpawnSyncCommand(cmds.name, [pm === "pnpm" ? "add" : "install", ...entry.deps], functionsPath);
407
+ }
408
+ if (entry.devDeps.length > 0) {
409
+ runSpawnSyncCommand(cmds.name, [pm === "pnpm" ? "add" : "install", "-D", ...entry.devDeps], functionsPath);
410
+ }
411
+ console.log(`✅ ${rdbDef.provider} driver installed`);
412
+ return;
413
+ }
414
+ if (backendLanguage === "csharp") {
415
+ const nugetMap = {
416
+ mysql: "MySqlConnector",
417
+ postgres: "Npgsql",
418
+ sqlserver: "Microsoft.Data.SqlClient",
419
+ };
420
+ const pkg = nugetMap[rdbDef.provider];
421
+ if (!pkg)
422
+ return;
423
+ console.log(`\n📦 Installing ${rdbDef.provider} NuGet package...`);
424
+ runSpawnSyncCommand("dotnet", ["add", path.join(functionsPath, "functions.csproj"), "package", pkg], functionsPath);
425
+ console.log(`✅ ${pkg} installed`);
426
+ return;
427
+ }
428
+ // Python — requirements.txt に追記
429
+ const pipMap = {
430
+ mysql: "mysql-connector-python",
431
+ postgres: "psycopg2-binary",
432
+ sqlserver: "pymssql",
433
+ };
434
+ const pipPkg = pipMap[rdbDef.provider];
435
+ if (!pipPkg)
436
+ return;
437
+ const reqPath = path.join(functionsPath, "requirements.txt");
438
+ if (fs.existsSync(reqPath)) {
439
+ const existing = fs.readFileSync(reqPath, "utf-8");
440
+ if (!existing.includes(pipPkg)) {
441
+ console.log(`\n📦 Adding ${pipPkg} to requirements.txt...`);
442
+ fs.appendFileSync(reqPath, `\n${pipPkg}\n`);
443
+ console.log(`✅ ${pipPkg} added`);
444
+ }
445
+ }
446
+ }
447
+ /**
448
+ * コネクタモデル用 BFF ルート生成(操作制限に対応)
449
+ */
450
+ async function generateConnectorBFFRoutesFiles(modelInfo, apiDir, sharedPackageName) {
451
+ console.log("\n🔌 Generating Connector BFF API routes...");
452
+ const modelCamel = (0, model_parser_1.toCamelCase)(modelInfo.name);
453
+ const operations = modelInfo.connectorConfig.operations;
454
+ const listRoutePath = path.join(process.cwd(), apiDir, modelCamel, "route.ts");
455
+ const detailRoutePath = path.join(process.cwd(), apiDir, modelCamel, "[id]", "route.ts");
456
+ fs.mkdirSync(path.dirname(listRoutePath), { recursive: true });
457
+ fs.mkdirSync(path.dirname(detailRoutePath), { recursive: true });
458
+ const routes = (0, nextjs_generator_1.generateConnectorBFFRoutes)(modelInfo, sharedPackageName, operations);
459
+ fs.writeFileSync(listRoutePath, routes.listRoute, "utf-8");
460
+ fs.writeFileSync(detailRoutePath, routes.detailRoute, "utf-8");
461
+ console.log(`✅ Created: ${listRoutePath}`);
462
+ console.log(`✅ Created: ${detailRoutePath}`);
463
+ }
464
+ async function collectModelGraph(modelPath, seen = new Map()) {
465
+ const resolvedPath = path.resolve(modelPath);
466
+ if (seen.has(resolvedPath)) {
467
+ return Array.from(seen.values());
468
+ }
469
+ const modelInfo = await (0, model_parser_1.parseModelFile)(resolvedPath);
470
+ seen.set(resolvedPath, modelInfo);
471
+ for (const ref of modelInfo.nestedSchemaRefs) {
472
+ const nestedPath = resolveNestedModelPath(resolvedPath, ref.importPath);
473
+ await collectModelGraph(nestedPath, seen);
474
+ }
475
+ return Array.from(seen.values());
476
+ }
477
+ function resolveNestedModelPath(modelPath, importPath) {
478
+ let resolvedPath = path.resolve(path.dirname(modelPath), importPath);
479
+ if (!resolvedPath.endsWith(".ts")) {
480
+ resolvedPath += ".ts";
481
+ }
482
+ return resolvedPath;
483
+ }
484
+ function describeFunctionsOutputPath(functionsDir, backendLanguage) {
485
+ if (backendLanguage === "typescript") {
486
+ return `${functionsDir}/src/`;
487
+ }
488
+ if (backendLanguage === "csharp") {
489
+ return `${functionsDir}/Crud/`;
490
+ }
491
+ return `${functionsDir}/blueprints/`;
492
+ }
493
+ function updatePythonFunctionRegistrations(functionAppPath, registration) {
494
+ if (!fs.existsSync(functionAppPath)) {
495
+ throw new Error(`Python Functions entrypoint not found: ${functionAppPath}`);
496
+ }
497
+ const content = fs.readFileSync(functionAppPath, "utf-8");
498
+ // Check if import line already exists (handles init-generated layout)
499
+ const importLine = registration.split("\n").find((l) => l.startsWith("from ") || l.startsWith("import "));
500
+ if (importLine && content.includes(importLine)) {
501
+ return;
502
+ }
503
+ if (content.includes(registration)) {
504
+ return;
505
+ }
506
+ const marker = "# SwallowKit scaffold registrations";
507
+ if (!content.includes(marker)) {
508
+ throw new Error(`Could not find scaffold registration marker in ${functionAppPath}`);
509
+ }
510
+ const updated = content.replace(marker, `${registration}\n${marker}`);
511
+ fs.writeFileSync(functionAppPath, updated, "utf-8");
192
512
  }
193
513
  /**
194
514
  * Next.js BFF API Routes を生成
195
515
  */
196
516
  async function generateBFFRoutes(modelInfo, apiDir, sharedPackageName) {
197
517
  console.log("\n🔨 Generating Next.js BFF API routes...");
198
- const modelKebab = (0, model_parser_1.toKebabCase)(modelInfo.name);
199
518
  const modelCamel = modelInfo.name.charAt(0).toLowerCase() + modelInfo.name.slice(1);
200
519
  // List route: app/api/[model]/route.ts
201
520
  const listRoutePath = path.join(process.cwd(), apiDir, modelCamel, "route.ts");
@@ -240,70 +559,70 @@ async function generateCosmosContainer(modelInfo) {
240
559
  // Generate container Bicep file
241
560
  const containerFileName = `${modelKebab}-container.bicep`;
242
561
  const containerFilePath = path.join(containersDir, containerFileName);
243
- const bicepContent = `@description('Cosmos DB account name')
244
- param cosmosAccountName string
245
-
246
- @description('Database name')
247
- param databaseName string
248
-
249
- @description('Container name')
250
- param containerName string = '${modelPascal}s'
251
-
252
- @description('Partition key path')
253
- param partitionKeyPath string = '/id'
254
-
255
- @description('Throughput (RU/s) - only used for Free Tier')
256
- param throughput int = 400
257
-
258
- @description('Cosmos DB mode: freetier or serverless')
259
- param cosmosDbMode string
260
-
261
- // Reference existing Cosmos DB account
262
- resource cosmosAccount 'Microsoft.DocumentDB/databaseAccounts@2023-11-15' existing = {
263
- name: cosmosAccountName
264
- }
265
-
266
- // Reference existing database
267
- resource database 'Microsoft.DocumentDB/databaseAccounts/sqlDatabases@2023-11-15' existing = {
268
- parent: cosmosAccount
269
- name: databaseName
270
- }
271
-
272
- // Container for ${modelPascal}
273
- resource container 'Microsoft.DocumentDB/databaseAccounts/sqlDatabases/containers@2023-11-15' = {
274
- parent: database
275
- name: containerName
276
- properties: {
277
- resource: {
278
- id: containerName
279
- partitionKey: {
280
- paths: [
281
- partitionKeyPath
282
- ]
283
- kind: 'Hash'
284
- }
285
- indexingPolicy: {
286
- automatic: true
287
- indexingMode: 'consistent'
288
- includedPaths: [
289
- {
290
- path: '/*'
291
- }
292
- ]
293
- excludedPaths: [
294
- {
295
- path: '/_etag/?'
296
- }
297
- ]
298
- }
299
- }
300
- options: cosmosDbMode == 'freetier' ? {
301
- throughput: throughput
302
- } : {}
303
- }
304
- }
305
-
306
- output containerName string = container.name
562
+ const bicepContent = `@description('Cosmos DB account name')
563
+ param cosmosAccountName string
564
+
565
+ @description('Database name')
566
+ param databaseName string
567
+
568
+ @description('Container name')
569
+ param containerName string = '${modelPascal}s'
570
+
571
+ @description('Partition key path')
572
+ param partitionKeyPath string = '${modelInfo.partitionKey}'
573
+
574
+ @description('Throughput (RU/s) - only used for Free Tier')
575
+ param throughput int = 400
576
+
577
+ @description('Cosmos DB mode: freetier or serverless')
578
+ param cosmosDbMode string
579
+
580
+ // Reference existing Cosmos DB account
581
+ resource cosmosAccount 'Microsoft.DocumentDB/databaseAccounts@2023-11-15' existing = {
582
+ name: cosmosAccountName
583
+ }
584
+
585
+ // Reference existing database
586
+ resource database 'Microsoft.DocumentDB/databaseAccounts/sqlDatabases@2023-11-15' existing = {
587
+ parent: cosmosAccount
588
+ name: databaseName
589
+ }
590
+
591
+ // Container for ${modelPascal}
592
+ resource container 'Microsoft.DocumentDB/databaseAccounts/sqlDatabases/containers@2023-11-15' = {
593
+ parent: database
594
+ name: containerName
595
+ properties: {
596
+ resource: {
597
+ id: containerName
598
+ partitionKey: {
599
+ paths: [
600
+ partitionKeyPath
601
+ ]
602
+ kind: 'Hash'
603
+ }
604
+ indexingPolicy: {
605
+ automatic: true
606
+ indexingMode: 'consistent'
607
+ includedPaths: [
608
+ {
609
+ path: '/*'
610
+ }
611
+ ]
612
+ excludedPaths: [
613
+ {
614
+ path: '/_etag/?'
615
+ }
616
+ ]
617
+ }
618
+ }
619
+ options: cosmosDbMode == 'freetier' ? {
620
+ throughput: throughput
621
+ } : {}
622
+ }
623
+ }
624
+
625
+ output containerName string = container.name
307
626
  `;
308
627
  // Write Bicep file (overwrite if exists)
309
628
  fs.writeFileSync(containerFilePath, bicepContent, "utf-8");
@@ -334,37 +653,37 @@ async function updateMainBicepWithContainer(modelKebab, modelPascal) {
334
653
  const cosmosModuleMatch = mainBicepContent.match(cosmosModulePattern);
335
654
  if (!cosmosModuleMatch) {
336
655
  console.log("⚠️ Could not find Cosmos DB Serverless module in main.bicep. Please manually add the container module:");
337
- console.log(`\nmodule ${containerModuleName} 'containers/${modelKebab}-container.bicep' = {
338
- name: '${modelKebab}-container'
339
- params: {
340
- cosmosAccountName: cosmosDbMode == 'freetier' ? cosmosDbFreeTier.outputs.accountName : cosmosDbServerless.outputs.accountName
341
- databaseName: databaseName
342
- cosmosDbMode: cosmosDbMode
343
- }
344
- dependsOn: [
345
- cosmosDbFreeTier
346
- cosmosDbServerless
347
- ]
656
+ console.log(`\nmodule ${containerModuleName} 'containers/${modelKebab}-container.bicep' = {
657
+ name: '${modelKebab}-container'
658
+ params: {
659
+ cosmosAccountName: cosmosDbMode == 'freetier' ? cosmosDbFreeTier.outputs.accountName : cosmosDbServerless.outputs.accountName
660
+ databaseName: databaseName
661
+ cosmosDbMode: cosmosDbMode
662
+ }
663
+ dependsOn: [
664
+ cosmosDbFreeTier
665
+ cosmosDbServerless
666
+ ]
348
667
  }\n`);
349
668
  return;
350
669
  }
351
670
  // Find the end of the cosmosDbServerless module
352
671
  const insertPosition = cosmosModuleMatch.index + cosmosModuleMatch[0].length;
353
672
  // Create the container module declaration
354
- const containerModule = `
355
-
356
- // ${modelPascal} Container
357
- module ${containerModuleName} 'containers/${modelKebab}-container.bicep' = {
358
- name: '${modelKebab}-container'
359
- params: {
360
- cosmosAccountName: cosmosDbMode == 'freetier' ? cosmosDbFreeTier.outputs.accountName : cosmosDbServerless.outputs.accountName
361
- databaseName: '$` + `{projectName}Database'
362
- cosmosDbMode: cosmosDbMode
363
- }
364
- dependsOn: [
365
- cosmosDbFreeTier
366
- cosmosDbServerless
367
- ]
673
+ const containerModule = `
674
+
675
+ // ${modelPascal} Container
676
+ module ${containerModuleName} 'containers/${modelKebab}-container.bicep' = {
677
+ name: '${modelKebab}-container'
678
+ params: {
679
+ cosmosAccountName: cosmosDbMode == 'freetier' ? cosmosDbFreeTier.outputs.accountName : cosmosDbServerless.outputs.accountName
680
+ databaseName: '$` + `{projectName}Database'
681
+ cosmosDbMode: cosmosDbMode
682
+ }
683
+ dependsOn: [
684
+ cosmosDbFreeTier
685
+ cosmosDbServerless
686
+ ]
368
687
  }`;
369
688
  // Insert the module
370
689
  mainBicepContent =
@@ -377,7 +696,7 @@ module ${containerModuleName} 'containers/${modelKebab}-container.bicep' = {
377
696
  /**
378
697
  * Generate UI components (list, detail, form, create, edit pages)
379
698
  */
380
- async function generateUIComponents(modelInfo, sharedPackageName) {
699
+ async function generateUIComponents(modelInfo, sharedPackageName, authOptions) {
381
700
  console.log("\n🎨 Generating UI components...");
382
701
  const modelKebab = (0, model_parser_1.toKebabCase)(modelInfo.name);
383
702
  const modelName = modelInfo.name;
@@ -395,11 +714,11 @@ async function generateUIComponents(modelInfo, sharedPackageName) {
395
714
  }
396
715
  });
397
716
  // Generate and write files
398
- const listPage = (0, ui_generator_1.generateListPage)(modelInfo, sharedPackageName);
399
- const detailPage = (0, ui_generator_1.generateDetailPage)(modelInfo, sharedPackageName);
717
+ const listPage = (0, ui_generator_1.generateListPage)(modelInfo, sharedPackageName, authOptions);
718
+ const detailPage = (0, ui_generator_1.generateDetailPage)(modelInfo, sharedPackageName, authOptions);
400
719
  const formComponent = (0, ui_generator_1.generateFormComponent)(modelInfo, sharedPackageName);
401
- const newPage = (0, ui_generator_1.generateNewPage)(modelInfo);
402
- const editPage = (0, ui_generator_1.generateEditPage)(modelInfo, sharedPackageName);
720
+ const newPage = (0, ui_generator_1.generateNewPage)(modelInfo, authOptions);
721
+ const editPage = (0, ui_generator_1.generateEditPage)(modelInfo, sharedPackageName, authOptions);
403
722
  fs.writeFileSync(path.join(modelDir, "page.tsx"), listPage, "utf-8");
404
723
  fs.writeFileSync(path.join(idDir, "page.tsx"), detailPage, "utf-8");
405
724
  fs.writeFileSync(path.join(componentsDir, `${modelName}Form.tsx`), formComponent, "utf-8");
@@ -454,17 +773,17 @@ async function updateNavigationMenu(modelInfo) {
454
773
  label: modelInfo.displayName
455
774
  });
456
775
  // Generate new scaffold-config.ts content
457
- const newConfigContent = `export interface ScaffoldModel {
458
- name: string;
459
- path: string;
460
- label: string;
461
- }
462
-
463
- export const scaffoldConfig = {
464
- models: [
465
- ${models.map(m => ` { name: '${m.name}', path: '${m.path}', label: '${m.label}' },`).join('\n')}
466
- ] as ScaffoldModel[]
467
- };
776
+ const newConfigContent = `export interface ScaffoldModel {
777
+ name: string;
778
+ path: string;
779
+ label: string;
780
+ }
781
+
782
+ export const scaffoldConfig = {
783
+ models: [
784
+ ${models.map(m => ` { name: '${m.name}', path: '${m.path}', label: '${m.label}' },`).join('\n')}
785
+ ] as ScaffoldModel[]
786
+ };
468
787
  `;
469
788
  fs.writeFileSync(configPath, newConfigContent, "utf-8");
470
789
  console.log(`✅ Added ${modelInfo.name} to navigation menu`);