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,81 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ *
7
+ * @flow strict
8
+ * @format
9
+ */
10
+
11
+ // flowlint ambiguous-object-type:error
12
+
13
+ 'use strict';
14
+
15
+ const chalk = require('chalk');
16
+
17
+ import type {Reporter} from './Reporter';
18
+
19
+ function getMemoryUsageString() {
20
+ return chalk.blue(
21
+ Math.round(process.memoryUsage().heapUsed / 1024 / 1024) + 'Mb',
22
+ );
23
+ }
24
+
25
+ class ConsoleReporter implements Reporter {
26
+ _verbose: boolean;
27
+ _quiet: boolean;
28
+
29
+ constructor(options: {verbose: boolean, quiet: boolean, ...}) {
30
+ this._verbose = options.verbose;
31
+ this._quiet = options.quiet;
32
+ }
33
+
34
+ reportMessage(message: string): void {
35
+ if (!this._quiet) {
36
+ process.stdout.write(message + '\n');
37
+ }
38
+ }
39
+
40
+ reportTime(name: string, ms: number): void {
41
+ if (this._verbose && !this._quiet) {
42
+ const time =
43
+ ms === 0
44
+ ? chalk.gray(' <1ms')
45
+ : ms < 1000
46
+ ? chalk.blue(leftPad(5, ms + 'ms'))
47
+ : chalk.red(Math.floor(ms / 10) / 100 + 's');
48
+ process.stdout.write(
49
+ ' ' +
50
+ time +
51
+ ' ' +
52
+ chalk.gray(name) +
53
+ ' [' +
54
+ getMemoryUsageString() +
55
+ ']\n',
56
+ );
57
+ }
58
+ }
59
+
60
+ reportError(caughtLocation: string, error: Error): void {
61
+ if (!this._quiet) {
62
+ process.stdout.write(chalk.red('ERROR:\n' + error.message + '\n'));
63
+ if (this._verbose) {
64
+ const frames = error.stack.match(/^ {4}at .*$/gm);
65
+ if (frames) {
66
+ process.stdout.write(
67
+ chalk.gray(
68
+ 'From: ' + caughtLocation + '\n' + frames.join('\n') + '\n',
69
+ ),
70
+ );
71
+ }
72
+ }
73
+ }
74
+ }
75
+ }
76
+
77
+ function leftPad(len, str) {
78
+ return new Array(len - str.length + 1).join(' ') + str;
79
+ }
80
+
81
+ module.exports = ConsoleReporter;
@@ -0,0 +1,43 @@
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
+ import type {Reporter} from './Reporter';
16
+
17
+ class MultiReporter implements Reporter {
18
+ _reporters: $ReadOnlyArray<Reporter>;
19
+
20
+ constructor(...reporters: $ReadOnlyArray<Reporter>) {
21
+ this._reporters = reporters;
22
+ }
23
+
24
+ reportMessage(message: string): void {
25
+ this._reporters.forEach(reporter => {
26
+ reporter.reportMessage(message);
27
+ });
28
+ }
29
+
30
+ reportTime(name: string, ms: number): void {
31
+ this._reporters.forEach(reporter => {
32
+ reporter.reportTime(name, ms);
33
+ });
34
+ }
35
+
36
+ reportError(caughtLocation: string, error: Error): void {
37
+ this._reporters.forEach(reporter => {
38
+ reporter.reportError(caughtLocation, error);
39
+ });
40
+ }
41
+ }
42
+
43
+ module.exports = MultiReporter;
@@ -0,0 +1,19 @@
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
+ export interface Reporter {
16
+ reportMessage(message: string): void;
17
+ reportTime(name: string, ms: number): void;
18
+ reportError(caughtLocation: string, error: Error): void;
19
+ }
@@ -0,0 +1,219 @@
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
+ * @format
8
+ * @flow strict-local
9
+ */
10
+
11
+ // flowlint ambiguous-object-type:error
12
+
13
+ 'use strict';
14
+
15
+ const crypto = require('crypto');
16
+ const nullthrows = require('nullthrows');
17
+
18
+ const {getName} = require('./GraphQLASTUtils');
19
+
20
+ import type CodegenDirectory, {Filesystem} from '../codegen/CodegenDirectory';
21
+ import type {SourceChanges} from './Sources';
22
+ import type {ExecutableDefinitionNode} from 'graphql';
23
+
24
+ type Filename = string;
25
+
26
+ type NodeName = string;
27
+
28
+ type FileSha1Hex = string;
29
+
30
+ type ArtifactFiles = Set<Filename>;
31
+
32
+ type ArtifactsMetadata = Map<Filename, FileSha1Hex>;
33
+
34
+ export type ArtifactMap = {|
35
+ // Map nodeName to the set of artifacts for this node
36
+ +artifacts: Map<NodeName, ArtifactFiles>,
37
+ // Artifacts metadata (for now it's just sha1 of the file)
38
+ +metadata: ArtifactsMetadata,
39
+ |};
40
+
41
+ export type ArtifactState = ArtifactMap;
42
+
43
+ export type SerializedArtifactState = $ReadOnlyArray<
44
+ [string, $ReadOnlyArray<[string, string]>],
45
+ >;
46
+
47
+ function createEmptyState(): ArtifactState {
48
+ return {
49
+ artifacts: new Map(),
50
+ metadata: new Map(),
51
+ };
52
+ }
53
+
54
+ function serializeState(state: ArtifactState): SerializedArtifactState {
55
+ const json = [];
56
+ for (const [name, artifacts] of state.artifacts) {
57
+ json.push([
58
+ name,
59
+ Array.from(artifacts).map(filename => {
60
+ return [filename, state.metadata.get(filename) ?? ''];
61
+ }),
62
+ ]);
63
+ }
64
+ return json;
65
+ }
66
+
67
+ function deserializeState(json: SerializedArtifactState): ArtifactState {
68
+ const metadata = new Map();
69
+ const artifacts = new Map();
70
+ json.forEach(([name, artifactArray]) => {
71
+ const artifactsFiles = new Set();
72
+ artifactArray.forEach(([filename, sha1hex]) => {
73
+ artifactsFiles.add(filename);
74
+ metadata.set(filename, sha1hex);
75
+ });
76
+ artifacts.set(name, artifactsFiles);
77
+ });
78
+ return {
79
+ artifacts,
80
+ metadata,
81
+ };
82
+ }
83
+
84
+ function updateState(
85
+ state: ArtifactState,
86
+ changes: SourceChanges<ExecutableDefinitionNode>,
87
+ generatedArtifacts: ArtifactMap,
88
+ filesystem: Filesystem,
89
+ resolveFullPath: (relativeFilePath: string) => string,
90
+ ): ArtifactState {
91
+ const nextState = {
92
+ artifacts: new Map(state.artifacts),
93
+ metadata: new Map(state.metadata),
94
+ };
95
+ const deletionCandidates = new Set();
96
+
97
+ const addedNames = new Set();
98
+ for (const {ast} of changes.added) {
99
+ addedNames.add(getName(ast));
100
+ }
101
+
102
+ // For every removed AST node, delete the generated artifacts tracked for that
103
+ // node, unless the AST node was also added when the file was moved or the
104
+ // AST changed which shows up as added and removed in changes.
105
+ for (const {ast} of changes.removed) {
106
+ const name = getName(ast);
107
+
108
+ if (addedNames.has(name)) {
109
+ // Update, we deal with that when iterating the added nodes.
110
+ continue;
111
+ }
112
+ const entry = nextState.artifacts.get(name);
113
+ if (entry == null) {
114
+ // No existing artifacts to delete
115
+ continue;
116
+ }
117
+ for (const outdatedFile of entry.keys()) {
118
+ deletionCandidates.add(outdatedFile);
119
+ }
120
+ nextState.artifacts.delete(name);
121
+ }
122
+
123
+ for (const [name, artifacts] of generatedArtifacts.artifacts) {
124
+ const oldEntry = nextState.artifacts.get(name);
125
+ if (oldEntry != null) {
126
+ for (const outdatedFile of oldEntry) {
127
+ if (!artifacts.has(outdatedFile)) {
128
+ deletionCandidates.add(outdatedFile);
129
+ }
130
+ }
131
+ }
132
+ nextState.artifacts.set(name, artifacts);
133
+ for (const filename of artifacts.keys()) {
134
+ nextState.metadata.set(
135
+ filename,
136
+ generatedArtifacts.metadata.get(filename) ?? '',
137
+ );
138
+ }
139
+ }
140
+
141
+ if (deletionCandidates.size === 0) {
142
+ return nextState;
143
+ }
144
+
145
+ const nextGeneratedArtifacts = new Set();
146
+ for (const [, artifact] of eachNameAndArtifact(nextState)) {
147
+ nextGeneratedArtifacts.add(artifact);
148
+ }
149
+ for (const candidate of deletionCandidates) {
150
+ const someoneElseArtifact = nextGeneratedArtifacts.has(candidate);
151
+ if (someoneElseArtifact) {
152
+ // Sometimes, there are artifacts that are generated by multiple files
153
+ // If this candidate is also generated by someone else in
154
+ // artifact map, we just skip it here
155
+ continue;
156
+ }
157
+
158
+ const candidatePath = resolveFullPath(candidate);
159
+ if (filesystem.existsSync(candidatePath)) {
160
+ filesystem.unlinkSync(candidatePath);
161
+ nextState.metadata.delete(candidate);
162
+ }
163
+ }
164
+
165
+ return nextState;
166
+ }
167
+
168
+ function producedFiles(
169
+ dirs: $ReadOnlyArray<{|
170
+ baseDir: string,
171
+ dir: CodegenDirectory,
172
+ |}>,
173
+ artifactsMetadata: ArtifactsMetadata,
174
+ ): ArtifactsMetadata {
175
+ const result = new Map();
176
+ dirs.forEach(({baseDir, dir}) => {
177
+ const {deleted, updated, created, unchanged} = dir.changes;
178
+ if (deleted.length > 0) {
179
+ throw new Error('Did not expect to see a deletion entry here.');
180
+ }
181
+ [...updated, ...created].forEach(filename => {
182
+ const name = dir.getPath(filename).substr(baseDir.length + 1);
183
+ const sha1hex = sha1(nullthrows(dir.read(filename)));
184
+ result.set(name, sha1hex);
185
+ });
186
+ unchanged.forEach(filename => {
187
+ const name = dir.getPath(filename).substr(baseDir.length + 1);
188
+ const sha1hex = artifactsMetadata.get(name);
189
+ result.set(name, sha1hex ?? sha1(nullthrows(dir.read(filename))));
190
+ });
191
+ });
192
+ return result;
193
+ }
194
+
195
+ function* eachNameAndArtifact(
196
+ artifacts: ArtifactState,
197
+ ): Iterator<[string, string]> {
198
+ for (const [name, artifactsForSource] of artifacts.artifacts) {
199
+ for (const artifactFile of artifactsForSource.keys()) {
200
+ yield [name, artifactFile];
201
+ }
202
+ }
203
+ }
204
+
205
+ function sha1(content: string): string {
206
+ return crypto
207
+ .createHash('sha1')
208
+ .update(content)
209
+ .digest('hex');
210
+ }
211
+
212
+ module.exports = {
213
+ createEmptyState,
214
+ serializeState,
215
+ deserializeState,
216
+ updateState,
217
+ producedFiles,
218
+ eachNameAndArtifact,
219
+ };
@@ -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
+ * @emails oncall+relay
8
+ * @flow strict
9
+ * @format
10
+ */
11
+
12
+ // flowlint ambiguous-object-type:error
13
+
14
+ 'use strict';
15
+
16
+ const fs = require('fs');
17
+ const invariant = require('invariant');
18
+
19
+ import type {Filesystem} from '../codegen/CodegenDirectory';
20
+ import type {SourceControl} from '../codegen/SourceControl';
21
+
22
+ /**
23
+ * A filesystem wrapper that buffers file reads and writes until `commit()` is
24
+ * called.
25
+ */
26
+ class BufferedFilesystem implements Filesystem {
27
+ buffer: Map<string, ?string> = new Map();
28
+ committed: boolean = false;
29
+
30
+ _assertNotComitted() {
31
+ invariant(
32
+ !this.committed,
33
+ 'BufferedFilesystem: no operations allowed after commit().',
34
+ );
35
+ }
36
+
37
+ async commit(sourceControl: ?SourceControl) {
38
+ this._assertNotComitted();
39
+ this.committed = true;
40
+
41
+ const removed = [];
42
+ const added = [];
43
+ for (const [path, data] of this.buffer) {
44
+ if (data == null) {
45
+ removed.push(path);
46
+ fs.unlinkSync(path);
47
+ } else {
48
+ const fileExisits = fs.existsSync(path);
49
+ const currentData = fileExisits ? fs.readFileSync(path, 'utf8') : null;
50
+ if (currentData !== data) {
51
+ added.push(path);
52
+ fs.writeFileSync(path, data, 'utf8');
53
+ }
54
+ }
55
+ }
56
+ if (sourceControl) {
57
+ await sourceControl.addRemove(added, removed);
58
+ }
59
+ }
60
+
61
+ hasChanges(): boolean {
62
+ this._assertNotComitted();
63
+ return this.buffer.size > 0;
64
+ }
65
+
66
+ getChangesSummary(): string {
67
+ this._assertNotComitted();
68
+ const added = [];
69
+ const updated = [];
70
+ const removed = [];
71
+ for (const [path, data] of this.buffer) {
72
+ if (data == null) {
73
+ removed.push(path);
74
+ } else {
75
+ if (!fs.existsSync(path)) {
76
+ added.push(path);
77
+ } else {
78
+ updated.push(path);
79
+ }
80
+ }
81
+ }
82
+ return [
83
+ added.length > 0 ? `Added:\n${added.map(formatFilepath).join('')}` : '',
84
+ updated.length > 0
85
+ ? `Updated:\n${updated.map(formatFilepath).join('')}`
86
+ : '',
87
+ removed.length > 0
88
+ ? `Removed:\n${removed.map(formatFilepath).join('')}`
89
+ : '',
90
+ ]
91
+ .filter(Boolean)
92
+ .join('\n');
93
+ }
94
+
95
+ getAddedRemovedFiles(): {|
96
+ +added: $ReadOnlyArray<string>,
97
+ +removed: $ReadOnlyArray<string>,
98
+ |} {
99
+ this._assertNotComitted();
100
+ const added = [];
101
+ const removed = [];
102
+ for (const [path, data] of this.buffer) {
103
+ if (data == null) {
104
+ removed.push(path);
105
+ } else {
106
+ if (!fs.existsSync(path)) {
107
+ added.push(path);
108
+ }
109
+ }
110
+ }
111
+ return {
112
+ added,
113
+ removed,
114
+ };
115
+ }
116
+
117
+ existsSync(path: string): boolean {
118
+ this._assertNotComitted();
119
+ return this.buffer.has(path)
120
+ ? Boolean(this.buffer.get(path))
121
+ : fs.existsSync(path);
122
+ }
123
+
124
+ mkdirSync(path: string): void {
125
+ this._assertNotComitted();
126
+ fs.mkdirSync(path);
127
+ }
128
+
129
+ readdirSync(path: string): Array<string> {
130
+ this._assertNotComitted();
131
+ throw new Error('BufferedFilesystem: readdirSync is not implemented.');
132
+ }
133
+
134
+ readFileSync(path: string, encoding: string): string {
135
+ this._assertNotComitted();
136
+ if (this.buffer.has(path)) {
137
+ const data = this.buffer.get(path);
138
+ invariant(
139
+ data != null,
140
+ 'BufferedFilesystem: trying to read deleted file.',
141
+ );
142
+ return data;
143
+ }
144
+ return fs.readFileSync(path, encoding);
145
+ }
146
+
147
+ statSync(path: string): {isDirectory(): boolean, ...} {
148
+ this._assertNotComitted();
149
+ return fs.statSync(path);
150
+ }
151
+
152
+ unlinkSync(path: string): void {
153
+ this._assertNotComitted();
154
+ this.buffer.set(path, null);
155
+ }
156
+
157
+ writeFileSync(filename: string, data: string, encoding: string): void {
158
+ this._assertNotComitted();
159
+ this.buffer.set(filename, data);
160
+ }
161
+
162
+ changedFilesToJSON(): {|
163
+ +changed: $ReadOnlyArray<{|
164
+ +path: string,
165
+ +data: string,
166
+ |}>,
167
+ +removed: $ReadOnlyArray<{|
168
+ +path: string,
169
+ |}>,
170
+ |} {
171
+ this._assertNotComitted();
172
+ const changed = [];
173
+ const removed = [];
174
+ for (const [path, data] of this.buffer) {
175
+ if (data == null) {
176
+ removed.push({path});
177
+ } else {
178
+ changed.push({path, data});
179
+ }
180
+ }
181
+ return {
182
+ removed,
183
+ changed,
184
+ };
185
+ }
186
+ }
187
+
188
+ function formatFilepath(filepath: string): string {
189
+ const startIndex = filepath.length - 80;
190
+ const prefix = startIndex > 0 ? `\t - ${filepath.substr(0, 8)}...` : '\t - ';
191
+ return prefix + filepath.substr(startIndex, filepath.length) + '\n';
192
+ }
193
+
194
+ module.exports = BufferedFilesystem;