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,223 @@
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 =
93
+ generatedNode.params.text != null ? generatedNode.params.text : null;
94
+ }
95
+
96
+ // Use `Promise.resolve` to work around a Babel 7.8/7.9 issue.
97
+ return Promise.resolve().then(async () => {
98
+ let hash = null;
99
+ if (generatedNode.kind === RelayConcreteNode.REQUEST) {
100
+ invariant(
101
+ docText != null,
102
+ 'writeRelayGeneratedFile: Expected `text` for operations to be set.',
103
+ );
104
+
105
+ const {
106
+ isRefetchableQuery: _ignored,
107
+ derivedFrom: _ignored2,
108
+ ...nextMetadata
109
+ } = generatedNode.params.metadata;
110
+
111
+ let nextRequestParams;
112
+ if (persistQuery != null) {
113
+ hash = md5(docText);
114
+
115
+ let id = null;
116
+ if (!shouldRepersist) {
117
+ // Unless we `shouldRepersist` the query, check if the @relayHash matches
118
+ // the operation text of the current text and re-use the persisted
119
+ // operation id.
120
+ const oldContent = codegenDir.read(filename);
121
+ const oldHash = extractHash(oldContent);
122
+ const oldRequestID = extractRelayRequestID(oldContent);
123
+ if (hash === oldHash && oldRequestID != null) {
124
+ id = oldRequestID;
125
+ }
126
+ }
127
+ if (id == null) {
128
+ id = await persistQuery(docText);
129
+ }
130
+ nextRequestParams = {
131
+ id,
132
+ metadata: nextMetadata,
133
+ name: generatedNode.params.name,
134
+ operationKind: generatedNode.params.operationKind,
135
+ text: null,
136
+ };
137
+ } else {
138
+ nextRequestParams = {
139
+ cacheID: md5(docText),
140
+ id: null,
141
+ metadata: nextMetadata,
142
+ name: generatedNode.params.name,
143
+ operationKind: generatedNode.params.operationKind,
144
+ text: docText,
145
+ };
146
+ }
147
+ generatedNode = {
148
+ ...generatedNode,
149
+ params: nextRequestParams,
150
+ };
151
+ }
152
+
153
+ // Strip metadata only used within the compiler
154
+ if (
155
+ generatedNode.kind === RelayConcreteNode.SPLIT_OPERATION &&
156
+ generatedNode.metadata?.derivedFrom != null
157
+ ) {
158
+ const {derivedFrom: _ignored, ...metadata} = generatedNode.metadata;
159
+ generatedNode = {
160
+ ...generatedNode,
161
+ metadata,
162
+ };
163
+ }
164
+
165
+ const moduleText = formatModule({
166
+ moduleName,
167
+ documentType: typeName,
168
+ definition,
169
+ kind: generatedNode.kind,
170
+ docText,
171
+ typeText,
172
+ hash: hash != null ? `@relayHash ${hash}` : null,
173
+ concreteText: CodeMarker.postProcess(
174
+ dedupeJSONStringify(generatedNode),
175
+ printModuleDependency,
176
+ ),
177
+ sourceHash,
178
+ node: generatedNode,
179
+ schema,
180
+ });
181
+ codegenDir.writeFile(filename, moduleText, shouldRepersist);
182
+ if (
183
+ writeQueryParameters &&
184
+ queryParametersFilename != null &&
185
+ generatedNode.kind === RelayConcreteNode.REQUEST &&
186
+ generatedNode.params.operationKind === 'query'
187
+ ) {
188
+ writeQueryParameters(
189
+ codegenDir,
190
+ queryParametersFilename,
191
+ moduleName,
192
+ generatedNode.params,
193
+ );
194
+ }
195
+ return generatedNode;
196
+ });
197
+ }
198
+
199
+ function extractHash(text: ?string): ?string {
200
+ if (text == null || text.length === 0) {
201
+ return null;
202
+ }
203
+ if (/<<<<<|>>>>>/.test(text)) {
204
+ // looks like a merge conflict
205
+ return null;
206
+ }
207
+ const match = text.match(/@relayHash (\w{32})\b/m);
208
+ return match && match[1];
209
+ }
210
+
211
+ function extractRelayRequestID(text: ?string): ?string {
212
+ if (text == null || text.length === 0) {
213
+ return null;
214
+ }
215
+ if (/<<<<<|>>>>>/.test(text)) {
216
+ // looks like a merge conflict
217
+ return null;
218
+ }
219
+ const match = text.match(/@relayRequestID (.+)/);
220
+ return match ? match[1] : null;
221
+ }
222
+
223
+ 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[untyped-import] : 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[untyped-import] - 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;