webpack 4.9.1 → 4.10.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.
- package/README.md +755 -755
- package/SECURITY.md +9 -9
- package/buildin/global.js +20 -20
- package/buildin/harmony-module.js +24 -24
- package/buildin/module.js +22 -22
- package/hot/dev-server.js +61 -61
- package/hot/log-apply-result.js +44 -44
- package/hot/log.js +45 -45
- package/hot/only-dev-server.js +105 -105
- package/hot/poll.js +37 -37
- package/hot/signal.js +62 -62
- package/lib/APIPlugin.js +84 -84
- package/lib/AmdMainTemplatePlugin.js +87 -75
- package/lib/AsyncDependenciesBlock.js +66 -66
- package/lib/AsyncDependencyToInitialChunkError.js +31 -21
- package/lib/AutomaticPrefetchPlugin.js +50 -50
- package/lib/BannerPlugin.js +3 -1
- package/lib/BasicEvaluatedExpression.js +211 -208
- package/lib/CachePlugin.js +102 -95
- package/lib/CaseSensitiveModulesWarning.js +71 -53
- package/lib/Chunk.js +750 -722
- package/lib/ChunkGroup.js +13 -5
- package/lib/ChunkRenderError.js +32 -32
- package/lib/CompatibilityPlugin.js +63 -63
- package/lib/Compilation.js +1947 -1905
- package/lib/Compiler.js +508 -496
- package/lib/ConcurrentCompilationError.js +19 -19
- package/lib/ConstPlugin.js +258 -242
- package/lib/ContextExclusionPlugin.js +17 -17
- package/lib/ContextModule.js +749 -710
- package/lib/ContextModuleFactory.js +256 -245
- package/lib/ContextReplacementPlugin.js +133 -126
- package/lib/DefinePlugin.js +206 -197
- package/lib/DelegatedModule.js +101 -101
- package/lib/DelegatedModuleFactoryPlugin.js +89 -89
- package/lib/DelegatedPlugin.js +39 -39
- package/lib/DependenciesBlock.js +89 -87
- package/lib/DependenciesBlockVariable.js +52 -51
- package/lib/Dependency.js +51 -51
- package/lib/DllEntryPlugin.js +51 -51
- package/lib/DllModule.js +54 -54
- package/lib/DllModuleFactory.js +29 -29
- package/lib/DllPlugin.js +44 -42
- package/lib/DllReferencePlugin.js +84 -84
- package/lib/DynamicEntryPlugin.js +73 -71
- package/lib/EntryOptionPlugin.js +33 -33
- package/lib/EnvironmentPlugin.js +65 -65
- package/lib/ErrorHelpers.js +60 -57
- package/lib/EvalDevToolModulePlugin.js +27 -27
- package/lib/EvalDevToolModuleTemplatePlugin.js +61 -61
- package/lib/EvalSourceMapDevToolPlugin.js +41 -40
- package/lib/ExportPropertyMainTemplatePlugin.js +53 -40
- package/lib/ExtendedAPIPlugin.js +84 -84
- package/lib/ExternalModule.js +159 -159
- package/lib/ExternalModuleFactoryPlugin.js +110 -110
- package/lib/ExternalsPlugin.js +23 -23
- package/lib/FlagDependencyExportsPlugin.js +146 -146
- package/lib/FlagDependencyUsagePlugin.js +110 -104
- package/lib/FlagInitialModulesAsUsedPlugin.js +36 -36
- package/lib/FunctionModulePlugin.js +19 -19
- package/lib/FunctionModuleTemplatePlugin.js +100 -98
- package/lib/GraphHelpers.js +64 -64
- package/lib/HarmonyLinkingError.js +18 -18
- package/lib/HashedModuleIdsPlugin.js +53 -53
- package/lib/HotModuleReplacement.runtime.js +7 -3
- package/lib/HotModuleReplacementPlugin.js +413 -406
- package/lib/HotUpdateChunk.js +16 -16
- package/lib/HotUpdateChunkTemplate.js +78 -78
- package/lib/IgnorePlugin.js +71 -71
- package/lib/JavascriptGenerator.js +229 -228
- package/lib/JavascriptModulesPlugin.js +179 -184
- package/lib/JsonGenerator.js +42 -42
- package/lib/JsonModulesPlugin.js +30 -30
- package/lib/JsonParser.js +27 -26
- package/lib/LibManifestPlugin.js +86 -86
- package/lib/LibraryTemplatePlugin.js +153 -119
- package/lib/LoaderOptionsPlugin.js +53 -52
- package/lib/LoaderTargetPlugin.js +24 -24
- package/lib/MainTemplate.js +34 -9
- package/lib/Module.js +381 -377
- package/lib/ModuleBuildError.js +42 -42
- package/lib/ModuleDependencyError.js +35 -25
- package/lib/ModuleDependencyWarning.js +25 -25
- package/lib/ModuleError.js +28 -28
- package/lib/ModuleFilenameHelpers.js +178 -166
- package/lib/ModuleParseError.js +44 -44
- package/lib/ModuleReason.js +40 -40
- package/lib/ModuleTemplate.js +84 -84
- package/lib/ModuleWarning.js +30 -30
- package/lib/MultiCompiler.js +283 -271
- package/lib/MultiEntryPlugin.js +58 -58
- package/lib/MultiModule.js +81 -78
- package/lib/MultiModuleFactory.js +23 -23
- package/lib/MultiStats.js +92 -92
- package/lib/MultiWatching.js +38 -38
- package/lib/NamedChunksPlugin.js +29 -29
- package/lib/NamedModulesPlugin.js +57 -57
- package/lib/NoEmitOnErrorsPlugin.js +20 -20
- package/lib/NoModeWarning.js +23 -23
- package/lib/NodeStuffPlugin.js +179 -178
- package/lib/NormalModule.js +497 -490
- package/lib/NormalModuleFactory.js +501 -483
- package/lib/NormalModuleReplacementPlugin.js +51 -51
- package/lib/OptionsDefaulter.js +84 -80
- package/lib/Parser.js +2164 -2086
- package/lib/ParserHelpers.js +103 -100
- package/lib/PrefetchPlugin.js +37 -37
- package/lib/ProgressPlugin.js +231 -231
- package/lib/ProvidePlugin.js +86 -86
- package/lib/RawModule.js +56 -54
- package/lib/RecordIdsPlugin.js +166 -162
- package/lib/RemovedPluginError.js +13 -13
- package/lib/RequestShortener.js +81 -74
- package/lib/RequireJsStuffPlugin.js +69 -69
- package/lib/ResolverFactory.js +64 -64
- package/lib/RuleSet.js +555 -534
- package/lib/RuntimeTemplate.js +320 -277
- package/lib/SetVarMainTemplatePlugin.js +69 -57
- package/lib/SingleEntryPlugin.js +44 -44
- package/lib/SizeFormatHelpers.js +24 -24
- package/lib/SourceMapDevToolModuleOptionsPlugin.js +49 -49
- package/lib/SourceMapDevToolPlugin.js +301 -300
- package/lib/Stats.js +1408 -1367
- package/lib/Template.js +4 -2
- package/lib/TemplatedPathPlugin.js +173 -170
- package/lib/UmdMainTemplatePlugin.js +304 -264
- package/lib/UseStrictPlugin.js +48 -48
- package/lib/WarnCaseSensitiveModulesPlugin.js +37 -36
- package/lib/WarnNoModeSetPlugin.js +17 -17
- package/lib/WatchIgnorePlugin.js +100 -100
- package/lib/Watching.js +194 -193
- package/lib/WebpackError.js +25 -19
- package/lib/WebpackOptionsApply.js +421 -405
- package/lib/WebpackOptionsDefaulter.js +347 -344
- package/lib/WebpackOptionsValidationError.js +345 -316
- package/lib/compareLocations.js +56 -56
- package/lib/dependencies/AMDDefineDependency.js +137 -137
- package/lib/dependencies/AMDDefineDependencyParserPlugin.js +336 -327
- package/lib/dependencies/AMDPlugin.js +250 -250
- package/lib/dependencies/AMDRequireArrayDependency.js +49 -49
- package/lib/dependencies/AMDRequireContextDependency.js +20 -20
- package/lib/dependencies/AMDRequireDependenciesBlock.js +43 -43
- package/lib/dependencies/AMDRequireDependenciesBlockParserPlugin.js +273 -270
- package/lib/dependencies/AMDRequireDependency.js +135 -135
- package/lib/dependencies/CommonJsPlugin.js +161 -161
- package/lib/dependencies/CommonJsRequireContextDependency.js +23 -23
- package/lib/dependencies/CommonJsRequireDependencyParserPlugin.js +130 -130
- package/lib/dependencies/ConstDependency.js +33 -33
- package/lib/dependencies/ContextDependency.js +68 -68
- package/lib/dependencies/ContextDependencyTemplateAsId.js +42 -42
- package/lib/dependencies/ContextDependencyTemplateAsRequireCall.js +38 -38
- package/lib/dependencies/ContextElementDependency.js +21 -21
- package/lib/dependencies/DelegatedExportsDependency.js +33 -33
- package/lib/dependencies/DependencyReference.js +18 -18
- package/lib/dependencies/HarmonyAcceptDependency.js +45 -45
- package/lib/dependencies/HarmonyAcceptImportDependency.js +23 -23
- package/lib/dependencies/HarmonyCompatibilityDependency.js +31 -31
- package/lib/dependencies/HarmonyDetectionParserPlugin.js +92 -90
- package/lib/dependencies/HarmonyExportDependencyParserPlugin.js +139 -139
- package/lib/dependencies/HarmonyExportExpressionDependency.js +53 -53
- package/lib/dependencies/HarmonyExportHeaderDependency.js +30 -30
- package/lib/dependencies/HarmonyExportImportedSpecifierDependency.js +620 -603
- package/lib/dependencies/HarmonyExportSpecifierDependency.js +54 -54
- package/lib/dependencies/HarmonyImportDependency.js +104 -94
- package/lib/dependencies/HarmonyImportDependencyParserPlugin.js +217 -214
- package/lib/dependencies/HarmonyImportSideEffectDependency.js +31 -31
- package/lib/dependencies/HarmonyImportSpecifierDependency.js +166 -156
- package/lib/dependencies/HarmonyInitDependency.js +60 -60
- package/lib/dependencies/HarmonyModulesPlugin.js +146 -146
- package/lib/dependencies/HarmonyTopLevelThisParserPlugin.js +26 -26
- package/lib/dependencies/ImportContextDependency.js +23 -23
- package/lib/dependencies/ImportDependenciesBlock.js +18 -18
- package/lib/dependencies/ImportDependency.js +34 -34
- package/lib/dependencies/ImportEagerDependency.js +32 -32
- package/lib/dependencies/ImportParserPlugin.js +233 -232
- package/lib/dependencies/ImportPlugin.js +82 -82
- package/lib/dependencies/ImportWeakDependency.js +34 -34
- package/lib/dependencies/JsonExportsDependency.js +26 -26
- package/lib/dependencies/LoaderPlugin.js +98 -93
- package/lib/dependencies/LocalModuleDependency.js +28 -28
- package/lib/dependencies/LocalModulesHelpers.js +52 -45
- package/lib/dependencies/ModuleDependency.js +20 -20
- package/lib/dependencies/ModuleDependencyTemplateAsId.js +17 -17
- package/lib/dependencies/ModuleDependencyTemplateAsRequireId.js +17 -17
- package/lib/dependencies/MultiEntryDependency.js +20 -20
- package/lib/dependencies/NullDependency.js +20 -20
- package/lib/dependencies/RequireContextDependency.js +22 -22
- package/lib/dependencies/RequireContextDependencyParserPlugin.js +56 -56
- package/lib/dependencies/RequireContextPlugin.js +143 -141
- package/lib/dependencies/RequireEnsureDependenciesBlock.js +33 -33
- package/lib/dependencies/RequireEnsureDependenciesBlockParserPlugin.js +116 -112
- package/lib/dependencies/RequireEnsureDependency.js +58 -58
- package/lib/dependencies/RequireEnsurePlugin.js +74 -74
- package/lib/dependencies/RequireHeaderDependency.js +26 -26
- package/lib/dependencies/RequireIncludeDependency.js +39 -39
- package/lib/dependencies/RequireIncludeDependencyParserPlugin.js +23 -23
- package/lib/dependencies/RequireIncludePlugin.js +61 -61
- package/lib/dependencies/RequireResolveContextDependency.js +23 -23
- package/lib/dependencies/RequireResolveDependencyParserPlugin.js +85 -85
- package/lib/dependencies/RequireResolveHeaderDependency.js +26 -26
- package/lib/dependencies/SingleEntryDependency.js +18 -18
- package/lib/dependencies/SystemPlugin.js +125 -125
- package/lib/dependencies/UnsupportedDependency.js +27 -27
- package/lib/dependencies/WebAssemblyImportDependency.js +48 -44
- package/lib/dependencies/WebpackMissingModule.js +20 -20
- package/lib/dependencies/getFunctionExpression.js +52 -52
- package/lib/formatLocation.js +61 -53
- package/lib/node/NodeChunkTemplatePlugin.js +31 -31
- package/lib/node/NodeEnvironmentPlugin.js +28 -28
- package/lib/node/NodeHotUpdateChunkTemplatePlugin.js +36 -36
- package/lib/node/NodeMainTemplate.runtime.js +27 -27
- package/lib/node/NodeMainTemplateAsync.runtime.js +44 -44
- package/lib/node/NodeMainTemplatePlugin.js +323 -320
- package/lib/node/NodeSourcePlugin.js +144 -140
- package/lib/node/NodeTargetPlugin.js +18 -18
- package/lib/node/NodeTemplatePlugin.js +31 -31
- package/lib/node/NodeWatchFileSystem.js +99 -82
- package/lib/node/ReadFileCompileWasmTemplatePlugin.js +52 -52
- package/lib/optimize/AggressiveMergingPlugin.js +87 -87
- package/lib/optimize/AggressiveSplittingPlugin.js +287 -281
- package/lib/optimize/ChunkModuleIdRangePlugin.js +68 -68
- package/lib/optimize/ConcatenatedModule.js +1420 -1413
- package/lib/optimize/EnsureChunkConditionsPlugin.js +70 -70
- package/lib/optimize/FlagIncludedChunksPlugin.js +99 -99
- package/lib/optimize/LimitChunkCountPlugin.js +66 -66
- package/lib/optimize/MergeDuplicateChunksPlugin.js +78 -75
- package/lib/optimize/MinChunkSizePlugin.js +77 -77
- package/lib/optimize/ModuleConcatenationPlugin.js +470 -457
- package/lib/optimize/OccurrenceOrderPlugin.js +133 -126
- package/lib/optimize/RemoveParentModulesPlugin.js +127 -117
- package/lib/optimize/RuntimeChunkPlugin.js +41 -41
- package/lib/optimize/SideEffectsFlagPlugin.js +158 -156
- package/lib/optimize/SplitChunksPlugin.js +709 -696
- package/lib/performance/AssetsOverSizeLimitWarning.js +30 -30
- package/lib/performance/EntrypointsOverSizeLimitWarning.js +31 -31
- package/lib/performance/NoAsyncChunksWarning.js +21 -21
- package/lib/performance/SizeLimitsPlugin.js +105 -105
- package/lib/util/Semaphore.js +41 -41
- package/lib/util/SortableSet.js +5 -2
- package/lib/util/StackedSetMap.js +12 -5
- package/lib/util/TrackingSet.js +35 -35
- package/lib/util/cachedMerge.js +35 -35
- package/lib/util/createHash.js +77 -77
- package/lib/util/identifier.js +76 -76
- package/lib/validateSchema.js +67 -67
- package/lib/wasm/UnsupportedWebAssemblyFeatureError.js +18 -18
- package/lib/wasm/WasmMainTemplatePlugin.js +310 -304
- package/lib/wasm/WebAssemblyGenerator.js +143 -19
- package/lib/wasm/WebAssemblyJavascriptGenerator.js +90 -107
- package/lib/wasm/WebAssemblyModulesPlugin.js +80 -80
- package/lib/wasm/WebAssemblyParser.js +28 -5
- package/lib/wasm/WebAssemblyUtils.js +48 -0
- package/lib/web/FetchCompileWasmTemplatePlugin.js +25 -25
- package/lib/web/JsonpChunkTemplatePlugin.js +47 -47
- package/lib/web/JsonpExportMainTemplatePlugin.js +47 -47
- package/lib/web/JsonpHotUpdateChunkTemplatePlugin.js +39 -39
- package/lib/web/JsonpMainTemplate.runtime.js +65 -64
- package/lib/web/JsonpMainTemplatePlugin.js +576 -574
- package/lib/web/JsonpTemplatePlugin.js +23 -23
- package/lib/webpack.js +183 -182
- package/lib/webpack.web.js +31 -31
- package/lib/webworker/WebWorkerChunkTemplatePlugin.js +35 -35
- package/lib/webworker/WebWorkerHotUpdateChunkTemplatePlugin.js +40 -40
- package/lib/webworker/WebWorkerMainTemplate.runtime.js +65 -64
- package/lib/webworker/WebWorkerMainTemplatePlugin.js +179 -179
- package/lib/webworker/WebWorkerTemplatePlugin.js +25 -25
- package/package.json +9 -8
- package/schemas/WebpackOptions.json +1988 -1988
- package/schemas/ajv.absolutePath.js +55 -55
- package/schemas/plugins/DllPlugin.json +32 -32
- package/schemas/plugins/DllReferencePlugin.json +99 -99
- package/schemas/plugins/HashedModuleIdsPlugin.json +24 -24
- package/schemas/plugins/LoaderOptionsPlugin.json +26 -26
- package/schemas/plugins/SourceMapDevToolPlugin.json +187 -187
- package/schemas/plugins/WatchIgnorePlugin.json +16 -16
- package/schemas/plugins/debug/ProfilingPlugin.json +12 -12
- package/schemas/plugins/optimize/AggressiveSplittingPlugin.json +22 -22
- package/schemas/plugins/optimize/LimitChunkCountPlugin.json +15 -15
- package/schemas/plugins/optimize/MinChunkSizePlugin.json +13 -13
@@ -1,457 +1,470 @@
|
|
1
|
-
/*
|
2
|
-
MIT License http://www.opensource.org/licenses/mit-license.php
|
3
|
-
Author Tobias Koppers @sokra
|
4
|
-
*/
|
5
|
-
"use strict";
|
6
|
-
|
7
|
-
const HarmonyImportDependency = require("../dependencies/HarmonyImportDependency");
|
8
|
-
const ModuleHotAcceptDependency = require("../dependencies/ModuleHotAcceptDependency");
|
9
|
-
const ModuleHotDeclineDependency = require("../dependencies/ModuleHotDeclineDependency");
|
10
|
-
const ConcatenatedModule = require("./ConcatenatedModule");
|
11
|
-
const HarmonyCompatibilityDependency = require("../dependencies/HarmonyCompatibilityDependency");
|
12
|
-
const StackedSetMap = require("../util/StackedSetMap");
|
13
|
-
|
14
|
-
const formatBailoutReason = msg => {
|
15
|
-
return "ModuleConcatenation bailout: " + msg;
|
16
|
-
};
|
17
|
-
|
18
|
-
class ModuleConcatenationPlugin {
|
19
|
-
constructor(options) {
|
20
|
-
if (typeof options !== "object") options = {};
|
21
|
-
this.options = options;
|
22
|
-
}
|
23
|
-
|
24
|
-
apply(compiler) {
|
25
|
-
compiler.hooks.compilation.tap(
|
26
|
-
"ModuleConcatenationPlugin",
|
27
|
-
(compilation, { normalModuleFactory }) => {
|
28
|
-
const handler = (parser, parserOptions) => {
|
29
|
-
parser.hooks.call.for("eval").tap("ModuleConcatenationPlugin", () => {
|
30
|
-
// Because of variable renaming we can't use modules with eval.
|
31
|
-
parser.state.module.buildMeta.moduleConcatenationBailout = "eval()";
|
32
|
-
});
|
33
|
-
};
|
34
|
-
|
35
|
-
normalModuleFactory.hooks.parser
|
36
|
-
.for("javascript/auto")
|
37
|
-
.tap("ModuleConcatenationPlugin", handler);
|
38
|
-
normalModuleFactory.hooks.parser
|
39
|
-
.for("javascript/dynamic")
|
40
|
-
.tap("ModuleConcatenationPlugin", handler);
|
41
|
-
normalModuleFactory.hooks.parser
|
42
|
-
.for("javascript/esm")
|
43
|
-
.tap("ModuleConcatenationPlugin", handler);
|
44
|
-
|
45
|
-
const bailoutReasonMap = new Map();
|
46
|
-
|
47
|
-
const setBailoutReason = (module, reason) => {
|
48
|
-
bailoutReasonMap.set(module, reason);
|
49
|
-
module.optimizationBailout.push(
|
50
|
-
typeof reason === "function"
|
51
|
-
? rs => formatBailoutReason(reason(rs))
|
52
|
-
: formatBailoutReason(reason)
|
53
|
-
);
|
54
|
-
};
|
55
|
-
|
56
|
-
const getBailoutReason = (module, requestShortener) => {
|
57
|
-
const reason = bailoutReasonMap.get(module);
|
58
|
-
if (typeof reason === "function") return reason(requestShortener);
|
59
|
-
return reason;
|
60
|
-
};
|
61
|
-
|
62
|
-
compilation.hooks.optimizeChunkModules.tap(
|
63
|
-
"ModuleConcatenationPlugin",
|
64
|
-
(chunks, modules) => {
|
65
|
-
const relevantModules = [];
|
66
|
-
const possibleInners = new Set();
|
67
|
-
for (const module of modules) {
|
68
|
-
// Only harmony modules are valid for optimization
|
69
|
-
if (
|
70
|
-
!module.buildMeta ||
|
71
|
-
module.buildMeta.exportsType !== "namespace" ||
|
72
|
-
!module.dependencies.some(
|
73
|
-
d => d instanceof HarmonyCompatibilityDependency
|
74
|
-
)
|
75
|
-
) {
|
76
|
-
setBailoutReason(module, "Module is not an ECMAScript module");
|
77
|
-
continue;
|
78
|
-
}
|
79
|
-
|
80
|
-
// Some expressions are not compatible with module concatenation
|
81
|
-
// because they may produce unexpected results. The plugin bails out
|
82
|
-
// if some were detected upfront.
|
83
|
-
if (
|
84
|
-
module.buildMeta &&
|
85
|
-
module.buildMeta.moduleConcatenationBailout
|
86
|
-
) {
|
87
|
-
setBailoutReason(
|
88
|
-
module,
|
89
|
-
`Module uses ${module.buildMeta.moduleConcatenationBailout}`
|
90
|
-
);
|
91
|
-
continue;
|
92
|
-
}
|
93
|
-
|
94
|
-
// Exports must be known (and not dynamic)
|
95
|
-
if (!Array.isArray(module.buildMeta.providedExports)) {
|
96
|
-
setBailoutReason(module, "Module exports are unknown");
|
97
|
-
continue;
|
98
|
-
}
|
99
|
-
|
100
|
-
// Using dependency variables is not possible as this wraps the code in a function
|
101
|
-
if (module.variables.length > 0) {
|
102
|
-
setBailoutReason(
|
103
|
-
module,
|
104
|
-
`Module uses injected variables (${module.variables
|
105
|
-
.map(v => v.name)
|
106
|
-
.join(", ")})`
|
107
|
-
);
|
108
|
-
continue;
|
109
|
-
}
|
110
|
-
|
111
|
-
// Hot Module Replacement need it's own module to work correctly
|
112
|
-
if (
|
113
|
-
module.dependencies.some(
|
114
|
-
dep =>
|
115
|
-
dep instanceof ModuleHotAcceptDependency ||
|
116
|
-
dep instanceof ModuleHotDeclineDependency
|
117
|
-
)
|
118
|
-
) {
|
119
|
-
setBailoutReason(module, "Module uses Hot Module Replacement");
|
120
|
-
continue;
|
121
|
-
}
|
122
|
-
|
123
|
-
relevantModules.push(module);
|
124
|
-
|
125
|
-
// Module must not be the entry points
|
126
|
-
if (module.isEntryModule()) {
|
127
|
-
setBailoutReason(module, "Module is an entry point");
|
128
|
-
continue;
|
129
|
-
}
|
130
|
-
|
131
|
-
// Module must be in any chunk (we don't want to do useless work)
|
132
|
-
if (module.getNumberOfChunks() === 0) {
|
133
|
-
setBailoutReason(module, "Module is not in any chunk");
|
134
|
-
continue;
|
135
|
-
}
|
136
|
-
|
137
|
-
// Module must only be used by Harmony Imports
|
138
|
-
const nonHarmonyReasons = module.reasons.filter(
|
139
|
-
reason =>
|
140
|
-
!reason.dependency ||
|
141
|
-
!(reason.dependency instanceof HarmonyImportDependency)
|
142
|
-
);
|
143
|
-
if (nonHarmonyReasons.length > 0) {
|
144
|
-
const importingModules = new Set(
|
145
|
-
nonHarmonyReasons.map(r => r.module).filter(Boolean)
|
146
|
-
);
|
147
|
-
const importingExplanations = new Set(
|
148
|
-
nonHarmonyReasons.map(r => r.explanation).filter(Boolean)
|
149
|
-
);
|
150
|
-
const importingModuleTypes = new Map(
|
151
|
-
Array.from(importingModules).map(
|
152
|
-
m => /** @type {[string, Set]} */ ([
|
153
|
-
m,
|
154
|
-
new Set(
|
155
|
-
nonHarmonyReasons
|
156
|
-
.filter(r => r.module === m)
|
157
|
-
.map(r => r.dependency.type)
|
158
|
-
.sort()
|
159
|
-
)
|
160
|
-
])
|
161
|
-
)
|
162
|
-
);
|
163
|
-
setBailoutReason(module, requestShortener => {
|
164
|
-
const names = Array.from(importingModules)
|
165
|
-
.map(
|
166
|
-
m =>
|
167
|
-
`${m.readableIdentifier(
|
168
|
-
requestShortener
|
169
|
-
)} (referenced with ${Array.from(
|
170
|
-
importingModuleTypes.get(m)
|
171
|
-
).join(", ")})`
|
172
|
-
)
|
173
|
-
.sort();
|
174
|
-
const explanations = Array.from(importingExplanations).sort();
|
175
|
-
if (names.length > 0 && explanations.length === 0)
|
176
|
-
return `Module is referenced from these modules with unsupported syntax: ${names.join(
|
177
|
-
", "
|
178
|
-
)}`;
|
179
|
-
else if (names.length === 0 && explanations.length > 0)
|
180
|
-
return `Module is referenced by: ${explanations.join(
|
181
|
-
", "
|
182
|
-
)}`;
|
183
|
-
else if (names.length > 0 && explanations.length > 0)
|
184
|
-
return `Module is referenced from these modules with unsupported syntax: ${names.join(
|
185
|
-
", "
|
186
|
-
)} and by: ${explanations.join(", ")}`;
|
187
|
-
else
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
//
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
//
|
238
|
-
|
239
|
-
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
|
284
|
-
|
285
|
-
|
286
|
-
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
|
298
|
-
|
299
|
-
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
304
|
-
|
305
|
-
|
306
|
-
|
307
|
-
|
308
|
-
|
309
|
-
|
310
|
-
|
311
|
-
|
312
|
-
|
313
|
-
|
314
|
-
|
315
|
-
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
|
324
|
-
|
325
|
-
|
326
|
-
|
327
|
-
|
328
|
-
|
329
|
-
)
|
330
|
-
|
331
|
-
//
|
332
|
-
.map(
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
337
|
-
|
338
|
-
|
339
|
-
|
340
|
-
|
341
|
-
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
348
|
-
|
349
|
-
|
350
|
-
|
351
|
-
|
352
|
-
|
353
|
-
|
354
|
-
|
355
|
-
|
356
|
-
|
357
|
-
|
358
|
-
|
359
|
-
|
360
|
-
|
361
|
-
|
362
|
-
|
363
|
-
|
364
|
-
|
365
|
-
|
366
|
-
|
367
|
-
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
|
372
|
-
|
373
|
-
|
374
|
-
|
375
|
-
|
376
|
-
|
377
|
-
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
384
|
-
|
385
|
-
|
386
|
-
|
387
|
-
|
388
|
-
|
389
|
-
|
390
|
-
|
391
|
-
|
392
|
-
|
393
|
-
|
394
|
-
|
395
|
-
|
396
|
-
|
397
|
-
|
398
|
-
|
399
|
-
|
400
|
-
|
401
|
-
|
402
|
-
|
403
|
-
|
404
|
-
|
405
|
-
|
406
|
-
|
407
|
-
|
408
|
-
|
409
|
-
|
410
|
-
|
411
|
-
|
412
|
-
|
413
|
-
|
414
|
-
|
415
|
-
|
416
|
-
|
417
|
-
|
418
|
-
|
419
|
-
|
420
|
-
|
421
|
-
|
422
|
-
|
423
|
-
|
424
|
-
|
425
|
-
|
426
|
-
|
427
|
-
|
428
|
-
|
429
|
-
|
430
|
-
|
431
|
-
|
432
|
-
|
433
|
-
|
434
|
-
|
435
|
-
|
436
|
-
|
437
|
-
|
438
|
-
|
439
|
-
|
440
|
-
|
441
|
-
|
442
|
-
|
443
|
-
|
444
|
-
|
445
|
-
|
446
|
-
|
447
|
-
|
448
|
-
|
449
|
-
|
450
|
-
|
451
|
-
|
452
|
-
|
453
|
-
|
454
|
-
|
455
|
-
|
456
|
-
|
457
|
-
|
1
|
+
/*
|
2
|
+
MIT License http://www.opensource.org/licenses/mit-license.php
|
3
|
+
Author Tobias Koppers @sokra
|
4
|
+
*/
|
5
|
+
"use strict";
|
6
|
+
|
7
|
+
const HarmonyImportDependency = require("../dependencies/HarmonyImportDependency");
|
8
|
+
const ModuleHotAcceptDependency = require("../dependencies/ModuleHotAcceptDependency");
|
9
|
+
const ModuleHotDeclineDependency = require("../dependencies/ModuleHotDeclineDependency");
|
10
|
+
const ConcatenatedModule = require("./ConcatenatedModule");
|
11
|
+
const HarmonyCompatibilityDependency = require("../dependencies/HarmonyCompatibilityDependency");
|
12
|
+
const StackedSetMap = require("../util/StackedSetMap");
|
13
|
+
|
14
|
+
const formatBailoutReason = msg => {
|
15
|
+
return "ModuleConcatenation bailout: " + msg;
|
16
|
+
};
|
17
|
+
|
18
|
+
class ModuleConcatenationPlugin {
|
19
|
+
constructor(options) {
|
20
|
+
if (typeof options !== "object") options = {};
|
21
|
+
this.options = options;
|
22
|
+
}
|
23
|
+
|
24
|
+
apply(compiler) {
|
25
|
+
compiler.hooks.compilation.tap(
|
26
|
+
"ModuleConcatenationPlugin",
|
27
|
+
(compilation, { normalModuleFactory }) => {
|
28
|
+
const handler = (parser, parserOptions) => {
|
29
|
+
parser.hooks.call.for("eval").tap("ModuleConcatenationPlugin", () => {
|
30
|
+
// Because of variable renaming we can't use modules with eval.
|
31
|
+
parser.state.module.buildMeta.moduleConcatenationBailout = "eval()";
|
32
|
+
});
|
33
|
+
};
|
34
|
+
|
35
|
+
normalModuleFactory.hooks.parser
|
36
|
+
.for("javascript/auto")
|
37
|
+
.tap("ModuleConcatenationPlugin", handler);
|
38
|
+
normalModuleFactory.hooks.parser
|
39
|
+
.for("javascript/dynamic")
|
40
|
+
.tap("ModuleConcatenationPlugin", handler);
|
41
|
+
normalModuleFactory.hooks.parser
|
42
|
+
.for("javascript/esm")
|
43
|
+
.tap("ModuleConcatenationPlugin", handler);
|
44
|
+
|
45
|
+
const bailoutReasonMap = new Map();
|
46
|
+
|
47
|
+
const setBailoutReason = (module, reason) => {
|
48
|
+
bailoutReasonMap.set(module, reason);
|
49
|
+
module.optimizationBailout.push(
|
50
|
+
typeof reason === "function"
|
51
|
+
? rs => formatBailoutReason(reason(rs))
|
52
|
+
: formatBailoutReason(reason)
|
53
|
+
);
|
54
|
+
};
|
55
|
+
|
56
|
+
const getBailoutReason = (module, requestShortener) => {
|
57
|
+
const reason = bailoutReasonMap.get(module);
|
58
|
+
if (typeof reason === "function") return reason(requestShortener);
|
59
|
+
return reason;
|
60
|
+
};
|
61
|
+
|
62
|
+
compilation.hooks.optimizeChunkModules.tap(
|
63
|
+
"ModuleConcatenationPlugin",
|
64
|
+
(chunks, modules) => {
|
65
|
+
const relevantModules = [];
|
66
|
+
const possibleInners = new Set();
|
67
|
+
for (const module of modules) {
|
68
|
+
// Only harmony modules are valid for optimization
|
69
|
+
if (
|
70
|
+
!module.buildMeta ||
|
71
|
+
module.buildMeta.exportsType !== "namespace" ||
|
72
|
+
!module.dependencies.some(
|
73
|
+
d => d instanceof HarmonyCompatibilityDependency
|
74
|
+
)
|
75
|
+
) {
|
76
|
+
setBailoutReason(module, "Module is not an ECMAScript module");
|
77
|
+
continue;
|
78
|
+
}
|
79
|
+
|
80
|
+
// Some expressions are not compatible with module concatenation
|
81
|
+
// because they may produce unexpected results. The plugin bails out
|
82
|
+
// if some were detected upfront.
|
83
|
+
if (
|
84
|
+
module.buildMeta &&
|
85
|
+
module.buildMeta.moduleConcatenationBailout
|
86
|
+
) {
|
87
|
+
setBailoutReason(
|
88
|
+
module,
|
89
|
+
`Module uses ${module.buildMeta.moduleConcatenationBailout}`
|
90
|
+
);
|
91
|
+
continue;
|
92
|
+
}
|
93
|
+
|
94
|
+
// Exports must be known (and not dynamic)
|
95
|
+
if (!Array.isArray(module.buildMeta.providedExports)) {
|
96
|
+
setBailoutReason(module, "Module exports are unknown");
|
97
|
+
continue;
|
98
|
+
}
|
99
|
+
|
100
|
+
// Using dependency variables is not possible as this wraps the code in a function
|
101
|
+
if (module.variables.length > 0) {
|
102
|
+
setBailoutReason(
|
103
|
+
module,
|
104
|
+
`Module uses injected variables (${module.variables
|
105
|
+
.map(v => v.name)
|
106
|
+
.join(", ")})`
|
107
|
+
);
|
108
|
+
continue;
|
109
|
+
}
|
110
|
+
|
111
|
+
// Hot Module Replacement need it's own module to work correctly
|
112
|
+
if (
|
113
|
+
module.dependencies.some(
|
114
|
+
dep =>
|
115
|
+
dep instanceof ModuleHotAcceptDependency ||
|
116
|
+
dep instanceof ModuleHotDeclineDependency
|
117
|
+
)
|
118
|
+
) {
|
119
|
+
setBailoutReason(module, "Module uses Hot Module Replacement");
|
120
|
+
continue;
|
121
|
+
}
|
122
|
+
|
123
|
+
relevantModules.push(module);
|
124
|
+
|
125
|
+
// Module must not be the entry points
|
126
|
+
if (module.isEntryModule()) {
|
127
|
+
setBailoutReason(module, "Module is an entry point");
|
128
|
+
continue;
|
129
|
+
}
|
130
|
+
|
131
|
+
// Module must be in any chunk (we don't want to do useless work)
|
132
|
+
if (module.getNumberOfChunks() === 0) {
|
133
|
+
setBailoutReason(module, "Module is not in any chunk");
|
134
|
+
continue;
|
135
|
+
}
|
136
|
+
|
137
|
+
// Module must only be used by Harmony Imports
|
138
|
+
const nonHarmonyReasons = module.reasons.filter(
|
139
|
+
reason =>
|
140
|
+
!reason.dependency ||
|
141
|
+
!(reason.dependency instanceof HarmonyImportDependency)
|
142
|
+
);
|
143
|
+
if (nonHarmonyReasons.length > 0) {
|
144
|
+
const importingModules = new Set(
|
145
|
+
nonHarmonyReasons.map(r => r.module).filter(Boolean)
|
146
|
+
);
|
147
|
+
const importingExplanations = new Set(
|
148
|
+
nonHarmonyReasons.map(r => r.explanation).filter(Boolean)
|
149
|
+
);
|
150
|
+
const importingModuleTypes = new Map(
|
151
|
+
Array.from(importingModules).map(
|
152
|
+
m => /** @type {[string, Set]} */ ([
|
153
|
+
m,
|
154
|
+
new Set(
|
155
|
+
nonHarmonyReasons
|
156
|
+
.filter(r => r.module === m)
|
157
|
+
.map(r => r.dependency.type)
|
158
|
+
.sort()
|
159
|
+
)
|
160
|
+
])
|
161
|
+
)
|
162
|
+
);
|
163
|
+
setBailoutReason(module, requestShortener => {
|
164
|
+
const names = Array.from(importingModules)
|
165
|
+
.map(
|
166
|
+
m =>
|
167
|
+
`${m.readableIdentifier(
|
168
|
+
requestShortener
|
169
|
+
)} (referenced with ${Array.from(
|
170
|
+
importingModuleTypes.get(m)
|
171
|
+
).join(", ")})`
|
172
|
+
)
|
173
|
+
.sort();
|
174
|
+
const explanations = Array.from(importingExplanations).sort();
|
175
|
+
if (names.length > 0 && explanations.length === 0) {
|
176
|
+
return `Module is referenced from these modules with unsupported syntax: ${names.join(
|
177
|
+
", "
|
178
|
+
)}`;
|
179
|
+
} else if (names.length === 0 && explanations.length > 0) {
|
180
|
+
return `Module is referenced by: ${explanations.join(
|
181
|
+
", "
|
182
|
+
)}`;
|
183
|
+
} else if (names.length > 0 && explanations.length > 0) {
|
184
|
+
return `Module is referenced from these modules with unsupported syntax: ${names.join(
|
185
|
+
", "
|
186
|
+
)} and by: ${explanations.join(", ")}`;
|
187
|
+
} else {
|
188
|
+
return "Module is referenced in a unsupported way";
|
189
|
+
}
|
190
|
+
});
|
191
|
+
continue;
|
192
|
+
}
|
193
|
+
|
194
|
+
possibleInners.add(module);
|
195
|
+
}
|
196
|
+
// sort by depth
|
197
|
+
// modules with lower depth are more likely suited as roots
|
198
|
+
// this improves performance, because modules already selected as inner are skipped
|
199
|
+
relevantModules.sort((a, b) => {
|
200
|
+
return a.depth - b.depth;
|
201
|
+
});
|
202
|
+
const concatConfigurations = [];
|
203
|
+
const usedAsInner = new Set();
|
204
|
+
for (const currentRoot of relevantModules) {
|
205
|
+
// when used by another configuration as inner:
|
206
|
+
// the other configuration is better and we can skip this one
|
207
|
+
if (usedAsInner.has(currentRoot)) continue;
|
208
|
+
|
209
|
+
// create a configuration with the root
|
210
|
+
const currentConfiguration = new ConcatConfiguration(currentRoot);
|
211
|
+
|
212
|
+
// cache failures to add modules
|
213
|
+
const failureCache = new Map();
|
214
|
+
|
215
|
+
// try to add all imports
|
216
|
+
for (const imp of this.getImports(currentRoot)) {
|
217
|
+
const problem = this.tryToAdd(
|
218
|
+
currentConfiguration,
|
219
|
+
imp,
|
220
|
+
possibleInners,
|
221
|
+
failureCache
|
222
|
+
);
|
223
|
+
if (problem) {
|
224
|
+
failureCache.set(imp, problem);
|
225
|
+
currentConfiguration.addWarning(imp, problem);
|
226
|
+
}
|
227
|
+
}
|
228
|
+
if (!currentConfiguration.isEmpty()) {
|
229
|
+
concatConfigurations.push(currentConfiguration);
|
230
|
+
for (const module of currentConfiguration.getModules()) {
|
231
|
+
if (module !== currentConfiguration.rootModule) {
|
232
|
+
usedAsInner.add(module);
|
233
|
+
}
|
234
|
+
}
|
235
|
+
}
|
236
|
+
}
|
237
|
+
// HACK: Sort configurations by length and start with the longest one
|
238
|
+
// to get the biggers groups possible. Used modules are marked with usedModules
|
239
|
+
// TODO: Allow to reuse existing configuration while trying to add dependencies.
|
240
|
+
// This would improve performance. O(n^2) -> O(n)
|
241
|
+
concatConfigurations.sort((a, b) => {
|
242
|
+
return b.modules.size - a.modules.size;
|
243
|
+
});
|
244
|
+
const usedModules = new Set();
|
245
|
+
for (const concatConfiguration of concatConfigurations) {
|
246
|
+
if (usedModules.has(concatConfiguration.rootModule)) continue;
|
247
|
+
const modules = concatConfiguration.getModules();
|
248
|
+
const newModule = new ConcatenatedModule(
|
249
|
+
concatConfiguration.rootModule,
|
250
|
+
modules
|
251
|
+
);
|
252
|
+
for (const warning of concatConfiguration.getWarningsSorted()) {
|
253
|
+
newModule.optimizationBailout.push(requestShortener => {
|
254
|
+
const reason = getBailoutReason(warning[0], requestShortener);
|
255
|
+
const reasonWithPrefix = reason ? ` (<- ${reason})` : "";
|
256
|
+
if (warning[0] === warning[1]) {
|
257
|
+
return formatBailoutReason(
|
258
|
+
`Cannot concat with ${warning[0].readableIdentifier(
|
259
|
+
requestShortener
|
260
|
+
)}${reasonWithPrefix}`
|
261
|
+
);
|
262
|
+
} else {
|
263
|
+
return formatBailoutReason(
|
264
|
+
`Cannot concat with ${warning[0].readableIdentifier(
|
265
|
+
requestShortener
|
266
|
+
)} because of ${warning[1].readableIdentifier(
|
267
|
+
requestShortener
|
268
|
+
)}${reasonWithPrefix}`
|
269
|
+
);
|
270
|
+
}
|
271
|
+
});
|
272
|
+
}
|
273
|
+
const chunks = concatConfiguration.rootModule.getChunks();
|
274
|
+
for (const m of modules) {
|
275
|
+
usedModules.add(m);
|
276
|
+
for (const chunk of chunks) {
|
277
|
+
chunk.removeModule(m);
|
278
|
+
}
|
279
|
+
}
|
280
|
+
for (const chunk of chunks) {
|
281
|
+
chunk.addModule(newModule);
|
282
|
+
newModule.addChunk(chunk);
|
283
|
+
if (chunk.entryModule === concatConfiguration.rootModule) {
|
284
|
+
chunk.entryModule = newModule;
|
285
|
+
}
|
286
|
+
}
|
287
|
+
compilation.modules.push(newModule);
|
288
|
+
for (const reason of newModule.reasons) {
|
289
|
+
if (reason.dependency.module === concatConfiguration.rootModule)
|
290
|
+
reason.dependency.module = newModule;
|
291
|
+
if (
|
292
|
+
reason.dependency.redirectedModule ===
|
293
|
+
concatConfiguration.rootModule
|
294
|
+
)
|
295
|
+
reason.dependency.redirectedModule = newModule;
|
296
|
+
}
|
297
|
+
// TODO: remove when LTS node version contains fixed v8 version
|
298
|
+
// @see https://github.com/webpack/webpack/pull/6613
|
299
|
+
// Turbofan does not correctly inline for-of loops with polymorphic input arrays.
|
300
|
+
// Work around issue by using a standard for loop and assigning dep.module.reasons
|
301
|
+
for (let i = 0; i < newModule.dependencies.length; i++) {
|
302
|
+
let dep = newModule.dependencies[i];
|
303
|
+
if (dep.module) {
|
304
|
+
let reasons = dep.module.reasons;
|
305
|
+
for (let j = 0; j < reasons.length; j++) {
|
306
|
+
let reason = reasons[j];
|
307
|
+
if (reason.dependency === dep) {
|
308
|
+
reason.module = newModule;
|
309
|
+
}
|
310
|
+
}
|
311
|
+
}
|
312
|
+
}
|
313
|
+
}
|
314
|
+
compilation.modules = compilation.modules.filter(
|
315
|
+
m => !usedModules.has(m)
|
316
|
+
);
|
317
|
+
}
|
318
|
+
);
|
319
|
+
}
|
320
|
+
);
|
321
|
+
}
|
322
|
+
|
323
|
+
getImports(module) {
|
324
|
+
return Array.from(
|
325
|
+
new Set(
|
326
|
+
module.dependencies
|
327
|
+
|
328
|
+
// Only harmony Dependencies
|
329
|
+
.filter(dep => dep instanceof HarmonyImportDependency)
|
330
|
+
|
331
|
+
// Get reference info for this dependency
|
332
|
+
.map(dep => dep.getReference())
|
333
|
+
|
334
|
+
// Reference is valid and has a module
|
335
|
+
.filter(ref => ref && ref.module)
|
336
|
+
|
337
|
+
// Dependencies are simple enough to concat them
|
338
|
+
.filter(
|
339
|
+
ref =>
|
340
|
+
Array.isArray(ref.importedNames) ||
|
341
|
+
Array.isArray(ref.module.buildMeta.providedExports)
|
342
|
+
)
|
343
|
+
|
344
|
+
// Take the imported module
|
345
|
+
.map(ref => ref.module)
|
346
|
+
)
|
347
|
+
);
|
348
|
+
}
|
349
|
+
|
350
|
+
tryToAdd(config, module, possibleModules, failureCache) {
|
351
|
+
const cacheEntry = failureCache.get(module);
|
352
|
+
if (cacheEntry) {
|
353
|
+
return cacheEntry;
|
354
|
+
}
|
355
|
+
|
356
|
+
// Already added?
|
357
|
+
if (config.has(module)) {
|
358
|
+
return null;
|
359
|
+
}
|
360
|
+
|
361
|
+
// Not possible to add?
|
362
|
+
if (!possibleModules.has(module)) {
|
363
|
+
failureCache.set(module, module); // cache failures for performance
|
364
|
+
return module;
|
365
|
+
}
|
366
|
+
|
367
|
+
// module must be in the same chunks
|
368
|
+
if (!config.rootModule.hasEqualsChunks(module)) {
|
369
|
+
failureCache.set(module, module); // cache failures for performance
|
370
|
+
return module;
|
371
|
+
}
|
372
|
+
|
373
|
+
// Clone config to make experimental changes
|
374
|
+
const testConfig = config.clone();
|
375
|
+
|
376
|
+
// Add the module
|
377
|
+
testConfig.add(module);
|
378
|
+
|
379
|
+
// Every module which depends on the added module must be in the configuration too.
|
380
|
+
for (const reason of module.reasons) {
|
381
|
+
// Modules that are not used can be ignored
|
382
|
+
if (
|
383
|
+
reason.module.factoryMeta.sideEffectFree &&
|
384
|
+
reason.module.used === false
|
385
|
+
)
|
386
|
+
continue;
|
387
|
+
|
388
|
+
const problem = this.tryToAdd(
|
389
|
+
testConfig,
|
390
|
+
reason.module,
|
391
|
+
possibleModules,
|
392
|
+
failureCache
|
393
|
+
);
|
394
|
+
if (problem) {
|
395
|
+
failureCache.set(module, problem); // cache failures for performance
|
396
|
+
return problem;
|
397
|
+
}
|
398
|
+
}
|
399
|
+
|
400
|
+
// Commit experimental changes
|
401
|
+
config.set(testConfig);
|
402
|
+
|
403
|
+
// Eagerly try to add imports too if possible
|
404
|
+
for (const imp of this.getImports(module)) {
|
405
|
+
const problem = this.tryToAdd(config, imp, possibleModules, failureCache);
|
406
|
+
if (problem) {
|
407
|
+
config.addWarning(imp, problem);
|
408
|
+
}
|
409
|
+
}
|
410
|
+
return null;
|
411
|
+
}
|
412
|
+
}
|
413
|
+
|
414
|
+
class ConcatConfiguration {
|
415
|
+
constructor(rootModule, cloneFrom) {
|
416
|
+
this.rootModule = rootModule;
|
417
|
+
if (cloneFrom) {
|
418
|
+
this.modules = cloneFrom.modules.createChild(5);
|
419
|
+
this.warnings = cloneFrom.warnings.createChild(5);
|
420
|
+
} else {
|
421
|
+
this.modules = new StackedSetMap();
|
422
|
+
this.modules.add(rootModule);
|
423
|
+
this.warnings = new StackedSetMap();
|
424
|
+
}
|
425
|
+
}
|
426
|
+
|
427
|
+
add(module) {
|
428
|
+
this.modules.add(module);
|
429
|
+
}
|
430
|
+
|
431
|
+
has(module) {
|
432
|
+
return this.modules.has(module);
|
433
|
+
}
|
434
|
+
|
435
|
+
isEmpty() {
|
436
|
+
return this.modules.size === 1;
|
437
|
+
}
|
438
|
+
|
439
|
+
addWarning(module, problem) {
|
440
|
+
this.warnings.set(module, problem);
|
441
|
+
}
|
442
|
+
|
443
|
+
getWarningsSorted() {
|
444
|
+
return new Map(
|
445
|
+
this.warnings.asPairArray().sort((a, b) => {
|
446
|
+
const ai = a[0].identifier();
|
447
|
+
const bi = b[0].identifier();
|
448
|
+
if (ai < bi) return -1;
|
449
|
+
if (ai > bi) return 1;
|
450
|
+
return 0;
|
451
|
+
})
|
452
|
+
);
|
453
|
+
}
|
454
|
+
|
455
|
+
getModules() {
|
456
|
+
return this.modules.asArray();
|
457
|
+
}
|
458
|
+
|
459
|
+
clone() {
|
460
|
+
return new ConcatConfiguration(this.rootModule, this);
|
461
|
+
}
|
462
|
+
|
463
|
+
set(config) {
|
464
|
+
this.rootModule = config.rootModule;
|
465
|
+
this.modules = config.modules;
|
466
|
+
this.warnings = config.warnings;
|
467
|
+
}
|
468
|
+
}
|
469
|
+
|
470
|
+
module.exports = ModuleConcatenationPlugin;
|