relay-compiler 8.0.0 → 10.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (220) hide show
  1. package/bin/RelayCompilerBin.js.flow +169 -0
  2. package/bin/RelayCompilerMain.js.flow +515 -0
  3. package/bin/__fixtures__/plugin-module.js.flow +17 -0
  4. package/bin/relay-compiler +8930 -8967
  5. package/codegen/CodegenDirectory.js.flow +375 -0
  6. package/codegen/CodegenRunner.js.flow +432 -0
  7. package/codegen/CodegenTypes.js.flow +28 -0
  8. package/codegen/CodegenWatcher.js.flow +254 -0
  9. package/codegen/NormalizationCodeGenerator.js.flow +563 -0
  10. package/codegen/ReaderCodeGenerator.js.flow +477 -0
  11. package/codegen/RelayCodeGenerator.js.flow +85 -0
  12. package/codegen/RelayFileWriter.js.flow +365 -0
  13. package/codegen/SourceControl.js.flow +58 -0
  14. package/codegen/compileRelayArtifacts.js.flow +182 -0
  15. package/codegen/createPrintRequireModuleDependency.js.flow +21 -0
  16. package/codegen/sortObjectByKey.js.flow +25 -0
  17. package/codegen/writeRelayGeneratedFile.js.flow +223 -0
  18. package/core/ASTCache.js.flow +73 -0
  19. package/core/ASTConvert.js.flow +233 -0
  20. package/core/CompilerContext.js.flow +190 -0
  21. package/core/CompilerError.js.flow +250 -0
  22. package/core/DotGraphQLParser.js.flow +39 -0
  23. package/core/GraphQLCompilerProfiler.js.flow +341 -0
  24. package/core/GraphQLDerivedFromMetadata.js.flow +36 -0
  25. package/core/GraphQLWatchmanClient.js.flow +111 -0
  26. package/core/IR.js.flow +327 -0
  27. package/core/IRPrinter.js.flow +482 -0
  28. package/core/IRTransformer.js.flow +377 -0
  29. package/core/IRValidator.js.flow +260 -0
  30. package/core/IRVisitor.js.flow +150 -0
  31. package/core/JSModuleParser.js.flow +24 -0
  32. package/core/RelayCompilerScope.js.flow +199 -0
  33. package/core/RelayFindGraphQLTags.js.flow +119 -0
  34. package/core/RelayGraphQLEnumsGenerator.js.flow +55 -0
  35. package/core/RelayIRTransforms.js.flow +131 -0
  36. package/core/RelayParser.js.flow +1731 -0
  37. package/core/RelaySourceModuleParser.js.flow +135 -0
  38. package/core/Schema.js.flow +1983 -0
  39. package/core/SchemaUtils.js.flow +120 -0
  40. package/core/filterContextForNode.js.flow +50 -0
  41. package/core/getFieldDefinition.js.flow +156 -0
  42. package/core/getIdentifierForArgumentValue.js.flow +49 -0
  43. package/core/getIdentifierForSelection.js.flow +69 -0
  44. package/core/getLiteralArgumentValues.js.flow +32 -0
  45. package/core/getNormalizationOperationName.js.flow +19 -0
  46. package/core/inferRootArgumentDefinitions.js.flow +323 -0
  47. package/index.js +1 -1
  48. package/index.js.flow +200 -0
  49. package/language/RelayLanguagePluginInterface.js.flow +283 -0
  50. package/language/javascript/FindGraphQLTags.js.flow +232 -0
  51. package/language/javascript/RelayFlowBabelFactories.js.flow +180 -0
  52. package/language/javascript/RelayFlowGenerator.js.flow +1042 -0
  53. package/language/javascript/RelayFlowTypeTransformers.js.flow +184 -0
  54. package/language/javascript/RelayLanguagePluginJavaScript.js.flow +34 -0
  55. package/language/javascript/formatGeneratedModule.js.flow +65 -0
  56. package/lib/bin/RelayCompilerBin.js +24 -7
  57. package/lib/bin/RelayCompilerMain.js +141 -136
  58. package/lib/codegen/CodegenDirectory.js +13 -8
  59. package/lib/codegen/CodegenRunner.js +37 -76
  60. package/lib/codegen/CodegenWatcher.js +13 -21
  61. package/lib/codegen/NormalizationCodeGenerator.js +117 -140
  62. package/lib/codegen/ReaderCodeGenerator.js +76 -117
  63. package/lib/codegen/RelayCodeGenerator.js +17 -6
  64. package/lib/codegen/RelayFileWriter.js +19 -40
  65. package/lib/codegen/compileRelayArtifacts.js +16 -30
  66. package/lib/codegen/sortObjectByKey.js +43 -0
  67. package/lib/codegen/writeRelayGeneratedFile.js +86 -95
  68. package/lib/core/ASTCache.js +2 -4
  69. package/lib/core/CompilerContext.js +2 -4
  70. package/lib/core/CompilerError.js +27 -54
  71. package/lib/core/GraphQLCompilerProfiler.js +8 -12
  72. package/lib/core/GraphQLDerivedFromMetadata.js +1 -10
  73. package/lib/core/GraphQLWatchmanClient.js +4 -12
  74. package/lib/core/IRPrinter.js +23 -21
  75. package/lib/core/IRTransformer.js +8 -16
  76. package/lib/core/IRValidator.js +1 -9
  77. package/lib/core/IRVisitor.js +0 -2
  78. package/lib/core/RelayCompilerScope.js +4 -4
  79. package/lib/core/RelayGraphQLEnumsGenerator.js +12 -15
  80. package/lib/core/RelayIRTransforms.js +16 -14
  81. package/lib/core/RelayParser.js +53 -89
  82. package/lib/core/RelaySourceModuleParser.js +3 -3
  83. package/lib/core/Schema.js +61 -73
  84. package/lib/core/SchemaUtils.js +15 -1
  85. package/lib/core/getFieldDefinition.js +12 -15
  86. package/lib/core/getIdentifierForSelection.js +2 -4
  87. package/lib/core/inferRootArgumentDefinitions.js +33 -73
  88. package/lib/index.js +4 -5
  89. package/lib/language/javascript/FindGraphQLTags.js +4 -3
  90. package/lib/language/javascript/RelayFlowGenerator.js +82 -171
  91. package/lib/language/javascript/RelayFlowTypeTransformers.js +1 -3
  92. package/lib/language/javascript/RelayLanguagePluginJavaScript.js +6 -4
  93. package/lib/language/javascript/formatGeneratedModule.js +11 -2
  94. package/lib/reporters/ConsoleReporter.js +1 -3
  95. package/lib/reporters/MultiReporter.js +1 -3
  96. package/lib/runner/Artifacts.js +69 -170
  97. package/lib/runner/BufferedFilesystem.js +32 -66
  98. package/lib/runner/GraphQLASTNodeGroup.js +54 -120
  99. package/lib/runner/GraphQLNodeMap.js +14 -19
  100. package/lib/runner/Sources.js +51 -85
  101. package/lib/runner/StrictMap.js +21 -37
  102. package/lib/runner/getChangedNodeNames.js +30 -62
  103. package/lib/transforms/ApplyFragmentArgumentTransform.js +73 -59
  104. package/lib/transforms/ClientExtensionsTransform.js +12 -16
  105. package/lib/transforms/ConnectionTransform.js +30 -37
  106. package/lib/transforms/DeclarativeConnectionMutationTransform.js +167 -0
  107. package/lib/transforms/DeferStreamTransform.js +30 -73
  108. package/lib/transforms/DisallowTypenameOnRoot.js +55 -0
  109. package/lib/transforms/FieldHandleTransform.js +6 -2
  110. package/lib/transforms/FlattenTransform.js +18 -45
  111. package/lib/transforms/GenerateIDFieldTransform.js +56 -35
  112. package/lib/transforms/GenerateTypeNameTransform.js +84 -10
  113. package/lib/transforms/InlineDataFragmentTransform.js +9 -4
  114. package/lib/transforms/MaskTransform.js +17 -17
  115. package/lib/transforms/MatchTransform.js +110 -32
  116. package/lib/transforms/RefetchableFragmentTransform.js +21 -38
  117. package/lib/transforms/RelayDirectiveTransform.js +8 -3
  118. package/lib/transforms/SkipClientExtensionsTransform.js +8 -0
  119. package/lib/transforms/SkipHandleFieldTransform.js +6 -2
  120. package/lib/transforms/SkipRedundantNodesTransform.js +7 -4
  121. package/lib/transforms/SkipSplitOperationTransform.js +32 -0
  122. package/lib/transforms/SkipUnreachableNodeTransform.js +9 -10
  123. package/lib/transforms/SkipUnusedVariablesTransform.js +18 -17
  124. package/lib/transforms/SplitModuleImportTransform.js +2 -2
  125. package/lib/transforms/TestOperationTransform.js +26 -22
  126. package/lib/transforms/ValidateGlobalVariablesTransform.js +18 -30
  127. package/lib/transforms/ValidateRequiredArgumentsTransform.js +12 -16
  128. package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +16 -30
  129. package/lib/transforms/ValidateUnusedVariablesTransform.js +18 -30
  130. package/lib/transforms/query-generators/FetchableQueryGenerator.js +161 -0
  131. package/lib/transforms/query-generators/NodeQueryGenerator.js +22 -3
  132. package/lib/transforms/query-generators/QueryQueryGenerator.js +2 -1
  133. package/lib/transforms/query-generators/ViewerQueryGenerator.js +1 -0
  134. package/lib/transforms/query-generators/index.js +23 -6
  135. package/lib/transforms/query-generators/utils.js +17 -16
  136. package/lib/util/RelayCompilerCache.js +2 -4
  137. package/lib/util/argumentContainsVariables.js +37 -0
  138. package/lib/util/dedupeJSONStringify.js +15 -12
  139. package/lib/util/generateAbstractTypeRefinementKey.js +24 -0
  140. package/lib/util/getModuleName.js +3 -5
  141. package/lib/util/joinArgumentDefinitions.js +3 -1
  142. package/package.json +6 -6
  143. package/relay-compiler.js +4 -4
  144. package/relay-compiler.min.js +4 -4
  145. package/reporters/ConsoleReporter.js.flow +81 -0
  146. package/reporters/MultiReporter.js.flow +43 -0
  147. package/reporters/Reporter.js.flow +19 -0
  148. package/runner/Artifacts.js.flow +219 -0
  149. package/runner/BufferedFilesystem.js.flow +194 -0
  150. package/runner/GraphQLASTNodeGroup.js.flow +176 -0
  151. package/runner/GraphQLASTUtils.js.flow +26 -0
  152. package/runner/GraphQLNodeMap.js.flow +55 -0
  153. package/runner/Sources.js.flow +214 -0
  154. package/runner/StrictMap.js.flow +96 -0
  155. package/runner/compileArtifacts.js.flow +76 -0
  156. package/runner/extractAST.js.flow +100 -0
  157. package/runner/getChangedNodeNames.js.flow +48 -0
  158. package/runner/getSchemaInstance.js.flow +36 -0
  159. package/runner/types.js.flow +37 -0
  160. package/transforms/ApplyFragmentArgumentTransform.js.flow +526 -0
  161. package/transforms/ClientExtensionsTransform.js.flow +222 -0
  162. package/transforms/ConnectionTransform.js.flow +856 -0
  163. package/transforms/DeclarativeConnectionMutationTransform.js.flow +157 -0
  164. package/transforms/DeferStreamTransform.js.flow +265 -0
  165. package/transforms/DisallowIdAsAlias.js.flow +47 -0
  166. package/transforms/DisallowTypenameOnRoot.js.flow +45 -0
  167. package/transforms/FieldHandleTransform.js.flow +80 -0
  168. package/transforms/FilterDirectivesTransform.js.flow +45 -0
  169. package/transforms/FlattenTransform.js.flow +453 -0
  170. package/transforms/GenerateIDFieldTransform.js.flow +152 -0
  171. package/transforms/GenerateTypeNameTransform.js.flow +161 -0
  172. package/transforms/InlineDataFragmentTransform.js.flow +125 -0
  173. package/transforms/InlineFragmentsTransform.js.flow +71 -0
  174. package/transforms/MaskTransform.js.flow +126 -0
  175. package/transforms/MatchTransform.js.flow +583 -0
  176. package/transforms/RefetchableFragmentTransform.js.flow +272 -0
  177. package/transforms/RelayDirectiveTransform.js.flow +97 -0
  178. package/transforms/SkipClientExtensionsTransform.js.flow +54 -0
  179. package/transforms/SkipHandleFieldTransform.js.flow +44 -0
  180. package/transforms/SkipRedundantNodesTransform.js.flow +254 -0
  181. package/transforms/SkipSplitOperationTransform.js.flow +37 -0
  182. package/transforms/SkipUnreachableNodeTransform.js.flow +149 -0
  183. package/transforms/SkipUnusedVariablesTransform.js.flow +59 -0
  184. package/transforms/SplitModuleImportTransform.js.flow +98 -0
  185. package/transforms/TestOperationTransform.js.flow +142 -0
  186. package/transforms/TransformUtils.js.flow +26 -0
  187. package/transforms/ValidateGlobalVariablesTransform.js.flow +81 -0
  188. package/transforms/ValidateRequiredArgumentsTransform.js.flow +127 -0
  189. package/transforms/ValidateServerOnlyDirectivesTransform.js.flow +112 -0
  190. package/transforms/ValidateUnusedVariablesTransform.js.flow +89 -0
  191. package/transforms/query-generators/FetchableQueryGenerator.js.flow +189 -0
  192. package/transforms/query-generators/NodeQueryGenerator.js.flow +219 -0
  193. package/transforms/query-generators/QueryQueryGenerator.js.flow +57 -0
  194. package/transforms/query-generators/ViewerQueryGenerator.js.flow +97 -0
  195. package/transforms/query-generators/index.js.flow +90 -0
  196. package/transforms/query-generators/utils.js.flow +76 -0
  197. package/util/CodeMarker.js.flow +79 -0
  198. package/{lib/core/GraphQLIR.js → util/DefaultHandleKey.js.flow} +9 -2
  199. package/util/RelayCompilerCache.js.flow +88 -0
  200. package/util/Rollout.js.flow +39 -0
  201. package/util/TimeReporter.js.flow +79 -0
  202. package/util/areEqualOSS.js.flow +123 -0
  203. package/util/argumentContainsVariables.js.flow +38 -0
  204. package/util/dedupeJSONStringify.js.flow +152 -0
  205. package/util/generateAbstractTypeRefinementKey.js.flow +29 -0
  206. package/util/getDefinitionNodeHash.js.flow +25 -0
  207. package/util/getModuleName.js.flow +39 -0
  208. package/util/joinArgumentDefinitions.js.flow +105 -0
  209. package/util/md5.js.flow +22 -0
  210. package/util/murmurHash.js.flow +94 -0
  211. package/util/nullthrowsOSS.js.flow +25 -0
  212. package/util/orList.js.flow +37 -0
  213. package/util/partitionArray.js.flow +37 -0
  214. package/lib/core/GraphQLCompilerContext.js +0 -165
  215. package/lib/core/GraphQLIRPrinter.js +0 -371
  216. package/lib/core/GraphQLIRTransformer.js +0 -344
  217. package/lib/core/GraphQLIRValidator.js +0 -218
  218. package/lib/core/GraphQLIRVisitor.js +0 -46
  219. package/lib/core/RelayCompilerError.js +0 -277
  220. package/lib/transforms/ConnectionFieldTransform.js +0 -276
