webpack 4.15.0 → 4.16.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (271) hide show
  1. package/LICENSE +20 -20
  2. package/README.md +761 -758
  3. package/SECURITY.md +9 -9
  4. package/buildin/amd-define.js +3 -3
  5. package/buildin/amd-options.js +2 -2
  6. package/buildin/global.js +20 -20
  7. package/buildin/harmony-module.js +24 -24
  8. package/buildin/module.js +22 -22
  9. package/buildin/system.js +7 -7
  10. package/hot/dev-server.js +61 -61
  11. package/hot/emitter.js +2 -2
  12. package/hot/log-apply-result.js +44 -44
  13. package/hot/log.js +45 -45
  14. package/hot/only-dev-server.js +105 -105
  15. package/hot/poll.js +40 -37
  16. package/hot/signal.js +62 -62
  17. package/lib/APIPlugin.js +84 -84
  18. package/lib/AmdMainTemplatePlugin.js +87 -87
  19. package/lib/AsyncDependencyToInitialChunkError.js +31 -31
  20. package/lib/AutomaticPrefetchPlugin.js +1 -1
  21. package/lib/BannerPlugin.js +117 -117
  22. package/lib/BasicEvaluatedExpression.js +211 -211
  23. package/lib/CachePlugin.js +102 -102
  24. package/lib/CaseSensitiveModulesWarning.js +67 -67
  25. package/lib/Chunk.js +833 -811
  26. package/lib/ChunkGroup.js +4 -4
  27. package/lib/ChunkRenderError.js +32 -32
  28. package/lib/CommentCompilationWarning.js +2 -2
  29. package/lib/CompatibilityPlugin.js +70 -70
  30. package/lib/Compilation.js +46 -15
  31. package/lib/ConcurrentCompilationError.js +19 -19
  32. package/lib/ConstPlugin.js +258 -258
  33. package/lib/ContextExclusionPlugin.js +28 -17
  34. package/lib/ContextModule.js +844 -739
  35. package/lib/ContextModuleFactory.js +262 -256
  36. package/lib/ContextReplacementPlugin.js +133 -133
  37. package/lib/DefinePlugin.js +49 -0
  38. package/lib/DelegatedModule.js +5 -0
  39. package/lib/DelegatedModuleFactoryPlugin.js +95 -89
  40. package/lib/DelegatedPlugin.js +39 -39
  41. package/lib/DependenciesBlock.js +1 -1
  42. package/lib/Dependency.js +10 -4
  43. package/lib/DllModule.js +60 -54
  44. package/lib/DllModuleFactory.js +29 -29
  45. package/lib/DllPlugin.js +44 -44
  46. package/lib/DllReferencePlugin.js +132 -84
  47. package/lib/EntryModuleNotFoundError.js +21 -21
  48. package/lib/Entrypoint.js +54 -54
  49. package/lib/EnvironmentPlugin.js +72 -65
  50. package/lib/ErrorHelpers.js +60 -60
  51. package/lib/EvalDevToolModulePlugin.js +27 -27
  52. package/lib/EvalSourceMapDevToolModuleTemplatePlugin.js +115 -115
  53. package/lib/EvalSourceMapDevToolPlugin.js +41 -41
  54. package/lib/ExportPropertyMainTemplatePlugin.js +53 -53
  55. package/lib/ExternalModule.js +165 -159
  56. package/lib/ExternalsPlugin.js +23 -23
  57. package/lib/FlagDependencyExportsPlugin.js +146 -146
  58. package/lib/FlagInitialModulesAsUsedPlugin.js +36 -36
  59. package/lib/FunctionModuleTemplatePlugin.js +100 -100
  60. package/lib/Generator.js +60 -52
  61. package/lib/HarmonyLinkingError.js +17 -17
  62. package/lib/HashedModuleIdsPlugin.js +53 -53
  63. package/lib/HotModuleReplacementPlugin.js +411 -413
  64. package/lib/IgnorePlugin.js +90 -90
  65. package/lib/JavascriptGenerator.js +229 -229
  66. package/lib/JavascriptModulesPlugin.js +179 -179
  67. package/lib/JsonGenerator.js +55 -55
  68. package/lib/JsonModulesPlugin.js +30 -30
  69. package/lib/JsonParser.js +27 -27
  70. package/lib/LibManifestPlugin.js +86 -86
  71. package/lib/LibraryTemplatePlugin.js +153 -153
  72. package/lib/LoaderOptionsPlugin.js +53 -53
  73. package/lib/LoaderTargetPlugin.js +24 -24
  74. package/lib/MemoryOutputFileSystem.js +5 -5
  75. package/lib/Module.js +431 -391
  76. package/lib/ModuleBuildError.js +52 -52
  77. package/lib/ModuleDependencyError.js +35 -35
  78. package/lib/ModuleDependencyWarning.js +25 -25
  79. package/lib/ModuleError.js +36 -36
  80. package/lib/ModuleFilenameHelpers.js +178 -178
  81. package/lib/ModuleNotFoundError.js +23 -23
  82. package/lib/ModuleParseError.js +57 -57
  83. package/lib/ModuleTemplate.js +93 -93
  84. package/lib/ModuleWarning.js +36 -36
  85. package/lib/MultiCompiler.js +283 -283
  86. package/lib/MultiModule.js +87 -81
  87. package/lib/MultiModuleFactory.js +23 -23
  88. package/lib/MultiStats.js +92 -92
  89. package/lib/MultiWatching.js +38 -38
  90. package/lib/NamedChunksPlugin.js +29 -29
  91. package/lib/NamedModulesPlugin.js +57 -57
  92. package/lib/NoEmitOnErrorsPlugin.js +20 -20
  93. package/lib/NoModeWarning.js +23 -23
  94. package/lib/NodeStuffPlugin.js +197 -179
  95. package/lib/NormalModule.js +542 -536
  96. package/lib/NormalModuleFactory.js +526 -526
  97. package/lib/NormalModuleReplacementPlugin.js +51 -51
  98. package/lib/NullFactory.js +12 -12
  99. package/lib/OptionsApply.js +10 -10
  100. package/lib/OptionsDefaulter.js +84 -84
  101. package/lib/Parser.js +2202 -2193
  102. package/lib/ParserHelpers.js +103 -103
  103. package/lib/PrefetchPlugin.js +37 -37
  104. package/lib/ProgressPlugin.js +246 -246
  105. package/lib/ProvidePlugin.js +86 -86
  106. package/lib/RawModule.js +56 -56
  107. package/lib/RecordIdsPlugin.js +230 -230
  108. package/lib/RemovedPluginError.js +11 -11
  109. package/lib/RequestShortener.js +83 -83
  110. package/lib/RequireJsStuffPlugin.js +69 -69
  111. package/lib/ResolverFactory.js +64 -64
  112. package/lib/RuntimeTemplate.js +12 -0
  113. package/lib/SetVarMainTemplatePlugin.js +69 -69
  114. package/lib/SingleEntryPlugin.js +6 -1
  115. package/lib/SizeFormatHelpers.js +24 -24
  116. package/lib/SourceMapDevToolModuleOptionsPlugin.js +49 -49
  117. package/lib/SourceMapDevToolPlugin.js +301 -301
  118. package/lib/Stats.js +28 -5
  119. package/lib/TemplatedPathPlugin.js +173 -173
  120. package/lib/UnsupportedFeatureWarning.js +22 -22
  121. package/lib/UseStrictPlugin.js +54 -54
  122. package/lib/WarnCaseSensitiveModulesPlugin.js +37 -37
  123. package/lib/WarnNoModeSetPlugin.js +17 -17
  124. package/lib/WatchIgnorePlugin.js +100 -100
  125. package/lib/Watching.js +194 -194
  126. package/lib/WebpackOptionsApply.js +92 -10
  127. package/lib/WebpackOptionsDefaulter.js +368 -354
  128. package/lib/debug/ProfilingPlugin.js +430 -430
  129. package/lib/dependencies/AMDPlugin.js +250 -250
  130. package/lib/dependencies/AMDRequireArrayDependency.js +49 -49
  131. package/lib/dependencies/AMDRequireContextDependency.js +20 -20
  132. package/lib/dependencies/AMDRequireDependency.js +135 -135
  133. package/lib/dependencies/AMDRequireItemDependency.js +22 -22
  134. package/lib/dependencies/CommonJsPlugin.js +161 -161
  135. package/lib/dependencies/CommonJsRequireContextDependency.js +23 -23
  136. package/lib/dependencies/CommonJsRequireDependency.js +22 -22
  137. package/lib/dependencies/CommonJsRequireDependencyParserPlugin.js +6 -0
  138. package/lib/dependencies/ConstDependency.js +33 -33
  139. package/lib/dependencies/ContextDependency.js +68 -68
  140. package/lib/dependencies/ContextDependencyHelpers.js +142 -142
  141. package/lib/dependencies/ContextDependencyTemplateAsId.js +42 -42
  142. package/lib/dependencies/ContextDependencyTemplateAsRequireCall.js +38 -38
  143. package/lib/dependencies/ContextElementDependency.js +21 -21
  144. package/lib/dependencies/CriticalDependencyWarning.js +20 -20
  145. package/lib/dependencies/DelegatedSourceDependency.js +18 -18
  146. package/lib/dependencies/DllEntryDependency.js +20 -20
  147. package/lib/dependencies/HarmonyAcceptDependency.js +45 -45
  148. package/lib/dependencies/HarmonyCompatibilityDependency.js +31 -31
  149. package/lib/dependencies/HarmonyDetectionParserPlugin.js +92 -92
  150. package/lib/dependencies/HarmonyExportDependencyParserPlugin.js +139 -139
  151. package/lib/dependencies/HarmonyExportHeaderDependency.js +30 -30
  152. package/lib/dependencies/HarmonyExportImportedSpecifierDependency.js +2 -1
  153. package/lib/dependencies/HarmonyImportDependency.js +109 -109
  154. package/lib/dependencies/HarmonyImportDependencyParserPlugin.js +222 -222
  155. package/lib/dependencies/HarmonyImportSideEffectDependency.js +31 -31
  156. package/lib/dependencies/HarmonyImportSpecifierDependency.js +167 -167
  157. package/lib/dependencies/HarmonyInitDependency.js +60 -60
  158. package/lib/dependencies/HarmonyModulesPlugin.js +146 -146
  159. package/lib/dependencies/HarmonyTopLevelThisParserPlugin.js +26 -26
  160. package/lib/dependencies/ImportContextDependency.js +23 -23
  161. package/lib/dependencies/ImportDependenciesBlock.js +18 -18
  162. package/lib/dependencies/ImportDependency.js +34 -34
  163. package/lib/dependencies/ImportEagerDependency.js +32 -32
  164. package/lib/dependencies/ImportParserPlugin.js +263 -263
  165. package/lib/dependencies/ImportPlugin.js +82 -82
  166. package/lib/dependencies/ImportWeakDependency.js +34 -34
  167. package/lib/dependencies/LoaderPlugin.js +18 -1
  168. package/lib/dependencies/LocalModule.js +23 -23
  169. package/lib/dependencies/LocalModulesHelpers.js +52 -52
  170. package/lib/dependencies/ModuleDependencyTemplateAsId.js +17 -17
  171. package/lib/dependencies/ModuleDependencyTemplateAsRequireId.js +17 -17
  172. package/lib/dependencies/ModuleHotAcceptDependency.js +23 -23
  173. package/lib/dependencies/ModuleHotDeclineDependency.js +23 -23
  174. package/lib/dependencies/NullDependency.js +20 -20
  175. package/lib/dependencies/PrefetchDependency.js +18 -18
  176. package/lib/dependencies/RequireContextDependency.js +22 -22
  177. package/lib/dependencies/RequireContextDependencyParserPlugin.js +56 -56
  178. package/lib/dependencies/RequireContextPlugin.js +143 -143
  179. package/lib/dependencies/RequireEnsureDependenciesBlock.js +33 -33
  180. package/lib/dependencies/RequireEnsureDependenciesBlockParserPlugin.js +116 -116
  181. package/lib/dependencies/RequireEnsureDependency.js +58 -58
  182. package/lib/dependencies/RequireEnsureItemDependency.js +21 -21
  183. package/lib/dependencies/RequireEnsurePlugin.js +74 -74
  184. package/lib/dependencies/RequireHeaderDependency.js +26 -26
  185. package/lib/dependencies/RequireIncludeDependencyParserPlugin.js +23 -23
  186. package/lib/dependencies/RequireIncludePlugin.js +61 -61
  187. package/lib/dependencies/RequireResolveContextDependency.js +23 -23
  188. package/lib/dependencies/RequireResolveDependency.js +22 -22
  189. package/lib/dependencies/RequireResolveDependencyParserPlugin.js +85 -85
  190. package/lib/dependencies/RequireResolveHeaderDependency.js +26 -26
  191. package/lib/dependencies/SystemPlugin.js +125 -125
  192. package/lib/dependencies/UnsupportedDependency.js +27 -27
  193. package/lib/dependencies/WebAssemblyExportImportedDependency.js +29 -29
  194. package/lib/dependencies/WebAssemblyImportDependency.js +48 -48
  195. package/lib/dependencies/WebpackMissingModule.js +20 -20
  196. package/lib/formatLocation.js +75 -61
  197. package/lib/node/NodeChunkTemplatePlugin.js +31 -31
  198. package/lib/node/NodeEnvironmentPlugin.js +28 -28
  199. package/lib/node/NodeHotUpdateChunkTemplatePlugin.js +36 -36
  200. package/lib/node/NodeMainTemplate.runtime.js +27 -27
  201. package/lib/node/NodeMainTemplatePlugin.js +323 -323
  202. package/lib/node/NodeOutputFileSystem.js +22 -22
  203. package/lib/node/NodeSourcePlugin.js +144 -144
  204. package/lib/node/NodeTargetPlugin.js +18 -18
  205. package/lib/node/NodeTemplatePlugin.js +31 -31
  206. package/lib/node/NodeWatchFileSystem.js +99 -99
  207. package/lib/node/ReadFileCompileWasmTemplatePlugin.js +61 -61
  208. package/lib/optimize/AggressiveMergingPlugin.js +87 -87
  209. package/lib/optimize/AggressiveSplittingPlugin.js +287 -287
  210. package/lib/optimize/ConcatenatedModule.js +5 -0
  211. package/lib/optimize/EnsureChunkConditionsPlugin.js +70 -70
  212. package/lib/optimize/FlagIncludedChunksPlugin.js +99 -99
  213. package/lib/optimize/LimitChunkCountPlugin.js +66 -66
  214. package/lib/optimize/MergeDuplicateChunksPlugin.js +78 -78
  215. package/lib/optimize/MinChunkSizePlugin.js +77 -77
  216. package/lib/optimize/NaturalChunkOrderPlugin.js +41 -0
  217. package/lib/optimize/OccurrenceChunkOrderPlugin.js +61 -0
  218. package/lib/optimize/OccurrenceModuleOrderPlugin.js +103 -0
  219. package/lib/optimize/OccurrenceOrderPlugin.js +135 -133
  220. package/lib/optimize/RemoveEmptyChunksPlugin.js +42 -42
  221. package/lib/optimize/RemoveParentModulesPlugin.js +127 -127
  222. package/lib/optimize/RuntimeChunkPlugin.js +41 -41
  223. package/lib/optimize/SideEffectsFlagPlugin.js +168 -168
  224. package/lib/optimize/SplitChunksPlugin.js +866 -850
  225. package/lib/performance/AssetsOverSizeLimitWarning.js +30 -30
  226. package/lib/performance/EntrypointsOverSizeLimitWarning.js +30 -30
  227. package/lib/performance/NoAsyncChunksWarning.js +21 -21
  228. package/lib/performance/SizeLimitsPlugin.js +105 -105
  229. package/lib/util/SortableSet.js +1 -0
  230. package/lib/util/StackedSetMap.js +144 -135
  231. package/lib/util/TrackingSet.js +35 -35
  232. package/lib/util/cachedMerge.js +35 -35
  233. package/lib/util/deterministicGrouping.js +251 -251
  234. package/lib/util/identifier.js +103 -103
  235. package/lib/util/objectToMap.js +16 -16
  236. package/lib/validateSchema.js +67 -67
  237. package/lib/wasm/UnsupportedWebAssemblyFeatureError.js +17 -17
  238. package/lib/wasm/WasmFinalizeExportsPlugin.js +1 -1
  239. package/lib/wasm/WebAssemblyGenerator.js +16 -2
  240. package/lib/wasm/WebAssemblyJavascriptGenerator.js +147 -133
  241. package/lib/wasm/WebAssemblyParser.js +174 -174
  242. package/lib/wasm/WebAssemblyUtils.js +59 -59
  243. package/lib/web/FetchCompileWasmTemplatePlugin.js +37 -37
  244. package/lib/web/JsonpExportMainTemplatePlugin.js +47 -47
  245. package/lib/web/JsonpHotUpdateChunkTemplatePlugin.js +39 -39
  246. package/lib/web/JsonpMainTemplate.runtime.js +65 -65
  247. package/lib/web/JsonpMainTemplatePlugin.js +1 -1
  248. package/lib/web/JsonpTemplatePlugin.js +23 -23
  249. package/lib/web/WebEnvironmentPlugin.js +18 -18
  250. package/lib/webpack.js +5 -0
  251. package/lib/webworker/WebWorkerChunkTemplatePlugin.js +35 -35
  252. package/lib/webworker/WebWorkerHotUpdateChunkTemplatePlugin.js +40 -40
  253. package/lib/webworker/WebWorkerMainTemplate.runtime.js +65 -65
  254. package/lib/webworker/WebWorkerMainTemplatePlugin.js +196 -196
  255. package/lib/webworker/WebWorkerTemplatePlugin.js +25 -25
  256. package/package.json +22 -13
  257. package/schemas/WebpackOptions.json +2062 -2036
  258. package/schemas/ajv.absolutePath.js +55 -55
  259. package/schemas/plugins/BannerPlugin.json +96 -96
  260. package/schemas/plugins/DllPlugin.json +32 -32
  261. package/schemas/plugins/DllReferencePlugin.json +99 -99
  262. package/schemas/plugins/HashedModuleIdsPlugin.json +24 -24
  263. package/schemas/plugins/LoaderOptionsPlugin.json +26 -26
  264. package/schemas/plugins/SourceMapDevToolPlugin.json +187 -187
  265. package/schemas/plugins/WatchIgnorePlugin.json +16 -16
  266. package/schemas/plugins/debug/ProfilingPlugin.json +12 -12
  267. package/schemas/plugins/optimize/AggressiveSplittingPlugin.json +22 -22
  268. package/schemas/plugins/optimize/LimitChunkCountPlugin.json +15 -15
  269. package/schemas/plugins/optimize/MinChunkSizePlugin.json +13 -13
  270. package/schemas/plugins/optimize/OccurrenceOrderChunkIdsPlugin.json +10 -0
  271. package/schemas/plugins/optimize/OccurrenceOrderModuleIdsPlugin.json +10 -0
