relay-compiler 9.0.0 → 10.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (217) hide show
  1. package/bin/RelayCompilerBin.js.flow +169 -0
  2. package/bin/RelayCompilerMain.js.flow +515 -0
  3. package/bin/__fixtures__/plugin-module.js.flow +17 -0
  4. package/bin/relay-compiler +3862 -2505
  5. package/codegen/CodegenDirectory.js.flow +375 -0
  6. package/codegen/CodegenRunner.js.flow +432 -0
  7. package/codegen/CodegenTypes.js.flow +28 -0
  8. package/codegen/CodegenWatcher.js.flow +254 -0
  9. package/codegen/NormalizationCodeGenerator.js.flow +571 -0
  10. package/codegen/ReaderCodeGenerator.js.flow +512 -0
  11. package/codegen/RelayCodeGenerator.js.flow +85 -0
  12. package/codegen/RelayFileWriter.js.flow +367 -0
  13. package/codegen/SourceControl.js.flow +58 -0
  14. package/codegen/compileRelayArtifacts.js.flow +182 -0
  15. package/codegen/createPrintRequireModuleDependency.js.flow +21 -0
  16. package/codegen/sortObjectByKey.js.flow +25 -0
  17. package/codegen/writeRelayGeneratedFile.js.flow +223 -0
  18. package/core/ASTCache.js.flow +74 -0
  19. package/core/ASTConvert.js.flow +233 -0
  20. package/core/CompilerContext.js.flow +191 -0
  21. package/core/CompilerError.js.flow +250 -0
  22. package/core/DotGraphQLParser.js.flow +39 -0
  23. package/core/GraphQLCompilerProfiler.js.flow +341 -0
  24. package/core/GraphQLDerivedFromMetadata.js.flow +36 -0
  25. package/core/GraphQLWatchmanClient.js.flow +111 -0
  26. package/core/IR.js.flow +326 -0
  27. package/core/IRPrinter.js.flow +477 -0
  28. package/core/IRTransformer.js.flow +377 -0
  29. package/core/IRValidator.js.flow +260 -0
  30. package/core/IRVisitor.js.flow +150 -0
  31. package/core/JSModuleParser.js.flow +24 -0
  32. package/core/RelayCompilerScope.js.flow +199 -0
  33. package/core/RelayFindGraphQLTags.js.flow +119 -0
  34. package/core/RelayGraphQLEnumsGenerator.js.flow +55 -0
  35. package/core/RelayIRTransforms.js.flow +138 -0
  36. package/core/RelayParser.js.flow +1731 -0
  37. package/core/RelaySourceModuleParser.js.flow +135 -0
  38. package/core/Schema.js.flow +2037 -0
  39. package/core/SchemaUtils.js.flow +120 -0
  40. package/core/filterContextForNode.js.flow +50 -0
  41. package/core/getFieldDefinition.js.flow +156 -0
  42. package/core/getIdentifierForArgumentValue.js.flow +49 -0
  43. package/core/getIdentifierForSelection.js.flow +69 -0
  44. package/core/getLiteralArgumentValues.js.flow +32 -0
  45. package/core/getNormalizationOperationName.js.flow +19 -0
  46. package/core/inferRootArgumentDefinitions.js.flow +323 -0
  47. package/index.js +1 -1
  48. package/index.js.flow +200 -0
  49. package/language/RelayLanguagePluginInterface.js.flow +283 -0
  50. package/language/javascript/FindGraphQLTags.js.flow +137 -0
  51. package/language/javascript/RelayFlowBabelFactories.js.flow +176 -0
  52. package/language/javascript/RelayFlowGenerator.js.flow +1099 -0
  53. package/language/javascript/RelayFlowTypeTransformers.js.flow +184 -0
  54. package/language/javascript/RelayLanguagePluginJavaScript.js.flow +34 -0
  55. package/language/javascript/formatGeneratedModule.js.flow +65 -0
  56. package/lib/bin/RelayCompilerBin.js +10 -0
  57. package/lib/bin/RelayCompilerMain.js +127 -130
  58. package/lib/codegen/CodegenDirectory.js +2 -6
  59. package/lib/codegen/CodegenRunner.js +37 -76
  60. package/lib/codegen/CodegenWatcher.js +13 -21
  61. package/lib/codegen/NormalizationCodeGenerator.js +131 -50
  62. package/lib/codegen/ReaderCodeGenerator.js +116 -49
  63. package/lib/codegen/RelayCodeGenerator.js +17 -6
  64. package/lib/codegen/RelayFileWriter.js +15 -37
  65. package/lib/codegen/compileRelayArtifacts.js +16 -30
  66. package/lib/codegen/sortObjectByKey.js +43 -0
  67. package/lib/codegen/writeRelayGeneratedFile.js +86 -96
  68. package/lib/core/ASTCache.js +3 -4
  69. package/lib/core/CompilerContext.js +3 -4
  70. package/lib/core/CompilerError.js +27 -54
  71. package/lib/core/GraphQLCompilerProfiler.js +8 -12
  72. package/lib/core/GraphQLDerivedFromMetadata.js +1 -10
  73. package/lib/core/GraphQLWatchmanClient.js +4 -12
  74. package/lib/core/IRPrinter.js +16 -21
  75. package/lib/core/IRTransformer.js +8 -6
  76. package/lib/core/IRValidator.js +1 -3
  77. package/lib/core/RelayCompilerScope.js +4 -4
  78. package/lib/core/RelayGraphQLEnumsGenerator.js +12 -15
  79. package/lib/core/RelayIRTransforms.js +23 -13
  80. package/lib/core/RelayParser.js +53 -89
  81. package/lib/core/RelaySourceModuleParser.js +1 -3
  82. package/lib/core/Schema.js +106 -77
  83. package/lib/core/SchemaUtils.js +15 -1
  84. package/lib/core/getFieldDefinition.js +12 -15
  85. package/lib/core/getIdentifierForSelection.js +1 -1
  86. package/lib/core/inferRootArgumentDefinitions.js +27 -36
  87. package/lib/index.js +1 -3
  88. package/lib/language/javascript/FindGraphQLTags.js +7 -72
  89. package/lib/language/javascript/RelayFlowBabelFactories.js +5 -5
  90. package/lib/language/javascript/RelayFlowGenerator.js +131 -108
  91. package/lib/language/javascript/RelayFlowTypeTransformers.js +1 -3
  92. package/lib/reporters/ConsoleReporter.js +1 -3
  93. package/lib/reporters/MultiReporter.js +1 -3
  94. package/lib/runner/Artifacts.js +69 -170
  95. package/lib/runner/BufferedFilesystem.js +32 -66
  96. package/lib/runner/GraphQLASTNodeGroup.js +54 -120
  97. package/lib/runner/GraphQLNodeMap.js +14 -19
  98. package/lib/runner/Sources.js +70 -85
  99. package/lib/runner/StrictMap.js +21 -37
  100. package/lib/runner/getChangedNodeNames.js +30 -62
  101. package/lib/transforms/ApplyFragmentArgumentTransform.js +71 -31
  102. package/lib/transforms/ClientExtensionsTransform.js +15 -15
  103. package/lib/transforms/ConnectionTransform.js +26 -38
  104. package/lib/transforms/DeclarativeConnectionMutationTransform.js +225 -0
  105. package/lib/transforms/DeferStreamTransform.js +27 -17
  106. package/lib/transforms/DisallowTypenameOnRoot.js +55 -0
  107. package/lib/transforms/FieldHandleTransform.js +7 -3
  108. package/lib/transforms/FilterCompilerDirectivesTransform.js +29 -0
  109. package/lib/transforms/FlattenTransform.js +23 -19
  110. package/lib/transforms/GenerateIDFieldTransform.js +56 -35
  111. package/lib/transforms/GenerateTypeNameTransform.js +84 -10
  112. package/lib/transforms/InlineDataFragmentTransform.js +9 -4
  113. package/lib/transforms/MaskTransform.js +17 -17
  114. package/lib/transforms/MatchTransform.js +114 -32
  115. package/lib/transforms/ReactFlightComponentTransform.js +162 -0
  116. package/lib/transforms/RefetchableFragmentTransform.js +21 -17
  117. package/lib/transforms/RelayDirectiveTransform.js +8 -3
  118. package/lib/transforms/RequiredFieldTransform.js +380 -0
  119. package/lib/transforms/SkipClientExtensionsTransform.js +8 -0
  120. package/lib/transforms/SkipHandleFieldTransform.js +6 -2
  121. package/lib/transforms/SkipRedundantNodesTransform.js +9 -2
  122. package/lib/transforms/SkipSplitOperationTransform.js +32 -0
  123. package/lib/transforms/SkipUnreachableNodeTransform.js +9 -2
  124. package/lib/transforms/SkipUnusedVariablesTransform.js +18 -17
  125. package/lib/transforms/SplitModuleImportTransform.js +2 -2
  126. package/lib/transforms/TestOperationTransform.js +26 -20
  127. package/lib/transforms/ValidateGlobalVariablesTransform.js +18 -30
  128. package/lib/transforms/ValidateRequiredArgumentsTransform.js +12 -15
  129. package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +16 -30
  130. package/lib/transforms/ValidateUnusedVariablesTransform.js +18 -30
  131. package/lib/transforms/query-generators/FetchableQueryGenerator.js +161 -0
  132. package/lib/transforms/query-generators/NodeQueryGenerator.js +22 -3
  133. package/lib/transforms/query-generators/QueryQueryGenerator.js +2 -1
  134. package/lib/transforms/query-generators/ViewerQueryGenerator.js +1 -0
  135. package/lib/transforms/query-generators/index.js +23 -6
  136. package/lib/transforms/query-generators/utils.js +17 -16
  137. package/lib/util/RelayCompilerCache.js +2 -4
  138. package/lib/util/argumentContainsVariables.js +37 -0
  139. package/lib/util/dedupeJSONStringify.js +15 -12
  140. package/lib/util/generateAbstractTypeRefinementKey.js +24 -0
  141. package/lib/util/getModuleName.js +1 -1
  142. package/lib/util/joinArgumentDefinitions.js +3 -1
  143. package/package.json +7 -7
  144. package/relay-compiler.js +4 -4
  145. package/relay-compiler.min.js +4 -4
  146. package/reporters/ConsoleReporter.js.flow +81 -0
  147. package/reporters/MultiReporter.js.flow +43 -0
  148. package/reporters/Reporter.js.flow +19 -0
  149. package/runner/Artifacts.js.flow +219 -0
  150. package/runner/BufferedFilesystem.js.flow +194 -0
  151. package/runner/GraphQLASTNodeGroup.js.flow +176 -0
  152. package/runner/GraphQLASTUtils.js.flow +26 -0
  153. package/runner/GraphQLNodeMap.js.flow +55 -0
  154. package/runner/Sources.js.flow +228 -0
  155. package/runner/StrictMap.js.flow +96 -0
  156. package/runner/compileArtifacts.js.flow +76 -0
  157. package/runner/extractAST.js.flow +100 -0
  158. package/runner/getChangedNodeNames.js.flow +48 -0
  159. package/runner/getSchemaInstance.js.flow +36 -0
  160. package/runner/types.js.flow +37 -0
  161. package/transforms/ApplyFragmentArgumentTransform.js.flow +526 -0
  162. package/transforms/ClientExtensionsTransform.js.flow +224 -0
  163. package/transforms/ConnectionTransform.js.flow +855 -0
  164. package/transforms/DeclarativeConnectionMutationTransform.js.flow +246 -0
  165. package/transforms/DeferStreamTransform.js.flow +265 -0
  166. package/transforms/DisallowIdAsAlias.js.flow +47 -0
  167. package/transforms/DisallowTypenameOnRoot.js.flow +45 -0
  168. package/transforms/FieldHandleTransform.js.flow +79 -0
  169. package/transforms/FilterCompilerDirectivesTransform.js.flow +33 -0
  170. package/transforms/FilterDirectivesTransform.js.flow +45 -0
  171. package/transforms/FlattenTransform.js.flow +454 -0
  172. package/transforms/GenerateIDFieldTransform.js.flow +152 -0
  173. package/transforms/GenerateTypeNameTransform.js.flow +161 -0
  174. package/transforms/InlineDataFragmentTransform.js.flow +125 -0
  175. package/transforms/InlineFragmentsTransform.js.flow +71 -0
  176. package/transforms/MaskTransform.js.flow +126 -0
  177. package/transforms/MatchTransform.js.flow +589 -0
  178. package/transforms/ReactFlightComponentTransform.js.flow +195 -0
  179. package/transforms/RefetchableFragmentTransform.js.flow +272 -0
  180. package/transforms/RelayDirectiveTransform.js.flow +97 -0
  181. package/transforms/RequiredFieldTransform.js.flow +415 -0
  182. package/transforms/SkipClientExtensionsTransform.js.flow +54 -0
  183. package/transforms/SkipHandleFieldTransform.js.flow +44 -0
  184. package/transforms/SkipRedundantNodesTransform.js.flow +257 -0
  185. package/transforms/SkipSplitOperationTransform.js.flow +37 -0
  186. package/transforms/SkipUnreachableNodeTransform.js.flow +149 -0
  187. package/transforms/SkipUnusedVariablesTransform.js.flow +59 -0
  188. package/transforms/SplitModuleImportTransform.js.flow +98 -0
  189. package/transforms/TestOperationTransform.js.flow +142 -0
  190. package/transforms/TransformUtils.js.flow +26 -0
  191. package/transforms/ValidateGlobalVariablesTransform.js.flow +81 -0
  192. package/transforms/ValidateRequiredArgumentsTransform.js.flow +127 -0
  193. package/transforms/ValidateServerOnlyDirectivesTransform.js.flow +112 -0
  194. package/transforms/ValidateUnusedVariablesTransform.js.flow +89 -0
  195. package/transforms/query-generators/FetchableQueryGenerator.js.flow +189 -0
  196. package/transforms/query-generators/NodeQueryGenerator.js.flow +219 -0
  197. package/transforms/query-generators/QueryQueryGenerator.js.flow +57 -0
  198. package/transforms/query-generators/ViewerQueryGenerator.js.flow +97 -0
  199. package/transforms/query-generators/index.js.flow +90 -0
  200. package/transforms/query-generators/utils.js.flow +76 -0
  201. package/util/CodeMarker.js.flow +79 -0
  202. package/util/DefaultHandleKey.js.flow +17 -0
  203. package/util/RelayCompilerCache.js.flow +88 -0
  204. package/util/Rollout.js.flow +39 -0
  205. package/util/TimeReporter.js.flow +79 -0
  206. package/util/areEqualOSS.js.flow +123 -0
  207. package/util/argumentContainsVariables.js.flow +38 -0
  208. package/util/dedupeJSONStringify.js.flow +152 -0
  209. package/util/generateAbstractTypeRefinementKey.js.flow +29 -0
  210. package/util/getDefinitionNodeHash.js.flow +25 -0
  211. package/util/getModuleName.js.flow +39 -0
  212. package/util/joinArgumentDefinitions.js.flow +105 -0
  213. package/util/md5.js.flow +22 -0
  214. package/util/murmurHash.js.flow +94 -0
  215. package/util/nullthrowsOSS.js.flow +25 -0
  216. package/util/orList.js.flow +37 -0
  217. package/util/partitionArray.js.flow +37 -0
