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,526 @@
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
8
+ * @format
9
+ */
10
+
11
+ // flowlint ambiguous-object-type:error
12
+
13
+ 'use strict';
14
+
15
+ const IRTransformer = require('../core/IRTransformer');
16
+ const RelayCompilerScope = require('../core/RelayCompilerScope');
17
+
18
+ const getIdentifierForArgumentValue = require('../core/getIdentifierForArgumentValue');
19
+ const murmurHash = require('../util/murmurHash');
20
+
21
+ const {
22
+ createCompilerError,
23
+ createNonRecoverableUserError,
24
+ } = require('../core/CompilerError');
25
+
26
+ import type CompilerContext from '../core/CompilerContext';
27
+ import type {
28
+ Argument,
29
+ ArgumentValue,
30
+ Condition,
31
+ Defer,
32
+ Directive,
33
+ Field,
34
+ Fragment,
35
+ FragmentSpread,
36
+ IR,
37
+ Node,
38
+ Selection,
39
+ Stream,
40
+ } from '../core/IR';
41
+ import type {Scope} from '../core/RelayCompilerScope';
42
+
43
+ const {getFragmentScope, getRootScope} = RelayCompilerScope;
44
+
45
+ type PendingFragment =
46
+ | {|kind: 'pending'|}
47
+ | {|kind: 'resolved', value: ?Fragment|};
48
+
49
+ /**
50
+ * A transform that converts a set of documents containing fragments/fragment
51
+ * spreads *with* arguments to one where all arguments have been inlined. This
52
+ * is effectively static currying of functions. Nodes are changed as follows:
53
+ * - Fragment spreads with arguments are replaced with references to an inlined
54
+ * version of the referenced fragment.
55
+ * - Fragments with argument definitions are cloned once per unique set of
56
+ * arguments, with the name changed to original name + hash and all nested
57
+ * variable references changed to the value of that variable given its
58
+ * arguments.
59
+ * - Field & directive argument variables are replaced with the value of those
60
+ * variables in context.
61
+ * - All nodes are cloned with updated children.
62
+ *
63
+ * The transform also handles statically passing/failing Condition nodes:
64
+ * - Literal Conditions with a passing value are elided and their selections
65
+ * inlined in their parent.
66
+ * - Literal Conditions with a failing value are removed.
67
+ * - Nodes that would become empty as a result of the above are removed.
68
+ *
69
+ * Note that unreferenced fragments are not added to the output.
70
+ */
71
+ function applyFragmentArgumentTransform(
72
+ context: CompilerContext,
73
+ ): CompilerContext {
74
+ const fragments: Map<string, PendingFragment> = new Map();
75
+ let nextContext = IRTransformer.transform(context, {
76
+ Root: node => {
77
+ const scope = getRootScope(node.argumentDefinitions);
78
+ return transformNode(context, fragments, scope, node, [node]);
79
+ },
80
+ SplitOperation: node => {
81
+ return transformNode(context, fragments, {}, node, [node]);
82
+ },
83
+ // Fragments are included below where referenced.
84
+ // Unreferenced fragments are not included.
85
+ Fragment: () => null,
86
+ });
87
+
88
+ for (const pendingFragment of fragments.values()) {
89
+ if (pendingFragment.kind === 'resolved' && pendingFragment.value) {
90
+ nextContext = nextContext.add(pendingFragment.value);
91
+ }
92
+ }
93
+ return nextContext;
94
+ }
95
+
96
+ function transformNode<T: Node>(
97
+ context: CompilerContext,
98
+ fragments: Map<string, PendingFragment>,
99
+ scope: Scope,
100
+ node: T,
101
+ errorContext: $ReadOnlyArray<IR>,
102
+ ): ?T {
103
+ const selections = transformSelections(
104
+ context,
105
+ fragments,
106
+ scope,
107
+ node.selections,
108
+ errorContext,
109
+ );
110
+ if (!selections) {
111
+ return null;
112
+ }
113
+ if (node.hasOwnProperty('directives')) {
114
+ const directives = transformDirectives(
115
+ scope,
116
+ (node: $FlowIssue).directives,
117
+ errorContext,
118
+ );
119
+ return ({
120
+ ...node,
121
+ directives,
122
+ selections,
123
+ }: any);
124
+ }
125
+ return ({
126
+ ...node,
127
+ selections,
128
+ }: $FlowIssue);
129
+ }
130
+
131
+ function transformDeferStreamNode<T: Defer | Stream>(
132
+ context: CompilerContext,
133
+ fragments: Map<string, PendingFragment>,
134
+ scope: Scope,
135
+ node: T,
136
+ errorContext: $ReadOnlyArray<IR>,
137
+ ): ?Selection {
138
+ const nextNode = transformNode(context, fragments, scope, node, errorContext);
139
+ if (!nextNode) {
140
+ return null;
141
+ }
142
+ (nextNode: T);
143
+ if (nextNode.if) {
144
+ const ifVal = transformValue(scope, nextNode.if, errorContext);
145
+ if (
146
+ ifVal.kind === 'Literal' &&
147
+ ifVal.value === false &&
148
+ node.selections &&
149
+ node.selections.length === 1
150
+ ) {
151
+ // Skip Defer/Stream wrapper with literal if: false
152
+ return node.selections[0];
153
+ }
154
+ // $FlowFixMe[cannot-write] nextNode is uniquely owned
155
+ nextNode.if = ifVal;
156
+ }
157
+ if (nextNode.useCustomizedBatch) {
158
+ // $FlowFixMe[cannot-write] nextNode is uniquely owned
159
+ nextNode.useCustomizedBatch = transformValue(
160
+ scope,
161
+ nextNode.useCustomizedBatch,
162
+ errorContext,
163
+ );
164
+ }
165
+ if (nextNode.initialCount) {
166
+ // $FlowFixMe[cannot-write] nextNode is uniquely owned
167
+ nextNode.initialCount = transformValue(
168
+ scope,
169
+ nextNode.initialCount,
170
+ errorContext,
171
+ );
172
+ }
173
+ return nextNode;
174
+ }
175
+
176
+ function transformFragmentSpread(
177
+ context: CompilerContext,
178
+ fragments: Map<string, PendingFragment>,
179
+ scope: Scope,
180
+ spread: FragmentSpread,
181
+ errorContext: $ReadOnlyArray<IR>,
182
+ ): ?FragmentSpread {
183
+ const directives = transformDirectives(
184
+ scope,
185
+ spread.directives,
186
+ errorContext,
187
+ );
188
+ const appliedFragment = transformFragment(
189
+ context,
190
+ fragments,
191
+ scope,
192
+ spread,
193
+ spread.args,
194
+ [...errorContext, spread],
195
+ );
196
+ if (!appliedFragment) {
197
+ return null;
198
+ }
199
+ const transformed: FragmentSpread = {
200
+ ...spread,
201
+ kind: 'FragmentSpread',
202
+ args: [],
203
+ directives,
204
+ name: appliedFragment.name,
205
+ };
206
+ return transformed;
207
+ }
208
+
209
+ function transformField<T: Field>(
210
+ context: CompilerContext,
211
+ fragments: Map<string, PendingFragment>,
212
+ scope: Scope,
213
+ field: T,
214
+ errorContext: $ReadOnlyArray<IR>,
215
+ ): ?T {
216
+ const args = transformArguments(scope, field.args, errorContext);
217
+ const directives = transformDirectives(scope, field.directives, errorContext);
218
+ if (field.kind === 'LinkedField') {
219
+ const selections = transformSelections(
220
+ context,
221
+ fragments,
222
+ scope,
223
+ field.selections,
224
+ errorContext,
225
+ );
226
+ if (!selections) {
227
+ return null;
228
+ }
229
+ return ({
230
+ ...field,
231
+ args,
232
+ directives,
233
+ selections,
234
+ }: $FlowFixMe);
235
+ } else {
236
+ return {
237
+ ...field,
238
+ args,
239
+ directives,
240
+ };
241
+ }
242
+ }
243
+
244
+ function transformCondition(
245
+ context: CompilerContext,
246
+ fragments: Map<string, PendingFragment>,
247
+ scope: Scope,
248
+ node: Condition,
249
+ errorContext: $ReadOnlyArray<IR>,
250
+ ): ?$ReadOnlyArray<Selection> {
251
+ const condition = transformValue(scope, node.condition, errorContext);
252
+ if (!(condition.kind === 'Literal' || condition.kind === 'Variable')) {
253
+ // This transform does whole-program optimization, errors in
254
+ // a single document could break invariants and/or cause
255
+ // additional spurious errors.
256
+ throw createNonRecoverableUserError(
257
+ 'A non-scalar value was applied to an @include or @skip directive, ' +
258
+ 'the `if` argument value must be a ' +
259
+ 'variable or a literal Boolean.',
260
+ [condition.loc],
261
+ );
262
+ }
263
+ if (condition.kind === 'Literal' && condition.value !== node.passingValue) {
264
+ // Dead code, no need to traverse further.
265
+ return null;
266
+ }
267
+ const selections = transformSelections(
268
+ context,
269
+ fragments,
270
+ scope,
271
+ node.selections,
272
+ errorContext,
273
+ );
274
+ if (!selections) {
275
+ return null;
276
+ }
277
+ if (condition.kind === 'Literal' && condition.value === node.passingValue) {
278
+ // Always passes, return inlined selections
279
+ return selections;
280
+ }
281
+ return [
282
+ {
283
+ ...node,
284
+ condition,
285
+ selections,
286
+ },
287
+ ];
288
+ }
289
+
290
+ function transformSelections(
291
+ context: CompilerContext,
292
+ fragments: Map<string, PendingFragment>,
293
+ scope: Scope,
294
+ selections: $ReadOnlyArray<Selection>,
295
+ errorContext: $ReadOnlyArray<IR>,
296
+ ): ?$ReadOnlyArray<Selection> {
297
+ let nextSelections = null;
298
+ selections.forEach(selection => {
299
+ let nextSelection;
300
+ if (
301
+ selection.kind === 'ClientExtension' ||
302
+ selection.kind === 'InlineDataFragmentSpread' ||
303
+ selection.kind === 'InlineFragment' ||
304
+ selection.kind === 'ModuleImport'
305
+ ) {
306
+ nextSelection = transformNode(
307
+ context,
308
+ fragments,
309
+ scope,
310
+ selection,
311
+ errorContext,
312
+ );
313
+ } else if (selection.kind === 'Defer' || selection.kind === 'Stream') {
314
+ nextSelection = transformDeferStreamNode(
315
+ context,
316
+ fragments,
317
+ scope,
318
+ selection,
319
+ errorContext,
320
+ );
321
+ } else if (selection.kind === 'FragmentSpread') {
322
+ nextSelection = transformFragmentSpread(
323
+ context,
324
+ fragments,
325
+ scope,
326
+ selection,
327
+ errorContext,
328
+ );
329
+ } else if (selection.kind === 'Condition') {
330
+ const conditionSelections = transformCondition(
331
+ context,
332
+ fragments,
333
+ scope,
334
+ selection,
335
+ errorContext,
336
+ );
337
+ if (conditionSelections) {
338
+ nextSelections = nextSelections || [];
339
+ nextSelections.push(...conditionSelections);
340
+ }
341
+ } else if (
342
+ selection.kind === 'LinkedField' ||
343
+ selection.kind === 'ScalarField'
344
+ ) {
345
+ nextSelection = transformField(
346
+ context,
347
+ fragments,
348
+ scope,
349
+ selection,
350
+ errorContext,
351
+ );
352
+ } else {
353
+ (selection: empty);
354
+ throw createCompilerError(
355
+ `ApplyFragmentArgumentTransform: Unsupported kind '${selection.kind}'.`,
356
+ [selection.loc],
357
+ );
358
+ }
359
+ if (nextSelection) {
360
+ nextSelections = nextSelections || [];
361
+ nextSelections.push(nextSelection);
362
+ }
363
+ });
364
+ return nextSelections;
365
+ }
366
+
367
+ function transformDirectives(
368
+ scope: Scope,
369
+ directives: $ReadOnlyArray<Directive>,
370
+ errorContext: $ReadOnlyArray<IR>,
371
+ ): $ReadOnlyArray<Directive> {
372
+ return directives.map(directive => {
373
+ const args = transformArguments(scope, directive.args, errorContext);
374
+ return {
375
+ ...directive,
376
+ args,
377
+ };
378
+ });
379
+ }
380
+
381
+ function transformArguments(
382
+ scope: Scope,
383
+ args: $ReadOnlyArray<Argument>,
384
+ errorContext: $ReadOnlyArray<IR>,
385
+ ): $ReadOnlyArray<Argument> {
386
+ return args.map(arg => {
387
+ const value = transformValue(scope, arg.value, errorContext);
388
+ return value === arg.value ? arg : {...arg, value};
389
+ });
390
+ }
391
+
392
+ function transformValue(
393
+ scope: Scope,
394
+ value: ArgumentValue,
395
+ errorContext: $ReadOnlyArray<IR>,
396
+ ): ArgumentValue {
397
+ if (value.kind === 'Variable') {
398
+ const scopeValue = scope[value.variableName];
399
+ if (scopeValue == null) {
400
+ // This transform does whole-program optimization, errors in
401
+ // a single document could break invariants and/or cause
402
+ // additional spurious errors.
403
+ throw createNonRecoverableUserError(
404
+ `Variable '$${value.variableName}' is not in scope.`,
405
+ [errorContext[0]?.loc, value.loc].filter(Boolean),
406
+ );
407
+ }
408
+ return scopeValue;
409
+ } else if (value.kind === 'ObjectValue') {
410
+ return {
411
+ ...value,
412
+ fields: value.fields.map(field => ({
413
+ ...field,
414
+ value: transformValue(scope, field.value, errorContext),
415
+ })),
416
+ };
417
+ } else if (value.kind === 'ListValue') {
418
+ return {
419
+ ...value,
420
+ items: value.items.map(item => transformValue(scope, item, errorContext)),
421
+ };
422
+ }
423
+ return value;
424
+ }
425
+
426
+ /**
427
+ * Apply arguments to a fragment, creating a new fragment (with the given name)
428
+ * with all values recursively applied.
429
+ */
430
+ function transformFragment(
431
+ context: CompilerContext,
432
+ fragments: Map<string, PendingFragment>,
433
+ parentScope: Scope,
434
+ spread: FragmentSpread,
435
+ args: $ReadOnlyArray<Argument>,
436
+ errorContext: $ReadOnlyArray<IR>,
437
+ ): ?Fragment {
438
+ const schema = context.getSchema();
439
+ const fragment = context.getFragment(spread.name, spread.loc);
440
+ const argumentsHash = hashArguments(args, parentScope, errorContext);
441
+ const fragmentName = argumentsHash
442
+ ? `${fragment.name}_${argumentsHash}`
443
+ : fragment.name;
444
+ const appliedFragment = fragments.get(fragmentName);
445
+ if (appliedFragment) {
446
+ if (appliedFragment.kind === 'resolved') {
447
+ return appliedFragment.value;
448
+ } else {
449
+ // This transform does whole-program optimization, errors in
450
+ // a single document could break invariants and/or cause
451
+ // additional spurious errors.
452
+ throw createNonRecoverableUserError(
453
+ `Found a circular reference from fragment '${fragment.name}'.`,
454
+ errorContext.map(node => node.loc),
455
+ );
456
+ }
457
+ }
458
+ const fragmentScope = getFragmentScope(
459
+ schema,
460
+ fragment.argumentDefinitions,
461
+ args,
462
+ parentScope,
463
+ spread,
464
+ );
465
+ // record that this fragment is pending to detect circular references
466
+ fragments.set(fragmentName, {kind: 'pending'});
467
+ let transformedFragment = null;
468
+ const selections = transformSelections(
469
+ context,
470
+ fragments,
471
+ fragmentScope,
472
+ fragment.selections,
473
+ errorContext,
474
+ );
475
+ if (selections) {
476
+ transformedFragment = {
477
+ ...fragment,
478
+ selections,
479
+ name: fragmentName,
480
+ argumentDefinitions: [],
481
+ };
482
+ }
483
+ fragments.set(fragmentName, {kind: 'resolved', value: transformedFragment});
484
+ return transformedFragment;
485
+ }
486
+
487
+ function hashArguments(
488
+ args: $ReadOnlyArray<Argument>,
489
+ scope: Scope,
490
+ errorContext: $ReadOnlyArray<IR>,
491
+ ): ?string {
492
+ if (!args.length) {
493
+ return null;
494
+ }
495
+ const sortedArgs = [...args].sort((a, b) => {
496
+ return a.name < b.name ? -1 : a.name > b.name ? 1 : 0;
497
+ });
498
+ const printedArgs = JSON.stringify(
499
+ sortedArgs.map(arg => {
500
+ let value;
501
+ if (arg.value.kind === 'Variable') {
502
+ value = scope[arg.value.variableName];
503
+ if (value == null) {
504
+ // This transform does whole-program optimization, errors in
505
+ // a single document could break invariants and/or cause
506
+ // additional spurious errors.
507
+ throw createNonRecoverableUserError(
508
+ `Variable '$${arg.value.variableName}' is not in scope.`,
509
+ [errorContext[0]?.loc, arg.value.loc].filter(Boolean),
510
+ );
511
+ }
512
+ } else {
513
+ value = arg.value;
514
+ }
515
+ return {
516
+ name: arg.name,
517
+ value: getIdentifierForArgumentValue(value),
518
+ };
519
+ }),
520
+ );
521
+ return murmurHash(printedArgs);
522
+ }
523
+
524
+ module.exports = {
525
+ transform: applyFragmentArgumentTransform,
526
+ };