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
|
|
|
|
2017-01-10 18:29:55 +08:00
|
|
|
"use strict";
|
|
|
|
|
2023-04-01 01:56:32 +08:00
|
|
|
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");
|
2017-01-10 18:29:55 +08:00
|
|
|
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 */
|
2025-08-28 18:34:30 +08:00
|
|
|
/** @typedef {import("./dependencies/ContextDependency")} ContextDependency */
|
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-09-09 23:41:52 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} CompatibilitySettingsDeclaration
|
|
|
|
* @property {boolean} updated
|
|
|
|
* @property {DependencyLocation} loc
|
|
|
|
* @property {Range} range
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} CompatibilitySettings
|
|
|
|
* @property {string} name
|
|
|
|
* @property {CompatibilitySettingsDeclaration} declaration
|
|
|
|
*/
|
2018-06-20 03:19:20 +08:00
|
|
|
|
2022-04-01 04:53:14 +08:00
|
|
|
const nestedWebpackIdentifierTag = Symbol("nested webpack identifier");
|
2023-04-01 01:56:32 +08:00
|
|
|
const PLUGIN_NAME = "CompatibilityPlugin";
|
2019-08-29 21:28:19 +08:00
|
|
|
|
2017-01-10 18:29:55 +08:00
|
|
|
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}
|
|
|
|
*/
|
2017-01-10 18:29:55 +08:00
|
|
|
apply(compiler) {
|
2018-02-25 09:00:20 +08:00
|
|
|
compiler.hooks.compilation.tap(
|
2023-04-01 01:56:32 +08:00
|
|
|
PLUGIN_NAME,
|
2018-02-25 09:00:20 +08:00
|
|
|
(compilation, { normalModuleFactory }) => {
|
|
|
|
compilation.dependencyTemplates.set(
|
|
|
|
ConstDependency,
|
|
|
|
new ConstDependency.Template()
|
|
|
|
);
|
2017-01-10 18:29:55 +08:00
|
|
|
|
2018-02-25 09:00:20 +08:00
|
|
|
normalModuleFactory.hooks.parser
|
2023-04-01 01:56:32 +08:00
|
|
|
.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
|
2025-07-02 20:10:54 +08:00
|
|
|
) {
|
2018-02-25 09:00:20 +08:00
|
|
|
return;
|
2025-07-02 20:10:54 +08:00
|
|
|
}
|
2017-01-10 18:29:55 +08:00
|
|
|
|
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
|
|
|
|
*/
|
2025-07-17 00:13:14 +08:00
|
|
|
(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 =
|
2025-08-28 18:34:30 +08:00
|
|
|
/** @type {ContextDependency} */
|
|
|
|
(
|
|
|
|
parser.state.current.dependencies[
|
|
|
|
parser.state.current.dependencies.length - 1
|
|
|
|
]
|
|
|
|
);
|
2023-06-17 01:13:03 +08:00
|
|
|
if (
|
|
|
|
last.critical &&
|
|
|
|
last.options &&
|
|
|
|
last.options.request === "." &&
|
|
|
|
last.userRequest === "." &&
|
|
|
|
last.options.recursive
|
2025-07-02 20:10:54 +08:00
|
|
|
) {
|
2023-06-17 01:13:03 +08:00
|
|
|
parser.state.current.dependencies.pop();
|
2025-07-02 20:10:54 +08:00
|
|
|
}
|
2023-06-17 01:13:03 +08:00
|
|
|
}
|
|
|
|
parser.state.module.addPresentationalDependency(dep);
|
|
|
|
return true;
|
2023-04-01 01:56:32 +08:00
|
|
|
}
|
2023-06-17 01:13:03 +08:00
|
|
|
);
|
2018-02-25 09:00:20 +08:00
|
|
|
});
|
2019-05-21 04:53:58 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {JavascriptParser} parser the parser
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2025-07-17 00:13:14 +08:00
|
|
|
const handler = (parser) => {
|
2021-08-10 17:23:30 +08:00
|
|
|
// Handle nested requires
|
2025-07-17 00:13:14 +08:00
|
|
|
parser.hooks.preStatement.tap(PLUGIN_NAME, (statement) => {
|
2019-09-30 16:08:08 +08:00
|
|
|
if (
|
|
|
|
statement.type === "FunctionDeclaration" &&
|
|
|
|
statement.id &&
|
2023-05-20 00:00:54 +08:00
|
|
|
statement.id.name === RuntimeGlobals.require
|
2019-09-30 16:08:08 +08:00
|
|
|
) {
|
2023-06-17 01:13:03 +08:00
|
|
|
const newName = `__nested_webpack_require_${
|
2025-09-09 23:41:52 +08:00
|
|
|
/** @type {Range} */
|
|
|
|
(statement.range)[0]
|
2023-06-17 01:13:03 +08:00
|
|
|
}__`;
|
2022-04-01 04:53:14 +08:00
|
|
|
parser.tagVariable(
|
|
|
|
statement.id.name,
|
|
|
|
nestedWebpackIdentifierTag,
|
|
|
|
{
|
|
|
|
name: newName,
|
|
|
|
declaration: {
|
|
|
|
updated: false,
|
2025-09-09 23:41:52 +08:00
|
|
|
loc: /** @type {DependencyLocation} */ (statement.id.loc),
|
|
|
|
range: /** @type {Range} */ (statement.id.range)
|
2022-04-01 04:53:14 +08:00
|
|
|
}
|
2021-01-04 22:30:11 +08:00
|
|
|
}
|
2022-04-01 04:53:14 +08:00
|
|
|
);
|
2019-09-30 16:08:08 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
2019-05-21 04:53:58 +08:00
|
|
|
parser.hooks.pattern
|
2023-05-20 00:00:54 +08:00
|
|
|
.for(RuntimeGlobals.require)
|
2025-07-17 00:13:14 +08:00
|
|
|
.tap(PLUGIN_NAME, (pattern) => {
|
2023-06-17 01:13:03 +08:00
|
|
|
const newName = `__nested_webpack_require_${
|
|
|
|
/** @type {Range} */ (pattern.range)[0]
|
|
|
|
}__`;
|
2022-04-01 04:53:14 +08:00
|
|
|
parser.tagVariable(pattern.name, nestedWebpackIdentifierTag, {
|
2021-01-04 22:30:11 +08:00
|
|
|
name: newName,
|
|
|
|
declaration: {
|
|
|
|
updated: false,
|
2025-09-09 23:41:52 +08:00
|
|
|
loc: /** @type {DependencyLocation} */ (pattern.loc),
|
|
|
|
range: /** @type {Range} */ (pattern.range)
|
2021-01-04 22:30:11 +08:00
|
|
|
}
|
|
|
|
});
|
2019-05-21 04:53:58 +08:00
|
|
|
return true;
|
|
|
|
});
|
2022-04-01 04:53:14 +08:00
|
|
|
parser.hooks.pattern
|
2023-05-26 23:34:26 +08:00
|
|
|
.for(RuntimeGlobals.exports)
|
2025-07-17 00:13:14 +08:00
|
|
|
.tap(PLUGIN_NAME, (pattern) => {
|
2022-04-01 04:53:14 +08:00
|
|
|
parser.tagVariable(pattern.name, nestedWebpackIdentifierTag, {
|
|
|
|
name: "__nested_webpack_exports__",
|
|
|
|
declaration: {
|
|
|
|
updated: false,
|
2025-09-09 23:41:52 +08:00
|
|
|
loc: /** @type {DependencyLocation} */ (pattern.loc),
|
|
|
|
range: /** @type {Range} */ (pattern.range)
|
2022-04-01 04:53:14 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
});
|
2019-05-21 04:53:58 +08:00
|
|
|
parser.hooks.expression
|
2022-04-01 04:53:14 +08:00
|
|
|
.for(nestedWebpackIdentifierTag)
|
2025-07-17 00:13:14 +08:00
|
|
|
.tap(PLUGIN_NAME, (expr) => {
|
2025-03-07 21:12:22 +08:00
|
|
|
const { name, declaration } =
|
2025-09-09 23:41:52 +08:00
|
|
|
/** @type {CompatibilitySettings} */
|
2025-03-07 21:12:22 +08:00
|
|
|
(parser.currentTagData);
|
2021-01-04 22:30:11 +08:00
|
|
|
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);
|
2019-10-30 13:40:40 +08:00
|
|
|
parser.state.module.addPresentationalDependency(dep);
|
2019-05-21 04:53:58 +08:00
|
|
|
return true;
|
|
|
|
});
|
2021-08-10 17:23:30 +08:00
|
|
|
|
|
|
|
// Handle hashbang
|
2023-04-01 01:56:32 +08:00
|
|
|
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) {
|
2023-04-01 01:56:32 +08:00
|
|
|
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);
|
2023-04-01 01:56:32 +08:00
|
|
|
parser.state.module.addPresentationalDependency(dep);
|
2021-08-10 17:23:30 +08:00
|
|
|
}
|
2023-04-01 01:56:32 +08:00
|
|
|
});
|
2019-05-21 04:53:58 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
normalModuleFactory.hooks.parser
|
2023-04-01 01:56:32 +08:00
|
|
|
.for(JAVASCRIPT_MODULE_TYPE_AUTO)
|
|
|
|
.tap(PLUGIN_NAME, handler);
|
2019-05-21 04:53:58 +08:00
|
|
|
normalModuleFactory.hooks.parser
|
2023-04-01 01:56:32 +08:00
|
|
|
.for(JAVASCRIPT_MODULE_TYPE_DYNAMIC)
|
|
|
|
.tap(PLUGIN_NAME, handler);
|
2019-05-21 04:53:58 +08:00
|
|
|
normalModuleFactory.hooks.parser
|
2023-04-01 01:56:32 +08:00
|
|
|
.for(JAVASCRIPT_MODULE_TYPE_ESM)
|
|
|
|
.tap(PLUGIN_NAME, handler);
|
2018-02-25 09:00:20 +08:00
|
|
|
}
|
|
|
|
);
|
2017-01-10 18:29:55 +08:00
|
|
|
}
|
|
|
|
}
|
2025-07-02 20:10:54 +08:00
|
|
|
|
2017-01-10 18:29:55 +08:00
|
|
|
module.exports = CompatibilityPlugin;
|