@gqlkit-ts/cli 0.1.0 → 0.2.0

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 (234) hide show
  1. package/dist/auto-type-generator/auto-type-generator.d.ts +19 -3
  2. package/dist/auto-type-generator/auto-type-generator.d.ts.map +1 -1
  3. package/dist/auto-type-generator/auto-type-generator.js +236 -49
  4. package/dist/auto-type-generator/auto-type-generator.js.map +1 -1
  5. package/dist/auto-type-generator/index.d.ts +1 -1
  6. package/dist/auto-type-generator/index.d.ts.map +1 -1
  7. package/dist/auto-type-generator/index.js.map +1 -1
  8. package/dist/auto-type-generator/inline-enum-collector.d.ts +31 -0
  9. package/dist/auto-type-generator/inline-enum-collector.d.ts.map +1 -0
  10. package/dist/auto-type-generator/inline-enum-collector.js +157 -0
  11. package/dist/auto-type-generator/inline-enum-collector.js.map +1 -0
  12. package/dist/auto-type-generator/naming-convention.d.ts +4 -0
  13. package/dist/auto-type-generator/naming-convention.d.ts.map +1 -1
  14. package/dist/auto-type-generator/naming-convention.js +8 -0
  15. package/dist/auto-type-generator/naming-convention.js.map +1 -1
  16. package/dist/gen-orchestrator/orchestrator.d.ts.map +1 -1
  17. package/dist/gen-orchestrator/orchestrator.js +106 -8
  18. package/dist/gen-orchestrator/orchestrator.js.map +1 -1
  19. package/dist/resolver-extractor/extract-resolvers.d.ts +10 -1
  20. package/dist/resolver-extractor/extract-resolvers.d.ts.map +1 -1
  21. package/dist/resolver-extractor/extractor/define-api-extractor.d.ts +11 -1
  22. package/dist/resolver-extractor/extractor/define-api-extractor.d.ts.map +1 -1
  23. package/dist/resolver-extractor/extractor/define-api-extractor.js +88 -264
  24. package/dist/resolver-extractor/extractor/define-api-extractor.js.map +1 -1
  25. package/dist/schema-generator/emitter/code-emitter.d.ts +10 -2
  26. package/dist/schema-generator/emitter/code-emitter.d.ts.map +1 -1
  27. package/dist/schema-generator/emitter/code-emitter.js +19 -4
  28. package/dist/schema-generator/emitter/code-emitter.js.map +1 -1
  29. package/dist/schema-generator/generate-schema.d.ts.map +1 -1
  30. package/dist/schema-generator/generate-schema.js +18 -1
  31. package/dist/schema-generator/generate-schema.js.map +1 -1
  32. package/dist/schema-generator/integrator/result-integrator.d.ts +20 -0
  33. package/dist/schema-generator/integrator/result-integrator.d.ts.map +1 -1
  34. package/dist/schema-generator/integrator/result-integrator.js +42 -0
  35. package/dist/schema-generator/integrator/result-integrator.js.map +1 -1
  36. package/dist/shared/constants.d.ts +0 -16
  37. package/dist/shared/constants.d.ts.map +1 -1
  38. package/dist/shared/constants.js +0 -19
  39. package/dist/shared/constants.js.map +1 -1
  40. package/dist/shared/directive-detector.d.ts.map +1 -1
  41. package/dist/shared/directive-detector.js +8 -15
  42. package/dist/shared/directive-detector.js.map +1 -1
  43. package/dist/shared/file-scanner.d.ts.map +1 -1
  44. package/dist/shared/file-scanner.js +5 -3
  45. package/dist/shared/file-scanner.js.map +1 -1
  46. package/dist/shared/index.d.ts +1 -1
  47. package/dist/shared/index.d.ts.map +1 -1
  48. package/dist/shared/index.js +1 -3
  49. package/dist/shared/index.js.map +1 -1
  50. package/dist/shared/interface-detector.d.ts +3 -2
  51. package/dist/shared/interface-detector.d.ts.map +1 -1
  52. package/dist/shared/interface-detector.js +54 -11
  53. package/dist/shared/interface-detector.js.map +1 -1
  54. package/dist/shared/path-utils.d.ts +2 -0
  55. package/dist/shared/path-utils.d.ts.map +1 -0
  56. package/dist/shared/path-utils.js +4 -0
  57. package/dist/shared/path-utils.js.map +1 -0
  58. package/dist/shared/type-converter.d.ts.map +1 -1
  59. package/dist/shared/type-converter.js +11 -0
  60. package/dist/shared/type-converter.js.map +1 -1
  61. package/dist/shared/typescript-utils.d.ts +34 -7
  62. package/dist/shared/typescript-utils.d.ts.map +1 -1
  63. package/dist/shared/typescript-utils.js +72 -24
  64. package/dist/shared/typescript-utils.js.map +1 -1
  65. package/dist/type-extractor/collector/scalar-collector.d.ts.map +1 -1
  66. package/dist/type-extractor/collector/scalar-collector.js +4 -14
  67. package/dist/type-extractor/collector/scalar-collector.js.map +1 -1
  68. package/dist/type-extractor/converter/graphql-converter.d.ts.map +1 -1
  69. package/dist/type-extractor/converter/graphql-converter.js +42 -3
  70. package/dist/type-extractor/converter/graphql-converter.js.map +1 -1
  71. package/dist/type-extractor/extractor/field-type-resolver.d.ts +28 -0
  72. package/dist/type-extractor/extractor/field-type-resolver.d.ts.map +1 -0
  73. package/dist/type-extractor/extractor/field-type-resolver.js +394 -0
  74. package/dist/type-extractor/extractor/field-type-resolver.js.map +1 -0
  75. package/dist/type-extractor/extractor/type-extractor.d.ts +12 -3
  76. package/dist/type-extractor/extractor/type-extractor.d.ts.map +1 -1
  77. package/dist/type-extractor/extractor/type-extractor.js +176 -210
  78. package/dist/type-extractor/extractor/type-extractor.js.map +1 -1
  79. package/dist/type-extractor/extractor/type-name-collector.d.ts +24 -0
  80. package/dist/type-extractor/extractor/type-name-collector.d.ts.map +1 -0
  81. package/dist/type-extractor/extractor/type-name-collector.js +102 -0
  82. package/dist/type-extractor/extractor/type-name-collector.js.map +1 -0
  83. package/dist/type-extractor/mapper/scalar-base-type-mapper.d.ts +89 -0
  84. package/dist/type-extractor/mapper/scalar-base-type-mapper.d.ts.map +1 -0
  85. package/dist/type-extractor/mapper/scalar-base-type-mapper.js +158 -0
  86. package/dist/type-extractor/mapper/scalar-base-type-mapper.js.map +1 -0
  87. package/dist/type-extractor/types/diagnostics.d.ts +1 -1
  88. package/dist/type-extractor/types/diagnostics.d.ts.map +1 -1
  89. package/dist/type-extractor/types/graphql.d.ts +2 -0
  90. package/dist/type-extractor/types/graphql.d.ts.map +1 -1
  91. package/dist/type-extractor/types/index.d.ts +2 -1
  92. package/dist/type-extractor/types/index.d.ts.map +1 -1
  93. package/dist/type-extractor/types/index.js +1 -1
  94. package/dist/type-extractor/types/index.js.map +1 -1
  95. package/dist/type-extractor/types/ts-type-reference-factory.d.ts +39 -0
  96. package/dist/type-extractor/types/ts-type-reference-factory.d.ts.map +1 -0
  97. package/dist/type-extractor/types/ts-type-reference-factory.js +69 -0
  98. package/dist/type-extractor/types/ts-type-reference-factory.js.map +1 -0
  99. package/dist/type-extractor/types/typescript.d.ts +21 -1
  100. package/dist/type-extractor/types/typescript.d.ts.map +1 -1
  101. package/dist/type-extractor/validator/type-validator.js +1 -1
  102. package/dist/type-extractor/validator/type-validator.js.map +1 -1
  103. package/docs/index.md +1 -0
  104. package/docs/integration/drizzle.md +191 -0
  105. package/docs/schema/enums.md +208 -0
  106. package/docs/schema/fields.md +2 -0
  107. package/docs/schema/inputs.md +2 -0
  108. package/docs/schema/objects.md +2 -0
  109. package/docs/schema/queries-mutations.md +2 -0
  110. package/package.json +10 -3
  111. package/src/auto-type-generator/auto-type-generator.ts +925 -0
  112. package/src/auto-type-generator/index.ts +21 -0
  113. package/src/auto-type-generator/inline-enum-collector.ts +290 -0
  114. package/src/auto-type-generator/name-collision-validator.ts +119 -0
  115. package/src/auto-type-generator/naming-convention.ts +126 -0
  116. package/src/cli.ts +11 -0
  117. package/src/commands/gen.ts +141 -0
  118. package/src/commands/main.ts +5 -0
  119. package/src/config/define-config.ts +28 -0
  120. package/src/config/index.ts +7 -0
  121. package/src/config/types.ts +144 -0
  122. package/src/config-loader/index.ts +14 -0
  123. package/src/config-loader/loader.ts +143 -0
  124. package/src/config-loader/validator.ts +672 -0
  125. package/src/gen-orchestrator/hook-executor/hook-executor.ts +117 -0
  126. package/src/gen-orchestrator/orchestrator.ts +784 -0
  127. package/src/gen-orchestrator/reporter/diagnostic-reporter.ts +44 -0
  128. package/src/gen-orchestrator/reporter/progress-reporter.ts +61 -0
  129. package/src/gen-orchestrator/writer/file-writer.ts +38 -0
  130. package/src/index.ts +2 -0
  131. package/src/resolver-extractor/extract-resolvers.ts +63 -0
  132. package/src/resolver-extractor/extractor/define-api-extractor.ts +806 -0
  133. package/src/resolver-extractor/index.ts +19 -0
  134. package/src/resolver-extractor/validator/abstract-resolver-validator.ts +251 -0
  135. package/src/resolver-extractor/validator/only-validator.ts +158 -0
  136. package/src/schema-generator/builder/ast-builder.ts +706 -0
  137. package/src/schema-generator/emitter/code-emitter.ts +351 -0
  138. package/src/schema-generator/emitter/sdl-emitter.ts +13 -0
  139. package/src/schema-generator/generate-schema.ts +170 -0
  140. package/src/schema-generator/index.ts +19 -0
  141. package/src/schema-generator/integrator/result-integrator.ts +690 -0
  142. package/src/schema-generator/pruner/schema-pruner.ts +112 -0
  143. package/src/schema-generator/resolver-collector/resolver-collector.ts +123 -0
  144. package/src/shared/constants.ts +122 -0
  145. package/src/shared/default-value-detector.ts +172 -0
  146. package/src/shared/diagnostics.ts +35 -0
  147. package/src/shared/directive-definition-extractor.ts +564 -0
  148. package/src/shared/directive-detector.ts +556 -0
  149. package/src/shared/file-scanner.ts +170 -0
  150. package/src/shared/index.ts +32 -0
  151. package/src/shared/inline-object-extractor.ts +102 -0
  152. package/src/shared/inline-object-utils.ts +23 -0
  153. package/src/shared/interface-detector.ts +176 -0
  154. package/src/shared/interface-validator.ts +211 -0
  155. package/src/shared/metadata-detector.ts +443 -0
  156. package/src/shared/path-utils.ts +3 -0
  157. package/src/shared/program-factory.ts +51 -0
  158. package/src/shared/source-location.ts +27 -0
  159. package/src/shared/tsconfig-loader.ts +126 -0
  160. package/src/shared/tsdoc-parser.ts +155 -0
  161. package/src/shared/type-converter.ts +99 -0
  162. package/src/shared/typescript-utils.ts +246 -0
  163. package/src/type-extractor/collector/result-collector.ts +57 -0
  164. package/src/type-extractor/collector/scalar-collector.ts +254 -0
  165. package/src/type-extractor/converter/field-eligibility.ts +112 -0
  166. package/src/type-extractor/converter/graphql-converter.ts +459 -0
  167. package/src/type-extractor/extract-types.ts +1 -0
  168. package/src/type-extractor/extractor/field-type-resolver.ts +569 -0
  169. package/src/type-extractor/extractor/type-extractor.ts +1567 -0
  170. package/src/type-extractor/extractor/type-name-collector.ts +130 -0
  171. package/src/type-extractor/index.ts +20 -0
  172. package/src/type-extractor/mapper/scalar-base-type-mapper.ts +265 -0
  173. package/src/type-extractor/types/diagnostics.ts +99 -0
  174. package/src/type-extractor/types/graphql.ts +55 -0
  175. package/src/type-extractor/types/index.ts +37 -0
  176. package/src/type-extractor/types/ts-type-reference-factory.ts +137 -0
  177. package/src/type-extractor/types/typescript.ts +133 -0
  178. package/src/type-extractor/validator/type-validator.ts +77 -0
  179. package/dist/auto-type-generator/auto-type-generator.test.d.ts +0 -2
  180. package/dist/auto-type-generator/auto-type-generator.test.d.ts.map +0 -1
  181. package/dist/auto-type-generator/auto-type-generator.test.js +0 -613
  182. package/dist/auto-type-generator/auto-type-generator.test.js.map +0 -1
  183. package/dist/auto-type-generator/name-collision-validator.test.d.ts +0 -2
  184. package/dist/auto-type-generator/name-collision-validator.test.d.ts.map +0 -1
  185. package/dist/auto-type-generator/name-collision-validator.test.js +0 -358
  186. package/dist/auto-type-generator/name-collision-validator.test.js.map +0 -1
  187. package/dist/auto-type-generator/naming-convention.test.d.ts +0 -2
  188. package/dist/auto-type-generator/naming-convention.test.d.ts.map +0 -1
  189. package/dist/auto-type-generator/naming-convention.test.js +0 -132
  190. package/dist/auto-type-generator/naming-convention.test.js.map +0 -1
  191. package/dist/commands/gen.test.d.ts +0 -2
  192. package/dist/commands/gen.test.d.ts.map +0 -1
  193. package/dist/commands/gen.test.js +0 -226
  194. package/dist/commands/gen.test.js.map +0 -1
  195. package/dist/config-loader/loader.test.d.ts +0 -2
  196. package/dist/config-loader/loader.test.d.ts.map +0 -1
  197. package/dist/config-loader/loader.test.js +0 -123
  198. package/dist/config-loader/loader.test.js.map +0 -1
  199. package/dist/config-loader/validator.test.d.ts +0 -2
  200. package/dist/config-loader/validator.test.d.ts.map +0 -1
  201. package/dist/config-loader/validator.test.js +0 -846
  202. package/dist/config-loader/validator.test.js.map +0 -1
  203. package/dist/gen-orchestrator/golden.test.d.ts +0 -2
  204. package/dist/gen-orchestrator/golden.test.d.ts.map +0 -1
  205. package/dist/gen-orchestrator/golden.test.js +0 -102
  206. package/dist/gen-orchestrator/golden.test.js.map +0 -1
  207. package/dist/gen-orchestrator/hook-executor/hook-executor.test.d.ts +0 -2
  208. package/dist/gen-orchestrator/hook-executor/hook-executor.test.d.ts.map +0 -1
  209. package/dist/gen-orchestrator/hook-executor/hook-executor.test.js +0 -167
  210. package/dist/gen-orchestrator/hook-executor/hook-executor.test.js.map +0 -1
  211. package/dist/gen-orchestrator/reporter/progress-reporter.test.d.ts +0 -2
  212. package/dist/gen-orchestrator/reporter/progress-reporter.test.d.ts.map +0 -1
  213. package/dist/gen-orchestrator/reporter/progress-reporter.test.js +0 -74
  214. package/dist/gen-orchestrator/reporter/progress-reporter.test.js.map +0 -1
  215. package/dist/resolver-extractor/validator/only-validator.test.d.ts +0 -8
  216. package/dist/resolver-extractor/validator/only-validator.test.d.ts.map +0 -1
  217. package/dist/resolver-extractor/validator/only-validator.test.js +0 -352
  218. package/dist/resolver-extractor/validator/only-validator.test.js.map +0 -1
  219. package/dist/schema-generator/builder/ast-builder.test.d.ts +0 -2
  220. package/dist/schema-generator/builder/ast-builder.test.d.ts.map +0 -1
  221. package/dist/schema-generator/builder/ast-builder.test.js +0 -469
  222. package/dist/schema-generator/builder/ast-builder.test.js.map +0 -1
  223. package/dist/shared/file-scanner.test.d.ts +0 -2
  224. package/dist/shared/file-scanner.test.d.ts.map +0 -1
  225. package/dist/shared/file-scanner.test.js +0 -138
  226. package/dist/shared/file-scanner.test.js.map +0 -1
  227. package/dist/shared/interface-validator.test.d.ts +0 -2
  228. package/dist/shared/interface-validator.test.d.ts.map +0 -1
  229. package/dist/shared/interface-validator.test.js +0 -145
  230. package/dist/shared/interface-validator.test.js.map +0 -1
  231. package/dist/type-extractor/types/typescript.test.d.ts +0 -2
  232. package/dist/type-extractor/types/typescript.test.d.ts.map +0 -1
  233. package/dist/type-extractor/types/typescript.test.js +0 -287
  234. package/dist/type-extractor/types/typescript.test.js.map +0 -1
