webpack/lib/dependencies/ImportMetaPlugin.js

254 lines
8.1 KiB
JavaScript
Raw Normal View History

2020-06-23 05:53:13 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Ivan Kopeykin @vankop
*/
"use strict";
2020-06-26 05:37:40 +08:00
const { pathToFileURL } = require("url");
const ModuleDependencyWarning = require("../ModuleDependencyWarning");
const {
JAVASCRIPT_MODULE_TYPE_AUTO,
JAVASCRIPT_MODULE_TYPE_ESM
} = require("../ModuleTypeConstants");
const Template = require("../Template");
const BasicEvaluatedExpression = require("../javascript/BasicEvaluatedExpression");
2020-06-23 05:53:13 +08:00
const {
evaluateToIdentifier,
toConstantDependency,
evaluateToString,
evaluateToNumber
2020-06-23 05:53:13 +08:00
} = require("../javascript/JavascriptParserHelpers");
const memoize = require("../util/memoize");
const propertyAccess = require("../util/propertyAccess");
const ConstDependency = require("./ConstDependency");
2020-06-23 05:53:13 +08:00
/** @typedef {import("estree").MemberExpression} MemberExpression */
/** @typedef {import("../../declarations/WebpackOptions").JavascriptParserOptions} JavascriptParserOptions */
2020-06-23 05:53:13 +08:00
/** @typedef {import("../Compiler")} Compiler */
2023-06-17 04:24:34 +08:00
/** @typedef {import("../Dependency").DependencyLocation} DependencyLocation */
/** @typedef {import("../NormalModule")} NormalModule */
2020-06-23 05:53:13 +08:00
/** @typedef {import("../javascript/JavascriptParser")} Parser */
2023-06-17 04:24:34 +08:00
/** @typedef {import("../javascript/JavascriptParser").Range} Range */
2020-06-23 05:53:13 +08:00
const getCriticalDependencyWarning = memoize(() =>
require("./CriticalDependencyWarning")
);
const PLUGIN_NAME = "ImportMetaPlugin";
2020-06-23 05:53:13 +08:00
class ImportMetaPlugin {
/**
* @param {Compiler} compiler compiler
*/
apply(compiler) {
compiler.hooks.compilation.tap(
PLUGIN_NAME,
2020-06-23 05:53:13 +08:00
(compilation, { normalModuleFactory }) => {
/**
* @param {NormalModule} module module
2020-06-26 20:04:18 +08:00
* @returns {string} file url
*/
2024-07-31 11:31:11 +08:00
const getUrl = module => pathToFileURL(module.resource).toString();
2020-06-23 05:53:13 +08:00
/**
* @param {Parser} parser parser parser
* @param {JavascriptParserOptions} parserOptions parserOptions
2020-06-23 05:53:13 +08:00
* @returns {void}
*/
const parserHandler = (parser, { importMeta }) => {
if (importMeta === false) {
const { importMetaName } = compilation.outputOptions;
2022-01-28 13:59:28 +08:00
if (importMetaName === "import.meta") return;
parser.hooks.expression
.for("import.meta")
.tap(PLUGIN_NAME, metaProperty => {
const dep = new ConstDependency(
2023-06-17 04:24:34 +08:00
/** @type {string} */ (importMetaName),
/** @type {Range} */ (metaProperty.range)
);
2023-06-17 04:24:34 +08:00
dep.loc = /** @type {DependencyLocation} */ (metaProperty.loc);
parser.state.module.addPresentationalDependency(dep);
return true;
});
return;
}
2024-07-31 12:23:44 +08:00
// import.meta direct
2024-08-02 02:36:27 +08:00
const webpackVersion = Number.parseInt(
require("../../package.json").version,
10
);
const importMetaUrl = () =>
JSON.stringify(getUrl(parser.state.module));
const importMetaWebpackVersion = () => JSON.stringify(webpackVersion);
2023-06-17 04:24:34 +08:00
/**
* @param {string[]} members members
* @returns {string} error message
*/
const importMetaUnknownProperty = members =>
`${Template.toNormalComment(
2024-07-31 10:39:30 +08:00
`unsupported import.meta.${members.join(".")}`
)} undefined${propertyAccess(members, 1)}`;
2020-06-23 05:53:13 +08:00
parser.hooks.typeof
.for("import.meta")
.tap(
PLUGIN_NAME,
2020-06-23 05:53:13 +08:00
toConstantDependency(parser, JSON.stringify("object"))
);
parser.hooks.expression
2020-08-16 15:26:14 +08:00
.for("import.meta")
.tap(PLUGIN_NAME, metaProperty => {
const referencedPropertiesInDestructuring =
parser.destructuringAssignmentPropertiesFor(metaProperty);
if (!referencedPropertiesInDestructuring) {
const CriticalDependencyWarning =
getCriticalDependencyWarning();
parser.state.module.addWarning(
new ModuleDependencyWarning(
parser.state.module,
new CriticalDependencyWarning(
2023-04-16 00:48:57 +08:00
"Accessing import.meta directly is unsupported (only property access or destructuring is supported)"
),
2023-06-17 04:24:34 +08:00
/** @type {DependencyLocation} */ (metaProperty.loc)
)
);
const dep = new ConstDependency(
`${
2023-06-17 04:24:34 +08:00
parser.isAsiPosition(
/** @type {Range} */ (metaProperty.range)[0]
)
? ";"
: ""
}({})`,
2023-06-17 04:24:34 +08:00
/** @type {Range} */ (metaProperty.range)
);
2023-06-17 04:24:34 +08:00
dep.loc = /** @type {DependencyLocation} */ (metaProperty.loc);
parser.state.module.addPresentationalDependency(dep);
return true;
}
let str = "";
2024-04-13 02:40:28 +08:00
for (const { id: prop } of referencedPropertiesInDestructuring) {
switch (prop) {
case "url":
str += `url: ${importMetaUrl()},`;
break;
case "webpack":
str += `webpack: ${importMetaWebpackVersion()},`;
break;
default:
str += `[${JSON.stringify(
prop
)}]: ${importMetaUnknownProperty([prop])},`;
break;
}
}
2023-06-17 04:24:34 +08:00
const dep = new ConstDependency(
`({${str}})`,
/** @type {Range} */ (metaProperty.range)
);
dep.loc = /** @type {DependencyLocation} */ (metaProperty.loc);
2020-08-16 15:26:14 +08:00
parser.state.module.addPresentationalDependency(dep);
return true;
});
parser.hooks.evaluateTypeof
.for("import.meta")
.tap(PLUGIN_NAME, evaluateToString("object"));
parser.hooks.evaluateIdentifier.for("import.meta").tap(
PLUGIN_NAME,
evaluateToIdentifier("import.meta", "import.meta", () => [], true)
);
2024-07-31 12:23:44 +08:00
// import.meta.url
2020-06-23 05:53:13 +08:00
parser.hooks.typeof
.for("import.meta.url")
.tap(
PLUGIN_NAME,
2020-06-23 05:53:13 +08:00
toConstantDependency(parser, JSON.stringify("string"))
);
parser.hooks.expression
.for("import.meta.url")
.tap(PLUGIN_NAME, expr => {
2023-06-17 04:24:34 +08:00
const dep = new ConstDependency(
importMetaUrl(),
/** @type {Range} */ (expr.range)
);
dep.loc = /** @type {DependencyLocation} */ (expr.loc);
parser.state.module.addPresentationalDependency(dep);
2020-06-23 05:53:13 +08:00
return true;
});
parser.hooks.evaluateTypeof
.for("import.meta.url")
.tap(PLUGIN_NAME, evaluateToString("string"));
2022-01-26 17:48:01 +08:00
parser.hooks.evaluateIdentifier
.for("import.meta.url")
2024-07-31 11:31:11 +08:00
.tap(PLUGIN_NAME, expr =>
new BasicEvaluatedExpression()
2022-01-26 17:48:01 +08:00
.setString(getUrl(parser.state.module))
2024-07-31 11:31:11 +08:00
.setRange(/** @type {Range} */ (expr.range))
);
2024-07-31 12:23:44 +08:00
// import.meta.webpack
parser.hooks.typeof
.for("import.meta.webpack")
.tap(
PLUGIN_NAME,
toConstantDependency(parser, JSON.stringify("number"))
);
parser.hooks.expression
.for("import.meta.webpack")
.tap(
PLUGIN_NAME,
toConstantDependency(parser, importMetaWebpackVersion())
);
parser.hooks.evaluateTypeof
.for("import.meta.webpack")
.tap(PLUGIN_NAME, evaluateToString("number"));
parser.hooks.evaluateIdentifier
.for("import.meta.webpack")
.tap(PLUGIN_NAME, evaluateToNumber(webpackVersion));
2024-07-31 12:23:44 +08:00
// Unknown properties
parser.hooks.unhandledExpressionMemberChain
.for("import.meta")
.tap(PLUGIN_NAME, (expr, members) => {
const dep = new ConstDependency(
importMetaUnknownProperty(members),
2023-06-17 04:24:34 +08:00
/** @type {Range} */ (expr.range)
);
2023-06-17 04:24:34 +08:00
dep.loc = /** @type {DependencyLocation} */ (expr.loc);
parser.state.module.addPresentationalDependency(dep);
return true;
});
parser.hooks.evaluate
.for("MemberExpression")
.tap(PLUGIN_NAME, expression => {
const expr = /** @type {MemberExpression} */ (expression);
if (
expr.object.type === "MetaProperty" &&
expr.object.meta.name === "import" &&
expr.object.property.name === "meta" &&
expr.property.type ===
(expr.computed ? "Literal" : "Identifier")
) {
return new BasicEvaluatedExpression()
.setUndefined()
2023-06-17 04:24:34 +08:00
.setRange(/** @type {Range} */ (expr.range));
}
});
2020-06-23 05:53:13 +08:00
};
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_AUTO)
.tap(PLUGIN_NAME, parserHandler);
2020-06-23 05:53:13 +08:00
normalModuleFactory.hooks.parser
.for(JAVASCRIPT_MODULE_TYPE_ESM)
.tap(PLUGIN_NAME, parserHandler);
2020-06-23 05:53:13 +08:00
}
);
}
}
module.exports = ImportMetaPlugin;