relay-compiler 7.0.0 → 9.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (242) hide show
  1. package/bin/RelayCompilerBin.js.flow +169 -0
  2. package/bin/RelayCompilerMain.js.flow +508 -0
  3. package/bin/__fixtures__/plugin-module.js.flow +17 -0
  4. package/bin/relay-compiler +8554 -8142
  5. package/codegen/CodegenDirectory.js.flow +375 -0
  6. package/codegen/CodegenRunner.js.flow +431 -0
  7. package/codegen/CodegenTypes.js.flow +28 -0
  8. package/codegen/CodegenWatcher.js.flow +254 -0
  9. package/codegen/NormalizationCodeGenerator.js.flow +499 -0
  10. package/codegen/ReaderCodeGenerator.js.flow +453 -0
  11. package/codegen/RelayCodeGenerator.js.flow +76 -0
  12. package/codegen/RelayFileWriter.js.flow +366 -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/writeRelayGeneratedFile.js.flow +194 -0
  17. package/core/ASTCache.js.flow +73 -0
  18. package/core/ASTConvert.js.flow +233 -0
  19. package/core/CompilerContext.js.flow +190 -0
  20. package/core/CompilerError.js.flow +250 -0
  21. package/core/DotGraphQLParser.js.flow +39 -0
  22. package/core/GraphQLCompilerProfiler.js.flow +341 -0
  23. package/core/GraphQLDerivedFromMetadata.js.flow +48 -0
  24. package/core/GraphQLWatchmanClient.js.flow +111 -0
  25. package/core/IR.js.flow +329 -0
  26. package/core/IRPrinter.js.flow +488 -0
  27. package/core/IRTransformer.js.flow +377 -0
  28. package/core/IRValidator.js.flow +260 -0
  29. package/core/IRVisitor.js.flow +150 -0
  30. package/core/JSModuleParser.js.flow +24 -0
  31. package/core/RelayCompilerScope.js.flow +199 -0
  32. package/core/RelayFindGraphQLTags.js.flow +119 -0
  33. package/core/RelayGraphQLEnumsGenerator.js.flow +55 -0
  34. package/core/RelayIRTransforms.js.flow +130 -0
  35. package/core/RelayParser.js.flow +1759 -0
  36. package/core/RelaySourceModuleParser.js.flow +135 -0
  37. package/core/Schema.js.flow +1985 -0
  38. package/core/SchemaUtils.js.flow +109 -0
  39. package/core/filterContextForNode.js.flow +50 -0
  40. package/core/getFieldDefinition.js.flow +156 -0
  41. package/core/getIdentifierForArgumentValue.js.flow +49 -0
  42. package/core/getIdentifierForSelection.js.flow +69 -0
  43. package/core/getLiteralArgumentValues.js.flow +32 -0
  44. package/core/getNormalizationOperationName.js.flow +19 -0
  45. package/core/inferRootArgumentDefinitions.js.flow +323 -0
  46. package/index.js +1 -1
  47. package/index.js.flow +202 -0
  48. package/language/RelayLanguagePluginInterface.js.flow +283 -0
  49. package/language/javascript/FindGraphQLTags.js.flow +233 -0
  50. package/language/javascript/RelayFlowBabelFactories.js.flow +180 -0
  51. package/language/javascript/RelayFlowGenerator.js.flow +1040 -0
  52. package/language/javascript/RelayFlowTypeTransformers.js.flow +184 -0
  53. package/language/javascript/RelayLanguagePluginJavaScript.js.flow +34 -0
  54. package/language/javascript/formatGeneratedModule.js.flow +65 -0
  55. package/lib/bin/RelayCompilerBin.js +25 -7
  56. package/lib/bin/RelayCompilerMain.js +134 -125
  57. package/lib/bin/__fixtures__/plugin-module.js +1 -0
  58. package/lib/codegen/CodegenDirectory.js +14 -8
  59. package/lib/codegen/CodegenRunner.js +35 -75
  60. package/lib/codegen/CodegenTypes.js +1 -0
  61. package/lib/codegen/CodegenWatcher.js +14 -21
  62. package/lib/codegen/NormalizationCodeGenerator.js +80 -127
  63. package/lib/codegen/ReaderCodeGenerator.js +85 -111
  64. package/lib/codegen/RelayCodeGenerator.js +9 -6
  65. package/lib/codegen/RelayFileWriter.js +22 -41
  66. package/lib/codegen/SourceControl.js +1 -0
  67. package/lib/codegen/compileRelayArtifacts.js +18 -31
  68. package/lib/codegen/createPrintRequireModuleDependency.js +1 -0
  69. package/lib/codegen/writeRelayGeneratedFile.js +62 -90
  70. package/lib/core/ASTCache.js +4 -4
  71. package/lib/core/ASTConvert.js +1 -0
  72. package/lib/core/{GraphQLCompilerContext.js → CompilerContext.js} +10 -11
  73. package/lib/core/{RelayCompilerError.js → CompilerError.js} +29 -55
  74. package/lib/core/DotGraphQLParser.js +1 -0
  75. package/lib/core/GraphQLCompilerProfiler.js +9 -12
  76. package/lib/core/GraphQLDerivedFromMetadata.js +1 -0
  77. package/lib/core/GraphQLWatchmanClient.js +5 -12
  78. package/lib/core/{GraphQLIR.js → IR.js} +1 -0
  79. package/lib/core/{GraphQLIRPrinter.js → IRPrinter.js} +39 -17
  80. package/lib/core/{GraphQLIRTransformer.js → IRTransformer.js} +21 -16
  81. package/lib/core/{GraphQLIRValidator.js → IRValidator.js} +18 -10
  82. package/lib/core/{GraphQLIRVisitor.js → IRVisitor.js} +1 -2
  83. package/lib/core/JSModuleParser.js +18 -0
  84. package/lib/core/RelayCompilerScope.js +6 -5
  85. package/lib/core/RelayFindGraphQLTags.js +1 -0
  86. package/lib/core/RelayGraphQLEnumsGenerator.js +26 -12
  87. package/lib/core/RelayIRTransforms.js +12 -9
  88. package/lib/core/RelayParser.js +113 -75
  89. package/lib/core/RelaySourceModuleParser.js +4 -3
  90. package/lib/core/Schema.js +808 -317
  91. package/lib/core/SchemaUtils.js +1 -0
  92. package/lib/core/filterContextForNode.js +5 -4
  93. package/lib/core/getFieldDefinition.js +14 -16
  94. package/lib/core/getIdentifierForArgumentValue.js +18 -0
  95. package/lib/core/getIdentifierForSelection.js +4 -5
  96. package/lib/core/getLiteralArgumentValues.js +1 -0
  97. package/lib/core/getNormalizationOperationName.js +1 -0
  98. package/lib/core/inferRootArgumentDefinitions.js +79 -99
  99. package/lib/index.js +69 -19
  100. package/lib/language/RelayLanguagePluginInterface.js +1 -0
  101. package/lib/language/javascript/FindGraphQLTags.js +1 -0
  102. package/lib/language/javascript/RelayFlowBabelFactories.js +15 -0
  103. package/lib/language/javascript/RelayFlowGenerator.js +94 -173
  104. package/lib/language/javascript/RelayFlowTypeTransformers.js +2 -3
  105. package/lib/language/javascript/RelayLanguagePluginJavaScript.js +7 -4
  106. package/lib/language/javascript/formatGeneratedModule.js +14 -4
  107. package/lib/reporters/ConsoleReporter.js +2 -3
  108. package/lib/reporters/MultiReporter.js +2 -3
  109. package/lib/reporters/Reporter.js +1 -0
  110. package/lib/runner/Artifacts.js +327 -0
  111. package/lib/runner/BufferedFilesystem.js +265 -0
  112. package/lib/runner/GraphQLASTNodeGroup.js +260 -0
  113. package/lib/runner/GraphQLASTUtils.js +23 -0
  114. package/lib/runner/GraphQLNodeMap.js +85 -0
  115. package/lib/runner/Sources.js +266 -0
  116. package/lib/runner/StrictMap.js +136 -0
  117. package/lib/runner/compileArtifacts.js +39 -0
  118. package/lib/runner/extractAST.js +77 -0
  119. package/lib/runner/getChangedNodeNames.js +84 -0
  120. package/lib/runner/getSchemaInstance.js +30 -0
  121. package/lib/runner/types.js +12 -0
  122. package/lib/transforms/ApplyFragmentArgumentTransform.js +49 -55
  123. package/lib/transforms/ClientExtensionsTransform.js +11 -17
  124. package/lib/transforms/ConnectionTransform.js +35 -28
  125. package/lib/transforms/DeferStreamTransform.js +26 -74
  126. package/lib/transforms/DisallowIdAsAlias.js +5 -4
  127. package/lib/transforms/DisallowTypenameOnRoot.js +55 -0
  128. package/lib/transforms/FieldHandleTransform.js +8 -3
  129. package/lib/transforms/FilterDirectivesTransform.js +4 -3
  130. package/lib/transforms/FlattenTransform.js +23 -47
  131. package/lib/transforms/GenerateIDFieldTransform.js +9 -4
  132. package/lib/transforms/GenerateTypeNameTransform.js +8 -3
  133. package/lib/transforms/InlineDataFragmentTransform.js +11 -6
  134. package/lib/transforms/InlineFragmentsTransform.js +3 -2
  135. package/lib/transforms/MaskTransform.js +20 -19
  136. package/lib/transforms/MatchTransform.js +113 -34
  137. package/lib/transforms/RefetchableFragmentTransform.js +25 -41
  138. package/lib/transforms/RelayDirectiveTransform.js +13 -4
  139. package/lib/transforms/SkipClientExtensionsTransform.js +11 -2
  140. package/lib/transforms/SkipHandleFieldTransform.js +8 -3
  141. package/lib/transforms/SkipRedundantNodesTransform.js +9 -6
  142. package/lib/transforms/SkipSplitOperationTransform.js +32 -0
  143. package/lib/transforms/SkipUnreachableNodeTransform.js +12 -12
  144. package/lib/transforms/SkipUnusedVariablesTransform.js +19 -17
  145. package/lib/transforms/SplitModuleImportTransform.js +4 -3
  146. package/lib/transforms/TestOperationTransform.js +9 -6
  147. package/lib/transforms/TransformUtils.js +1 -0
  148. package/lib/transforms/ValidateGlobalVariablesTransform.js +20 -31
  149. package/lib/transforms/ValidateRequiredArgumentsTransform.js +17 -20
  150. package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +20 -33
  151. package/lib/transforms/ValidateUnusedVariablesTransform.js +20 -31
  152. package/lib/transforms/query-generators/FetchableQueryGenerator.js +161 -0
  153. package/lib/transforms/query-generators/NodeQueryGenerator.js +9 -3
  154. package/lib/transforms/query-generators/QueryQueryGenerator.js +2 -0
  155. package/lib/transforms/query-generators/ViewerQueryGenerator.js +6 -3
  156. package/lib/transforms/query-generators/index.js +25 -7
  157. package/lib/transforms/query-generators/utils.js +13 -15
  158. package/lib/util/CodeMarker.js +1 -0
  159. package/lib/util/DefaultHandleKey.js +1 -0
  160. package/lib/util/RelayCompilerCache.js +2 -3
  161. package/lib/util/Rollout.js +1 -0
  162. package/lib/util/TimeReporter.js +83 -0
  163. package/lib/util/areEqualOSS.js +1 -0
  164. package/lib/util/dedupeJSONStringify.js +16 -12
  165. package/lib/util/getDefinitionNodeHash.js +22 -0
  166. package/lib/util/getModuleName.js +4 -5
  167. package/lib/util/joinArgumentDefinitions.js +2 -1
  168. package/lib/util/md5.js +17 -0
  169. package/lib/util/murmurHash.js +1 -0
  170. package/lib/util/nullthrowsOSS.js +1 -0
  171. package/lib/util/orList.js +2 -1
  172. package/lib/util/partitionArray.js +1 -0
  173. package/package.json +4 -4
  174. package/relay-compiler.js +4 -4
  175. package/relay-compiler.min.js +4 -4
  176. package/reporters/ConsoleReporter.js.flow +81 -0
  177. package/reporters/MultiReporter.js.flow +43 -0
  178. package/reporters/Reporter.js.flow +19 -0
  179. package/runner/Artifacts.js.flow +219 -0
  180. package/runner/BufferedFilesystem.js.flow +194 -0
  181. package/runner/GraphQLASTNodeGroup.js.flow +176 -0
  182. package/runner/GraphQLASTUtils.js.flow +26 -0
  183. package/runner/GraphQLNodeMap.js.flow +55 -0
  184. package/runner/Sources.js.flow +218 -0
  185. package/runner/StrictMap.js.flow +96 -0
  186. package/runner/compileArtifacts.js.flow +76 -0
  187. package/runner/extractAST.js.flow +100 -0
  188. package/runner/getChangedNodeNames.js.flow +48 -0
  189. package/runner/getSchemaInstance.js.flow +36 -0
  190. package/runner/types.js.flow +37 -0
  191. package/transforms/ApplyFragmentArgumentTransform.js.flow +474 -0
  192. package/transforms/ClientExtensionsTransform.js.flow +220 -0
  193. package/transforms/ConnectionTransform.js.flow +869 -0
  194. package/transforms/DeferStreamTransform.js.flow +258 -0
  195. package/transforms/DisallowIdAsAlias.js.flow +47 -0
  196. package/transforms/DisallowTypenameOnRoot.js.flow +45 -0
  197. package/transforms/FieldHandleTransform.js.flow +80 -0
  198. package/transforms/FilterDirectivesTransform.js.flow +45 -0
  199. package/transforms/FlattenTransform.js.flow +456 -0
  200. package/transforms/GenerateIDFieldTransform.js.flow +134 -0
  201. package/transforms/GenerateTypeNameTransform.js.flow +81 -0
  202. package/transforms/InlineDataFragmentTransform.js.flow +124 -0
  203. package/transforms/InlineFragmentsTransform.js.flow +71 -0
  204. package/transforms/MaskTransform.js.flow +126 -0
  205. package/transforms/MatchTransform.js.flow +583 -0
  206. package/transforms/RefetchableFragmentTransform.js.flow +272 -0
  207. package/transforms/RelayDirectiveTransform.js.flow +99 -0
  208. package/transforms/SkipClientExtensionsTransform.js.flow +54 -0
  209. package/transforms/SkipHandleFieldTransform.js.flow +44 -0
  210. package/transforms/SkipRedundantNodesTransform.js.flow +253 -0
  211. package/transforms/SkipSplitOperationTransform.js.flow +37 -0
  212. package/transforms/SkipUnreachableNodeTransform.js.flow +149 -0
  213. package/transforms/SkipUnusedVariablesTransform.js.flow +59 -0
  214. package/transforms/SplitModuleImportTransform.js.flow +98 -0
  215. package/transforms/TestOperationTransform.js.flow +138 -0
  216. package/transforms/TransformUtils.js.flow +26 -0
  217. package/transforms/ValidateGlobalVariablesTransform.js.flow +81 -0
  218. package/transforms/ValidateRequiredArgumentsTransform.js.flow +127 -0
  219. package/transforms/ValidateServerOnlyDirectivesTransform.js.flow +112 -0
  220. package/transforms/ValidateUnusedVariablesTransform.js.flow +89 -0
  221. package/transforms/query-generators/FetchableQueryGenerator.js.flow +190 -0
  222. package/transforms/query-generators/NodeQueryGenerator.js.flow +206 -0
  223. package/transforms/query-generators/QueryQueryGenerator.js.flow +57 -0
  224. package/transforms/query-generators/ViewerQueryGenerator.js.flow +97 -0
  225. package/transforms/query-generators/index.js.flow +90 -0
  226. package/transforms/query-generators/utils.js.flow +72 -0
  227. package/util/CodeMarker.js.flow +79 -0
  228. package/util/DefaultHandleKey.js.flow +17 -0
  229. package/util/RelayCompilerCache.js.flow +88 -0
  230. package/util/Rollout.js.flow +39 -0
  231. package/util/TimeReporter.js.flow +79 -0
  232. package/util/areEqualOSS.js.flow +123 -0
  233. package/util/dedupeJSONStringify.js.flow +152 -0
  234. package/util/getDefinitionNodeHash.js.flow +25 -0
  235. package/util/getModuleName.js.flow +39 -0
  236. package/util/joinArgumentDefinitions.js.flow +99 -0
  237. package/util/md5.js.flow +22 -0
  238. package/util/murmurHash.js.flow +94 -0
  239. package/util/nullthrowsOSS.js.flow +25 -0
  240. package/util/orList.js.flow +37 -0
  241. package/util/partitionArray.js.flow +37 -0
  242. package/lib/transforms/ConnectionFieldTransform.js +0 -275
