@ardatan/relay-compiler 12.0.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 (256) hide show
  1. package/LICENSE +21 -0
  2. package/bin/RelayCompilerBin.js.flow +169 -0
  3. package/bin/RelayCompilerMain.js.flow +515 -0
  4. package/bin/__fixtures__/plugin-module.js.flow +17 -0
  5. package/bin/relay-compiler +19066 -0
  6. package/codegen/CodegenDirectory.js.flow +375 -0
  7. package/codegen/CodegenRunner.js.flow +432 -0
  8. package/codegen/CodegenTypes.js.flow +28 -0
  9. package/codegen/CodegenWatcher.js.flow +254 -0
  10. package/codegen/NormalizationCodeGenerator.js.flow +566 -0
  11. package/codegen/ReaderCodeGenerator.js.flow +512 -0
  12. package/codegen/RelayCodeGenerator.js.flow +85 -0
  13. package/codegen/RelayFileWriter.js.flow +367 -0
  14. package/codegen/SourceControl.js.flow +58 -0
  15. package/codegen/compileRelayArtifacts.js.flow +182 -0
  16. package/codegen/createPrintRequireModuleDependency.js.flow +19 -0
  17. package/codegen/sortObjectByKey.js.flow +25 -0
  18. package/codegen/writeRelayGeneratedFile.js.flow +239 -0
  19. package/core/ASTCache.js.flow +74 -0
  20. package/core/ASTConvert.js.flow +233 -0
  21. package/core/CompilerContext.js.flow +191 -0
  22. package/core/CompilerError.js.flow +255 -0
  23. package/core/DotGraphQLParser.js.flow +39 -0
  24. package/core/GraphQLCompilerProfiler.js.flow +341 -0
  25. package/core/GraphQLDerivedFromMetadata.js.flow +36 -0
  26. package/core/GraphQLWatchmanClient.js.flow +111 -0
  27. package/core/IR.js.flow +326 -0
  28. package/core/IRPrinter.js.flow +478 -0
  29. package/core/IRTransformer.js.flow +377 -0
  30. package/core/IRValidator.js.flow +260 -0
  31. package/core/IRVisitor.js.flow +150 -0
  32. package/core/JSModuleParser.js.flow +24 -0
  33. package/core/RelayCompilerScope.js.flow +199 -0
  34. package/core/RelayFindGraphQLTags.js.flow +119 -0
  35. package/core/RelayGraphQLEnumsGenerator.js.flow +55 -0
  36. package/core/RelayIRTransforms.js.flow +138 -0
  37. package/core/RelayParser.js.flow +1734 -0
  38. package/core/RelaySourceModuleParser.js.flow +135 -0
  39. package/core/Schema.js.flow +2037 -0
  40. package/core/SchemaUtils.js.flow +120 -0
  41. package/core/filterContextForNode.js.flow +50 -0
  42. package/core/getFieldDefinition.js.flow +156 -0
  43. package/core/getIdentifierForArgumentValue.js.flow +49 -0
  44. package/core/getIdentifierForSelection.js.flow +69 -0
  45. package/core/getLiteralArgumentValues.js.flow +32 -0
  46. package/core/getNormalizationOperationName.js.flow +19 -0
  47. package/core/inferRootArgumentDefinitions.js.flow +323 -0
  48. package/index.js +10 -0
  49. package/index.js.flow +200 -0
  50. package/language/RelayLanguagePluginInterface.js.flow +283 -0
  51. package/language/javascript/FindGraphQLTags.js.flow +137 -0
  52. package/language/javascript/RelayFlowBabelFactories.js.flow +176 -0
  53. package/language/javascript/RelayFlowGenerator.js.flow +1100 -0
  54. package/language/javascript/RelayFlowTypeTransformers.js.flow +184 -0
  55. package/language/javascript/RelayLanguagePluginJavaScript.js.flow +34 -0
  56. package/language/javascript/formatGeneratedModule.js.flow +65 -0
  57. package/lib/bin/RelayCompilerBin.js +143 -0
  58. package/lib/bin/RelayCompilerMain.js +486 -0
  59. package/lib/bin/__fixtures__/plugin-module.js +16 -0
  60. package/lib/codegen/CodegenDirectory.js +336 -0
  61. package/lib/codegen/CodegenRunner.js +433 -0
  62. package/lib/codegen/CodegenTypes.js +11 -0
  63. package/lib/codegen/CodegenWatcher.js +271 -0
  64. package/lib/codegen/NormalizationCodeGenerator.js +480 -0
  65. package/lib/codegen/ReaderCodeGenerator.js +472 -0
  66. package/lib/codegen/RelayCodeGenerator.js +68 -0
  67. package/lib/codegen/RelayFileWriter.js +270 -0
  68. package/lib/codegen/SourceControl.js +60 -0
  69. package/lib/codegen/compileRelayArtifacts.js +157 -0
  70. package/lib/codegen/createPrintRequireModuleDependency.js +19 -0
  71. package/lib/codegen/sortObjectByKey.js +41 -0
  72. package/lib/codegen/writeRelayGeneratedFile.js +208 -0
  73. package/lib/core/ASTCache.js +70 -0
  74. package/lib/core/ASTConvert.js +198 -0
  75. package/lib/core/CompilerContext.js +165 -0
  76. package/lib/core/CompilerError.js +251 -0
  77. package/lib/core/DotGraphQLParser.js +40 -0
  78. package/lib/core/GraphQLCompilerProfiler.js +299 -0
  79. package/lib/core/GraphQLDerivedFromMetadata.js +31 -0
  80. package/lib/core/GraphQLWatchmanClient.js +150 -0
  81. package/lib/core/IR.js +11 -0
  82. package/lib/core/IRPrinter.js +389 -0
  83. package/lib/core/IRTransformer.js +345 -0
  84. package/lib/core/IRValidator.js +226 -0
  85. package/lib/core/IRVisitor.js +45 -0
  86. package/lib/core/JSModuleParser.js +18 -0
  87. package/lib/core/RelayCompilerScope.js +149 -0
  88. package/lib/core/RelayFindGraphQLTags.js +79 -0
  89. package/lib/core/RelayGraphQLEnumsGenerator.js +50 -0
  90. package/lib/core/RelayIRTransforms.js +109 -0
  91. package/lib/core/RelayParser.js +1382 -0
  92. package/lib/core/RelaySourceModuleParser.js +104 -0
  93. package/lib/core/Schema.js +1877 -0
  94. package/lib/core/SchemaUtils.js +98 -0
  95. package/lib/core/filterContextForNode.js +49 -0
  96. package/lib/core/getFieldDefinition.js +145 -0
  97. package/lib/core/getIdentifierForArgumentValue.js +53 -0
  98. package/lib/core/getIdentifierForSelection.js +48 -0
  99. package/lib/core/getLiteralArgumentValues.js +26 -0
  100. package/lib/core/getNormalizationOperationName.js +17 -0
  101. package/lib/core/inferRootArgumentDefinitions.js +351 -0
  102. package/lib/index.js +178 -0
  103. package/lib/language/RelayLanguagePluginInterface.js +14 -0
  104. package/lib/language/javascript/FindGraphQLTags.js +126 -0
  105. package/lib/language/javascript/RelayFlowBabelFactories.js +160 -0
  106. package/lib/language/javascript/RelayFlowGenerator.js +857 -0
  107. package/lib/language/javascript/RelayFlowTypeTransformers.js +119 -0
  108. package/lib/language/javascript/RelayLanguagePluginJavaScript.js +30 -0
  109. package/lib/language/javascript/formatGeneratedModule.js +36 -0
  110. package/lib/reporters/ConsoleReporter.js +61 -0
  111. package/lib/reporters/MultiReporter.js +45 -0
  112. package/lib/reporters/Reporter.js +11 -0
  113. package/lib/runner/Artifacts.js +323 -0
  114. package/lib/runner/BufferedFilesystem.js +261 -0
  115. package/lib/runner/GraphQLASTNodeGroup.js +256 -0
  116. package/lib/runner/GraphQLASTUtils.js +23 -0
  117. package/lib/runner/GraphQLNodeMap.js +81 -0
  118. package/lib/runner/Sources.js +271 -0
  119. package/lib/runner/StrictMap.js +134 -0
  120. package/lib/runner/compileArtifacts.js +39 -0
  121. package/lib/runner/extractAST.js +77 -0
  122. package/lib/runner/getChangedNodeNames.js +82 -0
  123. package/lib/runner/getSchemaInstance.js +30 -0
  124. package/lib/runner/types.js +12 -0
  125. package/lib/transforms/ApplyFragmentArgumentTransform.js +393 -0
  126. package/lib/transforms/ClientExtensionsTransform.js +222 -0
  127. package/lib/transforms/ConnectionTransform.js +643 -0
  128. package/lib/transforms/DeclarativeConnectionMutationTransform.js +221 -0
  129. package/lib/transforms/DeferStreamTransform.js +247 -0
  130. package/lib/transforms/DisallowIdAsAlias.js +41 -0
  131. package/lib/transforms/DisallowTypenameOnRoot.js +53 -0
  132. package/lib/transforms/FieldHandleTransform.js +81 -0
  133. package/lib/transforms/FilterCompilerDirectivesTransform.js +29 -0
  134. package/lib/transforms/FilterDirectivesTransform.js +41 -0
  135. package/lib/transforms/FlattenTransform.js +308 -0
  136. package/lib/transforms/GenerateIDFieldTransform.js +137 -0
  137. package/lib/transforms/GenerateTypeNameTransform.js +155 -0
  138. package/lib/transforms/InlineDataFragmentTransform.js +104 -0
  139. package/lib/transforms/InlineFragmentsTransform.js +63 -0
  140. package/lib/transforms/MaskTransform.js +121 -0
  141. package/lib/transforms/MatchTransform.js +438 -0
  142. package/lib/transforms/ReactFlightComponentTransform.js +161 -0
  143. package/lib/transforms/RefetchableFragmentTransform.js +249 -0
  144. package/lib/transforms/RelayDirectiveTransform.js +85 -0
  145. package/lib/transforms/RequiredFieldTransform.js +373 -0
  146. package/lib/transforms/SkipClientExtensionsTransform.js +49 -0
  147. package/lib/transforms/SkipHandleFieldTransform.js +45 -0
  148. package/lib/transforms/SkipRedundantNodesTransform.js +255 -0
  149. package/lib/transforms/SkipSplitOperationTransform.js +32 -0
  150. package/lib/transforms/SkipUnreachableNodeTransform.js +158 -0
  151. package/lib/transforms/SkipUnusedVariablesTransform.js +74 -0
  152. package/lib/transforms/SplitModuleImportTransform.js +85 -0
  153. package/lib/transforms/TestOperationTransform.js +145 -0
  154. package/lib/transforms/TransformUtils.js +21 -0
  155. package/lib/transforms/ValidateGlobalVariablesTransform.js +91 -0
  156. package/lib/transforms/ValidateRequiredArgumentsTransform.js +118 -0
  157. package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +111 -0
  158. package/lib/transforms/ValidateUnusedVariablesTransform.js +96 -0
  159. package/lib/transforms/query-generators/FetchableQueryGenerator.js +157 -0
  160. package/lib/transforms/query-generators/NodeQueryGenerator.js +166 -0
  161. package/lib/transforms/query-generators/QueryQueryGenerator.js +48 -0
  162. package/lib/transforms/query-generators/ViewerQueryGenerator.js +77 -0
  163. package/lib/transforms/query-generators/index.js +60 -0
  164. package/lib/transforms/query-generators/utils.js +92 -0
  165. package/lib/util/CodeMarker.js +80 -0
  166. package/lib/util/DefaultHandleKey.js +15 -0
  167. package/lib/util/RelayCompilerCache.js +98 -0
  168. package/lib/util/Rollout.js +40 -0
  169. package/lib/util/TimeReporter.js +83 -0
  170. package/lib/util/areEqualArgValues.js +135 -0
  171. package/lib/util/argumentContainsVariables.js +37 -0
  172. package/lib/util/dedupeJSONStringify.js +160 -0
  173. package/lib/util/generateAbstractTypeRefinementKey.js +24 -0
  174. package/lib/util/getDefinitionNodeHash.js +22 -0
  175. package/lib/util/getModuleName.js +32 -0
  176. package/lib/util/joinArgumentDefinitions.js +66 -0
  177. package/lib/util/md5.js +17 -0
  178. package/lib/util/murmurHash.js +86 -0
  179. package/lib/util/nullthrowsOSS.js +23 -0
  180. package/lib/util/orList.js +36 -0
  181. package/lib/util/partitionArray.js +35 -0
  182. package/package.json +42 -0
  183. package/relay-compiler.js +17 -0
  184. package/relay-compiler.min.js +22 -0
  185. package/reporters/ConsoleReporter.js.flow +81 -0
  186. package/reporters/MultiReporter.js.flow +43 -0
  187. package/reporters/Reporter.js.flow +19 -0
  188. package/runner/Artifacts.js.flow +219 -0
  189. package/runner/BufferedFilesystem.js.flow +194 -0
  190. package/runner/GraphQLASTNodeGroup.js.flow +176 -0
  191. package/runner/GraphQLASTUtils.js.flow +26 -0
  192. package/runner/GraphQLNodeMap.js.flow +55 -0
  193. package/runner/Sources.js.flow +228 -0
  194. package/runner/StrictMap.js.flow +96 -0
  195. package/runner/compileArtifacts.js.flow +76 -0
  196. package/runner/extractAST.js.flow +100 -0
  197. package/runner/getChangedNodeNames.js.flow +48 -0
  198. package/runner/getSchemaInstance.js.flow +36 -0
  199. package/runner/types.js.flow +37 -0
  200. package/transforms/ApplyFragmentArgumentTransform.js.flow +526 -0
  201. package/transforms/ClientExtensionsTransform.js.flow +226 -0
  202. package/transforms/ConnectionTransform.js.flow +859 -0
  203. package/transforms/DeclarativeConnectionMutationTransform.js.flow +250 -0
  204. package/transforms/DeferStreamTransform.js.flow +266 -0
  205. package/transforms/DisallowIdAsAlias.js.flow +48 -0
  206. package/transforms/DisallowTypenameOnRoot.js.flow +45 -0
  207. package/transforms/FieldHandleTransform.js.flow +81 -0
  208. package/transforms/FilterCompilerDirectivesTransform.js.flow +33 -0
  209. package/transforms/FilterDirectivesTransform.js.flow +45 -0
  210. package/transforms/FlattenTransform.js.flow +462 -0
  211. package/transforms/GenerateIDFieldTransform.js.flow +154 -0
  212. package/transforms/GenerateTypeNameTransform.js.flow +167 -0
  213. package/transforms/InlineDataFragmentTransform.js.flow +129 -0
  214. package/transforms/InlineFragmentsTransform.js.flow +73 -0
  215. package/transforms/MaskTransform.js.flow +130 -0
  216. package/transforms/MatchTransform.js.flow +593 -0
  217. package/transforms/ReactFlightComponentTransform.js.flow +198 -0
  218. package/transforms/RefetchableFragmentTransform.js.flow +272 -0
  219. package/transforms/RelayDirectiveTransform.js.flow +99 -0
  220. package/transforms/RequiredFieldTransform.js.flow +419 -0
  221. package/transforms/SkipClientExtensionsTransform.js.flow +57 -0
  222. package/transforms/SkipHandleFieldTransform.js.flow +45 -0
  223. package/transforms/SkipRedundantNodesTransform.js.flow +259 -0
  224. package/transforms/SkipSplitOperationTransform.js.flow +37 -0
  225. package/transforms/SkipUnreachableNodeTransform.js.flow +149 -0
  226. package/transforms/SkipUnusedVariablesTransform.js.flow +59 -0
  227. package/transforms/SplitModuleImportTransform.js.flow +101 -0
  228. package/transforms/TestOperationTransform.js.flow +143 -0
  229. package/transforms/TransformUtils.js.flow +26 -0
  230. package/transforms/ValidateGlobalVariablesTransform.js.flow +81 -0
  231. package/transforms/ValidateRequiredArgumentsTransform.js.flow +131 -0
  232. package/transforms/ValidateServerOnlyDirectivesTransform.js.flow +115 -0
  233. package/transforms/ValidateUnusedVariablesTransform.js.flow +89 -0
  234. package/transforms/query-generators/FetchableQueryGenerator.js.flow +189 -0
  235. package/transforms/query-generators/NodeQueryGenerator.js.flow +219 -0
  236. package/transforms/query-generators/QueryQueryGenerator.js.flow +57 -0
  237. package/transforms/query-generators/ViewerQueryGenerator.js.flow +97 -0
  238. package/transforms/query-generators/index.js.flow +90 -0
  239. package/transforms/query-generators/utils.js.flow +76 -0
  240. package/util/CodeMarker.js.flow +79 -0
  241. package/util/DefaultHandleKey.js.flow +17 -0
  242. package/util/RelayCompilerCache.js.flow +88 -0
  243. package/util/Rollout.js.flow +39 -0
  244. package/util/TimeReporter.js.flow +79 -0
  245. package/util/areEqualArgValues.js.flow +126 -0
  246. package/util/argumentContainsVariables.js.flow +38 -0
  247. package/util/dedupeJSONStringify.js.flow +152 -0
  248. package/util/generateAbstractTypeRefinementKey.js.flow +29 -0
  249. package/util/getDefinitionNodeHash.js.flow +25 -0
  250. package/util/getModuleName.js.flow +39 -0
  251. package/util/joinArgumentDefinitions.js.flow +105 -0
  252. package/util/md5.js.flow +22 -0
  253. package/util/murmurHash.js.flow +94 -0
  254. package/util/nullthrowsOSS.js.flow +25 -0
  255. package/util/orList.js.flow +37 -0
  256. package/util/partitionArray.js.flow +37 -0
@@ -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
+ };