webpack JS meta 信息插件 /lib/JavascriptMetaInfoPlugin.js

  • 插件名称定义为 JavascriptMetaInfoPlugin,并应用于 Webpack 的编译流程中。

  • 监听 compiler.hooks.compilation 钩子,在编译过程中注入逻辑。

  • 为三种 JavaScript 模块类型(AUTO、DYNAMIC、ESM)注册 parser 钩子,统一处理逻辑。

  • 在解析阶段监听 eval() 的调用

    • 标记 buildInfo.moduleConcatenationBailout = "eval()",阻止模块作用域连接优化。
    • 设置 buildInfo.usingEval = true,记录模块使用了 eval()
    • 使用 InnerGraph.getTopLevelSymbol() 尝试获取当前上下文的顶层符号。
    • 如果存在顶层符号,则调用 InnerGraph.addUsage() 记录引用关系。
    • 否则调用 InnerGraph.bailout(),中断内部图优化。
  • 在解析结束时(parser.finish)记录顶层变量声明

    • 初始化 buildInfo.topLevelDeclarations 为一个 Set
    • 遍历作用域中定义的变量名。
    • 如果变量没有自由变量信息(非闭包变量),视为顶层声明加入集合。
  • 最终导出插件类用于 Webpack 插件系统使用。

js 复制代码
/*
	MIT License http://www.opensource.org/licenses/mit-license.php
	Author Sergey Melyukov @smelukov
*/
// 这是插件的许可证信息和作者署名

"use strict";
// 启用严格模式,提高代码的规范性和安全性

const {
	JAVASCRIPT_MODULE_TYPE_AUTO,
	JAVASCRIPT_MODULE_TYPE_DYNAMIC,
	JAVASCRIPT_MODULE_TYPE_ESM
} = require("./ModuleTypeConstants");
// 引入三种 JavaScript 模块类型常量:自动识别、动态模块、ES 模块

const InnerGraph = require("./optimize/InnerGraph");
// 引入 InnerGraph,用于分析模块的内部依赖关系图(变量引用、作用域等)

/** @typedef {import("./Compiler")} Compiler */
// 引入类型定义:Compiler

/** @typedef {import("./Module").BuildInfo} BuildInfo */
// 引入类型定义:模块的构建信息 BuildInfo

/** @typedef {import("./javascript/JavascriptParser")} JavascriptParser */
// 引入类型定义:JavaScript 语法解析器 JavascriptParser

const PLUGIN_NAME = "JavascriptMetaInfoPlugin";
// 定义插件名称常量,后续钩子绑定中使用

class JavascriptMetaInfoPlugin {
	/**
	 * 插件应用函数
	 * @param {Compiler} compiler 编译器实例
	 * @returns {void}
	 */
	apply(compiler) {
		// 在 compilation 阶段注册钩子
		compiler.hooks.compilation.tap(
			PLUGIN_NAME,
			(compilation, { normalModuleFactory }) => {
				/**
				 * 处理 JavaScript 解析器的钩子函数
				 * @param {JavascriptParser} parser 解析器实例
				 * @returns {void}
				 */
				const handler = parser => {
					// 为 eval() 函数调用绑定钩子
					parser.hooks.call.for("eval").tap(PLUGIN_NAME, () => {
						const buildInfo =
							/** @type {BuildInfo} */
							(parser.state.module.buildInfo);
						// 使用 eval() 会导致模块无法进行作用域提升优化
						buildInfo.moduleConcatenationBailout = "eval()";
						// 标记当前模块使用了 eval
						buildInfo.usingEval = true;

						// 获取当前作用域的顶层符号
						const currentSymbol = InnerGraph.getTopLevelSymbol(parser.state);
						if (currentSymbol) {
							// 如果获取到了符号,则记录其引用关系
							InnerGraph.addUsage(parser.state, null, currentSymbol);
						} else {
							// 否则标记当前模块无法进行内联图分析(bailout)
							InnerGraph.bailout(parser.state);
						}
					});

					// 在解析完成时记录顶层声明
					parser.hooks.finish.tap(PLUGIN_NAME, () => {
						const buildInfo =
							/** @type {BuildInfo} */
							(parser.state.module.buildInfo);

						let topLevelDeclarations = buildInfo.topLevelDeclarations;
						if (topLevelDeclarations === undefined) {
							// 初始化 topLevelDeclarations 为 Set 集合
							topLevelDeclarations = buildInfo.topLevelDeclarations = new Set();
						}

						// 遍历当前作用域中定义的变量名
						for (const name of parser.scope.definitions.asSet()) {
							const freeInfo = parser.getFreeInfoFromVariable(name);
							// 如果变量没有被视为自由变量,认为它是顶层声明
							if (freeInfo === undefined) {
								topLevelDeclarations.add(name);
							}
						}
					});
				};

				// 为三种 JS 模块类型分别注册 parser 钩子
				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);
			}
		);
	}
}

module.exports = JavascriptMetaInfoPlugin;
// 导出插件类
相关推荐
读心悦7 分钟前
CSS 溢出内容处理、可见性控制与盒类型设置深度解析
前端·css
Minyy1115 分钟前
Vue3指令(二)--v-text、v-html数据渲染,计算属性
前端·javascript·vue.js·前端框架·vue·html
个人开发-胡涂涂18 分钟前
ECMAScript标准:JavaScript的核心
前端·javascript·ecmascript
GISer_Jing23 分钟前
React底层架构深度解析:从虚拟DOM到Fiber的演进之路
前端·react.js·架构
斯密码赛我是美女31 分钟前
ssti刷刷刷
java·服务器·前端
Mryan200542 分钟前
Angular | 利用 `ChangeDetectorRef` 解决 Angular 动态显示输入框的聚焦问题
前端·javascript·angular.js
郭尘帅66644 分钟前
Vue3中实现轮播图
开发语言·前端·javascript
众乐乐_20081 小时前
Java 后端给前端传Long值,精度丢失的问题与解决
java·前端·状态模式
一叶茶1 小时前
VsCode和AI的前端使用体验:分别使用了Copilot、通义灵码、iflyCode和Trae
前端·vscode·gpt·ai·chatgpt·copilot·deepseek
熊猫钓鱼>_>1 小时前
基于MCP的桥梁设计规范智能解析与校审系统构建实践
前端·easyui·设计规范