@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,250 @@
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 IRTransformer = require('../core/IRTransformer');
16
+
17
+ const {createUserError} = require('../core/CompilerError');
18
+ const {ConnectionInterface} = require('relay-runtime');
19
+
20
+ const DELETE_RECORD = 'deleteRecord';
21
+ const DELETE_EDGE = 'deleteEdge';
22
+ const APPEND_EDGE = 'appendEdge';
23
+ const PREPEND_EDGE = 'prependEdge';
24
+ const APPEND_NODE = 'appendNode';
25
+ const PREPEND_NODE = 'prependNode';
26
+ const EDGE_LINKED_FIELD_DIRECTIVES = [APPEND_EDGE, PREPEND_EDGE];
27
+ const NODE_LINKED_FIELD_DIRECTIVES = [APPEND_NODE, PREPEND_NODE];
28
+ const LINKED_FIELD_DIRECTIVES = [
29
+ ...EDGE_LINKED_FIELD_DIRECTIVES,
30
+ ...NODE_LINKED_FIELD_DIRECTIVES,
31
+ ];
32
+
33
+ const SCHEMA_EXTENSION = `
34
+ directive @${DELETE_RECORD} on FIELD
35
+ directive @${DELETE_EDGE}(
36
+ connections: [ID!]!
37
+ ) on FIELD
38
+ directive @${APPEND_EDGE}(
39
+ connections: [ID!]!
40
+ ) on FIELD
41
+ directive @${PREPEND_EDGE}(
42
+ connections: [ID!]!
43
+ ) on FIELD
44
+ directive @${APPEND_NODE}(
45
+ connections: [ID!]!
46
+ edgeTypeName: String!
47
+ ) on FIELD
48
+ directive @${PREPEND_NODE}(
49
+ connections: [ID!]!
50
+ edgeTypeName: String!
51
+ ) on FIELD
52
+ `;
53
+
54
+ import type CompilerContext from '../core/CompilerContext';
55
+ import type {ScalarField, LinkedField, Root, Handle} from '../core/IR';
56
+
57
+ function transform(context: CompilerContext): CompilerContext {
58
+ return IRTransformer.transform(context, {
59
+ ScalarField: visitScalarField,
60
+ LinkedField: visitLinkedField,
61
+ SplitOperation: skip,
62
+ Fragment: skip,
63
+ });
64
+ }
65
+
66
+ function skip<T>(node: T): T {
67
+ return node;
68
+ }
69
+
70
+ function visitScalarField(field: ScalarField): ScalarField {
71
+ const linkedFieldDirective = field.directives.find(
72
+ directive => LINKED_FIELD_DIRECTIVES.indexOf(directive.name) > -1,
73
+ );
74
+ if (linkedFieldDirective != null) {
75
+ throw createUserError(
76
+ `Invalid use of @${linkedFieldDirective.name} on scalar field '${field.name}'`,
77
+ [linkedFieldDirective.loc],
78
+ );
79
+ }
80
+ const deleteNodeDirective = field.directives.find(
81
+ directive => directive.name === DELETE_RECORD,
82
+ );
83
+ const deleteEdgeDirective = field.directives.find(
84
+ directive => directive.name === DELETE_EDGE,
85
+ );
86
+ if (deleteNodeDirective != null && deleteEdgeDirective != null) {
87
+ throw createUserError(
88
+ `Both @deleteNode and @deleteEdge are used on field '${field.name}'. Only one directive is supported for now.`,
89
+ [deleteNodeDirective.loc, deleteEdgeDirective.loc],
90
+ );
91
+ }
92
+ const targetDirective = deleteNodeDirective ?? deleteEdgeDirective;
93
+ if (targetDirective == null) {
94
+ return field;
95
+ }
96
+
97
+ // $FlowFixMe[incompatible-use]
98
+ const schema = this.getContext().getSchema();
99
+
100
+ if (!schema.isId(schema.getRawType(field.type))) {
101
+ throw createUserError(
102
+ `Invalid use of @${targetDirective.name} on field '${
103
+ field.name
104
+ }'. Expected field to return an ID or list of ID values, got ${schema.getTypeString(
105
+ field.type,
106
+ )}.`,
107
+ [targetDirective.loc],
108
+ );
109
+ }
110
+ const connectionsArg = targetDirective.args.find(
111
+ arg => arg.name === 'connections',
112
+ );
113
+ const handle: Handle = {
114
+ name: targetDirective.name,
115
+ key: '',
116
+ dynamicKey: null,
117
+ filters: null,
118
+ handleArgs: connectionsArg ? [connectionsArg] : undefined,
119
+ };
120
+ return {
121
+ ...field,
122
+ directives: field.directives.filter(
123
+ directive => directive !== targetDirective,
124
+ ),
125
+ handles: field.handles ? [...field.handles, handle] : [handle],
126
+ };
127
+ }
128
+
129
+ function visitLinkedField(field: LinkedField): LinkedField {
130
+ // $FlowFixMe[incompatible-use]
131
+ const transformedField = this.traverse(field);
132
+ const deleteDirective = transformedField.directives.find(
133
+ directive => directive.name === DELETE_RECORD,
134
+ );
135
+ if (deleteDirective != null) {
136
+ throw createUserError(
137
+ `Invalid use of @${deleteDirective.name} on scalar field '${transformedField.name}'.`,
138
+ [deleteDirective.loc],
139
+ );
140
+ }
141
+ const edgeDirective = transformedField.directives.find(
142
+ directive => EDGE_LINKED_FIELD_DIRECTIVES.indexOf(directive.name) > -1,
143
+ );
144
+ const nodeDirective = transformedField.directives.find(
145
+ directive => NODE_LINKED_FIELD_DIRECTIVES.indexOf(directive.name) > -1,
146
+ );
147
+
148
+ if (edgeDirective == null && nodeDirective == null) {
149
+ return transformedField;
150
+ }
151
+ if (edgeDirective != null && nodeDirective != null) {
152
+ throw createUserError(
153
+ `Invalid use of @${edgeDirective.name} and @${nodeDirective.name} on field '${transformedField.name}' - these directives cannot be used together.`,
154
+ [edgeDirective.loc],
155
+ );
156
+ }
157
+ const targetDirective = edgeDirective ?? nodeDirective;
158
+ const connectionsArg = targetDirective.args.find(
159
+ arg => arg.name === 'connections',
160
+ );
161
+ if (connectionsArg == null) {
162
+ throw createUserError(
163
+ `Expected the 'connections' argument to be defined on @${targetDirective.name}.`,
164
+ [targetDirective.loc],
165
+ );
166
+ }
167
+ // $FlowFixMe[incompatible-use]
168
+ const schema = this.getContext().getSchema();
169
+ if (edgeDirective) {
170
+ const fieldType = schema.getRawType(transformedField.type);
171
+ const fields = schema.getFields(fieldType);
172
+ let cursorFieldID;
173
+ let nodeFieldID;
174
+ for (const fieldID of fields) {
175
+ const fieldName = schema.getFieldName(fieldID);
176
+ if (fieldName === ConnectionInterface.get().CURSOR) {
177
+ cursorFieldID = fieldID;
178
+ } else if (fieldName === ConnectionInterface.get().NODE) {
179
+ nodeFieldID = fieldID;
180
+ }
181
+ }
182
+
183
+ // Edge
184
+ if (cursorFieldID != null && nodeFieldID != null) {
185
+ const handle: Handle = {
186
+ name: edgeDirective.name,
187
+ key: '',
188
+ dynamicKey: null,
189
+ filters: null,
190
+ handleArgs: [connectionsArg],
191
+ };
192
+ return {
193
+ ...transformedField,
194
+ directives: transformedField.directives.filter(
195
+ directive => directive !== edgeDirective,
196
+ ),
197
+ handles: transformedField.handles
198
+ ? [...transformedField.handles, handle]
199
+ : [handle],
200
+ };
201
+ }
202
+ throw createUserError(
203
+ `Unsupported use of @${edgeDirective.name} on field '${transformedField.name}', expected an edge field (a field with 'cursor' and 'node' selection).`,
204
+ [targetDirective.loc],
205
+ );
206
+ } else {
207
+ // Node
208
+ const edgeTypeNameArg = nodeDirective.args.find(
209
+ arg => arg.name === 'edgeTypeName',
210
+ );
211
+ if (!edgeTypeNameArg) {
212
+ throw createUserError(
213
+ `Unsupported use of @${nodeDirective.name} on field '${transformedField.name}', 'edgeTypeName' argument must be provided.`,
214
+ [targetDirective.loc],
215
+ );
216
+ }
217
+ const rawType = schema.getRawType(transformedField.type);
218
+ if (schema.canHaveSelections(rawType)) {
219
+ const handle: Handle = {
220
+ name: nodeDirective.name,
221
+ key: '',
222
+ dynamicKey: null,
223
+ filters: null,
224
+ handleArgs: [connectionsArg, edgeTypeNameArg],
225
+ };
226
+ return {
227
+ ...transformedField,
228
+ directives: transformedField.directives.filter(
229
+ directive => directive !== nodeDirective,
230
+ ),
231
+ handles: transformedField.handles
232
+ ? [...transformedField.handles, handle]
233
+ : [handle],
234
+ };
235
+ }
236
+ throw createUserError(
237
+ `Unsupported use of @${nodeDirective.name} on field '${
238
+ transformedField.name
239
+ }'. Expected an object, union or interface, but got '${schema.getTypeString(
240
+ transformedField.type,
241
+ )}'.`,
242
+ [nodeDirective.loc],
243
+ );
244
+ }
245
+ }
246
+
247
+ module.exports = {
248
+ SCHEMA_EXTENSION,
249
+ transform,
250
+ };
@@ -0,0 +1,266 @@
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 IRTransformer = require('../core/IRTransformer');
16
+
17
+ const getIdentifierForArgumentValue = require('../core/getIdentifierForArgumentValue');
18
+ const murmurHash = require('../util/murmurHash');
19
+
20
+ const {createUserError} = require('../core/CompilerError');
21
+
22
+ import type CompilerContext from '../core/CompilerContext';
23
+ import type {
24
+ Argument,
25
+ Defer,
26
+ Directive,
27
+ FragmentSpread,
28
+ InlineFragment,
29
+ LinkedField,
30
+ ScalarField,
31
+ Stream,
32
+ } from '../core/IR';
33
+
34
+ type State = {|
35
+ +documentName: string,
36
+ +recordLabel: (label: string, directive: Directive) => void,
37
+ |};
38
+
39
+ /**
40
+ * This transform finds usages of @defer and @stream, validates them, and
41
+ * converts the using node to specialized IR nodes (Defer/Stream).
42
+ */
43
+ function deferStreamTransform(context: CompilerContext): CompilerContext {
44
+ return IRTransformer.transform(
45
+ context,
46
+ {
47
+ // TODO: type IRTransformer to allow changing result type
48
+ FragmentSpread: (visitFragmentSpread: $FlowFixMe),
49
+ // TODO: type IRTransformer to allow changing result type
50
+ InlineFragment: (visitInlineFragment: $FlowFixMe),
51
+ // TODO: type IRTransformer to allow changing result type
52
+ LinkedField: (visitLinkedField: $FlowFixMe),
53
+ ScalarField: visitScalarField,
54
+ },
55
+ sourceNode => {
56
+ const labels = new Map();
57
+ return {
58
+ documentName: sourceNode.name,
59
+ recordLabel: (label, directive) => {
60
+ const prevDirective = labels.get(label);
61
+ if (prevDirective) {
62
+ const labelArg = directive.args.find(({name}) => name === 'label');
63
+ const prevLabelArg = prevDirective.args.find(
64
+ ({name}) => name === 'label',
65
+ );
66
+ const previousLocation = prevLabelArg?.loc ?? prevDirective.loc;
67
+ if (labelArg) {
68
+ throw createUserError(
69
+ `Invalid use of @${directive.name}, the provided label is ` +
70
+ "not unique. Specify a unique 'label' as a literal string.",
71
+ [labelArg?.loc, previousLocation],
72
+ );
73
+ } else {
74
+ throw createUserError(
75
+ `Invalid use of @${directive.name}, could not generate a ` +
76
+ "default label that is unique. Specify a unique 'label' " +
77
+ 'as a literal string.',
78
+ [directive.loc, previousLocation],
79
+ );
80
+ }
81
+ }
82
+ labels.set(label, directive);
83
+ },
84
+ };
85
+ },
86
+ );
87
+ }
88
+
89
+ function visitLinkedField(
90
+ field: LinkedField,
91
+ state: State,
92
+ ): LinkedField | Stream {
93
+ const context: CompilerContext = this.getContext();
94
+ const schema = context.getSchema();
95
+
96
+ let transformedField: LinkedField = this.traverse(field, state);
97
+ const streamDirective = transformedField.directives.find(
98
+ directive => directive.name === 'stream',
99
+ );
100
+ if (streamDirective == null) {
101
+ return transformedField;
102
+ }
103
+ const type = schema.getNullableType(field.type);
104
+ if (!schema.isList(type)) {
105
+ throw createUserError(
106
+ `Invalid use of @stream on non-plural field '${field.name}'`,
107
+ [streamDirective.loc],
108
+ );
109
+ }
110
+ transformedField = {
111
+ ...transformedField,
112
+ directives: transformedField.directives.filter(
113
+ directive => directive.name !== 'stream',
114
+ ),
115
+ };
116
+ const ifArg = streamDirective.args.find(arg => arg.name === 'if');
117
+ if (isLiteralFalse(ifArg)) {
118
+ return transformedField;
119
+ }
120
+ const initialCount = streamDirective.args.find(
121
+ arg => arg.name === 'initial_count',
122
+ );
123
+ if (initialCount == null) {
124
+ throw createUserError(
125
+ "Invalid use of @stream, the 'initial_count' argument is required.",
126
+ [streamDirective.loc],
127
+ );
128
+ }
129
+ const useCustomizedBatch = streamDirective.args.find(
130
+ arg => arg.name === 'use_customized_batch',
131
+ );
132
+
133
+ const label =
134
+ getLiteralStringArgument(streamDirective, 'label') ?? field.alias;
135
+ const transformedLabel = transformLabel(state.documentName, 'stream', label);
136
+ state.recordLabel(transformedLabel, streamDirective);
137
+ return {
138
+ if: ifArg?.value ?? null,
139
+ initialCount: initialCount.value,
140
+ useCustomizedBatch: useCustomizedBatch?.value ?? null,
141
+ kind: 'Stream',
142
+ label: transformedLabel,
143
+ loc: {kind: 'Derived', source: streamDirective.loc},
144
+ metadata: null,
145
+ selections: [transformedField],
146
+ };
147
+ }
148
+
149
+ function visitScalarField(field: ScalarField, state: State): ScalarField {
150
+ const streamDirective = field.directives.find(
151
+ directive => directive.name === 'stream',
152
+ );
153
+ if (streamDirective != null) {
154
+ throw createUserError(
155
+ `Invalid use of @stream on scalar field '${field.name}'`,
156
+ [streamDirective.loc],
157
+ );
158
+ }
159
+ // $FlowFixMe[incompatible-use]
160
+ return this.traverse(field, state);
161
+ }
162
+
163
+ function visitInlineFragment(
164
+ fragment: InlineFragment,
165
+ state: State,
166
+ ): InlineFragment | Defer {
167
+ const deferDirective = fragment.directives.find(
168
+ directive => directive.name === 'defer',
169
+ );
170
+ if (deferDirective != null) {
171
+ throw createUserError(
172
+ 'Invalid use of @defer on an inline fragment, @defer is only supported on fragment spreads.',
173
+ [fragment.loc],
174
+ );
175
+ }
176
+ return this.traverse(fragment, state);
177
+ }
178
+
179
+ function visitFragmentSpread(
180
+ spread: FragmentSpread,
181
+ state: State,
182
+ ): FragmentSpread | Defer {
183
+ let transformedSpread: FragmentSpread = this.traverse(spread, state);
184
+ const deferDirective = transformedSpread.directives.find(
185
+ directive => directive.name === 'defer',
186
+ );
187
+ if (deferDirective == null) {
188
+ return transformedSpread;
189
+ }
190
+ transformedSpread = {
191
+ ...transformedSpread,
192
+ directives: transformedSpread.directives.filter(
193
+ directive => directive.name !== 'defer',
194
+ ),
195
+ };
196
+ const ifArg = deferDirective.args.find(arg => arg.name === 'if');
197
+ if (isLiteralFalse(ifArg)) {
198
+ return transformedSpread;
199
+ }
200
+ const label =
201
+ getLiteralStringArgument(deferDirective, 'label') ??
202
+ getFragmentSpreadName(spread);
203
+ const transformedLabel = transformLabel(state.documentName, 'defer', label);
204
+ state.recordLabel(transformedLabel, deferDirective);
205
+ return {
206
+ if: ifArg?.value ?? null,
207
+ kind: 'Defer',
208
+ label: transformedLabel,
209
+ loc: {kind: 'Derived', source: deferDirective.loc},
210
+ selections: [transformedSpread],
211
+ };
212
+ }
213
+
214
+ function getLiteralStringArgument(
215
+ directive: Directive,
216
+ argName: string,
217
+ ): ?string {
218
+ const arg = directive.args.find(({name}) => name === argName);
219
+ if (arg == null) {
220
+ return null;
221
+ }
222
+ const value = arg.value.kind === 'Literal' ? arg.value.value : null;
223
+ if (value == null || typeof value !== 'string') {
224
+ throw createUserError(
225
+ `Expected the '${argName}' value to @${directive.name} to be a string literal if provided.`,
226
+ [arg.value.loc],
227
+ );
228
+ }
229
+ return value;
230
+ }
231
+
232
+ function transformLabel(
233
+ parentName: string,
234
+ directive: string,
235
+ label: string,
236
+ ): string {
237
+ return `${parentName}$${directive}$${label}`;
238
+ }
239
+
240
+ function isLiteralFalse(arg: ?Argument): boolean {
241
+ return (
242
+ arg != null && arg.value.kind === 'Literal' && arg.value.value === false
243
+ );
244
+ }
245
+
246
+ function getFragmentSpreadName(fragmentSpread: FragmentSpread): string {
247
+ if (fragmentSpread.args.length === 0) {
248
+ return fragmentSpread.name;
249
+ }
250
+ const sortedArgs = [...fragmentSpread.args]
251
+ .sort((a, b) => {
252
+ return a.name < b.name ? -1 : a.name > b.name ? 1 : 0;
253
+ })
254
+ .map(argument => {
255
+ return {
256
+ name: argument.name,
257
+ value: getIdentifierForArgumentValue(argument.value),
258
+ };
259
+ });
260
+ const hash = murmurHash(JSON.stringify(sortedArgs));
261
+ return `${fragmentSpread.name}_${hash}`;
262
+ }
263
+
264
+ module.exports = {
265
+ transform: deferStreamTransform,
266
+ };
@@ -0,0 +1,48 @@
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 IRTransformer = require('../core/IRTransformer');
16
+
17
+ const {createUserError} = require('../core/CompilerError');
18
+
19
+ import type CompilerContext from '../core/CompilerContext';
20
+ import type {ScalarField, LinkedField} from '../core/IR';
21
+
22
+ function visitField<T: ScalarField | LinkedField>(field: T): T {
23
+ if (field.alias === 'id' && field.name !== 'id') {
24
+ throw createUserError(
25
+ 'Relay does not allow aliasing fields to `id`. ' +
26
+ 'This name is reserved for the globally unique `id` field on ' +
27
+ '`Node`.',
28
+ [field.loc],
29
+ );
30
+ }
31
+ // $FlowFixMe[incompatible-use]
32
+ return this.traverse(field);
33
+ }
34
+
35
+ /**
36
+ * This is not an actual transform (but more a validation)
37
+ * Relay does not allow aliasing fields to `id`.
38
+ */
39
+ function disallowIdAsAlias(context: CompilerContext): CompilerContext {
40
+ return IRTransformer.transform(context, {
41
+ ScalarField: visitField,
42
+ LinkedField: visitField,
43
+ });
44
+ }
45
+
46
+ module.exports = {
47
+ transform: disallowIdAsAlias,
48
+ };
@@ -0,0 +1,45 @@
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 IRValidator = require('../core/IRValidator');
16
+
17
+ const {createUserError} = require('../core/CompilerError');
18
+
19
+ import type CompilerContext from '../core/CompilerContext';
20
+ import type {Root} from '../core/IR';
21
+
22
+ function visitRoot(node: Root) {
23
+ for (const selection of node.selections) {
24
+ if (selection.kind === 'ScalarField' && selection.name === '__typename') {
25
+ throw createUserError(
26
+ 'Relay does not allow `__typename` field on Query, Mutation or Subscription',
27
+ [selection.loc],
28
+ );
29
+ }
30
+ }
31
+ }
32
+
33
+ function stopVisit() {}
34
+
35
+ function disallowTypenameOnRoot(context: CompilerContext): CompilerContext {
36
+ IRValidator.validate(context, {
37
+ Root: visitRoot,
38
+ Fragment: stopVisit,
39
+ });
40
+ return context;
41
+ }
42
+
43
+ module.exports = {
44
+ transform: disallowTypenameOnRoot,
45
+ };
@@ -0,0 +1,81 @@
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 IRTransformer = require('../core/IRTransformer');
16
+ const SchemaUtils = require('../core/SchemaUtils');
17
+
18
+ const invariant = require('invariant');
19
+ const nullthrows = require('nullthrows');
20
+
21
+ const {getRelayHandleKey} = require('relay-runtime');
22
+
23
+ import type CompilerContext from '../core/CompilerContext';
24
+ import type {LinkedField, ScalarField} from '../core/IR';
25
+
26
+ function fieldHandleTransform(context: CompilerContext): CompilerContext {
27
+ return IRTransformer.transform(context, {
28
+ LinkedField: visitField,
29
+ ScalarField: visitField,
30
+ });
31
+ }
32
+
33
+ /**
34
+ * @internal
35
+ */
36
+ function visitField<F: LinkedField | ScalarField>(field: F): F {
37
+ // $FlowFixMe[incompatible-use]
38
+ const nextField = field.kind === 'LinkedField' ? this.traverse(field) : field;
39
+ const handles = nextField.handles;
40
+ if (!handles || !handles.length) {
41
+ return nextField;
42
+ }
43
+ // ensure exactly one handle
44
+ invariant(
45
+ handles.length === 1,
46
+ 'FieldHandleTransform: Expected fields to have at most one ' +
47
+ '"handle" property, got `%s`.',
48
+ handles.join(', '),
49
+ );
50
+ // $FlowFixMe[incompatible-use]
51
+ const context: CompilerContext = this.getContext();
52
+ const schema = context.getSchema();
53
+ const alias = nextField.alias;
54
+ const handle = handles[0];
55
+ const name = getRelayHandleKey(handle.name, handle.key, nextField.name);
56
+ const filters = handle.filters;
57
+ const args = filters
58
+ ? nextField.args.filter(arg => filters.indexOf(arg.name) !== -1)
59
+ : [];
60
+ if (handle.dynamicKey != null) {
61
+ args.push({
62
+ kind: 'Argument',
63
+ loc: handle.dynamicKey.loc,
64
+ name: '__dynamicKey',
65
+ type: SchemaUtils.getNullableStringInput(schema),
66
+ value: nullthrows(handle.dynamicKey),
67
+ });
68
+ }
69
+
70
+ return ({
71
+ ...nextField,
72
+ args,
73
+ alias,
74
+ name,
75
+ handles: null,
76
+ }: $FlowIssue);
77
+ }
78
+
79
+ module.exports = {
80
+ transform: fieldHandleTransform,
81
+ };