webpack/lib/dependencies/RequireEnsureDependenciesBl...

139 lines
4.5 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";
2013-01-31 01:49:25 +08:00
const RequireEnsureDependenciesBlock = require("./RequireEnsureDependenciesBlock");
const RequireEnsureDependency = require("./RequireEnsureDependency");
const RequireEnsureItemDependency = require("./RequireEnsureItemDependency");
const getFunctionExpression = require("./getFunctionExpression");
2023-06-17 02:49:43 +08:00
/** @typedef {import("../ChunkGroup").ChunkGroupOptions} ChunkGroupOptions */
2024-03-18 23:28:40 +08:00
/** @typedef {import("../Dependency").DependencyLocation} DependencyLocation */
/** @typedef {import("../javascript/BasicEvaluatedExpression")} BasicEvaluatedExpression */
2023-06-17 02:49:43 +08:00
/** @typedef {import("../javascript/JavascriptParser")} JavascriptParser */
2024-03-18 23:28:40 +08:00
/** @typedef {import("../javascript/JavascriptParser").Range} Range */
2023-06-17 02:49:43 +08:00
module.exports = class RequireEnsureDependenciesBlockParserPlugin {
2024-03-18 23:28:40 +08:00
/**
* @param {JavascriptParser} parser the parser
* @returns {void}
*/
apply(parser) {
2018-02-25 09:00:20 +08:00
parser.hooks.call
.for("require.ensure")
.tap("RequireEnsureDependenciesBlockParserPlugin", expr => {
let chunkName = null;
let errorExpressionArg = null;
let errorExpression = null;
switch (expr.arguments.length) {
case 4: {
2017-01-20 12:12:54 +08:00
const chunkNameExpr = parser.evaluateExpression(expr.arguments[3]);
2018-02-25 09:00:20 +08:00
if (!chunkNameExpr.isString()) return;
chunkName = chunkNameExpr.string;
}
// falls through
2018-02-25 09:00:20 +08:00
case 3: {
2017-01-20 12:12:54 +08:00
errorExpressionArg = expr.arguments[2];
errorExpression = getFunctionExpression(errorExpressionArg);
2018-02-25 09:00:20 +08:00
if (!errorExpression && !chunkName) {
const chunkNameExpr = parser.evaluateExpression(
expr.arguments[2]
);
if (!chunkNameExpr.isString()) return;
2017-01-20 12:12:54 +08:00
chunkName = chunkNameExpr.string;
}
}
// falls through
2018-02-25 09:00:20 +08:00
case 2: {
const dependenciesExpr = parser.evaluateExpression(
expr.arguments[0]
);
2024-03-18 23:28:40 +08:00
const dependenciesItems =
/** @type {BasicEvaluatedExpression[]} */ (
dependenciesExpr.isArray()
? dependenciesExpr.items
: [dependenciesExpr]
);
2017-01-20 12:12:54 +08:00
const successExpressionArg = expr.arguments[1];
2021-05-11 15:31:46 +08:00
const successExpression =
getFunctionExpression(successExpressionArg);
2018-02-25 09:00:20 +08:00
if (successExpression) {
2017-01-20 12:12:54 +08:00
parser.walkExpressions(successExpression.expressions);
}
2018-02-25 09:00:20 +08:00
if (errorExpression) {
2017-01-20 12:12:54 +08:00
parser.walkExpressions(errorExpression.expressions);
}
const depBlock = new RequireEnsureDependenciesBlock(
2024-03-18 23:28:40 +08:00
/** @type {ChunkGroupOptions & { entryOptions?: TODO }} */
(chunkName),
/** @type {DependencyLocation} */ (expr.loc)
2018-02-25 09:00:20 +08:00
);
const errorCallbackExists =
expr.arguments.length === 4 ||
(!chunkName && expr.arguments.length === 3);
const dep = new RequireEnsureDependency(
2024-03-18 23:28:40 +08:00
/** @type {Range} */ (expr.range),
/** @type {Range} */ (expr.arguments[1].range),
errorCallbackExists &&
/** @type {Range} */ (expr.arguments[2].range)
);
2024-03-18 23:28:40 +08:00
dep.loc = /** @type {DependencyLocation} */ (expr.loc);
depBlock.addDependency(dep);
const old = parser.state.current;
2024-03-18 23:28:40 +08:00
parser.state.current = /** @type {TODO} */ (depBlock);
try {
let failed = false;
parser.inScope([], () => {
2018-02-25 09:00:20 +08:00
for (const ee of dependenciesItems) {
if (ee.isString()) {
2020-03-13 00:51:26 +08:00
const ensureDependency = new RequireEnsureItemDependency(
2024-03-18 23:28:40 +08:00
/** @type {string} */ (ee.string)
2020-03-13 00:51:26 +08:00
);
2024-03-18 23:28:40 +08:00
ensureDependency.loc =
/** @type {DependencyLocation} */
(expr.loc);
2020-03-13 00:51:26 +08:00
depBlock.addDependency(ensureDependency);
} else {
failed = true;
}
2018-01-22 20:52:43 +08:00
}
});
2018-02-25 09:00:20 +08:00
if (failed) {
return;
2015-07-13 06:20:09 +08:00
}
2018-02-25 09:00:20 +08:00
if (successExpression) {
if (successExpression.fn.body.type === "BlockStatement") {
2017-01-20 12:12:54 +08:00
parser.walkStatement(successExpression.fn.body);
} else {
parser.walkExpression(successExpression.fn.body);
}
2017-01-20 12:12:54 +08:00
}
old.addBlock(depBlock);
} finally {
parser.state.current = old;
}
2018-02-25 09:00:20 +08:00
if (!successExpression) {
2017-01-20 12:12:54 +08:00
parser.walkExpression(successExpressionArg);
}
2018-02-25 09:00:20 +08:00
if (errorExpression) {
if (errorExpression.fn.body.type === "BlockStatement") {
parser.walkStatement(errorExpression.fn.body);
} else {
parser.walkExpression(errorExpression.fn.body);
}
2018-02-25 09:00:20 +08:00
} else if (errorExpressionArg) {
parser.walkExpression(errorExpressionArg);
}
return true;
2015-07-13 06:20:09 +08:00
}
2018-02-25 09:00:20 +08:00
}
});
2013-01-31 01:49:25 +08:00
}
};