@@ -0,0 +1,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,137 @@
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
+ // $FlowFixMe Discovered when typing babel/parser
50
+ const ast = babylon.parse(text, BABYLON_OPTIONS);
51
+
52
+ const visitors = {
53
+ TaggedTemplateExpression: node => {
54
+ if (isGraphQLTag(node.tag)) {
55
+ result.push({
56
+ keyName: null,
57
+ template: node.quasi.quasis[0].value.raw,
58
+ sourceLocationOffset: getSourceLocationOffset(node.quasi),
59
+ });
60
+ }
61
+ },
62
+ };
63
+ visit(ast, visitors);
64
+ return result;
65
+ }
66
+
67
+ const IGNORED_KEYS = {
68
+ comments: true,
69
+ end: true,
70
+ leadingComments: true,
71
+ loc: true,
72
+ name: true,
73
+ start: true,
74
+ trailingComments: true,
75
+ type: true,
76
+ };
77
+
78
+ function isGraphQLTag(tag): boolean {
79
+ return tag.type === 'Identifier' && tag.name === 'graphql';
80
+ }
81
+
82
+ function getTemplateNode(quasi) {
83
+ const quasis = quasi.quasis;
84
+ invariant(
85
+ quasis && quasis.length === 1,
86
+ 'FindGraphQLTags: Substitutions are not allowed in graphql tags.',
87
+ );
88
+ return quasis[0];
89
+ }
90
+
91
+ function getSourceLocationOffset(quasi) {
92
+ const loc = getTemplateNode(quasi).loc.start;
93
+ return {
94
+ line: loc.line,
95
+ column: loc.column + 1, // babylon is 0-indexed, graphql expects 1-indexed
96
+ };
97
+ }
98
+
99
+ function invariant(condition, msg, ...args) {
100
+ if (!condition) {
101
+ throw new Error(util.format(msg, ...args));
102
+ }
103
+ }
104
+
105
+ function visit(node, visitors) {
106
+ // $FlowFixMe Discovered when typing babel
107
+ const fn = visitors[node.type];
108
+ if (fn != null) {
109
+ fn(node);
110
+ return;
111
+ }
112
+ traverse(node, visitors);
113
+ }
114
+
115
+ function traverse(node, visitors) {
116
+ for (const key in node) {
117
+ if (IGNORED_KEYS[key]) {
118
+ continue;
119
+ }
120
+ const prop = node[key];
121
+ if (prop && typeof prop === 'object' && typeof prop.type === 'string') {
122
+ visit(prop, visitors);
123
+ } else if (Array.isArray(prop)) {
124
+ prop.forEach(item => {
125
+ if (item && typeof item === 'object' && typeof item.type === 'string') {
126
+ visit(item, visitors);
127
+ }
128
+ });
129
+ }
130
+ }
131
+ }
132
+
133
+ module.exports = {
134
+ find: (Profiler.instrument(find, 'FindGraphQLTags.find'): (
135
+ text: string,
136
+ ) => $ReadOnlyArray<GraphQLTag>),
137
+ };
@@ -0,0 +1,176 @@
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 = BabelNode_DEPRECATED;
19
+
20
+ /**
21
+ * type NAME = any;
22
+ */
23
+ function anyTypeAlias(name: string): BabelAST {
24
+ return t.typeAlias(t.identifier(name), undefined, t.anyTypeAnnotation());
25
+ }
26
+
27
+ /**
28
+ * {|
29
+ * PROPS
30
+ * |}
31
+ */
32
+ function exactObjectTypeAnnotation(props: Array<BabelAST>): $FlowFixMe {
33
+ const typeAnnotation = t.objectTypeAnnotation(props);
34
+ typeAnnotation.exact = true;
35
+ return typeAnnotation;
36
+ }
37
+
38
+ /**
39
+ * {
40
+ * PROPS
41
+ * ...
42
+ * }
43
+ */
44
+ function inexactObjectTypeAnnotation(props: Array<BabelAST>): $FlowFixMe {
45
+ const typeAnnotation = t.objectTypeAnnotation(props);
46
+ typeAnnotation.inexact = true;
47
+ return typeAnnotation;
48
+ }
49
+
50
+ /**
51
+ * export type NAME = TYPE
52
+ */
53
+ function exportType(name: string, type: BabelAST): $FlowFixMe {
54
+ return t.exportNamedDeclaration(
55
+ t.typeAlias(t.identifier(name), undefined, type),
56
+ [],
57
+ undefined,
58
+ );
59
+ }
60
+
61
+ /**
62
+ * export type {A, B, C}
63
+ */
64
+ function exportTypes(names: $ReadOnlyArray<string>): $FlowFixMe {
65
+ const res = t.exportNamedDeclaration(
66
+ undefined,
67
+ names.map(name =>
68
+ t.exportSpecifier(t.identifier(name), t.identifier(name)),
69
+ ),
70
+
71
+ undefined,
72
+ );
73
+ res.exportKind = 'type';
74
+ return res;
75
+ }
76
+
77
+ /**
78
+ * declare export type NAME = VALUE
79
+ */
80
+ function declareExportOpaqueType(name: string, value: string): $FlowFixMe {
81
+ return t.declareExportDeclaration(
82
+ t.declareOpaqueType(
83
+ t.identifier(name),
84
+ undefined,
85
+ t.genericTypeAnnotation(t.identifier(value)),
86
+ ),
87
+ );
88
+ }
89
+
90
+ /**
91
+ * import type {NAMES[0], NAMES[1], ...} from 'MODULE';
92
+ */
93
+ function importTypes(
94
+ names: $ReadOnlyArray<string>,
95
+ module: string,
96
+ ): $FlowFixMe {
97
+ const importDeclaration = t.importDeclaration(
98
+ names.map(name =>
99
+ t.importSpecifier(t.identifier(name), t.identifier(name)),
100
+ ),
101
+ t.stringLiteral(module),
102
+ );
103
+ importDeclaration.importKind = 'type';
104
+ return importDeclaration;
105
+ }
106
+
107
+ /**
108
+ * Create an intersection type if needed.
109
+ *
110
+ * TYPES[0] & TYPES[1] & ...
111
+ */
112
+ function intersectionTypeAnnotation(types: Array<BabelAST>): BabelAST {
113
+ invariant(
114
+ types.length > 0,
115
+ 'RelayFlowBabelFactories: cannot create an intersection of 0 types',
116
+ );
117
+ return types.length === 1 ? types[0] : t.intersectionTypeAnnotation(types);
118
+ }
119
+
120
+ function lineComments(
121
+ ...lines: $ReadOnlyArray<string>
122
+ ): $ReadOnlyArray<$FlowFixMe> {
123
+ return lines.map(line => ({type: 'CommentLine', value: ' ' + line}));
124
+ }
125
+
126
+ /**
127
+ * $ReadOnlyArray<TYPE>
128
+ */
129
+ function readOnlyArrayOfType(thing: BabelAST): $FlowFixMe {
130
+ return t.genericTypeAnnotation(
131
+ t.identifier('$ReadOnlyArray'),
132
+ t.typeParameterInstantiation([thing]),
133
+ );
134
+ }
135
+
136
+ /**
137
+ * +KEY: VALUE
138
+ */
139
+ function readOnlyObjectTypeProperty(key: string, value: BabelAST): $FlowFixMe {
140
+ const prop = t.objectTypeProperty(t.identifier(key), value);
141
+ prop.variance = t.variance('plus');
142
+ return prop;
143
+ }
144
+
145
+ function stringLiteralTypeAnnotation(value: string): $FlowFixMe {
146
+ return t.stringLiteralTypeAnnotation(value);
147
+ }
148
+
149
+ /**
150
+ * Create a union type if needed.
151
+ *
152
+ * TYPES[0] | TYPES[1] | ...
153
+ */
154
+ function unionTypeAnnotation(types: Array<BabelAST>): BabelAST {
155
+ invariant(
156
+ types.length > 0,
157
+ 'RelayFlowBabelFactories: cannot create a union of 0 types',
158
+ );
159
+ return types.length === 1 ? types[0] : t.unionTypeAnnotation(types);
160
+ }
161
+
162
+ module.exports = {
163
+ anyTypeAlias,
164
+ declareExportOpaqueType,
165
+ exactObjectTypeAnnotation,
166
+ inexactObjectTypeAnnotation,
167
+ exportType,
168
+ exportTypes,
169
+ importTypes,
170
+ intersectionTypeAnnotation,
171
+ lineComments,
172
+ readOnlyArrayOfType,
173
+ readOnlyObjectTypeProperty,
174
+ stringLiteralTypeAnnotation,
175
+ unionTypeAnnotation,
176
+ };