mirror of https://github.com/webpack/webpack.git
184 lines
5.3 KiB
JavaScript
184 lines
5.3 KiB
JavaScript
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
const glob2regexp = require("glob-to-regexp");
|
|
const { STAGE_DEFAULT } = require("../OptimizationStages");
|
|
const HarmonyExportImportedSpecifierDependency = require("../dependencies/HarmonyExportImportedSpecifierDependency");
|
|
const HarmonyImportSpecifierDependency = require("../dependencies/HarmonyImportSpecifierDependency");
|
|
|
|
/** @typedef {import("../Compiler")} Compiler */
|
|
/** @typedef {import("../Dependency")} Dependency */
|
|
/** @typedef {import("../Module")} Module */
|
|
|
|
/**
|
|
* @typedef {Object} ExportInModule
|
|
* @property {Module} module the module
|
|
* @property {string} exportName the name of the export
|
|
*/
|
|
|
|
/** @type {WeakMap<any, Map<string, RegExp>>} */
|
|
const globToRegexpCache = new WeakMap();
|
|
|
|
/**
|
|
* @param {string} glob the pattern
|
|
* @param {Map<string, RegExp>} cache the glob to RegExp cache
|
|
* @returns {RegExp} a regular expression
|
|
*/
|
|
const globToRegexp = (glob, cache) => {
|
|
const cacheEntry = cache.get(glob);
|
|
if (cacheEntry !== undefined) return cacheEntry;
|
|
if (!glob.includes("/")) {
|
|
glob = `**/${glob}`;
|
|
}
|
|
const baseRegexp = glob2regexp(glob, { globstar: true, extended: true });
|
|
const regexpSource = baseRegexp.source;
|
|
const regexp = new RegExp("^(\\./)?" + regexpSource.slice(1));
|
|
cache.set(glob, regexp);
|
|
return regexp;
|
|
};
|
|
|
|
class SideEffectsFlagPlugin {
|
|
/**
|
|
* @param {Compiler} compiler webpack compiler
|
|
* @returns {void}
|
|
*/
|
|
apply(compiler) {
|
|
let cache = globToRegexpCache.get(compiler.root);
|
|
if (cache === undefined) {
|
|
cache = new Map();
|
|
globToRegexpCache.set(compiler.root, cache);
|
|
}
|
|
compiler.hooks.normalModuleFactory.tap("SideEffectsFlagPlugin", nmf => {
|
|
nmf.hooks.module.tap("SideEffectsFlagPlugin", (module, data) => {
|
|
const resolveData = data.resourceResolveData;
|
|
if (
|
|
resolveData &&
|
|
resolveData.descriptionFileData &&
|
|
resolveData.relativePath
|
|
) {
|
|
const sideEffects = resolveData.descriptionFileData.sideEffects;
|
|
const hasSideEffects = SideEffectsFlagPlugin.moduleHasSideEffects(
|
|
resolveData.relativePath,
|
|
sideEffects,
|
|
cache
|
|
);
|
|
if (!hasSideEffects) {
|
|
module.factoryMeta.sideEffectFree = true;
|
|
}
|
|
}
|
|
|
|
return module;
|
|
});
|
|
nmf.hooks.module.tap("SideEffectsFlagPlugin", (module, data) => {
|
|
if (data.settings.sideEffects === false) {
|
|
module.factoryMeta.sideEffectFree = true;
|
|
} else if (data.settings.sideEffects === true) {
|
|
module.factoryMeta.sideEffectFree = false;
|
|
}
|
|
return module;
|
|
});
|
|
});
|
|
compiler.hooks.compilation.tap("SideEffectsFlagPlugin", compilation => {
|
|
const moduleGraph = compilation.moduleGraph;
|
|
compilation.hooks.optimizeDependencies.tap(
|
|
{
|
|
name: "SideEffectsFlagPlugin",
|
|
stage: STAGE_DEFAULT
|
|
},
|
|
modules => {
|
|
/** @type {Map<Module, Map<string, ExportInModule>>} */
|
|
const reexportMaps = new Map();
|
|
|
|
// Capture reexports of sideEffectFree modules
|
|
for (const module of modules) {
|
|
for (const dep of module.dependencies) {
|
|
if (dep instanceof HarmonyExportImportedSpecifierDependency) {
|
|
if (module.factoryMeta.sideEffectFree) {
|
|
const mode = dep.getMode(moduleGraph, true);
|
|
if (mode.type === "normal-reexport") {
|
|
let map = reexportMaps.get(module);
|
|
if (!map) {
|
|
reexportMaps.set(module, (map = new Map()));
|
|
}
|
|
for (const [key, id] of mode.map) {
|
|
if (!mode.checked.has(key)) {
|
|
map.set(key, {
|
|
module: mode.getModule(),
|
|
exportName: id
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Flatten reexports
|
|
for (const map of reexportMaps.values()) {
|
|
for (const pair of map) {
|
|
let mapping = pair[1];
|
|
while (mapping) {
|
|
const innerMap = reexportMaps.get(mapping.module);
|
|
if (!innerMap) break;
|
|
const newMapping = innerMap.get(mapping.exportName);
|
|
if (newMapping) {
|
|
map.set(pair[0], newMapping);
|
|
}
|
|
mapping = newMapping;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update imports along the reexports from sideEffectFree modules
|
|
for (const pair of reexportMaps) {
|
|
const module = pair[0];
|
|
const map = pair[1];
|
|
for (const connection of moduleGraph.getIncomingConnections(
|
|
module
|
|
)) {
|
|
const dep = connection.dependency;
|
|
if (
|
|
dep instanceof HarmonyExportImportedSpecifierDependency ||
|
|
(dep instanceof HarmonyImportSpecifierDependency &&
|
|
!dep.namespaceObjectAsContext)
|
|
) {
|
|
const mapping = map.get(dep.id);
|
|
if (mapping) {
|
|
moduleGraph.updateModule(dep, mapping.module);
|
|
moduleGraph.addExplanation(
|
|
dep,
|
|
"(skipped side-effect-free modules)"
|
|
);
|
|
dep.setId(moduleGraph, mapping.exportName);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
);
|
|
});
|
|
}
|
|
|
|
static moduleHasSideEffects(moduleName, flagValue, cache) {
|
|
switch (typeof flagValue) {
|
|
case "undefined":
|
|
return true;
|
|
case "boolean":
|
|
return flagValue;
|
|
case "string":
|
|
return globToRegexp(flagValue, cache).test(moduleName);
|
|
case "object":
|
|
return flagValue.some(glob =>
|
|
SideEffectsFlagPlugin.moduleHasSideEffects(moduleName, glob, cache)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
module.exports = SideEffectsFlagPlugin;
|