| 
									
										
										
										
											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; |