@@ -0,0 +1,459 @@
1
+ import {
2
+ BUILT_IN_SCALARS,
3
+ isBuiltInScalar,
4
+ PRIMITIVE_TYPE_MAP,
5
+ } from "../../shared/constants.js";
6
+ import { convertTsTypeToGraphQLType } from "../../shared/type-converter.js";
7
+ import type {
8
+ Diagnostic,
9
+ EnumMemberInfo,
10
+ EnumValueInfo,
11
+ ExtractedTypeInfo,
12
+ FieldInfo,
13
+ GraphQLTypeInfo,
14
+ InlineObjectMember,
15
+ InlineObjectProperty,
16
+ SourceLocation,
17
+ } from "../types/index.js";
18
+ import {
19
+ isEligibleAsEnumValue,
20
+ isEligibleAsInputObjectField,
21
+ isEligibleAsObjectField,
22
+ } from "./field-eligibility.js";
23
+
24
+ export interface ConversionResult {
25
+ readonly types: ReadonlyArray<GraphQLTypeInfo>;
26
+ readonly diagnostics: ReadonlyArray<Diagnostic>;
27
+ }
28
+
29
+ const RESERVED_TYPE_NAMES = new Set([
30
+ ...BUILT_IN_SCALARS,
31
+ "Query",
32
+ "Mutation",
33
+ "Subscription",
34
+ ]);
35
+
36
+ function isInputTypeName(name: string): boolean {
37
+ return name.endsWith("Input");
38
+ }
39
+
40
+ function toScreamingSnakeCase(value: string): string {
41
+ return value
42
+ .replace(/[-\s]+/g, "_")
43
+ .replace(/([a-z])([A-Z])/g, "$1_$2")
44
+ .replace(/([A-Z]+)([A-Z][a-z])/g, "$1_$2")
45
+ .toUpperCase();
46
+ }
47
+
48
+ interface ConvertEnumMembersParams {
49
+ readonly members: ReadonlyArray<EnumMemberInfo>;
50
+ readonly enumName: string;
51
+ readonly sourceFile: string;
52
+ readonly enumLocation: SourceLocation;
53
+ }
54
+
55
+ interface ConvertEnumMembersResult {
56
+ readonly values: ReadonlyArray<EnumValueInfo>;
57
+ readonly diagnostics: ReadonlyArray<Diagnostic>;
58
+ readonly isNumeric: boolean;
59
+ readonly needsMapping: boolean;
60
+ readonly hasError: boolean;
61
+ }
62
+
63
+ function convertEnumMembers(
64
+ params: ConvertEnumMembersParams,
65
+ ): ConvertEnumMembersResult {
66
+ const { members, enumName, sourceFile, enumLocation } = params;
67
+ const values: EnumValueInfo[] = [];
68
+ const diagnostics: Diagnostic[] = [];
69
+ let isNumeric = false;
70
+ let needsMapping = false;
71
+
72
+ const convertedNameToOriginals = new Map<string, string[]>();
73
+
74
+ for (const member of members) {
75
+ const convertedName = toScreamingSnakeCase(member.name);
76
+
77
+ const eligibility = isEligibleAsEnumValue(convertedName, member.name);
78
+ if (!eligibility.eligible) {
79
+ diagnostics.push({
80
+ code: "SKIPPED_ENUM_VALUE",
81
+ message: eligibility.skipReason!.message,
82
+ severity: "warning",
83
+ location: member.sourceLocation ?? {
84
+ file: sourceFile,
85
+ line: 1,
86
+ column: 1,
87
+ },
88
+ });
89
+ continue;
90
+ }
91
+
92
+ if (member.numericValue !== null) {
93
+ isNumeric = true;
94
+ }
95
+
96
+ if (convertedName !== member.value) {
97
+ needsMapping = true;
98
+ }
99
+
100
+ const existingOriginals = convertedNameToOriginals.get(convertedName) ?? [];
101
+ existingOriginals.push(member.value);
102
+ convertedNameToOriginals.set(convertedName, existingOriginals);
103
+
104
+ values.push({
105
+ name: convertedName,
106
+ originalValue: member.value,
107
+ numericValue: member.numericValue,
108
+ description: member.description,
109
+ deprecated: member.deprecated,
110
+ });
111
+ }
112
+
113
+ let hasError = false;
114
+ for (const [convertedName, originals] of convertedNameToOriginals) {
115
+ if (originals.length > 1) {
116
+ diagnostics.push({
117
+ code: "DUPLICATE_ENUM_VALUE_AFTER_CONVERSION",
118
+ message: `Enum '${enumName}' has duplicate value '${convertedName}' after conversion (from '${originals.join("' and '")}')`,
119
+ severity: "error",
120
+ location: enumLocation,
121
+ });
122
+ hasError = true;
123
+ }
124
+ }
125
+
126
+ return { values, diagnostics, isNumeric, needsMapping, hasError };
127
+ }
128
+
129
+ interface ConvertFieldsResult {
130
+ readonly fields: FieldInfo[];
131
+ readonly diagnostics: Diagnostic[];
132
+ }
133
+
134
+ function convertFields(
135
+ extracted: ExtractedTypeInfo,
136
+ isInput: boolean,
137
+ ): ConvertFieldsResult {
138
+ const fields: FieldInfo[] = [];
139
+ const diagnostics: Diagnostic[] = [];
140
+
141
+ for (const field of extracted.fields) {
142
+ const eligibility = isInput
143
+ ? isEligibleAsInputObjectField(field.name)
144
+ : isEligibleAsObjectField(field.name);
145
+
146
+ if (!eligibility.eligible) {
147
+ diagnostics.push({
148
+ code: "SKIPPED_FIELD",
149
+ message: eligibility.skipReason!.message,
150
+ severity: "warning",
151
+ location: field.sourceLocation ?? {
152
+ file: extracted.metadata.sourceFile,
153
+ line: 1,
154
+ column: 1,
155
+ },
156
+ });
157
+ continue;
158
+ }
159
+
160
+ fields.push({
161
+ name: field.name,
162
+ type: convertTsTypeToGraphQLType(field.tsType, field.optional),
163
+ description: field.description,
164
+ deprecated: field.deprecated,
165
+ directives: field.directives,
166
+ defaultValue: field.defaultValue,
167
+ });
168
+ }
169
+
170
+ return { fields, diagnostics };
171
+ }
172
+
173
+ function isValidOneOfFieldType(
174
+ typeName: string,
175
+ typeMap: Map<string, ExtractedTypeInfo>,
176
+ ): boolean {
177
+ if (isBuiltInScalar(typeName)) {
178
+ return true;
179
+ }
180
+ if (PRIMITIVE_TYPE_MAP[typeName]) {
181
+ return true;
182
+ }
183
+ const referencedType = typeMap.get(typeName);
184
+ if (referencedType) {
185
+ if (referencedType.metadata.kind === "enum") {
186
+ return true;
187
+ }
188
+ if (isInputTypeName(referencedType.metadata.name)) {
189
+ return true;
190
+ }
191
+ }
192
+ return false;
193
+ }
194
+
195
+ interface OneOfValidationResult {
196
+ readonly valid: boolean;
197
+ readonly diagnostics: Diagnostic[];
198
+ readonly fields: FieldInfo[];
199
+ }
200
+
201
+ function validateAndConvertInlineObjectMembers(
202
+ members: ReadonlyArray<InlineObjectMember>,
203
+ typeName: string,
204
+ location: SourceLocation,
205
+ typeMap: Map<string, ExtractedTypeInfo>,
206
+ ): OneOfValidationResult {
207
+ const diagnostics: Diagnostic[] = [];
208
+ const fields: FieldInfo[] = [];
209
+ const propertyNames = new Set<string>();
210
+ const allProperties: InlineObjectProperty[] = [];
211
+
212
+ for (let i = 0; i < members.length; i++) {
213
+ const member = members[i]!;
214
+ const props = member.properties;
215
+
216
+ if (props.length === 0) {
217
+ diagnostics.push({
218
+ code: "ONEOF_EMPTY_OBJECT",
219
+ message: `OneOf input '${typeName}' member at index ${i} is an empty object. Each member must have exactly one property.`,
220
+ severity: "error",
221
+ location,
222
+ });
223
+ continue;
224
+ }
225
+
226
+ if (props.length > 1) {
227
+ diagnostics.push({
228
+ code: "ONEOF_MULTIPLE_PROPERTIES",
229
+ message: `OneOf input '${typeName}' member at index ${i} has ${props.length} properties. Each member must have exactly one property.`,
230
+ severity: "error",
231
+ location,
232
+ });
233
+ continue;
234
+ }
235
+
236
+ allProperties.push(props[0]!);
237
+ }
238
+
239
+ for (const prop of allProperties) {
240
+ if (propertyNames.has(prop.propertyName)) {
241
+ diagnostics.push({
242
+ code: "ONEOF_DUPLICATE_PROPERTY",
243
+ message: `OneOf input '${typeName}' has duplicate property name '${prop.propertyName}'.`,
244
+ severity: "error",
245
+ location,
246
+ });
247
+ continue;
248
+ }
249
+ propertyNames.add(prop.propertyName);
250
+
251
+ const graphqlType = convertTsTypeToGraphQLType(prop.propertyType, false);
252
+ const referencedTypeName = graphqlType.typeName;
253
+
254
+ if (!isValidOneOfFieldType(referencedTypeName, typeMap)) {
255
+ diagnostics.push({
256
+ code: "ONEOF_INVALID_FIELD_TYPE",
257
+ message: `OneOf input '${typeName}' field '${prop.propertyName}' has invalid type '${referencedTypeName}'. Only scalar types and Input Object types are allowed.`,
258
+ severity: "error",
259
+ location,
260
+ });
261
+ continue;
262
+ }
263
+
264
+ fields.push({
265
+ name: prop.propertyName,
266
+ type: {
267
+ typeName: graphqlType.typeName,
268
+ nullable: true,
269
+ list: graphqlType.list,
270
+ listItemNullable: graphqlType.listItemNullable,
271
+ },
272
+ description: prop.description,
273
+ deprecated: prop.deprecated,
274
+ directives: null,
275
+ defaultValue: null,
276
+ });
277
+ }
278
+
279
+ return {
280
+ valid: diagnostics.length === 0,
281
+ diagnostics,
282
+ fields: fields.sort((a, b) => a.name.localeCompare(b.name)),
283
+ };
284
+ }
285
+
286
+ export function convertToGraphQL(
287
+ extractedTypes: ReadonlyArray<ExtractedTypeInfo>,
288
+ ): ConversionResult {
289
+ const types: GraphQLTypeInfo[] = [];
290
+ const diagnostics: Diagnostic[] = [];
291
+
292
+ const typeMap = new Map<string, ExtractedTypeInfo>();
293
+ for (const extracted of extractedTypes) {
294
+ typeMap.set(extracted.metadata.name, extracted);
295
+ }
296
+
297
+ for (const extracted of extractedTypes) {
298
+ const { metadata } = extracted;
299
+
300
+ if (RESERVED_TYPE_NAMES.has(metadata.name)) {
301
+ diagnostics.push({
302
+ code: "RESERVED_TYPE_NAME",
303
+ message: `Type name '${metadata.name}' conflicts with a GraphQL built-in type`,
304
+ severity: "error",
305
+ location: { file: metadata.sourceFile, line: 1, column: 1 },
306
+ });
307
+ }
308
+
309
+ if (metadata.kind === "enum") {
310
+ if (isInputTypeName(metadata.name)) {
311
+ diagnostics.push({
312
+ code: "INVALID_INPUT_TYPE",
313
+ message: `Type '${metadata.name}' ends with 'Input' but is an enum type. Input types must be object types.`,
314
+ severity: "error",
315
+ location: { file: metadata.sourceFile, line: 1, column: 1 },
316
+ });
317
+ }
318
+
319
+ const enumLocation: SourceLocation = metadata.sourceLocation ?? {
320
+ file: metadata.sourceFile,
321
+ line: 1,
322
+ column: 1,
323
+ };
324
+ const {
325
+ values: enumValues,
326
+ diagnostics: enumDiagnostics,
327
+ isNumeric,
328
+ needsMapping,
329
+ hasError,
330
+ } = convertEnumMembers({
331
+ members: extracted.enumMembers ?? [],
332
+ enumName: metadata.name,
333
+ sourceFile: metadata.sourceFile,
334
+ enumLocation,
335
+ });
336
+ diagnostics.push(...enumDiagnostics);
337
+
338
+ if (hasError) {
339
+ continue;
340
+ }
341
+
342
+ types.push({
343
+ name: metadata.name,
344
+ kind: "Enum",
345
+ fields: null,
346
+ unionMembers: null,
347
+ enumValues,
348
+ isNumericEnum: isNumeric,
349
+ needsStringEnumMapping: !isNumeric && needsMapping,
350
+ implementedInterfaces: null,
351
+ sourceFile: metadata.sourceFile,
352
+ description: metadata.description,
353
+ deprecated: metadata.deprecated,
354
+ directives: metadata.directives,
355
+ });
356
+ } else if (metadata.kind === "graphqlInterface") {
357
+ const { fields, diagnostics: fieldDiagnostics } = convertFields(
358
+ extracted,
359
+ false,
360
+ );
361
+ diagnostics.push(...fieldDiagnostics);
362
+
363
+ types.push({
364
+ name: metadata.name,
365
+ kind: "Interface",
366
+ fields,
367
+ unionMembers: null,
368
+ enumValues: null,
369
+ isNumericEnum: false,
370
+ needsStringEnumMapping: false,
371
+ implementedInterfaces: extracted.implementedInterfaces
372
+ ? [...extracted.implementedInterfaces]
373
+ : null,
374
+ sourceFile: metadata.sourceFile,
375
+ description: metadata.description,
376
+ deprecated: metadata.deprecated,
377
+ directives: metadata.directives,
378
+ });
379
+ } else if (metadata.kind === "union") {
380
+ if (isInputTypeName(metadata.name)) {
381
+ const inlineObjectMembers = extracted.inlineObjectMembers ?? [];
382
+
383
+ if (inlineObjectMembers.length > 0) {
384
+ const location = { file: metadata.sourceFile, line: 1, column: 1 };
385
+ const validationResult = validateAndConvertInlineObjectMembers(
386
+ inlineObjectMembers,
387
+ metadata.name,
388
+ location,
389
+ typeMap,
390
+ );
391
+
392
+ diagnostics.push(...validationResult.diagnostics);
393
+
394
+ if (validationResult.valid) {
395
+ types.push({
396
+ name: metadata.name,
397
+ kind: "OneOfInputObject",
398
+ fields: validationResult.fields,
399
+ unionMembers: null,
400
+ enumValues: null,
401
+ isNumericEnum: false,
402
+ needsStringEnumMapping: false,
403
+ implementedInterfaces: null,
404
+ sourceFile: metadata.sourceFile,
405
+ description: metadata.description,
406
+ deprecated: metadata.deprecated,
407
+ directives: metadata.directives,
408
+ });
409
+ }
410
+ }
411
+ } else {
412
+ const unionMembers = extracted.unionMembers
413
+ ? [...extracted.unionMembers].sort()
414
+ : [];
415
+
416
+ types.push({
417
+ name: metadata.name,
418
+ kind: "Union",
419
+ fields: null,
420
+ unionMembers,
421
+ enumValues: null,
422
+ isNumericEnum: false,
423
+ needsStringEnumMapping: false,
424
+ implementedInterfaces: null,
425
+ sourceFile: metadata.sourceFile,
426
+ description: metadata.description,
427
+ deprecated: null,
428
+ directives: metadata.directives,
429
+ });
430
+ }
431
+ } else {
432
+ const isInput = isInputTypeName(metadata.name);
433
+ const { fields, diagnostics: fieldDiagnostics } = convertFields(
434
+ extracted,
435
+ isInput,
436
+ );
437
+ diagnostics.push(...fieldDiagnostics);
438
+
439
+ types.push({
440
+ name: metadata.name,
441
+ kind: isInput ? "InputObject" : "Object",
442
+ fields,
443
+ unionMembers: null,
444
+ enumValues: null,
445
+ isNumericEnum: false,
446
+ needsStringEnumMapping: false,
447
+ implementedInterfaces: extracted.implementedInterfaces
448
+ ? [...extracted.implementedInterfaces]
449
+ : null,
450
+ sourceFile: metadata.sourceFile,
451
+ description: metadata.description,
452
+ deprecated: metadata.deprecated,
453
+ directives: metadata.directives,
454
+ });
455
+ }
456
+ }
457
+
458
+ return { types, diagnostics };
459
+ }
@@ -0,0 +1 @@
1
+ export type { ExtractTypesResult } from "./collector/result-collector.js";