mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			163 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			163 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const { UsageState } = require("./ModuleGraph");
 | |
| const { STAGE_DEFAULT } = require("./OptimizationStages");
 | |
| const { NO_IMPORTED_NAMES } = require("./dependencies/DependencyReference");
 | |
| const Queue = require("./util/Queue");
 | |
| 
 | |
| /** @typedef {import("./Compiler")} Compiler */
 | |
| /** @typedef {import("./DependenciesBlock")} DependenciesBlock */
 | |
| /** @typedef {import("./Dependency")} Dependency */
 | |
| /** @typedef {import("./Module")} Module */
 | |
| /** @typedef {import("./ModuleGraph").ExportsInfo} ExportsInfo */
 | |
| 
 | |
| class FlagDependencyUsagePlugin {
 | |
| 	/**
 | |
| 	 * @param {Compiler} compiler the compiler instance
 | |
| 	 * @returns {void}
 | |
| 	 */
 | |
| 	apply(compiler) {
 | |
| 		compiler.hooks.compilation.tap("FlagDependencyUsagePlugin", compilation => {
 | |
| 			const moduleGraph = compilation.moduleGraph;
 | |
| 			compilation.hooks.optimizeDependencies.tap(
 | |
| 				{
 | |
| 					name: "FlagDependencyUsagePlugin",
 | |
| 					stage: STAGE_DEFAULT
 | |
| 				},
 | |
| 				modules => {
 | |
| 					/** @type {Map<ExportsInfo, Module>} */
 | |
| 					const exportInfoToModuleMap = new Map();
 | |
| 
 | |
| 					/**
 | |
| 					 * @typedef {string[]} StringArray
 | |
| 					 * @param {Module} module module to process
 | |
| 					 * @param {StringArray[]} usedExports list of used exports
 | |
| 					 * @returns {void}
 | |
| 					 */
 | |
| 					const processModule = (module, usedExports) => {
 | |
| 						const exportsInfo = moduleGraph.getExportsInfo(module);
 | |
| 						if (usedExports.length > 0) {
 | |
| 							for (const usedExport of usedExports) {
 | |
| 								if (usedExport.length === 0) {
 | |
| 									if (exportsInfo.setUsedInUnknownWay()) {
 | |
| 										queue.enqueue(module);
 | |
| 									}
 | |
| 								} else {
 | |
| 									if (
 | |
| 										usedExport[0] === "default" &&
 | |
| 										module.buildMeta.exportsType === "named"
 | |
| 									) {
 | |
| 										if (exportsInfo.setUsedAsNamedExportType()) {
 | |
| 											queue.enqueue(module);
 | |
| 										}
 | |
| 									} else {
 | |
| 										let currentExportsInfo = exportsInfo;
 | |
| 										for (let i = 0; i < usedExport.length; i++) {
 | |
| 											const exportName = usedExport[i];
 | |
| 											const exportInfo = currentExportsInfo.getExportInfo(
 | |
| 												exportName
 | |
| 											);
 | |
| 											const lastOne = i === usedExport.length - 1;
 | |
| 											const nestedInfo = exportInfo.exportsInfo;
 | |
| 											if (!nestedInfo || lastOne) {
 | |
| 												if (exportInfo.used !== UsageState.Used) {
 | |
| 													exportInfo.used = UsageState.Used;
 | |
| 													const currentModule =
 | |
| 														currentExportsInfo === exportsInfo
 | |
| 															? module
 | |
| 															: exportInfoToModuleMap.get(currentExportsInfo);
 | |
| 													if (currentModule) {
 | |
| 														queue.enqueue(currentModule);
 | |
| 													}
 | |
| 												}
 | |
| 												break;
 | |
| 											} else {
 | |
| 												if (exportInfo.used === UsageState.Unused) {
 | |
| 													exportInfo.used = UsageState.OnlyPropertiesUsed;
 | |
| 													const currentModule =
 | |
| 														currentExportsInfo === exportsInfo
 | |
| 															? module
 | |
| 															: exportInfoToModuleMap.get(currentExportsInfo);
 | |
| 													if (currentModule) {
 | |
| 														queue.enqueue(currentModule);
 | |
| 													}
 | |
| 												}
 | |
| 												currentExportsInfo = nestedInfo;
 | |
| 											}
 | |
| 										}
 | |
| 									}
 | |
| 								}
 | |
| 							}
 | |
| 						} else {
 | |
| 							// for a module without side effects we stop tracking usage here when no export is used
 | |
| 							// This module won't be evaluated in this case
 | |
| 							// TODO webpack 6 remove this check
 | |
| 							if (module.factoryMeta.sideEffectFree) return;
 | |
| 							if (exportsInfo.setUsedForSideEffectsOnly()) {
 | |
| 								queue.enqueue(module);
 | |
| 							}
 | |
| 						}
 | |
| 					};
 | |
| 
 | |
| 					/**
 | |
| 					 * @param {DependenciesBlock} depBlock the dependencies block
 | |
| 					 * @returns {void}
 | |
| 					 */
 | |
| 					const processDependenciesBlock = depBlock => {
 | |
| 						for (const dep of depBlock.dependencies) {
 | |
| 							processDependency(dep);
 | |
| 						}
 | |
| 						for (const block of depBlock.blocks) {
 | |
| 							queue.enqueue(block);
 | |
| 						}
 | |
| 					};
 | |
| 
 | |
| 					/**
 | |
| 					 * @param {Dependency} dep the dependency
 | |
| 					 * @returns {void}
 | |
| 					 */
 | |
| 					const processDependency = dep => {
 | |
| 						const reference = compilation.getDependencyReference(dep);
 | |
| 						if (!reference) return;
 | |
| 						const referenceModule = reference.module;
 | |
| 						const importedNames = reference.importedNames;
 | |
| 
 | |
| 						processModule(referenceModule, importedNames);
 | |
| 					};
 | |
| 
 | |
| 					for (const module of modules) {
 | |
| 						const exportsInfo = moduleGraph.getExportsInfo(module);
 | |
| 						exportInfoToModuleMap.set(exportsInfo, module);
 | |
| 						exportsInfo.setHasUseInfo();
 | |
| 					}
 | |
| 
 | |
| 					/** @type {Queue<DependenciesBlock>} */
 | |
| 					const queue = new Queue();
 | |
| 
 | |
| 					for (const deps of compilation.entryDependencies.values()) {
 | |
| 						for (const dep of deps) {
 | |
| 							const module = moduleGraph.getModule(dep);
 | |
| 							if (module) {
 | |
| 								processModule(module, NO_IMPORTED_NAMES);
 | |
| 								queue.enqueue(module);
 | |
| 							}
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					while (queue.length) {
 | |
| 						const depBlock = queue.dequeue();
 | |
| 						processDependenciesBlock(depBlock);
 | |
| 					}
 | |
| 				}
 | |
| 			);
 | |
| 		});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| module.exports = FlagDependencyUsagePlugin;
 |