@@ -1,430 +1,430 @@
1
- const fs = require("fs");
2
- const { Tracer } = require("chrome-trace-event");
3
- const validateOptions = require("schema-utils");
4
- const schema = require("../../schemas/plugins/debug/ProfilingPlugin.json");
5
- let inspector = undefined;
6
-
7
- try {
8
- // eslint-disable-next-line node/no-missing-require
9
- inspector = require("inspector");
10
- } catch (e) {
11
- console.log("Unable to CPU profile in < node 8.0");
12
- }
13
-
14
- class Profiler {
15
- constructor(inspector) {
16
- this.session = undefined;
17
- this.inspector = inspector;
18
- }
19
-
20
- hasSession() {
21
- return this.session !== undefined;
22
- }
23
-
24
- startProfiling() {
25
- if (this.inspector === undefined) {
26
- return Promise.resolve();
27
- }
28
-
29
- try {
30
- this.session = new inspector.Session();
31
- this.session.connect();
32
- } catch (_) {
33
- this.session = undefined;
34
- return Promise.resolve();
35
- }
36
-
37
- return Promise.all([
38
- this.sendCommand("Profiler.setSamplingInterval", {
39
- interval: 100
40
- }),
41
- this.sendCommand("Profiler.enable"),
42
- this.sendCommand("Profiler.start")
43
- ]);
44
- }
45
-
46
- sendCommand(method, params) {
47
- if (this.hasSession()) {
48
- return new Promise((res, rej) => {
49
- return this.session.post(method, params, (err, params) => {
50
- if (err !== null) {
51
- rej(err);
52
- } else {
53
- res(params);
54
- }
55
- });
56
- });
57
- } else {
58
- return Promise.resolve();
59
- }
60
- }
61
-
62
- destroy() {
63
- if (this.hasSession()) {
64
- this.session.disconnect();
65
- }
66
-
67
- return Promise.resolve();
68
- }
69
-
70
- stopProfiling() {
71
- return this.sendCommand("Profiler.stop");
72
- }
73
- }
74
-
75
- /**
76
- * @typedef {Object} Trace
77
- * @description an object that wraps Tracer and Profiler with a counter
78
- * @property {Tracer} trace instance of Tracer
79
- * @property {number} counter Counter
80
- * @property {Profiler} profiler instance of Profiler
81
- * @property {Function} end the end function
82
- */
83
-
84
- /**
85
- * @param {string} outputPath The location where to write the log.
86
- * @returns {Trace} The trace object
87
- */
88
- const createTrace = outputPath => {
89
- const trace = new Tracer({
90
- noStream: true
91
- });
92
- const profiler = new Profiler(inspector);
93
- const fsStream = fs.createWriteStream(outputPath);
94
-
95
- let counter = 0;
96
-
97
- trace.pipe(fsStream);
98
- // These are critical events that need to be inserted so that tools like
99
- // chrome dev tools can load the profile.
100
- trace.instantEvent({
101
- name: "TracingStartedInPage",
102
- id: ++counter,
103
- cat: ["disabled-by-default-devtools.timeline"],
104
- args: {
105
- data: {
106
- sessionId: "-1",
107
- page: "0xfff",
108
- frames: [
109
- {
110
- frame: "0xfff",
111
- url: "webpack",
112
- name: ""
113
- }
114
- ]
115
- }
116
- }
117
- });
118
-
119
- trace.instantEvent({
120
- name: "TracingStartedInBrowser",
121
- id: ++counter,
122
- cat: ["disabled-by-default-devtools.timeline"],
123
- args: {
124
- data: {
125
- sessionId: "-1"
126
- }
127
- }
128
- });
129
-
130
- return {
131
- trace,
132
- counter,
133
- profiler,
134
- end: callback => {
135
- // Wait until the write stream finishes.
136
- fsStream.on("finish", () => {
137
- callback();
138
- });
139
- // Tear down the readable trace stream.
140
- trace.destroy();
141
- }
142
- };
143
- };
144
-
145
- const pluginName = "ProfilingPlugin";
146
-
147
- class ProfilingPlugin {
148
- constructor(opts) {
149
- validateOptions(schema, opts || {}, "Profiling plugin");
150
- opts = opts || {};
151
- this.outputPath = opts.outputPath || "events.json";
152
- }
153
-
154
- apply(compiler) {
155
- const tracer = createTrace(this.outputPath);
156
- tracer.profiler.startProfiling();
157
-
158
- // Compiler Hooks
159
- Object.keys(compiler.hooks).forEach(hookName => {
160
- compiler.hooks[hookName].intercept(
161
- makeInterceptorFor("Compiler", tracer)(hookName)
162
- );
163
- });
164
-
165
- Object.keys(compiler.resolverFactory.hooks).forEach(hookName => {
166
- compiler.resolverFactory.hooks[hookName].intercept(
167
- makeInterceptorFor("Resolver", tracer)(hookName)
168
- );
169
- });
170
-
171
- compiler.hooks.compilation.tap(
172
- pluginName,
173
- (compilation, { normalModuleFactory, contextModuleFactory }) => {
174
- interceptAllHooksFor(compilation, tracer, "Compilation");
175
- interceptAllHooksFor(
176
- normalModuleFactory,
177
- tracer,
178
- "Normal Module Factory"
179
- );
180
- interceptAllHooksFor(
181
- contextModuleFactory,
182
- tracer,
183
- "Context Module Factory"
184
- );
185
- interceptAllParserHooks(normalModuleFactory, tracer);
186
- interceptTemplateInstancesFrom(compilation, tracer);
187
- }
188
- );
189
-
190
- // We need to write out the CPU profile when we are all done.
191
- compiler.hooks.done.tapAsync(
192
- {
193
- name: pluginName,
194
- stage: Infinity
195
- },
196
- (stats, callback) => {
197
- tracer.profiler.stopProfiling().then(parsedResults => {
198
- if (parsedResults === undefined) {
199
- tracer.profiler.destroy();
200
- tracer.trace.flush();
201
- tracer.end(callback);
202
- return;
203
- }
204
-
205
- const cpuStartTime = parsedResults.profile.startTime;
206
- const cpuEndTime = parsedResults.profile.endTime;
207
-
208
- tracer.trace.completeEvent({
209
- name: "TaskQueueManager::ProcessTaskFromWorkQueue",
210
- id: ++tracer.counter,
211
- cat: ["toplevel"],
212
- ts: cpuStartTime,
213
- args: {
214
- src_file: "../../ipc/ipc_moji_bootstrap.cc",
215
- src_func: "Accept"
216
- }
217
- });
218
-
219
- tracer.trace.completeEvent({
220
- name: "EvaluateScript",
221
- id: ++tracer.counter,
222
- cat: ["devtools.timeline"],
223
- ts: cpuStartTime,
224
- dur: cpuEndTime - cpuStartTime,
225
- args: {
226
- data: {
227
- url: "webpack",
228
- lineNumber: 1,
229
- columnNumber: 1,
230
- frame: "0xFFF"
231
- }
232
- }
233
- });
234
-
235
- tracer.trace.instantEvent({
236
- name: "CpuProfile",
237
- id: ++tracer.counter,
238
- cat: ["disabled-by-default-devtools.timeline"],
239
- ts: cpuEndTime,
240
- args: {
241
- data: {
242
- cpuProfile: parsedResults.profile
243
- }
244
- }
245
- });
246
-
247
- tracer.profiler.destroy();
248
- tracer.trace.flush();
249
- tracer.end(callback);
250
- });
251
- }
252
- );
253
- }
254
- }
255
-
256
- const interceptTemplateInstancesFrom = (compilation, tracer) => {
257
- const {
258
- mainTemplate,
259
- chunkTemplate,
260
- hotUpdateChunkTemplate,
261
- moduleTemplates
262
- } = compilation;
263
-
264
- const { javascript, webassembly } = moduleTemplates;
265
-
266
- [
267
- {
268
- instance: mainTemplate,
269
- name: "MainTemplate"
270
- },
271
- {
272
- instance: chunkTemplate,
273
- name: "ChunkTemplate"
274
- },
275
- {
276
- instance: hotUpdateChunkTemplate,
277
- name: "HotUpdateChunkTemplate"
278
- },
279
- {
280
- instance: javascript,
281
- name: "JavaScriptModuleTemplate"
282
- },
283
- {
284
- instance: webassembly,
285
- name: "WebAssemblyModuleTemplate"
286
- }
287
- ].forEach(templateObject => {
288
- Object.keys(templateObject.instance.hooks).forEach(hookName => {
289
- templateObject.instance.hooks[hookName].intercept(
290
- makeInterceptorFor(templateObject.name, tracer)(hookName)
291
- );
292
- });
293
- });
294
- };
295
-
296
- const interceptAllHooksFor = (instance, tracer, logLabel) => {
297
- if (Reflect.has(instance, "hooks")) {
298
- Object.keys(instance.hooks).forEach(hookName => {
299
- instance.hooks[hookName].intercept(
300
- makeInterceptorFor(logLabel, tracer)(hookName)
301
- );
302
- });
303
- }
304
- };
305
-
306
- const interceptAllParserHooks = (moduleFactory, tracer) => {
307
- const moduleTypes = [
308
- "javascript/auto",
309
- "javascript/dynamic",
310
- "javascript/esm",
311
- "json",
312
- "webassembly/experimental"
313
- ];
314
-
315
- moduleTypes.forEach(moduleType => {
316
- moduleFactory.hooks.parser
317
- .for(moduleType)
318
- .tap("ProfilingPlugin", (parser, parserOpts) => {
319
- interceptAllHooksFor(parser, tracer, "Parser");
320
- });
321
- });
322
- };
323
-
324
- const makeInterceptorFor = (instance, tracer) => hookName => ({
325
- register: ({ name, type, context, fn }) => {
326
- const newFn = makeNewProfiledTapFn(hookName, tracer, {
327
- name,
328
- type,
329
- fn
330
- });
331
- return {
332
- name,
333
- type,
334
- context,
335
- fn: newFn
336
- };
337
- }
338
- });
339
-
340
- // TODO improve typing
341
- /** @typedef {(...args: TODO[]) => void | Promise<TODO>} PluginFunction */
342
-
343
- /**
344
- * @param {string} hookName Name of the hook to profile.
345
- * @param {Trace} tracer The trace object.
346
- * @param {object} options Options for the profiled fn.
347
- * @param {string} options.name Plugin name
348
- * @param {string} options.type Plugin type (sync | async | promise)
349
- * @param {PluginFunction} options.fn Plugin function
350
- * @returns {PluginFunction} Chainable hooked function.
351
- */
352
- const makeNewProfiledTapFn = (hookName, tracer, { name, type, fn }) => {
353
- const defaultCategory = ["blink.user_timing"];
354
-
355
- switch (type) {
356
- case "promise":
357
- return (...args) => {
358
- const id = ++tracer.counter;
359
- tracer.trace.begin({
360
- name,
361
- id,
362
- cat: defaultCategory
363
- });
364
- const promise = /** @type {Promise<*>} */ (fn(...args));
365
- return promise.then(r => {
366
- tracer.trace.end({
367
- name,
368
- id,
369
- cat: defaultCategory
370
- });
371
- return r;
372
- });
373
- };
374
- case "async":
375
- return (...args) => {
376
- const id = ++tracer.counter;
377
- tracer.trace.begin({
378
- name,
379
- id,
380
- cat: defaultCategory
381
- });
382
- fn(...args, (...r) => {
383
- const callback = args.pop();
384
- tracer.trace.end({
385
- name,
386
- id,
387
- cat: defaultCategory
388
- });
389
- callback(...r);
390
- });
391
- };
392
- case "sync":
393
- return (...args) => {
394
- const id = ++tracer.counter;
395
- // Do not instrument ourself due to the CPU
396
- // profile needing to be the last event in the trace.
397
- if (name === pluginName) {
398
- return fn(...args);
399
- }
400
-
401
- tracer.trace.begin({
402
- name,
403
- id,
404
- cat: defaultCategory
405
- });
406
- let r;
407
- try {
408
- r = fn(...args);
409
- } catch (error) {
410
- tracer.trace.end({
411
- name,
412
- id,
413
- cat: defaultCategory
414
- });
415
- throw error;
416
- }
417
- tracer.trace.end({
418
- name,
419
- id,
420
- cat: defaultCategory
421
- });
422
- return r;
423
- };
424
- default:
425
- break;
426
- }
427
- };
428
-
429
- module.exports = ProfilingPlugin;
430
- module.exports.Profiler = Profiler;
1
+ const fs = require("fs");
2
+ const { Tracer } = require("chrome-trace-event");
3
+ const validateOptions = require("schema-utils");
4
+ const schema = require("../../schemas/plugins/debug/ProfilingPlugin.json");
5
+ let inspector = undefined;
6
+
7
+ try {
8
+ // eslint-disable-next-line node/no-missing-require
9
+ inspector = require("inspector");
10
+ } catch (e) {
11
+ console.log("Unable to CPU profile in < node 8.0");
12
+ }
13
+
14
+ class Profiler {
15
+ constructor(inspector) {
16
+ this.session = undefined;
17
+ this.inspector = inspector;
18
+ }
19
+
20
+ hasSession() {
21
+ return this.session !== undefined;
22
+ }
23
+
24
+ startProfiling() {
25
+ if (this.inspector === undefined) {
26
+ return Promise.resolve();
27
+ }
28
+
29
+ try {
30
+ this.session = new inspector.Session();
31
+ this.session.connect();
32
+ } catch (_) {
33
+ this.session = undefined;
34
+ return Promise.resolve();
35
+ }
36
+
37
+ return Promise.all([
38
+ this.sendCommand("Profiler.setSamplingInterval", {
39
+ interval: 100
40
+ }),
41
+ this.sendCommand("Profiler.enable"),
42
+ this.sendCommand("Profiler.start")
43
+ ]);
44
+ }
45
+
46
+ sendCommand(method, params) {
47
+ if (this.hasSession()) {
48
+ return new Promise((res, rej) => {
49
+ return this.session.post(method, params, (err, params) => {
50
+ if (err !== null) {
51
+ rej(err);
52
+ } else {
53
+ res(params);
54
+ }
55
+ });
56
+ });
57
+ } else {
58
+ return Promise.resolve();
59
+ }
60
+ }
61
+
62
+ destroy() {
63
+ if (this.hasSession()) {
64
+ this.session.disconnect();
65
+ }
66
+
67
+ return Promise.resolve();
68
+ }
69
+
70
+ stopProfiling() {
71
+ return this.sendCommand("Profiler.stop");
72
+ }
73
+ }
74
+
75
+ /**
76
+ * @typedef {Object} Trace
77
+ * @description an object that wraps Tracer and Profiler with a counter
78
+ * @property {Tracer} trace instance of Tracer
79
+ * @property {number} counter Counter
80
+ * @property {Profiler} profiler instance of Profiler
81
+ * @property {Function} end the end function
82
+ */
83
+
84
+ /**
85
+ * @param {string} outputPath The location where to write the log.
86
+ * @returns {Trace} The trace object
87
+ */
88
+ const createTrace = outputPath => {
89
+ const trace = new Tracer({
90
+ noStream: true
91
+ });
92
+ const profiler = new Profiler(inspector);
93
+ const fsStream = fs.createWriteStream(outputPath);
94
+
95
+ let counter = 0;
96
+
97
+ trace.pipe(fsStream);
98
+ // These are critical events that need to be inserted so that tools like
99
+ // chrome dev tools can load the profile.
100
+ trace.instantEvent({
101
+ name: "TracingStartedInPage",
102
+ id: ++counter,
103
+ cat: ["disabled-by-default-devtools.timeline"],
104
+ args: {
105
+ data: {
106
+ sessionId: "-1",
107
+ page: "0xfff",
108
+ frames: [
109
+ {
110
+ frame: "0xfff",
111
+ url: "webpack",
112
+ name: ""
113
+ }
114
+ ]
115
+ }
116
+ }
117
+ });
118
+
119
+ trace.instantEvent({
120
+ name: "TracingStartedInBrowser",
121
+ id: ++counter,
122
+ cat: ["disabled-by-default-devtools.timeline"],
123
+ args: {
124
+ data: {
125
+ sessionId: "-1"
126
+ }
127
+ }
128
+ });
129
+
130
+ return {
131
+ trace,
132
+ counter,
133
+ profiler,
134
+ end: callback => {
135
+ // Wait until the write stream finishes.
136
+ fsStream.on("finish", () => {
137
+ callback();
138
+ });
139
+ // Tear down the readable trace stream.
140
+ trace.push(null);
141
+ }
142
+ };
143
+ };
144
+
145
+ const pluginName = "ProfilingPlugin";
146
+
147
+ class ProfilingPlugin {
148
+ constructor(opts) {
149
+ validateOptions(schema, opts || {}, "Profiling plugin");
150
+ opts = opts || {};
151
+ this.outputPath = opts.outputPath || "events.json";
152
+ }
153
+
154
+ apply(compiler) {
155
+ const tracer = createTrace(this.outputPath);
156
+ tracer.profiler.startProfiling();
157
+
158
+ // Compiler Hooks
159
+ Object.keys(compiler.hooks).forEach(hookName => {
160
+ compiler.hooks[hookName].intercept(
161
+ makeInterceptorFor("Compiler", tracer)(hookName)
162
+ );
163
+ });
164
+
165
+ Object.keys(compiler.resolverFactory.hooks).forEach(hookName => {
166
+ compiler.resolverFactory.hooks[hookName].intercept(
167
+ makeInterceptorFor("Resolver", tracer)(hookName)
168
+ );
169
+ });
170
+
171
+ compiler.hooks.compilation.tap(
172
+ pluginName,
173
+ (compilation, { normalModuleFactory, contextModuleFactory }) => {
174
+ interceptAllHooksFor(compilation, tracer, "Compilation");
175
+ interceptAllHooksFor(
176
+ normalModuleFactory,
177
+ tracer,
178
+ "Normal Module Factory"
179
+ );
180
+ interceptAllHooksFor(
181
+ contextModuleFactory,
182
+ tracer,
183
+ "Context Module Factory"
184
+ );
185
+ interceptAllParserHooks(normalModuleFactory, tracer);
186
+ interceptTemplateInstancesFrom(compilation, tracer);
187
+ }
188
+ );
189
+
190
+ // We need to write out the CPU profile when we are all done.
191
+ compiler.hooks.done.tapAsync(
192
+ {
193
+ name: pluginName,
194
+ stage: Infinity
195
+ },
196
+ (stats, callback) => {
197
+ tracer.profiler.stopProfiling().then(parsedResults => {
198
+ if (parsedResults === undefined) {
199
+ tracer.profiler.destroy();
200
+ tracer.trace.flush();
201
+ tracer.end(callback);
202
+ return;
203
+ }
204
+
205
+ const cpuStartTime = parsedResults.profile.startTime;
206
+ const cpuEndTime = parsedResults.profile.endTime;
207
+
208
+ tracer.trace.completeEvent({
209
+ name: "TaskQueueManager::ProcessTaskFromWorkQueue",
210
+ id: ++tracer.counter,
211
+ cat: ["toplevel"],
212
+ ts: cpuStartTime,
213
+ args: {
214
+ src_file: "../../ipc/ipc_moji_bootstrap.cc",
215
+ src_func: "Accept"
216
+ }
217
+ });
218
+
219
+ tracer.trace.completeEvent({
220
+ name: "EvaluateScript",
221
+ id: ++tracer.counter,
222
+ cat: ["devtools.timeline"],
223
+ ts: cpuStartTime,
224
+ dur: cpuEndTime - cpuStartTime,
225
+ args: {
226
+ data: {
227
+ url: "webpack",
228
+ lineNumber: 1,
229
+ columnNumber: 1,
230
+ frame: "0xFFF"
231
+ }
232
+ }
233
+ });
234
+
235
+ tracer.trace.instantEvent({
236
+ name: "CpuProfile",
237
+ id: ++tracer.counter,
238
+ cat: ["disabled-by-default-devtools.timeline"],
239
+ ts: cpuEndTime,
240
+ args: {
241
+ data: {
242
+ cpuProfile: parsedResults.profile
243
+ }
244
+ }
245
+ });
246
+
247
+ tracer.profiler.destroy();
248
+ tracer.trace.flush();
249
+ tracer.end(callback);
250
+ });
251
+ }
252
+ );
253
+ }
254
+ }
255
+
256
+ const interceptTemplateInstancesFrom = (compilation, tracer) => {
257
+ const {
258
+ mainTemplate,
259
+ chunkTemplate,
260
+ hotUpdateChunkTemplate,
261
+ moduleTemplates
262
+ } = compilation;
263
+
264
+ const { javascript, webassembly } = moduleTemplates;
265
+
266
+ [
267
+ {
268
+ instance: mainTemplate,
269
+ name: "MainTemplate"
270
+ },
271
+ {
272
+ instance: chunkTemplate,
273
+ name: "ChunkTemplate"
274
+ },
275
+ {
276
+ instance: hotUpdateChunkTemplate,
277
+ name: "HotUpdateChunkTemplate"
278
+ },
279
+ {
280
+ instance: javascript,
281
+ name: "JavaScriptModuleTemplate"
282
+ },
283
+ {
284
+ instance: webassembly,
285
+ name: "WebAssemblyModuleTemplate"
286
+ }
287
+ ].forEach(templateObject => {
288
+ Object.keys(templateObject.instance.hooks).forEach(hookName => {
289
+ templateObject.instance.hooks[hookName].intercept(
290
+ makeInterceptorFor(templateObject.name, tracer)(hookName)
291
+ );
292
+ });
293
+ });
294
+ };
295
+
296
+ const interceptAllHooksFor = (instance, tracer, logLabel) => {
297
+ if (Reflect.has(instance, "hooks")) {
298
+ Object.keys(instance.hooks).forEach(hookName => {
299
+ instance.hooks[hookName].intercept(
300
+ makeInterceptorFor(logLabel, tracer)(hookName)
301
+ );
302
+ });
303
+ }
304
+ };
305
+
306
+ const interceptAllParserHooks = (moduleFactory, tracer) => {
307
+ const moduleTypes = [
308
+ "javascript/auto",
309
+ "javascript/dynamic",
310
+ "javascript/esm",
311
+ "json",
312
+ "webassembly/experimental"
313
+ ];
314
+
315
+ moduleTypes.forEach(moduleType => {
316
+ moduleFactory.hooks.parser
317
+ .for(moduleType)
318
+ .tap("ProfilingPlugin", (parser, parserOpts) => {
319
+ interceptAllHooksFor(parser, tracer, "Parser");
320
+ });
321
+ });
322
+ };
323
+
324
+ const makeInterceptorFor = (instance, tracer) => hookName => ({
325
+ register: ({ name, type, context, fn }) => {
326
+ const newFn = makeNewProfiledTapFn(hookName, tracer, {
327
+ name,
328
+ type,
329
+ fn
330
+ });
331
+ return {
332
+ name,
333
+ type,
334
+ context,
335
+ fn: newFn
336
+ };
337
+ }
338
+ });
339
+
340
+ // TODO improve typing
341
+ /** @typedef {(...args: TODO[]) => void | Promise<TODO>} PluginFunction */
342
+
343
+ /**
344
+ * @param {string} hookName Name of the hook to profile.
345
+ * @param {Trace} tracer The trace object.
346
+ * @param {object} options Options for the profiled fn.
347
+ * @param {string} options.name Plugin name
348
+ * @param {string} options.type Plugin type (sync | async | promise)
349
+ * @param {PluginFunction} options.fn Plugin function
350
+ * @returns {PluginFunction} Chainable hooked function.
351
+ */
352
+ const makeNewProfiledTapFn = (hookName, tracer, { name, type, fn }) => {
353
+ const defaultCategory = ["blink.user_timing"];
354
+
355
+ switch (type) {
356
+ case "promise":
357
+ return (...args) => {
358
+ const id = ++tracer.counter;
359
+ tracer.trace.begin({
360
+ name,
361
+ id,
362
+ cat: defaultCategory
363
+ });
364
+ const promise = /** @type {Promise<*>} */ (fn(...args));
365
+ return promise.then(r => {
366
+ tracer.trace.end({
367
+ name,
368
+ id,
369
+ cat: defaultCategory
370
+ });
371
+ return r;
372
+ });
373
+ };
374
+ case "async":
375
+ return (...args) => {
376
+ const id = ++tracer.counter;
377
+ tracer.trace.begin({
378
+ name,
379
+ id,
380
+ cat: defaultCategory
381
+ });
382
+ fn(...args, (...r) => {
383
+ const callback = args.pop();
384
+ tracer.trace.end({
385
+ name,
386
+ id,
387
+ cat: defaultCategory
388
+ });
389
+ callback(...r);
390
+ });
391
+ };
392
+ case "sync":
393
+ return (...args) => {
394
+ const id = ++tracer.counter;
395
+ // Do not instrument ourself due to the CPU
396
+ // profile needing to be the last event in the trace.
397
+ if (name === pluginName) {
398
+ return fn(...args);
399
+ }
400
+
401
+ tracer.trace.begin({
402
+ name,
403
+ id,
404
+ cat: defaultCategory
405
+ });
406
+ let r;
407
+ try {
408
+ r = fn(...args);
409
+ } catch (error) {
410
+ tracer.trace.end({
411
+ name,
412
+ id,
413
+ cat: defaultCategory
414
+ });
415
+ throw error;
416
+ }
417
+ tracer.trace.end({
418
+ name,
419
+ id,
420
+ cat: defaultCategory
421
+ });
422
+ return r;
423
+ };
424
+ default:
425
+ break;
426
+ }
427
+ };
428
+
429
+ module.exports = ProfilingPlugin;
430
+ module.exports.Profiler = Profiler;