relay-compiler 9.0.0 → 10.1.0

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