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.
Files changed (191) hide show
  1. package/LICENSE +21 -21
  2. package/README.ja.md +312 -215
  3. package/README.md +369 -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 +35 -0
  21. package/dist/cli/commands/dev-seeds.d.ts.map +1 -0
  22. package/dist/cli/commands/dev-seeds.js +292 -0
  23. package/dist/cli/commands/dev-seeds.js.map +1 -0
  24. package/dist/cli/commands/dev.d.ts +19 -0
  25. package/dist/cli/commands/dev.d.ts.map +1 -1
  26. package/dist/cli/commands/dev.js +476 -117
  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 +13 -0
  33. package/dist/cli/commands/init.d.ts.map +1 -1
  34. package/dist/cli/commands/init.js +2627 -1708
  35. package/dist/cli/commands/init.js.map +1 -1
  36. package/dist/cli/commands/scaffold.d.ts +3 -0
  37. package/dist/cli/commands/scaffold.d.ts.map +1 -1
  38. package/dist/cli/commands/scaffold.js +617 -129
  39. package/dist/cli/commands/scaffold.js.map +1 -1
  40. package/dist/cli/index.d.ts +5 -1
  41. package/dist/cli/index.d.ts.map +1 -1
  42. package/dist/cli/index.js +164 -42
  43. package/dist/cli/index.js.map +1 -1
  44. package/dist/core/config.d.ts +8 -2
  45. package/dist/core/config.d.ts.map +1 -1
  46. package/dist/core/config.js +90 -4
  47. package/dist/core/config.js.map +1 -1
  48. package/dist/core/mock/connector-mock-server.d.ts +101 -0
  49. package/dist/core/mock/connector-mock-server.d.ts.map +1 -0
  50. package/dist/core/mock/connector-mock-server.js +480 -0
  51. package/dist/core/mock/connector-mock-server.js.map +1 -0
  52. package/dist/core/mock/zod-mock-generator.d.ts +14 -0
  53. package/dist/core/mock/zod-mock-generator.d.ts.map +1 -0
  54. package/dist/core/mock/zod-mock-generator.js +163 -0
  55. package/dist/core/mock/zod-mock-generator.js.map +1 -0
  56. package/dist/core/operations/create-model.d.ts +15 -0
  57. package/dist/core/operations/create-model.d.ts.map +1 -0
  58. package/dist/core/operations/create-model.js +171 -0
  59. package/dist/core/operations/create-model.js.map +1 -0
  60. package/dist/core/operations/runtime.d.ts +32 -0
  61. package/dist/core/operations/runtime.d.ts.map +1 -0
  62. package/dist/core/operations/runtime.js +225 -0
  63. package/dist/core/operations/runtime.js.map +1 -0
  64. package/dist/core/operations/scaffold-machine.d.ts +16 -0
  65. package/dist/core/operations/scaffold-machine.d.ts.map +1 -0
  66. package/dist/core/operations/scaffold-machine.js +63 -0
  67. package/dist/core/operations/scaffold-machine.js.map +1 -0
  68. package/dist/core/project/manifest.d.ts +92 -0
  69. package/dist/core/project/manifest.d.ts.map +1 -0
  70. package/dist/core/project/manifest.js +321 -0
  71. package/dist/core/project/manifest.js.map +1 -0
  72. package/dist/core/project/validation.d.ts +20 -0
  73. package/dist/core/project/validation.d.ts.map +1 -0
  74. package/dist/core/project/validation.js +204 -0
  75. package/dist/core/project/validation.js.map +1 -0
  76. package/dist/core/scaffold/auth-generator.d.ts +38 -0
  77. package/dist/core/scaffold/auth-generator.d.ts.map +1 -0
  78. package/dist/core/scaffold/auth-generator.js +1244 -0
  79. package/dist/core/scaffold/auth-generator.js.map +1 -0
  80. package/dist/core/scaffold/connector-functions-generator.d.ts +41 -0
  81. package/dist/core/scaffold/connector-functions-generator.d.ts.map +1 -0
  82. package/dist/core/scaffold/connector-functions-generator.js +1027 -0
  83. package/dist/core/scaffold/connector-functions-generator.js.map +1 -0
  84. package/dist/core/scaffold/functions-generator.d.ts +7 -1
  85. package/dist/core/scaffold/functions-generator.d.ts.map +1 -1
  86. package/dist/core/scaffold/functions-generator.js +920 -213
  87. package/dist/core/scaffold/functions-generator.js.map +1 -1
  88. package/dist/core/scaffold/model-parser.d.ts +20 -1
  89. package/dist/core/scaffold/model-parser.d.ts.map +1 -1
  90. package/dist/core/scaffold/model-parser.js +329 -135
  91. package/dist/core/scaffold/model-parser.js.map +1 -1
  92. package/dist/core/scaffold/nextjs-generator.d.ts +8 -0
  93. package/dist/core/scaffold/nextjs-generator.d.ts.map +1 -1
  94. package/dist/core/scaffold/nextjs-generator.js +314 -182
  95. package/dist/core/scaffold/nextjs-generator.js.map +1 -1
  96. package/dist/core/scaffold/openapi-generator.d.ts +3 -0
  97. package/dist/core/scaffold/openapi-generator.d.ts.map +1 -0
  98. package/dist/core/scaffold/openapi-generator.js +190 -0
  99. package/dist/core/scaffold/openapi-generator.js.map +1 -0
  100. package/dist/core/scaffold/ui-generator.d.ts +10 -4
  101. package/dist/core/scaffold/ui-generator.d.ts.map +1 -1
  102. package/dist/core/scaffold/ui-generator.js +768 -663
  103. package/dist/core/scaffold/ui-generator.js.map +1 -1
  104. package/dist/database/base-model.d.ts +3 -3
  105. package/dist/database/base-model.js +3 -3
  106. package/dist/index.d.ts +2 -2
  107. package/dist/index.d.ts.map +1 -1
  108. package/dist/index.js +2 -1
  109. package/dist/index.js.map +1 -1
  110. package/dist/machine/contracts.d.ts +16 -0
  111. package/dist/machine/contracts.d.ts.map +1 -0
  112. package/dist/machine/contracts.js +3 -0
  113. package/dist/machine/contracts.js.map +1 -0
  114. package/dist/machine/errors.d.ts +11 -0
  115. package/dist/machine/errors.d.ts.map +1 -0
  116. package/dist/machine/errors.js +34 -0
  117. package/dist/machine/errors.js.map +1 -0
  118. package/dist/machine/index.d.ts +3 -0
  119. package/dist/machine/index.d.ts.map +1 -0
  120. package/dist/machine/index.js +156 -0
  121. package/dist/machine/index.js.map +1 -0
  122. package/dist/mcp/index.d.ts +25 -0
  123. package/dist/mcp/index.d.ts.map +1 -0
  124. package/dist/mcp/index.js +184 -0
  125. package/dist/mcp/index.js.map +1 -0
  126. package/dist/types/index.d.ts +65 -0
  127. package/dist/types/index.d.ts.map +1 -1
  128. package/dist/utils/package-manager.d.ts +109 -0
  129. package/dist/utils/package-manager.d.ts.map +1 -0
  130. package/dist/utils/package-manager.js +215 -0
  131. package/dist/utils/package-manager.js.map +1 -0
  132. package/package.json +85 -73
  133. package/src/__tests__/__snapshots__/functions-generator.test.ts.snap +1139 -0
  134. package/src/__tests__/__snapshots__/nextjs-generator.test.ts.snap +194 -0
  135. package/src/__tests__/__snapshots__/ui-generator.test.ts.snap +532 -0
  136. package/src/__tests__/auth.test.ts +654 -0
  137. package/src/__tests__/config.test.ts +263 -0
  138. package/src/__tests__/connector-functions-generator.test.ts +288 -0
  139. package/src/__tests__/connector-mock-server.test.ts +439 -0
  140. package/src/__tests__/connector-model-bff.test.ts +162 -0
  141. package/src/__tests__/dev-seeds.test.ts +112 -0
  142. package/src/__tests__/dev.test.ts +154 -0
  143. package/src/__tests__/fixtures.ts +144 -0
  144. package/src/__tests__/functions-generator.test.ts +237 -0
  145. package/src/__tests__/init.test.ts +80 -0
  146. package/src/__tests__/machine.test.ts +212 -0
  147. package/src/__tests__/mcp.test.ts +56 -0
  148. package/src/__tests__/model-parser.test.ts +72 -0
  149. package/src/__tests__/nextjs-generator.test.ts +97 -0
  150. package/src/__tests__/openapi-generator.test.ts +43 -0
  151. package/src/__tests__/package-manager.test.ts +189 -0
  152. package/src/__tests__/scaffold.test.ts +39 -0
  153. package/src/__tests__/string-utils.test.ts +75 -0
  154. package/src/__tests__/ui-generator.test.ts +144 -0
  155. package/src/__tests__/zod-mock-generator.test.ts +132 -0
  156. package/src/cli/commands/add-auth.ts +500 -0
  157. package/src/cli/commands/add-connector.ts +158 -0
  158. package/src/cli/commands/create-model.ts +62 -0
  159. package/src/cli/commands/dev-seeds.ts +358 -0
  160. package/src/cli/commands/dev.ts +962 -0
  161. package/src/cli/commands/index.ts +9 -0
  162. package/src/cli/commands/init.ts +3371 -0
  163. package/src/cli/commands/provision.ts +193 -0
  164. package/src/cli/commands/scaffold.ts +1211 -0
  165. package/src/cli/index.ts +193 -0
  166. package/src/core/config.ts +308 -0
  167. package/src/core/mock/connector-mock-server.ts +555 -0
  168. package/src/core/mock/zod-mock-generator.ts +205 -0
  169. package/src/core/operations/create-model.ts +174 -0
  170. package/src/core/operations/runtime.ts +235 -0
  171. package/src/core/operations/scaffold-machine.ts +91 -0
  172. package/src/core/project/manifest.ts +402 -0
  173. package/src/core/project/validation.ts +221 -0
  174. package/src/core/scaffold/auth-generator.ts +1284 -0
  175. package/src/core/scaffold/connector-functions-generator.ts +1128 -0
  176. package/src/core/scaffold/functions-generator.ts +970 -0
  177. package/src/core/scaffold/model-parser.ts +841 -0
  178. package/src/core/scaffold/nextjs-generator.ts +370 -0
  179. package/src/core/scaffold/openapi-generator.ts +212 -0
  180. package/src/core/scaffold/ui-generator.ts +1061 -0
  181. package/src/database/base-model.ts +184 -0
  182. package/src/database/client.ts +140 -0
  183. package/src/database/repository.ts +104 -0
  184. package/src/database/runtime-check.ts +25 -0
  185. package/src/index.ts +27 -0
  186. package/src/machine/contracts.ts +17 -0
  187. package/src/machine/errors.ts +34 -0
  188. package/src/machine/index.ts +173 -0
  189. package/src/mcp/index.ts +185 -0
  190. package/src/types/index.ts +134 -0
  191. 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;;AAQH,8EA0NC;AAhOD,iDAAqE;AAErE;;;GAGG;AACH,SAAgB,iCAAiC,CAAC,KAAgB,EAAE,iBAAyB;IAC3F,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC;IAC7B,MAAM,UAAU,GAAG,IAAA,0BAAW,EAAC,SAAS,CAAC,CAAC;IAC1C,MAAM,UAAU,GAAG,IAAA,0BAAW,EAAC,SAAS,CAAC,CAAC;IAC1C,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;IAEpC,OAAO;;;WAGE,UAAU,YAAY,iBAAiB;;yBAEzB,SAAS;;cAEpB,UAAU;YACZ,UAAU;;YAEV,UAAU;;;;;;;;;;;;;;;;;;;;kCAoBY,UAAU;;;;;;;;;;;cAW9B,UAAU;YACZ,UAAU;;YAEV,UAAU;;;;;;;;;;;;;;;;;;;;;0BAqBI,UAAU;;;;;;;;;eASrB,UAAU;YACb,UAAU;;YAEV,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;uBAyBC,UAAU;;;;;;;;;;;;;;;;cAgBnB,UAAU;YACZ,UAAU;;YAEV,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;uBA4CC,UAAU;;;;;;;;;;;;;;;;iBAgBhB,UAAU;YACf,UAAU;;YAEV,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;CA2BrB,CAAC;AACF,CAAC"}
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 モデルディレクトリのパス(デフォルト: "lib/models")
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;CACrC;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,CAqD1E;AA+cD;;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,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"}
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
- let schemaMatch = content.match(/export\s+const\s+(\w+Schema)\s*=/);
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 = content.match(/export\s+const\s+(\w+)\s*=\s*z\.object\s*\(/);
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
- // ローカル相対インポートを絶対パスの .mjs import 文に変換して保持
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
- if (fs.existsSync(resolvedPath)) {
290
- // インポート先の内容を読み込み、インライン化する
291
- let refContent = readFileSync(resolvedPath, 'utf8');
292
- // インポート先の import 文を除去
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
- const result = execSync(`npx tsx "${tempScript}"`, {
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 モデルディレクトリのパス(デフォルト: "lib/models")
727
+ * @param modelsDir モデルディレクトリのパス(デフォルト: "shared/models")
534
728
  * @returns モデル情報の配列
535
729
  */
536
730
  async function getAllModels(modelsDir = "shared/models") {