@@ -0,0 +1,563 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ *
7
+ * @flow strict-local
8
+ * @format
9
+ */
10
+
11
+ // flowlint ambiguous-object-type:error
12
+
13
+ 'use strict';
14
+
15
+ const argumentContainsVariables = require('../util/argumentContainsVariables');
16
+ const generateAbstractTypeRefinementKey = require('../util/generateAbstractTypeRefinementKey');
17
+ const partitionArray = require('../util/partitionArray');
18
+ const sortObjectByKey = require('./sortObjectByKey');
19
+
20
+ const {createCompilerError, createUserError} = require('../core/CompilerError');
21
+ const {getStorageKey, stableCopy} = require('relay-runtime');
22
+
23
+ import type {
24
+ Argument,
25
+ ArgumentValue,
26
+ ClientExtension,
27
+ Metadata,
28
+ Root,
29
+ Selection,
30
+ SplitOperation,
31
+ LinkedField,
32
+ Defer,
33
+ Stream,
34
+ Condition,
35
+ InlineFragment,
36
+ ModuleImport,
37
+ LocalArgumentDefinition,
38
+ } from '../core/IR';
39
+ import type {Schema, TypeID} from '../core/Schema';
40
+ import type {
41
+ NormalizationArgument,
42
+ NormalizationDefer,
43
+ NormalizationField,
44
+ NormalizationLinkedField,
45
+ NormalizationLinkedHandle,
46
+ NormalizationLocalArgumentDefinition,
47
+ NormalizationModuleImport,
48
+ NormalizationOperation,
49
+ NormalizationScalarField,
50
+ NormalizationSelection,
51
+ NormalizationSplitOperation,
52
+ NormalizationStream,
53
+ NormalizationTypeDiscriminator,
54
+ } from 'relay-runtime';
55
+
56
+ /**
57
+ * @public
58
+ *
59
+ * Converts an IR node into a plain JS object representation that can be
60
+ * used at runtime.
61
+ */
62
+ declare function generate(schema: Schema, node: Root): NormalizationOperation;
63
+ declare function generate(
64
+ schema: Schema,
65
+ node: SplitOperation,
66
+ ): NormalizationSplitOperation;
67
+ function generate(
68
+ schema: Schema,
69
+ node: Root | SplitOperation,
70
+ ): NormalizationOperation | NormalizationSplitOperation {
71
+ switch (node.kind) {
72
+ case 'Root':
73
+ return generateRoot(schema, node);
74
+ case 'SplitOperation':
75
+ return generateSplitOperation(schema, node);
76
+ default:
77
+ throw createCompilerError(
78
+ `NormalizationCodeGenerator: Unsupported AST kind '${node.kind}'.`,
79
+ [node.loc],
80
+ );
81
+ }
82
+ }
83
+
84
+ function generateRoot(schema: Schema, node: Root): NormalizationOperation {
85
+ return {
86
+ argumentDefinitions: generateArgumentDefinitions(
87
+ schema,
88
+ node.argumentDefinitions,
89
+ ),
90
+ kind: 'Operation',
91
+ name: node.name,
92
+ selections: generateSelections(schema, node.selections),
93
+ };
94
+ }
95
+
96
+ function generateSplitOperation(
97
+ schema: Schema,
98
+ node: SplitOperation,
99
+ ): NormalizationSplitOperation {
100
+ return {
101
+ kind: 'SplitOperation',
102
+ metadata: sortObjectByKey(node.metadata),
103
+ name: node.name,
104
+ selections: generateSelections(schema, node.selections),
105
+ };
106
+ }
107
+
108
+ function generateSelections(
109
+ schema: Schema,
110
+ selections: $ReadOnlyArray<Selection>,
111
+ ): $ReadOnlyArray<NormalizationSelection> {
112
+ const normalizationSelections: Array<NormalizationSelection> = [];
113
+ selections.forEach(selection => {
114
+ switch (selection.kind) {
115
+ case 'Condition':
116
+ normalizationSelections.push(generateCondition(schema, selection));
117
+ break;
118
+ case 'ClientExtension':
119
+ normalizationSelections.push(
120
+ generateClientExtension(schema, selection),
121
+ );
122
+ break;
123
+ case 'ScalarField':
124
+ // NOTE: Inline fragments in normalization ast have the abstractKey
125
+ // but we skip the corresponding ScalarField for the type discriminator
126
+ // selection, since it's guaranteed to be a duplicate of a parent __typename
127
+ // selection.
128
+ const abstractKey = selection.metadata?.abstractKey;
129
+ if (typeof abstractKey === 'string') {
130
+ normalizationSelections.push(generateTypeDiscriminator(abstractKey));
131
+ } else {
132
+ normalizationSelections.push(...generateScalarField(selection));
133
+ }
134
+ break;
135
+ case 'ModuleImport':
136
+ normalizationSelections.push(generateModuleImport(selection));
137
+ break;
138
+ case 'InlineFragment':
139
+ normalizationSelections.push(generateInlineFragment(schema, selection));
140
+ break;
141
+ case 'LinkedField':
142
+ normalizationSelections.push(...generateLinkedField(schema, selection));
143
+ break;
144
+ case 'Defer':
145
+ normalizationSelections.push(generateDefer(schema, selection));
146
+ break;
147
+ case 'Stream':
148
+ normalizationSelections.push(generateStream(schema, selection));
149
+ break;
150
+ case 'InlineDataFragmentSpread':
151
+ case 'FragmentSpread':
152
+ throw new createCompilerError(
153
+ `NormalizationCodeGenerator: Unexpected IR node ${selection.kind}.`,
154
+ [selection.loc],
155
+ );
156
+ default:
157
+ (selection: empty);
158
+ throw new Error();
159
+ }
160
+ });
161
+ return normalizationSelections;
162
+ }
163
+
164
+ function generateArgumentDefinitions(
165
+ schema: Schema,
166
+ nodes: $ReadOnlyArray<LocalArgumentDefinition>,
167
+ ): $ReadOnlyArray<NormalizationLocalArgumentDefinition> {
168
+ return nodes.map(node => {
169
+ return {
170
+ defaultValue: stableCopy(node.defaultValue),
171
+ kind: 'LocalArgument',
172
+ name: node.name,
173
+ };
174
+ });
175
+ }
176
+
177
+ function generateClientExtension(
178
+ schema: Schema,
179
+ node: ClientExtension,
180
+ ): NormalizationSelection {
181
+ return {
182
+ kind: 'ClientExtension',
183
+ selections: generateSelections(schema, node.selections),
184
+ };
185
+ }
186
+
187
+ function generateCondition(
188
+ schema: Schema,
189
+ node: Condition,
190
+ ): NormalizationSelection {
191
+ if (node.condition.kind !== 'Variable') {
192
+ throw createCompilerError(
193
+ "NormalizationCodeGenerator: Expected 'Condition' with static " +
194
+ 'value to be pruned or inlined',
195
+ [node.condition.loc],
196
+ );
197
+ }
198
+ return {
199
+ condition: node.condition.variableName,
200
+ kind: 'Condition',
201
+ passingValue: node.passingValue,
202
+ selections: generateSelections(schema, node.selections),
203
+ };
204
+ }
205
+
206
+ function generateDefer(schema: Schema, node: Defer): NormalizationDefer {
207
+ if (
208
+ !(
209
+ node.if == null ||
210
+ node.if.kind === 'Variable' ||
211
+ (node.if.kind === 'Literal' && node.if.value === true)
212
+ )
213
+ ) {
214
+ throw createCompilerError(
215
+ 'NormalizationCodeGenerator: Expected @defer `if` condition to be ' +
216
+ 'a variable, unspecified, or the literal `true`.',
217
+ [node.if?.loc ?? node.loc],
218
+ );
219
+ }
220
+ return {
221
+ if:
222
+ node.if != null && node.if.kind === 'Variable'
223
+ ? node.if.variableName
224
+ : null,
225
+ kind: 'Defer',
226
+ label: node.label,
227
+ selections: generateSelections(schema, node.selections),
228
+ };
229
+ }
230
+
231
+ function generateInlineFragment(
232
+ schema: Schema,
233
+ node: InlineFragment,
234
+ ): NormalizationSelection {
235
+ const rawType = schema.getRawType(node.typeCondition);
236
+ const isAbstractType = schema.isAbstractType(rawType);
237
+ const abstractKey = isAbstractType
238
+ ? generateAbstractTypeRefinementKey(schema, rawType)
239
+ : null;
240
+ let selections = generateSelections(schema, node.selections);
241
+
242
+ if (isAbstractType) {
243
+ // Maintain a few invariants:
244
+ // - InlineFragment (and `selections` arrays generally) cannot be empty
245
+ // - Don't emit a TypeDiscriminator under an InlineFragment unless it has
246
+ // a different abstractKey
247
+ // This means we have to handle two cases:
248
+ // - The inline fragment only contains a TypeDiscriminator with the same
249
+ // abstractKey: replace the Fragment w the Discriminator
250
+ // - The inline fragment contains other selections: return all the selections
251
+ // minus any Discriminators w the same key
252
+ const [discriminators, otherSelections] = partitionArray(
253
+ selections,
254
+ selection =>
255
+ selection.kind === 'TypeDiscriminator' &&
256
+ selection.abstractKey === abstractKey,
257
+ );
258
+ const discriminator = discriminators[0];
259
+ if (discriminator != null && otherSelections.length === 0) {
260
+ return discriminator;
261
+ } else {
262
+ selections = otherSelections;
263
+ }
264
+ }
265
+
266
+ return {
267
+ kind: 'InlineFragment',
268
+ selections,
269
+ type: schema.getTypeString(rawType),
270
+ abstractKey,
271
+ };
272
+ }
273
+
274
+ function generateLinkedField(
275
+ schema: Schema,
276
+ node: LinkedField,
277
+ ): $ReadOnlyArray<NormalizationSelection> {
278
+ // Note: it is important that the arguments of this field be sorted to
279
+ // ensure stable generation of storage keys for equivalent arguments
280
+ // which may have originally appeared in different orders across an app.
281
+ const handles =
282
+ (node.handles &&
283
+ node.handles.map(handle => {
284
+ let handleNode: NormalizationLinkedHandle = {
285
+ alias: node.alias === node.name ? null : node.alias,
286
+ args: generateArgs(node.args),
287
+ filters: handle.filters,
288
+ handle: handle.name,
289
+ key: handle.key,
290
+ kind: 'LinkedHandle',
291
+ name: node.name,
292
+ };
293
+ // T45504512: new connection model
294
+ // NOTE: this intentionally adds a dynamic key in order to avoid
295
+ // triggering updates to existing queries that do not use dynamic
296
+ // keys.
297
+ if (handle.dynamicKey != null) {
298
+ const dynamicKeyArgName = '__dynamicKey';
299
+ handleNode = {
300
+ ...handleNode,
301
+ dynamicKey: {
302
+ kind: 'Variable',
303
+ name: dynamicKeyArgName,
304
+ variableName: handle.dynamicKey.variableName,
305
+ },
306
+ };
307
+ }
308
+ if (handle.handleArgs != null) {
309
+ const handleArgs = generateArgs(handle.handleArgs);
310
+ if (handleArgs != null) {
311
+ handleNode = {
312
+ ...handleNode,
313
+ handleArgs,
314
+ };
315
+ }
316
+ }
317
+ return handleNode;
318
+ })) ||
319
+ [];
320
+ const type = schema.getRawType(node.type);
321
+ let field: NormalizationLinkedField = {
322
+ alias: node.alias === node.name ? null : node.alias,
323
+ args: generateArgs(node.args),
324
+ concreteType: !schema.isAbstractType(type)
325
+ ? schema.getTypeString(type)
326
+ : null,
327
+ kind: 'LinkedField',
328
+ name: node.name,
329
+ plural: isPlural(schema, node.type),
330
+ selections: generateSelections(schema, node.selections),
331
+ storageKey: null,
332
+ };
333
+ // Precompute storageKey if possible
334
+ const storageKey = getStaticStorageKey(field, node.metadata);
335
+ if (storageKey != null) {
336
+ field = {...field, storageKey};
337
+ }
338
+ return [field].concat(handles);
339
+ }
340
+
341
+ function generateModuleImport(node: ModuleImport): NormalizationModuleImport {
342
+ const fragmentName = node.name;
343
+ const regExpMatch = fragmentName.match(
344
+ /^([a-zA-Z][a-zA-Z0-9]*)(?:_([a-zA-Z][_a-zA-Z0-9]*))?$/,
345
+ );
346
+ if (!regExpMatch) {
347
+ throw createCompilerError(
348
+ 'NormalizationCodeGenerator: @module fragments should be named ' +
349
+ `'FragmentName_propName', got '${fragmentName}'.`,
350
+ [node.loc],
351
+ );
352
+ }
353
+ const fragmentPropName = regExpMatch[2];
354
+ if (typeof fragmentPropName !== 'string') {
355
+ throw createCompilerError(
356
+ 'NormalizationCodeGenerator: @module fragments should be named ' +
357
+ `'FragmentName_propName', got '${fragmentName}'.`,
358
+ [node.loc],
359
+ );
360
+ }
361
+ return {
362
+ documentName: node.key,
363
+ fragmentName,
364
+ fragmentPropName,
365
+ kind: 'ModuleImport',
366
+ };
367
+ }
368
+
369
+ function generateTypeDiscriminator(
370
+ abstractKey: string,
371
+ ): NormalizationTypeDiscriminator {
372
+ return {
373
+ kind: 'TypeDiscriminator',
374
+ abstractKey,
375
+ };
376
+ }
377
+
378
+ function generateScalarField(node): Array<NormalizationSelection> {
379
+ // flowlint-next-line sketchy-null-mixed:off
380
+ if (node.metadata?.skipNormalizationNode) {
381
+ return [];
382
+ }
383
+ // Note: it is important that the arguments of this field be sorted to
384
+ // ensure stable generation of storage keys for equivalent arguments
385
+ // which may have originally appeared in different orders across an app.
386
+ const handles =
387
+ (node.handles &&
388
+ node.handles.map(handle => {
389
+ if (handle.dynamicKey != null) {
390
+ throw createUserError(
391
+ 'Dynamic key values are not supported on scalar fields.',
392
+ [handle.dynamicKey.loc],
393
+ );
394
+ }
395
+ return {
396
+ alias: node.alias === node.name ? null : node.alias,
397
+ args: generateArgs(node.args),
398
+ filters: handle.filters,
399
+ handle: handle.name,
400
+ key: handle.key,
401
+ kind: 'ScalarHandle',
402
+ name: node.name,
403
+ };
404
+ })) ||
405
+ [];
406
+ let field: NormalizationScalarField = {
407
+ alias: node.alias === node.name ? null : node.alias,
408
+ args: generateArgs(node.args),
409
+ kind: 'ScalarField',
410
+ name: node.name,
411
+ storageKey: null,
412
+ };
413
+ // Precompute storageKey if possible
414
+ const storageKey = getStaticStorageKey(field, node.metadata);
415
+ if (storageKey != null) {
416
+ field = {...field, storageKey};
417
+ }
418
+ return [field].concat(handles);
419
+ }
420
+
421
+ function generateStream(schema: Schema, node: Stream): NormalizationStream {
422
+ if (
423
+ !(
424
+ node.if == null ||
425
+ node.if.kind === 'Variable' ||
426
+ (node.if.kind === 'Literal' && node.if.value === true)
427
+ )
428
+ ) {
429
+ throw createCompilerError(
430
+ 'NormalizationCodeGenerator: Expected @stream `if` condition to be ' +
431
+ 'a variable, unspecified, or the literal `true`.',
432
+ [node.if?.loc ?? node.loc],
433
+ );
434
+ }
435
+ return {
436
+ if:
437
+ node.if != null && node.if.kind === 'Variable'
438
+ ? node.if.variableName
439
+ : null,
440
+ kind: 'Stream',
441
+ label: node.label,
442
+ metadata: sortObjectByKey(node.metadata),
443
+ selections: generateSelections(schema, node.selections),
444
+ useCustomizedBatch:
445
+ node.useCustomizedBatch != null &&
446
+ node.useCustomizedBatch.kind === 'Variable'
447
+ ? node.useCustomizedBatch.variableName
448
+ : null,
449
+ };
450
+ }
451
+
452
+ function generateArgumentValue(
453
+ name: string,
454
+ value: ArgumentValue,
455
+ ): NormalizationArgument | null {
456
+ switch (value.kind) {
457
+ case 'Variable':
458
+ return {
459
+ kind: 'Variable',
460
+ name: name,
461
+ variableName: value.variableName,
462
+ };
463
+ case 'Literal':
464
+ return value.value === null
465
+ ? null
466
+ : {
467
+ kind: 'Literal',
468
+ name: name,
469
+ value: stableCopy(value.value),
470
+ };
471
+ case 'ObjectValue': {
472
+ const objectKeys = value.fields.map(field => field.name).sort();
473
+ const objectValues = new Map(
474
+ value.fields.map(field => {
475
+ return [field.name, field.value];
476
+ }),
477
+ );
478
+ return {
479
+ fields: objectKeys.map(fieldName => {
480
+ const fieldValue = objectValues.get(fieldName);
481
+ if (fieldValue == null) {
482
+ throw createCompilerError('Expected to have object field value');
483
+ }
484
+ return (
485
+ generateArgumentValue(fieldName, fieldValue) ?? {
486
+ kind: 'Literal',
487
+ name: fieldName,
488
+ value: null,
489
+ }
490
+ );
491
+ }),
492
+ kind: 'ObjectValue',
493
+ name: name,
494
+ };
495
+ }
496
+ case 'ListValue': {
497
+ return {
498
+ items: value.items.map((item, index) => {
499
+ return generateArgumentValue(`${name}.${index}`, item);
500
+ }),
501
+ kind: 'ListValue',
502
+ name: name,
503
+ };
504
+ }
505
+ default:
506
+ throw createUserError(
507
+ 'NormalizationCodeGenerator: Complex argument values (Lists or ' +
508
+ 'InputObjects with nested variables) are not supported.',
509
+ [value.loc],
510
+ );
511
+ }
512
+ }
513
+
514
+ function generateArgs(
515
+ args: $ReadOnlyArray<Argument>,
516
+ ): ?$ReadOnlyArray<NormalizationArgument> {
517
+ const concreteArguments = [];
518
+ args.forEach(arg => {
519
+ const concreteArgument = generateArgumentValue(arg.name, arg.value);
520
+ if (concreteArgument !== null) {
521
+ concreteArguments.push(concreteArgument);
522
+ }
523
+ });
524
+ return concreteArguments.length === 0
525
+ ? null
526
+ : concreteArguments.sort(nameComparator);
527
+ }
528
+
529
+ function nameComparator(
530
+ a: {+name: string, ...},
531
+ b: {+name: string, ...},
532
+ ): number {
533
+ return a.name < b.name ? -1 : a.name > b.name ? 1 : 0;
534
+ }
535
+
536
+ /**
537
+ * Pre-computes storage key if possible and advantageous. Storage keys are
538
+ * generated for fields with supplied arguments that are all statically known
539
+ * (ie. literals, no variables) at build time.
540
+ */
541
+ function getStaticStorageKey(
542
+ field: NormalizationField,
543
+ metadata: Metadata,
544
+ ): ?string {
545
+ const metadataStorageKey = metadata?.storageKey;
546
+ if (typeof metadataStorageKey === 'string') {
547
+ return metadataStorageKey;
548
+ }
549
+ if (
550
+ !field.args ||
551
+ field.args.length === 0 ||
552
+ field.args.some(argumentContainsVariables)
553
+ ) {
554
+ return null;
555
+ }
556
+ return getStorageKey(field, {});
557
+ }
558
+
559
+ function isPlural(schema: Schema, type: TypeID): boolean {
560
+ return schema.isList(schema.getNullableType(type));
561
+ }
562
+
563
+ module.exports = {generate};