swallowkit 1.0.0-beta.3 → 1.0.0-beta.30
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -21
- package/README.ja.md +321 -215
- package/README.md +388 -216
- package/dist/__tests__/fixtures.d.ts +22 -0
- package/dist/__tests__/fixtures.d.ts.map +1 -0
- package/dist/__tests__/fixtures.js +146 -0
- package/dist/__tests__/fixtures.js.map +1 -0
- package/dist/cli/commands/add-auth.d.ts +10 -0
- package/dist/cli/commands/add-auth.d.ts.map +1 -0
- package/dist/cli/commands/add-auth.js +444 -0
- package/dist/cli/commands/add-auth.js.map +1 -0
- package/dist/cli/commands/add-connector.d.ts +20 -0
- package/dist/cli/commands/add-connector.d.ts.map +1 -0
- package/dist/cli/commands/add-connector.js +163 -0
- package/dist/cli/commands/add-connector.js.map +1 -0
- package/dist/cli/commands/create-model.d.ts +1 -4
- package/dist/cli/commands/create-model.d.ts.map +1 -1
- package/dist/cli/commands/create-model.js +21 -82
- package/dist/cli/commands/create-model.js.map +1 -1
- package/dist/cli/commands/dev-seeds.d.ts +57 -0
- package/dist/cli/commands/dev-seeds.d.ts.map +1 -0
- package/dist/cli/commands/dev-seeds.js +470 -0
- package/dist/cli/commands/dev-seeds.js.map +1 -0
- package/dist/cli/commands/dev.d.ts +33 -0
- package/dist/cli/commands/dev.d.ts.map +1 -1
- package/dist/cli/commands/dev.js +628 -146
- package/dist/cli/commands/dev.js.map +1 -1
- package/dist/cli/commands/index.d.ts +1 -0
- package/dist/cli/commands/index.d.ts.map +1 -1
- package/dist/cli/commands/index.js +3 -1
- package/dist/cli/commands/index.js.map +1 -1
- package/dist/cli/commands/init.d.ts +15 -0
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +2693 -1706
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/scaffold.d.ts.map +1 -1
- package/dist/cli/commands/scaffold.js +448 -129
- package/dist/cli/commands/scaffold.js.map +1 -1
- package/dist/cli/index.d.ts +5 -1
- package/dist/cli/index.d.ts.map +1 -1
- package/dist/cli/index.js +200 -42
- package/dist/cli/index.js.map +1 -1
- package/dist/core/config.d.ts +8 -2
- package/dist/core/config.d.ts.map +1 -1
- package/dist/core/config.js +94 -5
- package/dist/core/config.js.map +1 -1
- package/dist/core/mock/connector-mock-server.d.ts +101 -0
- package/dist/core/mock/connector-mock-server.d.ts.map +1 -0
- package/dist/core/mock/connector-mock-server.js +480 -0
- package/dist/core/mock/connector-mock-server.js.map +1 -0
- package/dist/core/mock/zod-mock-generator.d.ts +14 -0
- package/dist/core/mock/zod-mock-generator.d.ts.map +1 -0
- package/dist/core/mock/zod-mock-generator.js +163 -0
- package/dist/core/mock/zod-mock-generator.js.map +1 -0
- package/dist/core/operations/create-model.d.ts +15 -0
- package/dist/core/operations/create-model.d.ts.map +1 -0
- package/dist/core/operations/create-model.js +171 -0
- package/dist/core/operations/create-model.js.map +1 -0
- package/dist/core/operations/runtime.d.ts +32 -0
- package/dist/core/operations/runtime.d.ts.map +1 -0
- package/dist/core/operations/runtime.js +225 -0
- package/dist/core/operations/runtime.js.map +1 -0
- package/dist/core/operations/scaffold-machine.d.ts +16 -0
- package/dist/core/operations/scaffold-machine.d.ts.map +1 -0
- package/dist/core/operations/scaffold-machine.js +63 -0
- package/dist/core/operations/scaffold-machine.js.map +1 -0
- package/dist/core/project/manifest.d.ts +92 -0
- package/dist/core/project/manifest.d.ts.map +1 -0
- package/dist/core/project/manifest.js +321 -0
- package/dist/core/project/manifest.js.map +1 -0
- package/dist/core/project/validation.d.ts +20 -0
- package/dist/core/project/validation.d.ts.map +1 -0
- package/dist/core/project/validation.js +209 -0
- package/dist/core/project/validation.js.map +1 -0
- package/dist/core/scaffold/auth-generator.d.ts +38 -0
- package/dist/core/scaffold/auth-generator.d.ts.map +1 -0
- package/dist/core/scaffold/auth-generator.js +1244 -0
- package/dist/core/scaffold/auth-generator.js.map +1 -0
- package/dist/core/scaffold/connector-functions-generator.d.ts +41 -0
- package/dist/core/scaffold/connector-functions-generator.d.ts.map +1 -0
- package/dist/core/scaffold/connector-functions-generator.js +1027 -0
- package/dist/core/scaffold/connector-functions-generator.js.map +1 -0
- package/dist/core/scaffold/functions-generator.d.ts +7 -1
- package/dist/core/scaffold/functions-generator.d.ts.map +1 -1
- package/dist/core/scaffold/functions-generator.js +920 -213
- package/dist/core/scaffold/functions-generator.js.map +1 -1
- package/dist/core/scaffold/model-parser.d.ts +20 -1
- package/dist/core/scaffold/model-parser.d.ts.map +1 -1
- package/dist/core/scaffold/model-parser.js +328 -135
- package/dist/core/scaffold/model-parser.js.map +1 -1
- package/dist/core/scaffold/native-schema-generator.d.ts +13 -0
- package/dist/core/scaffold/native-schema-generator.d.ts.map +1 -0
- package/dist/core/scaffold/native-schema-generator.js +677 -0
- package/dist/core/scaffold/native-schema-generator.js.map +1 -0
- package/dist/core/scaffold/nextjs-generator.d.ts +8 -0
- package/dist/core/scaffold/nextjs-generator.d.ts.map +1 -1
- package/dist/core/scaffold/nextjs-generator.js +314 -182
- package/dist/core/scaffold/nextjs-generator.js.map +1 -1
- package/dist/core/scaffold/openapi-generator.d.ts +3 -0
- package/dist/core/scaffold/openapi-generator.d.ts.map +1 -0
- package/dist/core/scaffold/openapi-generator.js +190 -0
- package/dist/core/scaffold/openapi-generator.js.map +1 -0
- package/dist/core/scaffold/ui-generator.d.ts +10 -4
- package/dist/core/scaffold/ui-generator.d.ts.map +1 -1
- package/dist/core/scaffold/ui-generator.js +768 -663
- package/dist/core/scaffold/ui-generator.js.map +1 -1
- package/dist/database/base-model.d.ts +3 -3
- package/dist/database/base-model.js +3 -3
- package/dist/index.d.ts +2 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -1
- package/dist/index.js.map +1 -1
- package/dist/machine/contracts.d.ts +16 -0
- package/dist/machine/contracts.d.ts.map +1 -0
- package/dist/machine/contracts.js +3 -0
- package/dist/machine/contracts.js.map +1 -0
- package/dist/machine/errors.d.ts +11 -0
- package/dist/machine/errors.d.ts.map +1 -0
- package/dist/machine/errors.js +34 -0
- package/dist/machine/errors.js.map +1 -0
- package/dist/machine/index.d.ts +3 -0
- package/dist/machine/index.d.ts.map +1 -0
- package/dist/machine/index.js +156 -0
- package/dist/machine/index.js.map +1 -0
- package/dist/mcp/index.d.ts +25 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +184 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/types/index.d.ts +65 -0
- package/dist/types/index.d.ts.map +1 -1
- package/dist/utils/package-manager.d.ts +109 -0
- package/dist/utils/package-manager.d.ts.map +1 -0
- package/dist/utils/package-manager.js +215 -0
- package/dist/utils/package-manager.js.map +1 -0
- package/dist/utils/python-uv.d.ts +21 -0
- package/dist/utils/python-uv.d.ts.map +1 -0
- package/dist/utils/python-uv.js +111 -0
- package/dist/utils/python-uv.js.map +1 -0
- package/package.json +85 -73
- package/src/__tests__/__snapshots__/functions-generator.test.ts.snap +1139 -0
- package/src/__tests__/__snapshots__/nextjs-generator.test.ts.snap +194 -0
- package/src/__tests__/__snapshots__/ui-generator.test.ts.snap +532 -0
- package/src/__tests__/auth.test.ts +654 -0
- package/src/__tests__/config.test.ts +274 -0
- package/src/__tests__/connector-functions-generator.test.ts +288 -0
- package/src/__tests__/connector-mock-server.test.ts +439 -0
- package/src/__tests__/connector-model-bff.test.ts +162 -0
- package/src/__tests__/dev-seeds.test.ts +173 -0
- package/src/__tests__/dev.test.ts +252 -0
- package/src/__tests__/fixtures.ts +144 -0
- package/src/__tests__/functions-generator.test.ts +237 -0
- package/src/__tests__/init.test.ts +109 -0
- package/src/__tests__/machine.test.ts +251 -0
- package/src/__tests__/mcp.test.ts +117 -0
- package/src/__tests__/model-parser.test.ts +52 -0
- package/src/__tests__/nextjs-generator.test.ts +97 -0
- package/src/__tests__/openapi-generator.test.ts +43 -0
- package/src/__tests__/package-manager.test.ts +189 -0
- package/src/__tests__/python-uv.test.ts +48 -0
- package/src/__tests__/scaffold.test.ts +67 -0
- package/src/__tests__/string-utils.test.ts +75 -0
- package/src/__tests__/ui-generator.test.ts +144 -0
- package/src/__tests__/zod-mock-generator.test.ts +132 -0
- package/src/cli/commands/add-auth.ts +500 -0
- package/src/cli/commands/add-connector.ts +158 -0
- package/src/cli/commands/create-model.ts +62 -0
- package/src/cli/commands/dev-seeds.ts +614 -0
- package/src/cli/commands/dev.ts +1134 -0
- package/src/cli/commands/index.ts +9 -0
- package/src/cli/commands/init.ts +3477 -0
- package/src/cli/commands/provision.ts +193 -0
- package/src/cli/commands/scaffold.ts +1001 -0
- package/src/cli/index.ts +196 -0
- package/src/core/config.ts +312 -0
- package/src/core/mock/connector-mock-server.ts +555 -0
- package/src/core/mock/zod-mock-generator.ts +205 -0
- package/src/core/operations/create-model.ts +174 -0
- package/src/core/operations/runtime.ts +235 -0
- package/src/core/operations/scaffold-machine.ts +91 -0
- package/src/core/project/manifest.ts +402 -0
- package/src/core/project/validation.ts +229 -0
- package/src/core/scaffold/auth-generator.ts +1284 -0
- package/src/core/scaffold/connector-functions-generator.ts +1128 -0
- package/src/core/scaffold/functions-generator.ts +970 -0
- package/src/core/scaffold/model-parser.ts +841 -0
- package/src/core/scaffold/native-schema-generator.ts +798 -0
- package/src/core/scaffold/nextjs-generator.ts +370 -0
- package/src/core/scaffold/openapi-generator.ts +212 -0
- package/src/core/scaffold/ui-generator.ts +1061 -0
- package/src/database/base-model.ts +184 -0
- package/src/database/client.ts +140 -0
- package/src/database/repository.ts +104 -0
- package/src/database/runtime-check.ts +25 -0
- package/src/index.ts +27 -0
- package/src/machine/contracts.ts +17 -0
- package/src/machine/errors.ts +34 -0
- package/src/machine/index.ts +173 -0
- package/src/mcp/index.ts +185 -0
- package/src/types/index.ts +134 -0
- package/src/utils/package-manager.ts +229 -0
- 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
|
|
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
|
-
|
|
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
|
-
|
|
83
|
-
|
|
84
|
-
|
|
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
|
-
|
|
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}
|
|
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
|
-
|
|
98
|
-
|
|
99
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
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
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
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 = '
|
|
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`);
|