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,283 @@
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 {RelayConcreteNode} = require('relay-runtime');
16
+
17
+ import type {
18
+ IsGeneratedFileFn,
19
+ KeepExtraFileFn,
20
+ } from '../codegen/CodegenRunner';
21
+ import type {IRTransform} from '../core/CompilerContext';
22
+ import type {GeneratedDefinition, Root, Fragment} from '../core/IR';
23
+ import type {GetFileFilter} from '../core/RelaySourceModuleParser';
24
+ import type {Schema} from '../core/Schema';
25
+ import type {ScalarTypeMapping} from './javascript/RelayFlowTypeTransformers';
26
+ import type {GeneratedNode} from 'relay-runtime';
27
+
28
+ /**
29
+ * A language plugin allows relay-compiler to both read and write files for any
30
+ * language.
31
+ *
32
+ * When reading, the plugin is expected to parse and return GraphQL tags; and
33
+ * when writing the plugin is responsible for generating type information about
34
+ * the GraphQL selections made as well as generating the contents of the
35
+ * artifact file.
36
+ *
37
+ * This interface describes the details relay-compiler requires to be able to
38
+ * use the plugin and is expected to be returned by a {PluginInitializer}.
39
+ */
40
+ export type PluginInterface = {
41
+ inputExtensions: string[],
42
+ outputExtension: string,
43
+ findGraphQLTags: GraphQLTagFinder,
44
+ formatModule: FormatModule,
45
+ typeGenerator: TypeGenerator,
46
+ isGeneratedFile?: IsGeneratedFileFn,
47
+ keepExtraFile?: KeepExtraFileFn,
48
+ schemaExtensions?: $ReadOnlyArray<string>,
49
+ getModuleName?: (operationName: string) => string,
50
+ getFileFilter?: GetFileFilter,
51
+ ...
52
+ };
53
+
54
+ /**
55
+ * The plugin is expected to have as its main default export a function that
56
+ * returns an object conforming to the plugin interface.
57
+ *
58
+ * For now a plugin doesn’t take any arguments, but may do so in the future.
59
+ */
60
+ export type PluginInitializer = () => PluginInterface;
61
+
62
+ export type GraphQLTag = {
63
+ /**
64
+ * Should hold the string content of the `graphql` tagged template literal,
65
+ * which is either an operation or fragment.
66
+ *
67
+ * @example
68
+ *
69
+ * grapqhl`query MyQuery { … }`
70
+ * grapqhl`fragment MyFragment on MyType { … }`
71
+ */
72
+ template: string,
73
+ /**
74
+ * In the case this tag was part of a fragment container and it used a node
75
+ * map as fragment spec, rather than a single tagged node, this should hold
76
+ * the prop key to which the node is assigned.
77
+ *
78
+ * @example
79
+ *
80
+ * createFragmentContainer(
81
+ * MyComponent,
82
+ * {
83
+ * keyName: graphql`fragment MyComponent_keyName { … }`
84
+ * }
85
+ * )
86
+ *
87
+ */
88
+ keyName: ?string,
89
+ /**
90
+ * The location in the source file that the tag is placed at.
91
+ */
92
+ sourceLocationOffset: {|
93
+ /**
94
+ * The line in the source file that the tag is placed on.
95
+ *
96
+ * Lines use 1-based indexing.
97
+ */
98
+ line: number,
99
+
100
+ /**
101
+ * The column in the source file that the tag starts on.
102
+ *
103
+ * Columns use 1-based indexing.
104
+ */
105
+ column: number,
106
+ |},
107
+ ...
108
+ };
109
+
110
+ /**
111
+ * This function is responsible for extracting `GraphQLTag` objects from source
112
+ * files.
113
+ *
114
+ * @param {string} text The source file contents.
115
+ * @param {string} filePath The path to the source file on disk.
116
+ * @return {Array<GraphQLTag>} All extracted `GraphQLTag` objects.
117
+ *
118
+ * @see {@link javascript/FindGraphQLTags.js}
119
+ */
120
+ export type GraphQLTagFinder = (
121
+ text: string,
122
+ filePath: string,
123
+ ) => $ReadOnlyArray<GraphQLTag>;
124
+
125
+ /**
126
+ * The function that is responsible for generating the contents of the artifact
127
+ * file.
128
+ *
129
+ * @see {@link javascript/formatGeneratedModule.js}
130
+ */
131
+ export type FormatModule = ({|
132
+ /**
133
+ * The filename of the module.
134
+ */
135
+ moduleName: string,
136
+
137
+ /**
138
+ * The type of artifact that this module represents.
139
+ *
140
+ * @todo Document when this can be `empty`.
141
+ */
142
+ documentType:
143
+ | typeof RelayConcreteNode.FRAGMENT
144
+ | typeof RelayConcreteNode.REQUEST
145
+ | null,
146
+
147
+ /**
148
+ * The actual document that this module represents.
149
+ */
150
+ docText: ?string,
151
+
152
+ /**
153
+ * The IR for the document that this module represents.
154
+ */
155
+ concreteText: string,
156
+
157
+ /**
158
+ * The type information generated for the GraphQL selections made.
159
+ */
160
+ typeText: string,
161
+
162
+ /**
163
+ * A hash of the concrete node including the query text.
164
+ *
165
+ * @todo Document how this is different from `sourceHash`.
166
+ */
167
+ hash: ?string,
168
+
169
+ /**
170
+ * The 'kind' of the generated node.
171
+ */
172
+ kind: string,
173
+
174
+ /**
175
+ * The IR node from which the generated node is derived.
176
+ */
177
+ definition: GeneratedDefinition,
178
+
179
+ /**
180
+ * A hash of the document, which is used by relay-compiler to know if it needs
181
+ * to write a new version of the artifact.
182
+ *
183
+ * @todo Is this correct? And document how this is different from `hash`.
184
+ */
185
+ sourceHash: string,
186
+
187
+ /**
188
+ * The generated node being written.
189
+ */
190
+ node: GeneratedNode,
191
+
192
+ /**
193
+ * GraphQL Schema Interface
194
+ */
195
+ schema: Schema,
196
+ |}) => string;
197
+
198
+ /**
199
+ * The options that will be passed to the `generate` function of your plugin’s
200
+ * type generator.
201
+ */
202
+ export type TypeGeneratorOptions = {|
203
+ /**
204
+ * A map of custom scalars to scalars that the plugin knows about and emits
205
+ * type information for.
206
+ *
207
+ * @example
208
+ *
209
+ * // The URL custom scalar is essentially a string and should be treated as
210
+ * // such by the language’s type system.
211
+ * { URL: 'String' }
212
+ */
213
+ +customScalars: ScalarTypeMapping,
214
+
215
+ /**
216
+ * Whether or not relay-compiler will store artifacts next to the module that
217
+ * they originate from or all together in a single directory.
218
+ *
219
+ * Storing all artifacts in a single directory makes it easy to import and
220
+ * reference fragments defined in other artifacts without needing to use the
221
+ * Haste module system.
222
+ *
223
+ * This defaults to `false`.
224
+ */
225
+ +useSingleArtifactDirectory: boolean,
226
+
227
+ /**
228
+ * This option controls whether or not a catch-all entry is added to enum type
229
+ * definitions for values that may be added in the future. Enabling this means
230
+ * you will have to update your application whenever the GraphQL server schema
231
+ * adds new enum values to prevent it from breaking.
232
+ *
233
+ * This defaults to `false`.
234
+ */
235
+ +noFutureProofEnums: boolean,
236
+
237
+ /**
238
+ * @todo Document this.
239
+ */
240
+ +optionalInputFields: $ReadOnlyArray<string>,
241
+
242
+ /**
243
+ * Whether or not the Haste module system is being used. This will currently
244
+ * always be `false` for OSS users.
245
+ */
246
+ +useHaste: boolean,
247
+
248
+ /**
249
+ * Import flow types from the Haste-style global module name or per-enum
250
+ * global module name given by the function variant.
251
+ */
252
+ +enumsHasteModule?: string | ((enumName: string) => string),
253
+
254
+ /**
255
+ * Optional normalization IR for generating raw response
256
+ */
257
+ +normalizationIR?: Root,
258
+ |};
259
+
260
+ /**
261
+ * This object should hold the implementation required to generate types for the
262
+ * GraphQL selections made.
263
+ *
264
+ * @see {@link javascript/RelayFlowGenerator.js}
265
+ */
266
+ export type TypeGenerator = {
267
+ /**
268
+ * Transforms that should be applied to the intermediate representation of the
269
+ * GraphQL document before passing to the `generate` function.
270
+ */
271
+ transforms: $ReadOnlyArray<IRTransform>,
272
+ /**
273
+ * Given GraphQL document IR, this function should generate type information
274
+ * for e.g. the selections made. It can, however, also generate any other
275
+ * content such as importing other files, including other artifacts.
276
+ */
277
+ generate: (
278
+ schema: Schema,
279
+ node: Root | Fragment,
280
+ options: TypeGeneratorOptions,
281
+ ) => string,
282
+ ...
283
+ };
@@ -0,0 +1,232 @@
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 Profiler = require('../../core/GraphQLCompilerProfiler');
16
+
17
+ const babylon = require('@babel/parser');
18
+ const util = require('util');
19
+
20
+ import type {GraphQLTag} from '../RelayLanguagePluginInterface';
21
+
22
+ // Attempt to be as inclusive as possible of source text.
23
+ const BABYLON_OPTIONS = {
24
+ allowImportExportEverywhere: true,
25
+ allowReturnOutsideFunction: true,
26
+ allowSuperOutsideMethod: true,
27
+ sourceType: 'module',
28
+ plugins: [
29
+ 'asyncGenerators',
30
+ 'classProperties',
31
+ ['decorators', {decoratorsBeforeExport: true}],
32
+ 'doExpressions',
33
+ 'dynamicImport',
34
+ 'exportExtensions',
35
+ ['flow', {enums: true}],
36
+ 'functionBind',
37
+ 'functionSent',
38
+ 'jsx',
39
+ 'nullishCoalescingOperator',
40
+ 'objectRestSpread',
41
+ 'optionalChaining',
42
+ 'optionalCatchBinding',
43
+ ],
44
+ strictMode: false,
45
+ };
46
+
47
+ function find(text: string): $ReadOnlyArray<GraphQLTag> {
48
+ const result: Array<GraphQLTag> = [];
49
+ const ast = babylon.parse(text, BABYLON_OPTIONS);
50
+
51
+ const visitors = {
52
+ CallExpression: node => {
53
+ const callee = node.callee;
54
+ if (
55
+ !(
56
+ (callee.type === 'Identifier' &&
57
+ CREATE_CONTAINER_FUNCTIONS[callee.name]) ||
58
+ (callee.kind === 'MemberExpression' &&
59
+ callee.object.type === 'Identifier' &&
60
+ callee.object.value === 'Relay' &&
61
+ callee.property.type === 'Identifier' &&
62
+ CREATE_CONTAINER_FUNCTIONS[callee.property.name])
63
+ )
64
+ ) {
65
+ traverse(node, visitors);
66
+ return;
67
+ }
68
+ const fragments = node.arguments[1];
69
+ if (fragments.type === 'ObjectExpression') {
70
+ fragments.properties.forEach(property => {
71
+ invariant(
72
+ property.type === 'ObjectProperty' &&
73
+ property.key.type === 'Identifier' &&
74
+ property.value.type === 'TaggedTemplateExpression',
75
+ 'FindGraphQLTags: `%s` expects fragment definitions to be ' +
76
+ '`key: graphql`.',
77
+ node.callee.name,
78
+ );
79
+ invariant(
80
+ isGraphQLModernOrDeprecatedTag(property.value.tag),
81
+ 'FindGraphQLTags: `%s` expects fragment definitions to be tagged ' +
82
+ 'with `graphql`, got `%s`.',
83
+ node.callee.name,
84
+ getSourceTextForLocation(text, property.value.tag.loc),
85
+ );
86
+ if (isGraphQLTag(property.value.tag)) {
87
+ result.push({
88
+ keyName: property.key.name,
89
+ template: getGraphQLText(property.value.quasi),
90
+ sourceLocationOffset: getSourceLocationOffset(
91
+ property.value.quasi,
92
+ ),
93
+ });
94
+ }
95
+ });
96
+ } else {
97
+ invariant(
98
+ fragments && fragments.type === 'TaggedTemplateExpression',
99
+ 'FindGraphQLTags: `%s` expects a second argument of fragment ' +
100
+ 'definitions.',
101
+ node.callee.name,
102
+ );
103
+ invariant(
104
+ isGraphQLModernOrDeprecatedTag(fragments.tag),
105
+ 'FindGraphQLTags: `%s` expects fragment definitions to be tagged ' +
106
+ 'with `graphql`, got `%s`.',
107
+ node.callee.name,
108
+ getSourceTextForLocation(text, fragments.tag.loc),
109
+ );
110
+ result.push({
111
+ keyName: null,
112
+ template: getGraphQLText(fragments.quasi),
113
+ sourceLocationOffset: getSourceLocationOffset(fragments.quasi),
114
+ });
115
+ }
116
+
117
+ // Visit remaining arguments
118
+ for (let ii = 2; ii < node.arguments.length; ii++) {
119
+ visit(node.arguments[ii], visitors);
120
+ }
121
+ },
122
+ TaggedTemplateExpression: node => {
123
+ if (isGraphQLTag(node.tag)) {
124
+ result.push({
125
+ keyName: null,
126
+ template: node.quasi.quasis[0].value.raw,
127
+ sourceLocationOffset: getSourceLocationOffset(node.quasi),
128
+ });
129
+ }
130
+ },
131
+ };
132
+ visit(ast, visitors);
133
+ return result;
134
+ }
135
+
136
+ const CREATE_CONTAINER_FUNCTIONS = Object.create(null, {
137
+ createFragmentContainer: {value: true},
138
+ createPaginationContainer: {value: true},
139
+ createRefetchContainer: {value: true},
140
+ });
141
+
142
+ const IGNORED_KEYS = {
143
+ comments: true,
144
+ end: true,
145
+ leadingComments: true,
146
+ loc: true,
147
+ name: true,
148
+ start: true,
149
+ trailingComments: true,
150
+ type: true,
151
+ };
152
+
153
+ function isGraphQLTag(tag): boolean {
154
+ return tag.type === 'Identifier' && tag.name === 'graphql';
155
+ }
156
+
157
+ function isGraphQLModernOrDeprecatedTag(tag): boolean {
158
+ return (
159
+ tag.type === 'Identifier' &&
160
+ (tag.name === 'graphql' || tag.name === 'graphql_DEPRECATED')
161
+ );
162
+ }
163
+
164
+ function getTemplateNode(quasi) {
165
+ const quasis = quasi.quasis;
166
+ invariant(
167
+ quasis && quasis.length === 1,
168
+ 'FindGraphQLTags: Substitutions are not allowed in graphql tags.',
169
+ );
170
+ return quasis[0];
171
+ }
172
+
173
+ function getGraphQLText(quasi): string {
174
+ return getTemplateNode(quasi).value.raw;
175
+ }
176
+
177
+ function getSourceLocationOffset(quasi) {
178
+ const loc = getTemplateNode(quasi).loc.start;
179
+ return {
180
+ line: loc.line,
181
+ column: loc.column + 1, // babylon is 0-indexed, graphql expects 1-indexed
182
+ };
183
+ }
184
+
185
+ function getSourceTextForLocation(text, loc) {
186
+ if (loc == null) {
187
+ return '(source unavailable)';
188
+ }
189
+ const lines = text.split('\n').slice(loc.start.line - 1, loc.end.line);
190
+ lines[0] = lines[0].slice(loc.start.column);
191
+ lines[lines.length - 1] = lines[lines.length - 1].slice(0, loc.end.column);
192
+ return lines.join('\n');
193
+ }
194
+
195
+ function invariant(condition, msg, ...args) {
196
+ if (!condition) {
197
+ throw new Error(util.format(msg, ...args));
198
+ }
199
+ }
200
+
201
+ function visit(node, visitors) {
202
+ const fn = visitors[node.type];
203
+ if (fn != null) {
204
+ fn(node);
205
+ return;
206
+ }
207
+ traverse(node, visitors);
208
+ }
209
+
210
+ function traverse(node, visitors) {
211
+ for (const key in node) {
212
+ if (IGNORED_KEYS[key]) {
213
+ continue;
214
+ }
215
+ const prop = node[key];
216
+ if (prop && typeof prop === 'object' && typeof prop.type === 'string') {
217
+ visit(prop, visitors);
218
+ } else if (Array.isArray(prop)) {
219
+ prop.forEach(item => {
220
+ if (item && typeof item === 'object' && typeof item.type === 'string') {
221
+ visit(item, visitors);
222
+ }
223
+ });
224
+ }
225
+ }
226
+ }
227
+
228
+ module.exports = {
229
+ find: (Profiler.instrument(find, 'FindGraphQLTags.find'): (
230
+ text: string,
231
+ ) => $ReadOnlyArray<GraphQLTag>),
232
+ };
@@ -0,0 +1,180 @@
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
8
+ * @format
9
+ */
10
+
11
+ // flowlint ambiguous-object-type:error
12
+
13
+ 'use strict';
14
+
15
+ const invariant = require('invariant');
16
+ const t = require('@babel/types');
17
+
18
+ type BabelAST = mixed;
19
+
20
+ /**
21
+ * type NAME = any;
22
+ */
23
+ function anyTypeAlias(name: string): BabelAST {
24
+ return t.typeAlias(t.identifier(name), null, t.anyTypeAnnotation());
25
+ }
26
+
27
+ /**
28
+ * {|
29
+ * PROPS
30
+ * |}
31
+ */
32
+ function exactObjectTypeAnnotation(
33
+ props: $ReadOnlyArray<BabelAST>,
34
+ ): $FlowFixMe {
35
+ const typeAnnotation = t.objectTypeAnnotation(props);
36
+ typeAnnotation.exact = true;
37
+ return typeAnnotation;
38
+ }
39
+
40
+ /**
41
+ * {
42
+ * PROPS
43
+ * ...
44
+ * }
45
+ */
46
+ function inexactObjectTypeAnnotation(
47
+ props: $ReadOnlyArray<BabelAST>,
48
+ ): $FlowFixMe {
49
+ const typeAnnotation = t.objectTypeAnnotation(props);
50
+ typeAnnotation.inexact = true;
51
+ return typeAnnotation;
52
+ }
53
+
54
+ /**
55
+ * export type NAME = TYPE
56
+ */
57
+ function exportType(name: string, type: BabelAST): $FlowFixMe {
58
+ return t.exportNamedDeclaration(
59
+ t.typeAlias(t.identifier(name), null, type),
60
+ [],
61
+ null,
62
+ );
63
+ }
64
+
65
+ /**
66
+ * export type {A, B, C}
67
+ */
68
+ function exportTypes(names: $ReadOnlyArray<string>): $FlowFixMe {
69
+ const res = t.exportNamedDeclaration(
70
+ null,
71
+ names.map(name =>
72
+ t.exportSpecifier(t.identifier(name), t.identifier(name)),
73
+ ),
74
+
75
+ null,
76
+ );
77
+ res.exportKind = 'type';
78
+ return res;
79
+ }
80
+
81
+ /**
82
+ * declare export type NAME = VALUE
83
+ */
84
+ function declareExportOpaqueType(name: string, value: string): $FlowFixMe {
85
+ return t.declareExportDeclaration(
86
+ t.declareOpaqueType(
87
+ t.identifier(name),
88
+ null,
89
+ t.genericTypeAnnotation(t.identifier(value)),
90
+ ),
91
+ );
92
+ }
93
+
94
+ /**
95
+ * import type {NAMES[0], NAMES[1], ...} from 'MODULE';
96
+ */
97
+ function importTypes(
98
+ names: $ReadOnlyArray<string>,
99
+ module: string,
100
+ ): $FlowFixMe {
101
+ const importDeclaration = t.importDeclaration(
102
+ names.map(name =>
103
+ t.importSpecifier(t.identifier(name), t.identifier(name)),
104
+ ),
105
+ t.stringLiteral(module),
106
+ );
107
+ importDeclaration.importKind = 'type';
108
+ return importDeclaration;
109
+ }
110
+
111
+ /**
112
+ * Create an intersection type if needed.
113
+ *
114
+ * TYPES[0] & TYPES[1] & ...
115
+ */
116
+ function intersectionTypeAnnotation(types: $ReadOnlyArray<BabelAST>): BabelAST {
117
+ invariant(
118
+ types.length > 0,
119
+ 'RelayFlowBabelFactories: cannot create an intersection of 0 types',
120
+ );
121
+ return types.length === 1 ? types[0] : t.intersectionTypeAnnotation(types);
122
+ }
123
+
124
+ function lineComments(
125
+ ...lines: $ReadOnlyArray<string>
126
+ ): $ReadOnlyArray<$FlowFixMe> {
127
+ return lines.map(line => ({type: 'CommentLine', value: ' ' + line}));
128
+ }
129
+
130
+ /**
131
+ * $ReadOnlyArray<TYPE>
132
+ */
133
+ function readOnlyArrayOfType(thing: BabelAST): $FlowFixMe {
134
+ return t.genericTypeAnnotation(
135
+ t.identifier('$ReadOnlyArray'),
136
+ t.typeParameterInstantiation([thing]),
137
+ );
138
+ }
139
+
140
+ /**
141
+ * +KEY: VALUE
142
+ */
143
+ function readOnlyObjectTypeProperty(key: string, value: BabelAST): $FlowFixMe {
144
+ const prop = t.objectTypeProperty(t.identifier(key), value);
145
+ prop.variance = t.variance('plus');
146
+ return prop;
147
+ }
148
+
149
+ function stringLiteralTypeAnnotation(value: string): $FlowFixMe {
150
+ return t.stringLiteralTypeAnnotation(value);
151
+ }
152
+
153
+ /**
154
+ * Create a union type if needed.
155
+ *
156
+ * TYPES[0] | TYPES[1] | ...
157
+ */
158
+ function unionTypeAnnotation(types: $ReadOnlyArray<BabelAST>): BabelAST {
159
+ invariant(
160
+ types.length > 0,
161
+ 'RelayFlowBabelFactories: cannot create a union of 0 types',
162
+ );
163
+ return types.length === 1 ? types[0] : t.unionTypeAnnotation(types);
164
+ }
165
+
166
+ module.exports = {
167
+ anyTypeAlias,
168
+ declareExportOpaqueType,
169
+ exactObjectTypeAnnotation,
170
+ inexactObjectTypeAnnotation,
171
+ exportType,
172
+ exportTypes,
173
+ importTypes,
174
+ intersectionTypeAnnotation,
175
+ lineComments,
176
+ readOnlyArrayOfType,
177
+ readOnlyObjectTypeProperty,
178
+ stringLiteralTypeAnnotation,
179
+ unionTypeAnnotation,
180
+ };