webpack/lib/CompatibilityPlugin.js

198 lines
5.9 KiB
JavaScript
Raw Normal View History

2013-01-31 01:49:25 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2018-07-30 23:08:51 +08:00
"use strict";
const {
JAVASCRIPT_MODULE_TYPE_AUTO,
JAVASCRIPT_MODULE_TYPE_DYNAMIC,
JAVASCRIPT_MODULE_TYPE_ESM
} = require("./ModuleTypeConstants");
2023-05-20 00:27:52 +08:00
const RuntimeGlobals = require("./RuntimeGlobals");
const ConstDependency = require("./dependencies/ConstDependency");
2023-06-17 01:13:03 +08:00
/** @typedef {import("estree").CallExpression} CallExpression */
2018-07-09 20:48:28 +08:00
/** @typedef {import("./Compiler")} Compiler */
2023-06-17 01:13:03 +08:00
/** @typedef {import("./Dependency").DependencyLocation} DependencyLocation */
2019-10-11 21:46:57 +08:00
/** @typedef {import("./javascript/JavascriptParser")} JavascriptParser */
2023-06-17 01:13:03 +08:00
/** @typedef {import("./javascript/JavascriptParser").Range} Range */
2025-03-07 21:12:22 +08:00
/** @typedef {import("./javascript/JavascriptParser").TagData} TagData */
2018-06-20 03:19:20 +08:00
const nestedWebpackIdentifierTag = Symbol("nested webpack identifier");
const PLUGIN_NAME = "CompatibilityPlugin";
class CompatibilityPlugin {
2018-06-16 02:15:27 +08:00
/**
* Apply the plugin
2018-11-03 04:05:46 +08:00
* @param {Compiler} compiler the compiler instance
2018-06-16 02:15:27 +08:00
* @returns {void}
*/
apply(compiler) {
2018-02-25 09:00:20 +08:00
compiler.hooks.compilation.tap(
PLUGIN_NAME,
2018-02-25 09:00:20 +08:00
(compilation, { normalModuleFactory }) => {
compilation.dependencyTemplates.set(
ConstDependency,
new ConstDependency.Template()
);
2018-02-25 09:00:20 +08:00
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_AUTO)
.tap(PLUGIN_NAME, (parser, parserOptions) => {
2018-02-25 09:00:20 +08:00
if (
2018-08-21 08:26:50 +08:00
parserOptions.browserify !== undefined &&
2018-02-25 09:00:20 +08:00
!parserOptions.browserify
) {
2018-02-25 09:00:20 +08:00
return;
}
2023-06-17 01:13:03 +08:00
parser.hooks.call.for("require").tap(
PLUGIN_NAME,
/**
* @param {CallExpression} expr call expression
* @returns {boolean | void} true when need to handle
*/
(expr) => {
2023-06-17 01:13:03 +08:00
// support for browserify style require delegator: "require(o, !0)"
if (expr.arguments.length !== 2) return;
const second = parser.evaluateExpression(expr.arguments[1]);
if (!second.isBoolean()) return;
if (second.asBool() !== true) return;
const dep = new ConstDependency(
"require",
/** @type {Range} */ (expr.callee.range)
);
dep.loc = /** @type {DependencyLocation} */ (expr.loc);
if (parser.state.current.dependencies.length > 0) {
const last =
parser.state.current.dependencies[
parser.state.current.dependencies.length - 1
];
if (
last.critical &&
last.options &&
last.options.request === "." &&
last.userRequest === "." &&
last.options.recursive
) {
2023-06-17 01:13:03 +08:00
parser.state.current.dependencies.pop();
}
2023-06-17 01:13:03 +08:00
}
parser.state.module.addPresentationalDependency(dep);
return true;
}
2023-06-17 01:13:03 +08:00
);
2018-02-25 09:00:20 +08:00
});
/**
* @param {JavascriptParser} parser the parser
* @returns {void}
*/
const handler = (parser) => {
// Handle nested requires
parser.hooks.preStatement.tap(PLUGIN_NAME, (statement) => {
if (
statement.type === "FunctionDeclaration" &&
statement.id &&
2023-05-20 00:00:54 +08:00
statement.id.name === RuntimeGlobals.require
) {
2023-06-17 01:13:03 +08:00
const newName = `__nested_webpack_require_${
/** @type {Range} */ (statement.range)[0]
}__`;
parser.tagVariable(
statement.id.name,
nestedWebpackIdentifierTag,
{
name: newName,
declaration: {
updated: false,
loc: statement.id.loc,
range: statement.id.range
}
}
);
return true;
}
});
parser.hooks.pattern
2023-05-20 00:00:54 +08:00
.for(RuntimeGlobals.require)
.tap(PLUGIN_NAME, (pattern) => {
2023-06-17 01:13:03 +08:00
const newName = `__nested_webpack_require_${
/** @type {Range} */ (pattern.range)[0]
}__`;
parser.tagVariable(pattern.name, nestedWebpackIdentifierTag, {
name: newName,
declaration: {
updated: false,
loc: pattern.loc,
range: pattern.range
}
});
return true;
});
parser.hooks.pattern
.for(RuntimeGlobals.exports)
.tap(PLUGIN_NAME, (pattern) => {
parser.tagVariable(pattern.name, nestedWebpackIdentifierTag, {
name: "__nested_webpack_exports__",
declaration: {
updated: false,
loc: pattern.loc,
range: pattern.range
}
});
return true;
});
parser.hooks.expression
.for(nestedWebpackIdentifierTag)
.tap(PLUGIN_NAME, (expr) => {
2025-03-07 21:12:22 +08:00
const { name, declaration } =
/** @type {TagData} */
(parser.currentTagData);
if (!declaration.updated) {
const dep = new ConstDependency(name, declaration.range);
dep.loc = declaration.loc;
parser.state.module.addPresentationalDependency(dep);
declaration.updated = true;
}
2023-06-17 01:13:03 +08:00
const dep = new ConstDependency(
name,
/** @type {Range} */ (expr.range)
);
dep.loc = /** @type {DependencyLocation} */ (expr.loc);
parser.state.module.addPresentationalDependency(dep);
return true;
});
// Handle hashbang
parser.hooks.program.tap(PLUGIN_NAME, (program, comments) => {
if (comments.length === 0) return;
const c = comments[0];
2023-06-17 01:13:03 +08:00
if (c.type === "Line" && /** @type {Range} */ (c.range)[0] === 0) {
if (parser.state.source.slice(0, 2).toString() !== "#!") return;
// this is a hashbang comment
const dep = new ConstDependency("//", 0);
2023-06-17 01:13:03 +08:00
dep.loc = /** @type {DependencyLocation} */ (c.loc);
parser.state.module.addPresentationalDependency(dep);
}
});
};
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_AUTO)
.tap(PLUGIN_NAME, handler);
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_DYNAMIC)
.tap(PLUGIN_NAME, handler);
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_ESM)
.tap(PLUGIN_NAME, handler);
2018-02-25 09:00:20 +08:00
}
);
}
}
module.exports = CompatibilityPlugin;