@@ -0,0 +1,194 @@
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 CodeMarker = require('../util/CodeMarker');
16
+
17
+ const createPrintRequireModuleDependency = require('./createPrintRequireModuleDependency');
18
+ const dedupeJSONStringify = require('../util/dedupeJSONStringify');
19
+ const invariant = require('invariant');
20
+ const md5 = require('../util/md5');
21
+
22
+ const {RelayConcreteNode} = require('relay-runtime');
23
+
24
+ import type {GeneratedDefinition} from '../core/IR';
25
+ import type {Schema} from '../core/Schema';
26
+ import type {
27
+ FormatModule,
28
+ PluginInterface,
29
+ } from '../language/RelayLanguagePluginInterface';
30
+ import type CodegenDirectory from './CodegenDirectory';
31
+ import type {GeneratedNode, RequestParameters} from 'relay-runtime';
32
+
33
+ function getConcreteType(node: GeneratedNode): string {
34
+ switch (node.kind) {
35
+ case RelayConcreteNode.FRAGMENT:
36
+ return 'ReaderFragment';
37
+ case RelayConcreteNode.REQUEST:
38
+ return 'ConcreteRequest';
39
+ case RelayConcreteNode.SPLIT_OPERATION:
40
+ return 'NormalizationSplitOperation';
41
+ case RelayConcreteNode.INLINE_DATA_FRAGMENT:
42
+ return 'ReaderInlineDataFragment';
43
+ default:
44
+ (node: empty);
45
+ invariant(false, 'Unexpected GeneratedNode kind: `%s`.', node.kind);
46
+ }
47
+ }
48
+
49
+ function writeRelayGeneratedFile(
50
+ schema: Schema,
51
+ codegenDir: CodegenDirectory,
52
+ definition: GeneratedDefinition,
53
+ _generatedNode: GeneratedNode,
54
+ formatModule: FormatModule,
55
+ typeText: string,
56
+ _persistQuery: ?(text: string) => Promise<string>,
57
+ sourceHash: string,
58
+ extension: string,
59
+ printModuleDependency: (
60
+ moduleName: string,
61
+ ) => string = createPrintRequireModuleDependency(extension),
62
+ shouldRepersist: boolean,
63
+ writeQueryParameters: (
64
+ dir: CodegenDirectory,
65
+ filename: string,
66
+ moduleName: string,
67
+ params: RequestParameters,
68
+ ) => void,
69
+ languagePlugin: ?PluginInterface,
70
+ ): Promise<?GeneratedNode> {
71
+ let generatedNode = _generatedNode;
72
+ // Copy to const so Flow can refine.
73
+ const persistQuery = _persistQuery;
74
+ const operationName =
75
+ generatedNode.kind === 'Request'
76
+ ? generatedNode.params.name
77
+ : generatedNode.name;
78
+ const moduleName = languagePlugin?.getModuleName
79
+ ? languagePlugin.getModuleName(operationName)
80
+ : operationName + '.graphql';
81
+
82
+ const filename = moduleName + '.' + extension;
83
+ const queryParametersFilename =
84
+ generatedNode.kind === 'Request'
85
+ ? `${generatedNode.params.name}$Parameters.${extension}`
86
+ : null;
87
+
88
+ const typeName = getConcreteType(generatedNode);
89
+
90
+ let docText;
91
+ if (generatedNode.kind === RelayConcreteNode.REQUEST) {
92
+ docText = generatedNode.params.text;
93
+ }
94
+
95
+ // Use `Promise.resolve` to work around a Babel 7.8/7.9 issue.
96
+ return Promise.resolve().then(async () => {
97
+ let hash = null;
98
+ if (generatedNode.kind === RelayConcreteNode.REQUEST && persistQuery) {
99
+ const {text} = generatedNode.params;
100
+ invariant(
101
+ text != null,
102
+ 'writeRelayGeneratedFile: Expected `text` in order to persist query',
103
+ );
104
+
105
+ hash = md5(text);
106
+
107
+ let id = null;
108
+ if (!shouldRepersist) {
109
+ // Unless we `shouldRepersist` the query, check if the @relayHash matches
110
+ // the operation text of the current text and re-use the persisted
111
+ // operation id.
112
+ const oldContent = codegenDir.read(filename);
113
+ const oldHash = extractHash(oldContent);
114
+ const oldRequestID = extractRelayRequestID(oldContent);
115
+
116
+ if (hash === oldHash && oldRequestID != null) {
117
+ id = oldRequestID;
118
+ }
119
+ }
120
+ if (id == null) {
121
+ id = await persistQuery(text);
122
+ }
123
+
124
+ generatedNode = {
125
+ ...generatedNode,
126
+ params: {
127
+ id,
128
+ metadata: generatedNode.params.metadata,
129
+ name: generatedNode.params.name,
130
+ operationKind: generatedNode.params.operationKind,
131
+ text: null,
132
+ },
133
+ };
134
+ }
135
+
136
+ const moduleText = formatModule({
137
+ moduleName,
138
+ documentType: typeName,
139
+ definition,
140
+ kind: generatedNode.kind,
141
+ docText,
142
+ typeText,
143
+ hash: hash != null ? `@relayHash ${hash}` : null,
144
+ concreteText: CodeMarker.postProcess(
145
+ dedupeJSONStringify(generatedNode),
146
+ printModuleDependency,
147
+ ),
148
+ sourceHash,
149
+ node: generatedNode,
150
+ schema,
151
+ });
152
+ codegenDir.writeFile(filename, moduleText, shouldRepersist);
153
+ if (
154
+ writeQueryParameters &&
155
+ queryParametersFilename != null &&
156
+ generatedNode.kind === RelayConcreteNode.REQUEST &&
157
+ generatedNode.params.operationKind === 'query'
158
+ ) {
159
+ writeQueryParameters(
160
+ codegenDir,
161
+ queryParametersFilename,
162
+ moduleName,
163
+ generatedNode.params,
164
+ );
165
+ }
166
+ return generatedNode;
167
+ });
168
+ }
169
+
170
+ function extractHash(text: ?string): ?string {
171
+ if (text == null || text.length === 0) {
172
+ return null;
173
+ }
174
+ if (/<<<<<|>>>>>/.test(text)) {
175
+ // looks like a merge conflict
176
+ return null;
177
+ }
178
+ const match = text.match(/@relayHash (\w{32})\b/m);
179
+ return match && match[1];
180
+ }
181
+
182
+ function extractRelayRequestID(text: ?string): ?string {
183
+ if (text == null || text.length === 0) {
184
+ return null;
185
+ }
186
+ if (/<<<<<|>>>>>/.test(text)) {
187
+ // looks like a merge conflict
188
+ return null;
189
+ }
190
+ const match = text.match(/@relayRequestID (.+)/);
191
+ return match ? match[1] : null;
192
+ }
193
+
194
+ module.exports = writeRelayGeneratedFile;
@@ -0,0 +1,73 @@
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 Profiler = require('./GraphQLCompilerProfiler');
16
+
17
+ // $FlowFixMe: Immutable is not typed
18
+ const {Map: ImmutableMap} = require('immutable');
19
+
20
+ import type {File} from '../codegen/CodegenTypes';
21
+ import type {DocumentNode} from 'graphql';
22
+
23
+ type ParseFn = (baseDir: string, file: File) => ?DocumentNode;
24
+
25
+ class ASTCache {
26
+ _documents: Map<string, DocumentNode>;
27
+
28
+ _baseDir: string;
29
+ _parse: ParseFn;
30
+
31
+ constructor(config: {baseDir: string, parse: ParseFn, ...}) {
32
+ this._documents = new Map();
33
+ this._baseDir = config.baseDir;
34
+ this._parse = Profiler.instrument(config.parse, 'ASTCache.parseFn');
35
+ }
36
+
37
+ // Short-term: we don't do subscriptions/delta updates, instead always use all definitions
38
+ documents(): ImmutableMap<string, DocumentNode> {
39
+ return ImmutableMap(this._documents);
40
+ }
41
+
42
+ // parse should return the set of changes
43
+ parseFiles(files: Set<File>): ImmutableMap<string, DocumentNode> {
44
+ let documents = ImmutableMap();
45
+
46
+ files.forEach(file => {
47
+ if (!file.exists) {
48
+ this._documents.delete(file.relPath);
49
+ return;
50
+ }
51
+
52
+ const doc = (() => {
53
+ try {
54
+ return this._parse(this._baseDir, file);
55
+ } catch (error) {
56
+ throw new Error(`Parse error: ${error} in "${file.relPath}"`);
57
+ }
58
+ })();
59
+
60
+ if (!doc) {
61
+ this._documents.delete(file.relPath);
62
+ return;
63
+ }
64
+
65
+ documents = documents.set(file.relPath, doc);
66
+ this._documents.set(file.relPath, doc);
67
+ });
68
+
69
+ return documents;
70
+ }
71
+ }
72
+
73
+ module.exports = ASTCache;
@@ -0,0 +1,233 @@
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 Profiler = require('./GraphQLCompilerProfiler');
16
+
17
+ const {
18
+ isExecutableDefinitionAST,
19
+ isSchemaDefinitionAST,
20
+ } = require('./SchemaUtils');
21
+ const {extendSchema, parse, print, visit} = require('graphql');
22
+
23
+ import type {Fragment, Root} from './IR';
24
+ import type {Schema} from './Schema';
25
+ import type {
26
+ DefinitionNode,
27
+ DocumentNode,
28
+ FragmentDefinitionNode,
29
+ FragmentSpreadNode,
30
+ GraphQLSchema,
31
+ OperationDefinitionNode,
32
+ TypeSystemDefinitionNode,
33
+ TypeSystemExtensionNode,
34
+ } from 'graphql';
35
+
36
+ type ASTDefinitionNode = FragmentDefinitionNode | OperationDefinitionNode;
37
+ type TransformFn = (
38
+ schema: Schema,
39
+ definitions: $ReadOnlyArray<ASTDefinitionNode>,
40
+ ) => $ReadOnlyArray<Root | Fragment>;
41
+
42
+ function convertASTDocuments(
43
+ schema: Schema,
44
+ documents: $ReadOnlyArray<DocumentNode>,
45
+ transform: TransformFn,
46
+ ): $ReadOnlyArray<Fragment | Root> {
47
+ return Profiler.run('ASTConvert.convertASTDocuments', () => {
48
+ const definitions = definitionsFromDocuments(documents);
49
+
50
+ const astDefinitions: Array<ASTDefinitionNode> = [];
51
+ documents.forEach(doc => {
52
+ doc.definitions.forEach(definition => {
53
+ if (isExecutableDefinitionAST(definition)) {
54
+ astDefinitions.push(definition);
55
+ }
56
+ });
57
+ });
58
+
59
+ return convertASTDefinitions(schema, definitions, transform);
60
+ });
61
+ }
62
+
63
+ function convertASTDocumentsWithBase(
64
+ schema: Schema,
65
+ baseDocuments: $ReadOnlyArray<DocumentNode>,
66
+ documents: $ReadOnlyArray<DocumentNode>,
67
+ transform: TransformFn,
68
+ ): $ReadOnlyArray<Fragment | Root> {
69
+ return Profiler.run('ASTConvert.convertASTDocumentsWithBase', () => {
70
+ const baseDefinitions = definitionsFromDocuments(baseDocuments);
71
+ const definitions = definitionsFromDocuments(documents);
72
+
73
+ const requiredDefinitions = new Map();
74
+ const baseMap: Map<string, ASTDefinitionNode> = new Map();
75
+ baseDefinitions.forEach(definition => {
76
+ if (isExecutableDefinitionAST(definition)) {
77
+ const definitionName = definition.name && definition.name.value;
78
+ // If there's no name, no reason to put in the map
79
+ if (definitionName != null) {
80
+ if (baseMap.has(definitionName)) {
81
+ throw new Error(`Duplicate definition of '${definitionName}'.`);
82
+ }
83
+ baseMap.set(definitionName, definition);
84
+ }
85
+ }
86
+ });
87
+
88
+ const definitionsToVisit: Array<ASTDefinitionNode> = [];
89
+ definitions.forEach(definition => {
90
+ if (isExecutableDefinitionAST(definition)) {
91
+ definitionsToVisit.push(definition);
92
+ }
93
+ });
94
+ while (definitionsToVisit.length > 0) {
95
+ const definition = definitionsToVisit.pop();
96
+ const name = definition.name && definition.name.value;
97
+ if (name == null) {
98
+ continue;
99
+ }
100
+ if (requiredDefinitions.has(name)) {
101
+ if (requiredDefinitions.get(name) !== definition) {
102
+ throw new Error(`Duplicate definition of '${name}'.`);
103
+ }
104
+ continue;
105
+ }
106
+ requiredDefinitions.set(name, definition);
107
+ visit(definition, {
108
+ FragmentSpread(spread: FragmentSpreadNode) {
109
+ const baseDefinition = baseMap.get(spread.name.value);
110
+ if (baseDefinition) {
111
+ // We only need to add those definitions not already included
112
+ // in definitions
113
+ definitionsToVisit.push(baseDefinition);
114
+ }
115
+ },
116
+ });
117
+ }
118
+
119
+ const definitionsToConvert = [];
120
+ requiredDefinitions.forEach(definition =>
121
+ definitionsToConvert.push(definition),
122
+ );
123
+ return convertASTDefinitions(schema, definitionsToConvert, transform);
124
+ });
125
+ }
126
+
127
+ function convertASTDefinitions(
128
+ schema: Schema,
129
+ definitions: $ReadOnlyArray<DefinitionNode>,
130
+ transform: TransformFn,
131
+ ): $ReadOnlyArray<Fragment | Root> {
132
+ const operationDefinitions: Array<ASTDefinitionNode> = [];
133
+ definitions.forEach(definition => {
134
+ if (isExecutableDefinitionAST(definition)) {
135
+ operationDefinitions.push(definition);
136
+ }
137
+ });
138
+ return transform(schema, operationDefinitions);
139
+ }
140
+
141
+ function definitionsFromDocuments(
142
+ documents: $ReadOnlyArray<DocumentNode>,
143
+ ): $ReadOnlyArray<DefinitionNode> {
144
+ const definitions = [];
145
+ documents.forEach(doc => {
146
+ doc.definitions.forEach(definition => definitions.push(definition));
147
+ });
148
+ return definitions;
149
+ }
150
+
151
+ /**
152
+ * Extends a GraphQLSchema with a list of schema extensions in string form.
153
+ */
154
+ function transformASTSchema(
155
+ schema: GraphQLSchema,
156
+ schemaExtensions: $ReadOnlyArray<string>,
157
+ ): GraphQLSchema {
158
+ return Profiler.run('ASTConvert.transformASTSchema', () => {
159
+ if (schemaExtensions.length === 0) {
160
+ return schema;
161
+ }
162
+ const extension = schemaExtensions.join('\n');
163
+ return cachedExtend(schema, extension, () =>
164
+ extendSchema(schema, parse(extension)),
165
+ );
166
+ });
167
+ }
168
+
169
+ /**
170
+ * Extends a GraphQLSchema with a list of schema extensions in AST form.
171
+ */
172
+ function extendASTSchema(
173
+ baseSchema: GraphQLSchema,
174
+ documents: $ReadOnlyArray<DocumentNode>,
175
+ ): GraphQLSchema {
176
+ return Profiler.run('ASTConvert.extendASTSchema', () => {
177
+ const schemaExtensions: Array<
178
+ TypeSystemDefinitionNode | TypeSystemExtensionNode,
179
+ > = [];
180
+ documents.forEach(doc => {
181
+ doc.definitions.forEach(definition => {
182
+ if (isSchemaDefinitionAST(definition)) {
183
+ schemaExtensions.push(definition);
184
+ }
185
+ });
186
+ });
187
+ if (schemaExtensions.length === 0) {
188
+ return baseSchema;
189
+ }
190
+ const key = schemaExtensions.map(print).join('\n');
191
+ return cachedExtend(baseSchema, key, () =>
192
+ extendSchema(
193
+ baseSchema,
194
+ {
195
+ kind: 'Document',
196
+ definitions: schemaExtensions,
197
+ },
198
+ // TODO T24511737 figure out if this is dangerous
199
+ {assumeValid: true},
200
+ ),
201
+ );
202
+ });
203
+ }
204
+
205
+ const extendedSchemas: Map<
206
+ GraphQLSchema,
207
+ {[key: string]: GraphQLSchema, ...},
208
+ > = new Map();
209
+
210
+ function cachedExtend(
211
+ schema: GraphQLSchema,
212
+ key: string,
213
+ compute: () => GraphQLSchema,
214
+ ): GraphQLSchema {
215
+ let cache = extendedSchemas.get(schema);
216
+ if (!cache) {
217
+ cache = {};
218
+ extendedSchemas.set(schema, cache);
219
+ }
220
+ let extendedSchema = cache[key];
221
+ if (!extendedSchema) {
222
+ extendedSchema = compute();
223
+ cache[key] = extendedSchema;
224
+ }
225
+ return extendedSchema;
226
+ }
227
+
228
+ module.exports = {
229
+ convertASTDocuments,
230
+ convertASTDocumentsWithBase,
231
+ extendASTSchema,
232
+ transformASTSchema,
233
+ };
@@ -0,0 +1,190 @@
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 Profiler = require('./GraphQLCompilerProfiler');
16
+
17
+ const invariant = require('invariant');
18
+
19
+ const {createUserError} = require('./CompilerError');
20
+ // $FlowFixMe - immutable.js is not flow-typed
21
+ const {OrderedMap: ImmutableOrderedMap} = require('immutable');
22
+
23
+ import type {Reporter} from '../reporters/Reporter';
24
+ import type {Fragment, Location, Root, SplitOperation} from './IR';
25
+ import type {Schema} from './Schema';
26
+
27
+ export type IRTransform = CompilerContext => CompilerContext;
28
+
29
+ export type CompilerContextDocument = Fragment | Root | SplitOperation;
30
+
31
+ /**
32
+ * An immutable representation of a corpus of documents being compiled together.
33
+ * For each document, the context stores the IR and any validation errors.
34
+ */
35
+ class CompilerContext {
36
+ _isMutable: boolean;
37
+ _documents: ImmutableOrderedMap<string, CompilerContextDocument>;
38
+ _withTransform: WeakMap<IRTransform, CompilerContext>;
39
+ +_schema: Schema;
40
+
41
+ constructor(schema: Schema) {
42
+ this._isMutable = false;
43
+ this._documents = new ImmutableOrderedMap();
44
+ this._withTransform = new WeakMap();
45
+ this._schema = schema;
46
+ }
47
+
48
+ /**
49
+ * Returns the documents for the context in the order they were added.
50
+ */
51
+ documents(): Array<CompilerContextDocument> {
52
+ return this._documents.toArray();
53
+ }
54
+
55
+ forEachDocument(fn: CompilerContextDocument => void): void {
56
+ this._documents.forEach(fn);
57
+ }
58
+
59
+ replace(node: CompilerContextDocument): CompilerContext {
60
+ return this._update(
61
+ this._documents.update(node.name, existing => {
62
+ invariant(
63
+ existing,
64
+ 'CompilerContext: Expected to replace existing node %s, but ' +
65
+ 'one was not found in the context.',
66
+ node.name,
67
+ );
68
+ return node;
69
+ }),
70
+ );
71
+ }
72
+
73
+ add(node: CompilerContextDocument): CompilerContext {
74
+ return this._update(
75
+ this._documents.update(node.name, existing => {
76
+ invariant(
77
+ !existing,
78
+ 'CompilerContext: Duplicate document named `%s`. GraphQL ' +
79
+ 'fragments and roots must have unique names.',
80
+ node.name,
81
+ );
82
+ return node;
83
+ }),
84
+ );
85
+ }
86
+
87
+ addAll(nodes: $ReadOnlyArray<CompilerContextDocument>): CompilerContext {
88
+ return this.withMutations(mutable =>
89
+ nodes.reduce((ctx, definition) => ctx.add(definition), mutable),
90
+ );
91
+ }
92
+
93
+ /**
94
+ * Apply a list of compiler transforms and return a new compiler context.
95
+ */
96
+ applyTransforms(
97
+ transforms: $ReadOnlyArray<IRTransform>,
98
+ reporter?: Reporter,
99
+ ): CompilerContext {
100
+ return Profiler.run('applyTransforms', () =>
101
+ transforms.reduce(
102
+ (ctx, transform) => ctx.applyTransform(transform, reporter),
103
+ this,
104
+ ),
105
+ );
106
+ }
107
+
108
+ /**
109
+ * Applies a transform to this context, returning a new context.
110
+ *
111
+ * This is memoized such that applying the same sequence of transforms will
112
+ * not result in duplicated work.
113
+ */
114
+ applyTransform(transform: IRTransform, reporter?: Reporter): CompilerContext {
115
+ let transformed = this._withTransform.get(transform);
116
+ if (!transformed) {
117
+ const start = process.hrtime();
118
+ transformed = Profiler.instrument(transform)(this);
119
+ const delta = process.hrtime(start);
120
+ const deltaMs = Math.round((delta[0] * 1e9 + delta[1]) / 1e6);
121
+ reporter && reporter.reportTime(transform.name, deltaMs);
122
+ this._withTransform.set(transform, transformed);
123
+ }
124
+ return transformed;
125
+ }
126
+
127
+ get(name: string): ?CompilerContextDocument {
128
+ return this._documents.get(name);
129
+ }
130
+
131
+ getFragment(name: string, referencedFrom?: ?Location): Fragment {
132
+ const node = this._documents.get(name);
133
+ if (node == null) {
134
+ throw createUserError(
135
+ `Cannot find fragment '${name}'.`,
136
+ referencedFrom != null ? [referencedFrom] : null,
137
+ );
138
+ } else if (node.kind !== 'Fragment') {
139
+ throw createUserError(
140
+ `Cannot find fragment '${name}', a document with this name exists ` +
141
+ 'but is not a fragment.',
142
+ [node.loc, referencedFrom].filter(Boolean),
143
+ );
144
+ }
145
+ return node;
146
+ }
147
+
148
+ getRoot(name: string): Root {
149
+ const node = this._documents.get(name);
150
+ if (node == null) {
151
+ throw createUserError(`Cannot find root '${name}'.`);
152
+ } else if (node.kind !== 'Root') {
153
+ throw createUserError(
154
+ `Cannot find root '${name}', a document with this name exists but ` +
155
+ 'is not a root.',
156
+ [node.loc],
157
+ );
158
+ }
159
+ return node;
160
+ }
161
+
162
+ remove(name: string): CompilerContext {
163
+ return this._update(this._documents.delete(name));
164
+ }
165
+
166
+ withMutations(fn: CompilerContext => CompilerContext): CompilerContext {
167
+ const mutableCopy = this._update(this._documents.asMutable());
168
+ mutableCopy._isMutable = true;
169
+ const result = fn(mutableCopy);
170
+ result._isMutable = false;
171
+ result._documents = result._documents.asImmutable();
172
+ return this._documents === result._documents ? this : result;
173
+ }
174
+
175
+ _update(
176
+ documents: ImmutableOrderedMap<string, CompilerContextDocument>,
177
+ ): CompilerContext {
178
+ const context = this._isMutable
179
+ ? this
180
+ : new CompilerContext(this.getSchema());
181
+ context._documents = documents;
182
+ return context;
183
+ }
184
+
185
+ getSchema(): Schema {
186
+ return this._schema;
187
+ }
188
+ }
189
+
190
+ module.exports = CompilerContext;