swallowkit 1.0.0-beta.2 → 1.0.0-beta.21
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 +312 -215
- package/README.md +369 -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 +35 -0
- package/dist/cli/commands/dev-seeds.d.ts.map +1 -0
- package/dist/cli/commands/dev-seeds.js +292 -0
- package/dist/cli/commands/dev-seeds.js.map +1 -0
- package/dist/cli/commands/dev.d.ts +19 -0
- package/dist/cli/commands/dev.d.ts.map +1 -1
- package/dist/cli/commands/dev.js +476 -117
- 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 +13 -0
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +2627 -1708
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/scaffold.d.ts +3 -0
- package/dist/cli/commands/scaffold.d.ts.map +1 -1
- package/dist/cli/commands/scaffold.js +617 -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 +164 -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 +90 -4
- 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 +204 -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 +329 -135
- package/dist/core/scaffold/model-parser.js.map +1 -1
- 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/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 +263 -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 +112 -0
- package/src/__tests__/dev.test.ts +154 -0
- package/src/__tests__/fixtures.ts +144 -0
- package/src/__tests__/functions-generator.test.ts +237 -0
- package/src/__tests__/init.test.ts +80 -0
- package/src/__tests__/machine.test.ts +212 -0
- package/src/__tests__/mcp.test.ts +56 -0
- package/src/__tests__/model-parser.test.ts +72 -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__/scaffold.test.ts +39 -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 +358 -0
- package/src/cli/commands/dev.ts +962 -0
- package/src/cli/commands/index.ts +9 -0
- package/src/cli/commands/init.ts +3371 -0
- package/src/cli/commands/provision.ts +193 -0
- package/src/cli/commands/scaffold.ts +1211 -0
- package/src/cli/index.ts +193 -0
- package/src/core/config.ts +308 -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 +221 -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/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
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"functions-generator.js","sourceRoot":"","sources":["../../../src/core/scaffold/functions-generator.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;
|
|
1
|
+
{"version":3,"file":"functions-generator.js","sourceRoot":"","sources":["../../../src/core/scaffold/functions-generator.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Zod モデルファイルを解析して、スキーマ情報を抽出する
|
|
3
3
|
*/
|
|
4
|
+
import { ModelConnectorConfig, ModelAuthPolicy } from "../../types";
|
|
4
5
|
export interface ModelInfo {
|
|
5
6
|
name: string;
|
|
6
7
|
displayName: string;
|
|
@@ -11,6 +12,9 @@ export interface ModelInfo {
|
|
|
11
12
|
hasCreatedAt: boolean;
|
|
12
13
|
hasUpdatedAt: boolean;
|
|
13
14
|
nestedSchemaRefs: NestedSchemaRef[];
|
|
15
|
+
connectorConfig?: ModelConnectorConfig;
|
|
16
|
+
authPolicy?: ModelAuthPolicy;
|
|
17
|
+
partitionKey: string;
|
|
14
18
|
}
|
|
15
19
|
export interface FieldInfo {
|
|
16
20
|
name: string;
|
|
@@ -49,13 +53,28 @@ export declare function toPascalCase(str: string): string;
|
|
|
49
53
|
* 文字列を camelCase に変換
|
|
50
54
|
*/
|
|
51
55
|
export declare function toCamelCase(str: string): string;
|
|
56
|
+
/**
|
|
57
|
+
* モデルファイルから connectorConfig エクスポートを静的解析で抽出する
|
|
58
|
+
*/
|
|
59
|
+
export declare function parseConnectorConfig(content: string): ModelConnectorConfig | undefined;
|
|
60
|
+
/**
|
|
61
|
+
* authPolicy をモデルファイルから抽出
|
|
62
|
+
* パターン: export const authPolicy = { roles: [...], read: [...], write: [...] }
|
|
63
|
+
*/
|
|
64
|
+
export declare function parseAuthPolicy(content: string): ModelAuthPolicy | undefined;
|
|
65
|
+
/**
|
|
66
|
+
* パーティションキーを抽出する
|
|
67
|
+
* export const partitionKey = '/tenantId' のようなエクスポートを検出
|
|
68
|
+
* 未指定の場合はデフォルト '/id' を返す
|
|
69
|
+
*/
|
|
70
|
+
export declare function parsePartitionKey(content: string): string;
|
|
52
71
|
/**
|
|
53
72
|
* 文字列を kebab-case に変換
|
|
54
73
|
*/
|
|
55
74
|
export declare function toKebabCase(str: string): string;
|
|
56
75
|
/**
|
|
57
76
|
* models ディレクトリから全てのモデル情報を取得
|
|
58
|
-
* @param modelsDir モデルディレクトリのパス(デフォルト: "
|
|
77
|
+
* @param modelsDir モデルディレクトリのパス(デフォルト: "shared/models")
|
|
59
78
|
* @returns モデル情報の配列
|
|
60
79
|
*/
|
|
61
80
|
export declare function getAllModels(modelsDir?: string): Promise<ModelInfo[]>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"model-parser.d.ts","sourceRoot":"","sources":["../../../src/core/scaffold/model-parser.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,SAAS,EAAE,CAAC;IACpB,KAAK,EAAE,OAAO,CAAC;IACf,YAAY,EAAE,OAAO,CAAC;IACtB,YAAY,EAAE,OAAO,CAAC;IACtB,gBAAgB,EAAE,eAAe,EAAE,CAAC;
|
|
1
|
+
{"version":3,"file":"model-parser.d.ts","sourceRoot":"","sources":["../../../src/core/scaffold/model-parser.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,OAAO,EAAE,oBAAoB,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAGpE,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,SAAS,EAAE,CAAC;IACpB,KAAK,EAAE,OAAO,CAAC;IACf,YAAY,EAAE,OAAO,CAAC;IACtB,YAAY,EAAE,OAAO,CAAC;IACtB,gBAAgB,EAAE,eAAe,EAAE,CAAC;IACpC,eAAe,CAAC,EAAE,oBAAoB,CAAC;IACvC,UAAU,CAAC,EAAE,eAAe,CAAC;IAC7B,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,OAAO,CAAC;IACpB,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,cAAc,CAAC,EAAE,OAAO,CAAC;IACzB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,kBAAkB,CAAC,EAAE,MAAM,CAAC;CAC7B;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,OAAO,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED;;GAEG;AACH,wBAAsB,cAAc,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC,CAgF1E;AAkfD;;GAEG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAKhD;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAG/C;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,MAAM,GAAG,oBAAoB,GAAG,SAAS,CA2FtF;AAED;;;GAGG;AACH,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,eAAe,GAAG,SAAS,CAsC5E;AAED;;;;GAIG;AACH,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAGzD;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAI/C;AAED;;;;GAIG;AACH,wBAAsB,YAAY,CAAC,SAAS,GAAE,MAAwB,GAAG,OAAO,CAAC,SAAS,EAAE,CAAC,CAwB5F"}
|
|
@@ -39,10 +39,15 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
39
39
|
exports.parseModelFile = parseModelFile;
|
|
40
40
|
exports.toPascalCase = toPascalCase;
|
|
41
41
|
exports.toCamelCase = toCamelCase;
|
|
42
|
+
exports.parseConnectorConfig = parseConnectorConfig;
|
|
43
|
+
exports.parseAuthPolicy = parseAuthPolicy;
|
|
44
|
+
exports.parsePartitionKey = parsePartitionKey;
|
|
42
45
|
exports.toKebabCase = toKebabCase;
|
|
43
46
|
exports.getAllModels = getAllModels;
|
|
44
47
|
const fs = __importStar(require("fs"));
|
|
48
|
+
const child_process_1 = require("child_process");
|
|
45
49
|
const path = __importStar(require("path"));
|
|
50
|
+
const package_manager_1 = require("../../utils/package-manager");
|
|
46
51
|
/**
|
|
47
52
|
* モデルファイルを解析して ModelInfo を返す
|
|
48
53
|
*/
|
|
@@ -55,11 +60,15 @@ async function parseModelFile(modelPath) {
|
|
|
55
60
|
// モデル名を推定(ファイル名を PascalCase に変換)
|
|
56
61
|
const modelName = toPascalCase(fileName);
|
|
57
62
|
// スキーマ変数名を抽出
|
|
63
|
+
// コメント内のコードパターンを誤認識しないよう、先にコメントを除去してから正規表現を適用する
|
|
58
64
|
// パターン1: export const todoSchema = z.object({ ... }) (camelCase + Schema接尾辞)
|
|
59
65
|
// パターン2: export const Todo = z.object({ ... }) (Zod公式パターン)
|
|
60
|
-
|
|
66
|
+
const contentWithoutComments = content
|
|
67
|
+
.replace(/\/\*[\s\S]*?\*\//g, '') // ブロックコメントを除去
|
|
68
|
+
.replace(/\/\/.*/g, ''); // 行コメントを除去
|
|
69
|
+
let schemaMatch = contentWithoutComments.match(/export\s+const\s+(\w+Schema)\s*=/);
|
|
61
70
|
if (!schemaMatch) {
|
|
62
|
-
schemaMatch =
|
|
71
|
+
schemaMatch = contentWithoutComments.match(/export\s+const\s+(\w+)\s*=\s*z\.object\s*\(/);
|
|
63
72
|
}
|
|
64
73
|
if (!schemaMatch) {
|
|
65
74
|
throw new Error(`Could not find exported schema in ${modelPath}. Expected patterns:\n - export const xxxSchema = z.object({ ... })\n - export const Xxx = z.object({ ... })`);
|
|
@@ -68,6 +77,12 @@ async function parseModelFile(modelPath) {
|
|
|
68
77
|
// displayName を抽出(例: export const displayName = 'Task')
|
|
69
78
|
const displayNameMatch = content.match(/export\s+const\s+displayName\s*=\s*['"]([^'"]+)['"]/);
|
|
70
79
|
const displayName = displayNameMatch ? displayNameMatch[1] : modelName;
|
|
80
|
+
// connectorConfig を抽出(外部データソース用メタデータ)
|
|
81
|
+
const connectorConfig = parseConnectorConfig(content);
|
|
82
|
+
// authPolicy を抽出(ロールベースアクセス制御用メタデータ)
|
|
83
|
+
const authPolicy = parseAuthPolicy(content);
|
|
84
|
+
// partitionKey を抽出(Cosmos DB パーティションキー)
|
|
85
|
+
const partitionKey = parsePartitionKey(content);
|
|
71
86
|
// ネストしたスキーマ参照を検出
|
|
72
87
|
const nestedSchemaRefs = detectNestedSchemaRefs(modelPath, content, schemaName);
|
|
73
88
|
// フィールド情報を抽出(動的インポートを使用)
|
|
@@ -78,6 +93,16 @@ async function parseModelFile(modelPath) {
|
|
|
78
93
|
const hasId = fields.some(f => f.name === "id");
|
|
79
94
|
const hasCreatedAt = fields.some(f => f.name === "createdAt");
|
|
80
95
|
const hasUpdatedAt = fields.some(f => f.name === "updatedAt");
|
|
96
|
+
// パーティションキーのバリデーション
|
|
97
|
+
if (partitionKey !== '/id') {
|
|
98
|
+
if (!partitionKey.startsWith('/')) {
|
|
99
|
+
console.warn(`⚠️ [${modelName}] partitionKey should start with '/': got '${partitionKey}'`);
|
|
100
|
+
}
|
|
101
|
+
const pkField = partitionKey.startsWith('/') ? partitionKey.slice(1) : partitionKey;
|
|
102
|
+
if (fields.length > 0 && !fields.some(f => f.name === pkField)) {
|
|
103
|
+
console.warn(`⚠️ [${modelName}] partitionKey field '${pkField}' not found in schema fields`);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
81
106
|
return {
|
|
82
107
|
name: modelName,
|
|
83
108
|
displayName,
|
|
@@ -88,6 +113,9 @@ async function parseModelFile(modelPath) {
|
|
|
88
113
|
hasCreatedAt,
|
|
89
114
|
hasUpdatedAt,
|
|
90
115
|
nestedSchemaRefs,
|
|
116
|
+
partitionKey,
|
|
117
|
+
...(connectorConfig ? { connectorConfig } : {}),
|
|
118
|
+
...(authPolicy ? { authPolicy } : {}),
|
|
91
119
|
};
|
|
92
120
|
}
|
|
93
121
|
/**
|
|
@@ -257,53 +285,74 @@ function mergeNestedSchemaInfo(fields, nestedRefs) {
|
|
|
257
285
|
}
|
|
258
286
|
}
|
|
259
287
|
}
|
|
288
|
+
/**
|
|
289
|
+
* ローカル依存ファイルを再帰的にインライン化する(動的インポート用一時スクリプト生成用)。
|
|
290
|
+
* seen によって循環依存を防ぐ。
|
|
291
|
+
* importedNames が指定された場合はその名前の const だけを残し、他を除去する。
|
|
292
|
+
*/
|
|
293
|
+
function inlineLocalDeps(filePath, importedNames, seen) {
|
|
294
|
+
const resolvedPath = path.resolve(filePath);
|
|
295
|
+
if (seen.has(resolvedPath))
|
|
296
|
+
return '';
|
|
297
|
+
seen.add(resolvedPath);
|
|
298
|
+
if (!fs.existsSync(resolvedPath))
|
|
299
|
+
return '';
|
|
300
|
+
let content = fs.readFileSync(resolvedPath, 'utf8');
|
|
301
|
+
const fileDir = path.dirname(resolvedPath);
|
|
302
|
+
const transitiveParts = [];
|
|
303
|
+
// ローカル(相対)インポートを再帰的にインライン化
|
|
304
|
+
content = content.replace(/import\s*\{([^}]+)\}\s*from\s*['"](\.[^'"]+)['"]\s*;?/g, (_match, _imports, importPath) => {
|
|
305
|
+
let depPath = path.resolve(fileDir, importPath);
|
|
306
|
+
if (!depPath.endsWith('.ts') && !depPath.endsWith('.js'))
|
|
307
|
+
depPath += '.ts';
|
|
308
|
+
const inlined = inlineLocalDeps(depPath, null, seen);
|
|
309
|
+
if (inlined)
|
|
310
|
+
transitiveParts.push(inlined);
|
|
311
|
+
return '';
|
|
312
|
+
});
|
|
313
|
+
// 外部パッケージのインポートを除去
|
|
314
|
+
content = content.replace(/import\s+.*?\s+from\s+['"](?!\.).*?['"];?\s*/g, '');
|
|
315
|
+
// export キーワードを除去
|
|
316
|
+
content = content.replace(/export\s+(const|type|interface|class|function)\s+/g, '$1 ');
|
|
317
|
+
// 型宣言を除去(ランタイムでは不要)
|
|
318
|
+
content = content.replace(/^type\s+\w+\s*=\s*[^;]+;/gm, '');
|
|
319
|
+
content = content.replace(/^interface\s+\w+\s*\{[\s\S]*?\}/gm, '');
|
|
320
|
+
// コメントを除去
|
|
321
|
+
content = content.replace(/\/\*[\s\S]*?\*\//g, '');
|
|
322
|
+
content = content.replace(/\/\/.*/g, '');
|
|
323
|
+
// 直接インポートされていない const 宣言を除去(displayName 等の重複を防ぐ)
|
|
324
|
+
if (importedNames) {
|
|
325
|
+
content = content.replace(/^const\s+(\w+)\s*=\s*[^;]+;/gm, (constMatch, varName) => {
|
|
326
|
+
return importedNames.has(varName) ? constMatch : '';
|
|
327
|
+
});
|
|
328
|
+
}
|
|
329
|
+
return [...transitiveParts, content.trim()].filter(Boolean).join('\n\n');
|
|
330
|
+
}
|
|
260
331
|
/**
|
|
261
332
|
* Zodスキーマから動的にフィールド情報を抽出
|
|
262
333
|
*/
|
|
263
334
|
async function extractFieldsFromSchema(modelPath, schemaName) {
|
|
264
335
|
const fields = [];
|
|
265
336
|
try {
|
|
266
|
-
// child_processでtsxを実行
|
|
267
|
-
const { execSync } = require('child_process');
|
|
268
|
-
const { tmpdir } = require('os');
|
|
269
|
-
const { join } = require('path');
|
|
270
|
-
const { writeFileSync, unlinkSync, readFileSync } = require('fs');
|
|
271
337
|
// モデルファイルの内容を読み込む
|
|
272
|
-
let modelContent = readFileSync(modelPath, 'utf8');
|
|
338
|
+
let modelContent = fs.readFileSync(modelPath, 'utf8');
|
|
273
339
|
const modelDir = path.dirname(path.resolve(modelPath));
|
|
274
|
-
//
|
|
340
|
+
// ローカル相対インポートを再帰的にインライン化して保持
|
|
275
341
|
// パターン: import { categorySchema } from './category'
|
|
342
|
+
const seenPaths = new Set([path.resolve(modelPath)]);
|
|
276
343
|
const localImports = [];
|
|
277
344
|
modelContent = modelContent.replace(/import\s*\{([^}]+)\}\s*from\s*['"](\.[^'"]+)['"]\s*;?/g, (_match, imports, importPath) => {
|
|
278
345
|
// インポートされた変数名を取得
|
|
279
|
-
const importedNames = imports.split(',').map(s =>
|
|
280
|
-
const trimmed = s.trim();
|
|
281
|
-
// 'as' エイリアスに対応
|
|
282
|
-
return trimmed.split(/\s+as\s+/).pop().trim();
|
|
283
|
-
}).filter(s => s.length > 0);
|
|
346
|
+
const importedNames = new Set(imports.split(',').map(s => s.trim().split(/\s+as\s+/).pop().trim()).filter(s => s.length > 0));
|
|
284
347
|
// 相対パスを絶対パスに解決
|
|
285
348
|
let resolvedPath = path.resolve(modelDir, importPath);
|
|
286
349
|
if (!resolvedPath.endsWith('.ts') && !resolvedPath.endsWith('.js')) {
|
|
287
350
|
resolvedPath += '.ts';
|
|
288
351
|
}
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
refContent = refContent.replace(/import\s+.*?\s+from\s+['"](?!\.).*?['"];?\s*/g, '');
|
|
294
|
-
refContent = refContent.replace(/import\s+.*?\s+from\s+['"]\..*?['"];?\s*/g, '');
|
|
295
|
-
refContent = refContent.replace(/export\s+(const|type|interface|class|function)\s+/g, '$1 ');
|
|
296
|
-
refContent = refContent.replace(/^type\s+\w+\s*=\s*[^;]+;/gm, '');
|
|
297
|
-
refContent = refContent.replace(/^interface\s+\w+\s*\{[\s\S]*?\}/gm, '');
|
|
298
|
-
refContent = refContent.replace(/\/\*[\s\S]*?\*\//g, '');
|
|
299
|
-
refContent = refContent.replace(/\/\/.*/g, '');
|
|
300
|
-
// インポートされていない const 宣言を除去(displayName 等の重複を防ぐ)
|
|
301
|
-
const importedNameSet = new Set(importedNames);
|
|
302
|
-
refContent = refContent.replace(/^const\s+(\w+)\s*=\s*[^;]+;/gm, (constMatch, varName) => {
|
|
303
|
-
return importedNameSet.has(varName) ? constMatch : '';
|
|
304
|
-
});
|
|
305
|
-
localImports.push(refContent.trim());
|
|
306
|
-
}
|
|
352
|
+
// 推移的なローカル依存を含めて再帰的にインライン化
|
|
353
|
+
const inlined = inlineLocalDeps(resolvedPath, importedNames, seenPaths);
|
|
354
|
+
if (inlined)
|
|
355
|
+
localImports.push(inlined);
|
|
307
356
|
return ''; // 元のインポート文は除去
|
|
308
357
|
});
|
|
309
358
|
// zod 以外のパッケージインポートを除去
|
|
@@ -316,6 +365,8 @@ async function extractFieldsFromSchema(modelPath, schemaName) {
|
|
|
316
365
|
// コメントを削除
|
|
317
366
|
modelContent = modelContent.replace(/\/\*[\s\S]*?\*\//g, '');
|
|
318
367
|
modelContent = modelContent.replace(/\/\/.*/g, '');
|
|
368
|
+
// TypeScript の `as const` アサーションを削除(.mjs では構文エラーになる)
|
|
369
|
+
modelContent = modelContent.replace(/\s+as\s+const\b/g, '');
|
|
319
370
|
// インライン化したローカルインポートを先頭に追加
|
|
320
371
|
const inlinedDeps = localImports.length > 0 ? localImports.join('\n\n') + '\n\n' : '';
|
|
321
372
|
// プロジェクトルートを探す(package.jsonがある場所)
|
|
@@ -328,109 +379,112 @@ async function extractFieldsFromSchema(modelPath, schemaName) {
|
|
|
328
379
|
}
|
|
329
380
|
// プロジェクトルート内に一時スクリプトファイルを作成
|
|
330
381
|
const tempScript = path.join(projectRoot, `.swallowkit-parser-${Date.now()}.mjs`);
|
|
331
|
-
const scriptCode = `
|
|
332
|
-
import { z } from 'zod/v4';
|
|
333
|
-
|
|
334
|
-
// インライン化した依存スキーマ
|
|
335
|
-
${inlinedDeps}
|
|
336
|
-
// モデルファイルの内容を評価
|
|
337
|
-
${modelContent}
|
|
338
|
-
|
|
339
|
-
const schema = ${schemaName};
|
|
340
|
-
|
|
341
|
-
// Zod v3とv4の両方に対応
|
|
342
|
-
const isObject = (schema && schema._def &&
|
|
343
|
-
(schema._def.typeName === 'ZodObject' || schema.constructor?.name === 'ZodObject' || typeof schema._def.shape === 'function'));
|
|
344
|
-
|
|
345
|
-
if (isObject) {
|
|
346
|
-
const shape = typeof schema._def.shape === 'function' ? schema._def.shape() : schema._def.shape;
|
|
347
|
-
const fields = Object.keys(shape).map(key => {
|
|
348
|
-
const field = shape[key];
|
|
349
|
-
let type = 'string';
|
|
350
|
-
let isOptional = false;
|
|
351
|
-
let isArray = false;
|
|
352
|
-
let enumValues = undefined;
|
|
353
|
-
|
|
354
|
-
// ZodOptional, ZodDefault, ZodEffects を unwrap
|
|
355
|
-
let fieldDef = field;
|
|
356
|
-
const getTypeName = (def) => def?._def?.typeName || def?.constructor?.name || '';
|
|
357
|
-
|
|
358
|
-
// 繰り返し unwrap(複数のラッパーがある場合に対応)
|
|
359
|
-
let unwrapped = false;
|
|
360
|
-
do {
|
|
361
|
-
unwrapped = false;
|
|
362
|
-
const typeName = getTypeName(fieldDef);
|
|
363
|
-
|
|
364
|
-
if (typeName === 'ZodOptional') {
|
|
365
|
-
isOptional = true;
|
|
366
|
-
fieldDef = fieldDef._def.innerType;
|
|
367
|
-
unwrapped = true;
|
|
368
|
-
} else if (typeName === 'ZodDefault') {
|
|
369
|
-
// .default() は optional と同様に扱う
|
|
370
|
-
isOptional = true;
|
|
371
|
-
fieldDef = fieldDef._def.innerType;
|
|
372
|
-
unwrapped = true;
|
|
373
|
-
} else if (typeName === 'ZodEffects') {
|
|
374
|
-
// .min(), .max(), .regex() などの Effects を unwrap
|
|
375
|
-
fieldDef = fieldDef._def.schema;
|
|
376
|
-
unwrapped = true;
|
|
377
|
-
}
|
|
378
|
-
} while (unwrapped);
|
|
379
|
-
|
|
380
|
-
// ZodArray をチェック
|
|
381
|
-
if (getTypeName(fieldDef) === 'ZodArray') {
|
|
382
|
-
isArray = true;
|
|
383
|
-
fieldDef = fieldDef._def.type || fieldDef._def.element;
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
// 基本型を判定
|
|
387
|
-
const typeName = getTypeName(fieldDef);
|
|
388
|
-
if (typeName === 'ZodString') type = 'string';
|
|
389
|
-
else if (typeName === 'ZodNumber') type = 'number';
|
|
390
|
-
else if (typeName === 'ZodBoolean') type = 'boolean';
|
|
391
|
-
else if (typeName === 'ZodDate') type = 'date';
|
|
392
|
-
else if (typeName === 'ZodObject') type = 'object';
|
|
393
|
-
else if (typeName === 'ZodEnum' || typeName === 'ZodNativeEnum') {
|
|
394
|
-
type = 'string';
|
|
395
|
-
// enum の選択肢を取得(複数のZodバージョンに対応)
|
|
396
|
-
if (fieldDef.options) {
|
|
397
|
-
// Zod v3.23+ では options プロパティを使用
|
|
398
|
-
enumValues = Array.isArray(fieldDef.options)
|
|
399
|
-
? fieldDef.options
|
|
400
|
-
: Object.values(fieldDef.options);
|
|
401
|
-
} else if (fieldDef._def.values) {
|
|
402
|
-
// 古いバージョンでは _def.values を使用
|
|
403
|
-
enumValues = Array.isArray(fieldDef._def.values)
|
|
404
|
-
? fieldDef._def.values
|
|
405
|
-
: Object.values(fieldDef._def.values);
|
|
406
|
-
} else if (fieldDef._def.entries) {
|
|
407
|
-
// さらに古いバージョンでは _def.entries を使用
|
|
408
|
-
enumValues = Array.isArray(fieldDef._def.entries)
|
|
409
|
-
? fieldDef._def.entries
|
|
410
|
-
: Object.values(fieldDef._def.entries);
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
// 外部キー検出: フィールド名が <ModelName>Id のパターンの場合
|
|
415
|
-
let isForeignKey = false;
|
|
416
|
-
let referencedModel = undefined;
|
|
417
|
-
if (key.endsWith('Id') && key.length > 2 && type === 'string') {
|
|
418
|
-
// categoryId -> Category, userId -> User など
|
|
419
|
-
const modelName = key.slice(0, -2); // "Id" を除去
|
|
420
|
-
referencedModel = modelName.charAt(0).toUpperCase() + modelName.slice(1); // 先頭を大文字に
|
|
421
|
-
isForeignKey = true;
|
|
422
|
-
}
|
|
423
|
-
|
|
424
|
-
return { name: key, type, isOptional, isArray, enumValues, isForeignKey, referencedModel };
|
|
425
|
-
});
|
|
426
|
-
|
|
427
|
-
console.log(JSON.stringify(fields));
|
|
428
|
-
}
|
|
382
|
+
const scriptCode = `
|
|
383
|
+
import { z } from 'zod/v4';
|
|
384
|
+
|
|
385
|
+
// インライン化した依存スキーマ
|
|
386
|
+
${inlinedDeps}
|
|
387
|
+
// モデルファイルの内容を評価
|
|
388
|
+
${modelContent}
|
|
389
|
+
|
|
390
|
+
const schema = ${schemaName};
|
|
391
|
+
|
|
392
|
+
// Zod v3とv4の両方に対応
|
|
393
|
+
const isObject = (schema && schema._def &&
|
|
394
|
+
(schema._def.typeName === 'ZodObject' || schema.constructor?.name === 'ZodObject' || typeof schema._def.shape === 'function'));
|
|
395
|
+
|
|
396
|
+
if (isObject) {
|
|
397
|
+
const shape = typeof schema._def.shape === 'function' ? schema._def.shape() : schema._def.shape;
|
|
398
|
+
const fields = Object.keys(shape).map(key => {
|
|
399
|
+
const field = shape[key];
|
|
400
|
+
let type = 'string';
|
|
401
|
+
let isOptional = false;
|
|
402
|
+
let isArray = false;
|
|
403
|
+
let enumValues = undefined;
|
|
404
|
+
|
|
405
|
+
// ZodOptional, ZodDefault, ZodEffects を unwrap
|
|
406
|
+
let fieldDef = field;
|
|
407
|
+
const getTypeName = (def) => def?._def?.typeName || def?.constructor?.name || '';
|
|
408
|
+
|
|
409
|
+
// 繰り返し unwrap(複数のラッパーがある場合に対応)
|
|
410
|
+
let unwrapped = false;
|
|
411
|
+
do {
|
|
412
|
+
unwrapped = false;
|
|
413
|
+
const typeName = getTypeName(fieldDef);
|
|
414
|
+
|
|
415
|
+
if (typeName === 'ZodOptional') {
|
|
416
|
+
isOptional = true;
|
|
417
|
+
fieldDef = fieldDef._def.innerType;
|
|
418
|
+
unwrapped = true;
|
|
419
|
+
} else if (typeName === 'ZodDefault') {
|
|
420
|
+
// .default() は optional と同様に扱う
|
|
421
|
+
isOptional = true;
|
|
422
|
+
fieldDef = fieldDef._def.innerType;
|
|
423
|
+
unwrapped = true;
|
|
424
|
+
} else if (typeName === 'ZodEffects') {
|
|
425
|
+
// .min(), .max(), .regex() などの Effects を unwrap
|
|
426
|
+
fieldDef = fieldDef._def.schema;
|
|
427
|
+
unwrapped = true;
|
|
428
|
+
}
|
|
429
|
+
} while (unwrapped);
|
|
430
|
+
|
|
431
|
+
// ZodArray をチェック
|
|
432
|
+
if (getTypeName(fieldDef) === 'ZodArray') {
|
|
433
|
+
isArray = true;
|
|
434
|
+
fieldDef = fieldDef._def.type || fieldDef._def.element;
|
|
435
|
+
}
|
|
436
|
+
|
|
437
|
+
// 基本型を判定
|
|
438
|
+
const typeName = getTypeName(fieldDef);
|
|
439
|
+
if (typeName === 'ZodString') type = 'string';
|
|
440
|
+
else if (typeName === 'ZodNumber') type = 'number';
|
|
441
|
+
else if (typeName === 'ZodBoolean') type = 'boolean';
|
|
442
|
+
else if (typeName === 'ZodDate') type = 'date';
|
|
443
|
+
else if (typeName === 'ZodObject') type = 'object';
|
|
444
|
+
else if (typeName === 'ZodEnum' || typeName === 'ZodNativeEnum') {
|
|
445
|
+
type = 'string';
|
|
446
|
+
// enum の選択肢を取得(複数のZodバージョンに対応)
|
|
447
|
+
if (fieldDef.options) {
|
|
448
|
+
// Zod v3.23+ では options プロパティを使用
|
|
449
|
+
enumValues = Array.isArray(fieldDef.options)
|
|
450
|
+
? fieldDef.options
|
|
451
|
+
: Object.values(fieldDef.options);
|
|
452
|
+
} else if (fieldDef._def.values) {
|
|
453
|
+
// 古いバージョンでは _def.values を使用
|
|
454
|
+
enumValues = Array.isArray(fieldDef._def.values)
|
|
455
|
+
? fieldDef._def.values
|
|
456
|
+
: Object.values(fieldDef._def.values);
|
|
457
|
+
} else if (fieldDef._def.entries) {
|
|
458
|
+
// さらに古いバージョンでは _def.entries を使用
|
|
459
|
+
enumValues = Array.isArray(fieldDef._def.entries)
|
|
460
|
+
? fieldDef._def.entries
|
|
461
|
+
: Object.values(fieldDef._def.entries);
|
|
462
|
+
}
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
// 外部キー検出: フィールド名が <ModelName>Id のパターンの場合
|
|
466
|
+
let isForeignKey = false;
|
|
467
|
+
let referencedModel = undefined;
|
|
468
|
+
if (key.endsWith('Id') && key.length > 2 && type === 'string') {
|
|
469
|
+
// categoryId -> Category, userId -> User など
|
|
470
|
+
const modelName = key.slice(0, -2); // "Id" を除去
|
|
471
|
+
referencedModel = modelName.charAt(0).toUpperCase() + modelName.slice(1); // 先頭を大文字に
|
|
472
|
+
isForeignKey = true;
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
return { name: key, type, isOptional, isArray, enumValues, isForeignKey, referencedModel };
|
|
476
|
+
});
|
|
477
|
+
|
|
478
|
+
console.log(JSON.stringify(fields));
|
|
479
|
+
}
|
|
429
480
|
`;
|
|
430
|
-
writeFileSync(tempScript, scriptCode, 'utf8');
|
|
481
|
+
fs.writeFileSync(tempScript, scriptCode, 'utf8');
|
|
431
482
|
try {
|
|
432
|
-
// プロジェクトルートでtsxを実行
|
|
433
|
-
|
|
483
|
+
// プロジェクトルートでtsxを実行 (npm/pnpm 両対応)
|
|
484
|
+
// Detect package manager from project lockfile
|
|
485
|
+
const pm = (0, package_manager_1.detectFromProject)(projectRoot);
|
|
486
|
+
const pmCmd = (0, package_manager_1.getCommands)(pm);
|
|
487
|
+
const result = (0, child_process_1.execSync)(`${pmCmd.exec} tsx "${tempScript}"`, {
|
|
434
488
|
encoding: 'utf8',
|
|
435
489
|
cwd: projectRoot,
|
|
436
490
|
});
|
|
@@ -443,7 +497,7 @@ if (isObject) {
|
|
|
443
497
|
finally {
|
|
444
498
|
// 一時ファイルを削除
|
|
445
499
|
try {
|
|
446
|
-
unlinkSync(tempScript);
|
|
500
|
+
fs.unlinkSync(tempScript);
|
|
447
501
|
}
|
|
448
502
|
catch (e) {
|
|
449
503
|
// ファイル削除失敗は無視
|
|
@@ -520,6 +574,146 @@ function toCamelCase(str) {
|
|
|
520
574
|
const pascal = toPascalCase(str);
|
|
521
575
|
return pascal.charAt(0).toLowerCase() + pascal.slice(1);
|
|
522
576
|
}
|
|
577
|
+
/**
|
|
578
|
+
* モデルファイルから connectorConfig エクスポートを静的解析で抽出する
|
|
579
|
+
*/
|
|
580
|
+
function parseConnectorConfig(content) {
|
|
581
|
+
// export const connectorConfig = { ... } を検出
|
|
582
|
+
const connectorMatch = content.match(/export\s+const\s+connectorConfig\s*=\s*\{/);
|
|
583
|
+
if (!connectorMatch) {
|
|
584
|
+
return undefined;
|
|
585
|
+
}
|
|
586
|
+
// connectorConfig オブジェクトの内容を抽出
|
|
587
|
+
const startIdx = content.indexOf('{', connectorMatch.index);
|
|
588
|
+
let braceCount = 1;
|
|
589
|
+
let endIdx = startIdx + 1;
|
|
590
|
+
while (braceCount > 0 && endIdx < content.length) {
|
|
591
|
+
if (content[endIdx] === '{')
|
|
592
|
+
braceCount++;
|
|
593
|
+
if (content[endIdx] === '}')
|
|
594
|
+
braceCount--;
|
|
595
|
+
endIdx++;
|
|
596
|
+
}
|
|
597
|
+
const objectStr = content.substring(startIdx, endIdx);
|
|
598
|
+
// connector 名を抽出
|
|
599
|
+
const connectorNameMatch = objectStr.match(/connector\s*:\s*['"]([^'"]+)['"]/);
|
|
600
|
+
if (!connectorNameMatch) {
|
|
601
|
+
return undefined;
|
|
602
|
+
}
|
|
603
|
+
const connector = connectorNameMatch[1];
|
|
604
|
+
// operations を抽出
|
|
605
|
+
const opsMatch = objectStr.match(/operations\s*:\s*\[([^\]]*)\]/);
|
|
606
|
+
const operations = [];
|
|
607
|
+
if (opsMatch) {
|
|
608
|
+
const opsStr = opsMatch[1];
|
|
609
|
+
const opEntries = opsStr.match(/['"]([^'"]+)['"]/g);
|
|
610
|
+
if (opEntries) {
|
|
611
|
+
for (const entry of opEntries) {
|
|
612
|
+
operations.push(entry.replace(/['"]/g, ''));
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
}
|
|
616
|
+
// table を抽出(RDB 固有)
|
|
617
|
+
const tableMatch = objectStr.match(/table\s*:\s*['"]([^'"]+)['"]/);
|
|
618
|
+
// idColumn を抽出(RDB 固有)
|
|
619
|
+
const idColumnMatch = objectStr.match(/idColumn\s*:\s*['"]([^'"]+)['"]/);
|
|
620
|
+
// endpoints を抽出(API 固有)— endpoint値に {id} 等のプレースホルダが含まれるためbrace-countingで抽出
|
|
621
|
+
let endpoints;
|
|
622
|
+
const endpointsStart = objectStr.match(/endpoints\s*:\s*\{/);
|
|
623
|
+
if (endpointsStart) {
|
|
624
|
+
const epStartIdx = objectStr.indexOf('{', endpointsStart.index);
|
|
625
|
+
let epBraceCount = 1;
|
|
626
|
+
let epEndIdx = epStartIdx + 1;
|
|
627
|
+
while (epBraceCount > 0 && epEndIdx < objectStr.length) {
|
|
628
|
+
const ch = objectStr[epEndIdx];
|
|
629
|
+
// 文字列内の波括弧をスキップ
|
|
630
|
+
if (ch === "'" || ch === '"') {
|
|
631
|
+
epEndIdx++;
|
|
632
|
+
while (epEndIdx < objectStr.length && objectStr[epEndIdx] !== ch) {
|
|
633
|
+
epEndIdx++;
|
|
634
|
+
}
|
|
635
|
+
}
|
|
636
|
+
else if (ch === '{') {
|
|
637
|
+
epBraceCount++;
|
|
638
|
+
}
|
|
639
|
+
else if (ch === '}') {
|
|
640
|
+
epBraceCount--;
|
|
641
|
+
}
|
|
642
|
+
epEndIdx++;
|
|
643
|
+
}
|
|
644
|
+
const epStr = objectStr.substring(epStartIdx + 1, epEndIdx - 1);
|
|
645
|
+
endpoints = {};
|
|
646
|
+
const endpointEntries = epStr.matchAll(/(\w+)\s*:\s*['"]([^'"]+)['"]/g);
|
|
647
|
+
for (const entry of endpointEntries) {
|
|
648
|
+
endpoints[entry[1]] = entry[2];
|
|
649
|
+
}
|
|
650
|
+
}
|
|
651
|
+
if (tableMatch) {
|
|
652
|
+
// RDB コネクタ
|
|
653
|
+
return {
|
|
654
|
+
connector,
|
|
655
|
+
operations: operations,
|
|
656
|
+
table: tableMatch[1],
|
|
657
|
+
...(idColumnMatch ? { idColumn: idColumnMatch[1] } : {}),
|
|
658
|
+
};
|
|
659
|
+
}
|
|
660
|
+
// API コネクタ
|
|
661
|
+
return {
|
|
662
|
+
connector,
|
|
663
|
+
operations: operations,
|
|
664
|
+
...(endpoints ? { endpoints } : {}),
|
|
665
|
+
};
|
|
666
|
+
}
|
|
667
|
+
/**
|
|
668
|
+
* authPolicy をモデルファイルから抽出
|
|
669
|
+
* パターン: export const authPolicy = { roles: [...], read: [...], write: [...] }
|
|
670
|
+
*/
|
|
671
|
+
function parseAuthPolicy(content) {
|
|
672
|
+
const policyMatch = content.match(/export\s+const\s+authPolicy\s*=\s*\{/);
|
|
673
|
+
if (!policyMatch) {
|
|
674
|
+
return undefined;
|
|
675
|
+
}
|
|
676
|
+
const startIdx = content.indexOf('{', policyMatch.index);
|
|
677
|
+
let braceCount = 1;
|
|
678
|
+
let endIdx = startIdx + 1;
|
|
679
|
+
while (braceCount > 0 && endIdx < content.length) {
|
|
680
|
+
if (content[endIdx] === '{')
|
|
681
|
+
braceCount++;
|
|
682
|
+
if (content[endIdx] === '}')
|
|
683
|
+
braceCount--;
|
|
684
|
+
endIdx++;
|
|
685
|
+
}
|
|
686
|
+
const objectStr = content.substring(startIdx, endIdx);
|
|
687
|
+
const extractRoles = (key) => {
|
|
688
|
+
const match = objectStr.match(new RegExp(`${key}\\s*:\\s*\\[([^\\]]*)\\]`));
|
|
689
|
+
if (!match)
|
|
690
|
+
return undefined;
|
|
691
|
+
const entries = match[1].match(/['"]([^'"]+)['"]/g);
|
|
692
|
+
if (!entries)
|
|
693
|
+
return [];
|
|
694
|
+
return entries.map(e => e.replace(/['"]/g, ''));
|
|
695
|
+
};
|
|
696
|
+
const roles = extractRoles('roles');
|
|
697
|
+
const read = extractRoles('read');
|
|
698
|
+
const write = extractRoles('write');
|
|
699
|
+
if (!roles && !read && !write) {
|
|
700
|
+
return undefined;
|
|
701
|
+
}
|
|
702
|
+
return {
|
|
703
|
+
...(roles ? { roles } : {}),
|
|
704
|
+
...(read ? { read } : {}),
|
|
705
|
+
...(write ? { write } : {}),
|
|
706
|
+
};
|
|
707
|
+
}
|
|
708
|
+
/**
|
|
709
|
+
* パーティションキーを抽出する
|
|
710
|
+
* export const partitionKey = '/tenantId' のようなエクスポートを検出
|
|
711
|
+
* 未指定の場合はデフォルト '/id' を返す
|
|
712
|
+
*/
|
|
713
|
+
function parsePartitionKey(content) {
|
|
714
|
+
const match = content.match(/export\s+const\s+partitionKey\s*=\s*['"]([^'"]+)['"]/);
|
|
715
|
+
return match ? match[1] : '/id';
|
|
716
|
+
}
|
|
523
717
|
/**
|
|
524
718
|
* 文字列を kebab-case に変換
|
|
525
719
|
*/
|
|
@@ -530,7 +724,7 @@ function toKebabCase(str) {
|
|
|
530
724
|
}
|
|
531
725
|
/**
|
|
532
726
|
* models ディレクトリから全てのモデル情報を取得
|
|
533
|
-
* @param modelsDir モデルディレクトリのパス(デフォルト: "
|
|
727
|
+
* @param modelsDir モデルディレクトリのパス(デフォルト: "shared/models")
|
|
534
728
|
* @returns モデル情報の配列
|
|
535
729
|
*/
|
|
536
730
|
async function getAllModels(modelsDir = "shared/models") {
|