mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			304 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			304 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const { ConcatSource } = require("webpack-sources");
 | |
| const { UsageState } = require("../ExportsInfo");
 | |
| const RuntimeGlobals = require("../RuntimeGlobals");
 | |
| const Template = require("../Template");
 | |
| const ConcatenatedModule = require("../optimize/ConcatenatedModule");
 | |
| const propertyAccess = require("../util/propertyAccess");
 | |
| const { getEntryRuntime } = require("../util/runtime");
 | |
| const AbstractLibraryPlugin = require("./AbstractLibraryPlugin");
 | |
| 
 | |
| /** @typedef {import("webpack-sources").Source} Source */
 | |
| /** @typedef {import("../../declarations/WebpackOptions").LibraryOptions} LibraryOptions */
 | |
| /** @typedef {import("../../declarations/WebpackOptions").LibraryType} LibraryType */
 | |
| /** @typedef {import("../../declarations/WebpackOptions").LibraryExport} LibraryExport */
 | |
| /** @typedef {import("../Chunk")} Chunk */
 | |
| /** @typedef {import("../Compiler")} Compiler */
 | |
| /** @typedef {import("../Module")} Module */
 | |
| /** @typedef {import("../Module").RuntimeRequirements} RuntimeRequirements */
 | |
| /** @typedef {import("../javascript/JavascriptModulesPlugin").StartupRenderContext} StartupRenderContext */
 | |
| /** @typedef {import("../javascript/JavascriptModulesPlugin").ModuleRenderContext} ModuleRenderContext */
 | |
| 
 | |
| /**
 | |
|  * @template T
 | |
|  * @typedef {import("./AbstractLibraryPlugin").LibraryContext<T>} LibraryContext<T>
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @typedef {object} ModuleLibraryPluginOptions
 | |
|  * @property {LibraryType} type
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @typedef {object} ModuleLibraryPluginParsed
 | |
|  * @property {string} name
 | |
|  * @property {LibraryExport=} export
 | |
|  */
 | |
| 
 | |
| const PLUGIN_NAME = "ModuleLibraryPlugin";
 | |
| 
 | |
| /**
 | |
|  * @typedef {ModuleLibraryPluginParsed} T
 | |
|  * @extends {AbstractLibraryPlugin<ModuleLibraryPluginParsed>}
 | |
|  */
 | |
| class ModuleLibraryPlugin extends AbstractLibraryPlugin {
 | |
| 	/**
 | |
| 	 * @param {ModuleLibraryPluginOptions} options the plugin options
 | |
| 	 */
 | |
| 	constructor(options) {
 | |
| 		super({
 | |
| 			pluginName: "ModuleLibraryPlugin",
 | |
| 			type: options.type
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Apply the plugin
 | |
| 	 * @param {Compiler} compiler the compiler instance
 | |
| 	 * @returns {void}
 | |
| 	 */
 | |
| 	apply(compiler) {
 | |
| 		super.apply(compiler);
 | |
| 
 | |
| 		compiler.hooks.thisCompilation.tap(PLUGIN_NAME, (compilation) => {
 | |
| 			const { onDemandExportsGeneration } =
 | |
| 				ConcatenatedModule.getCompilationHooks(compilation);
 | |
| 			onDemandExportsGeneration.tap(PLUGIN_NAME, (_module) => true);
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Module} module the exporting entry module
 | |
| 	 * @param {string} entryName the name of the entrypoint
 | |
| 	 * @param {LibraryContext<T>} libraryContext context
 | |
| 	 * @returns {void}
 | |
| 	 */
 | |
| 	finishEntryModule(
 | |
| 		module,
 | |
| 		entryName,
 | |
| 		{ options, compilation, compilation: { moduleGraph } }
 | |
| 	) {
 | |
| 		const runtime = getEntryRuntime(compilation, entryName);
 | |
| 		if (options.export) {
 | |
| 			const exportsInfo = moduleGraph.getExportInfo(
 | |
| 				module,
 | |
| 				Array.isArray(options.export) ? options.export[0] : options.export
 | |
| 			);
 | |
| 			exportsInfo.setUsed(UsageState.Used, runtime);
 | |
| 			exportsInfo.canMangleUse = false;
 | |
| 		} else {
 | |
| 			const exportsInfo = moduleGraph.getExportsInfo(module);
 | |
| 			// If the entry module is commonjs, its exports cannot be mangled
 | |
| 			if (module.buildMeta && module.buildMeta.treatAsCommonJs) {
 | |
| 				exportsInfo.setUsedInUnknownWay(runtime);
 | |
| 			} else {
 | |
| 				exportsInfo.setAllKnownExportsUsed(runtime);
 | |
| 			}
 | |
| 		}
 | |
| 		moduleGraph.addExtraReason(module, "used as library export");
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Chunk} chunk the chunk
 | |
| 	 * @param {RuntimeRequirements} set runtime requirements
 | |
| 	 * @param {LibraryContext<T>} libraryContext context
 | |
| 	 * @returns {void}
 | |
| 	 */
 | |
| 	runtimeRequirements(chunk, set, libraryContext) {
 | |
| 		set.add(RuntimeGlobals.exports);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {LibraryOptions} library normalized library option
 | |
| 	 * @returns {T | false} preprocess as needed by overriding
 | |
| 	 */
 | |
| 	parseOptions(library) {
 | |
| 		const { name } = library;
 | |
| 		if (name) {
 | |
| 			throw new Error(
 | |
| 				`Library name must be unset. ${AbstractLibraryPlugin.COMMON_LIBRARY_NAME_MESSAGE}`
 | |
| 			);
 | |
| 		}
 | |
| 		const _name = /** @type {string} */ (name);
 | |
| 		return {
 | |
| 			name: _name,
 | |
| 			export: library.export
 | |
| 		};
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Source} source source
 | |
| 	 * @param {Module} module module
 | |
| 	 * @param {StartupRenderContext} renderContext render context
 | |
| 	 * @param {LibraryContext<T>} libraryContext context
 | |
| 	 * @returns {Source} source with library export
 | |
| 	 */
 | |
| 	renderStartup(
 | |
| 		source,
 | |
| 		module,
 | |
| 		{
 | |
| 			moduleGraph,
 | |
| 			chunk,
 | |
| 			codeGenerationResults,
 | |
| 			inlined,
 | |
| 			inlinedInIIFE,
 | |
| 			runtimeTemplate
 | |
| 		},
 | |
| 		{ options, compilation }
 | |
| 	) {
 | |
| 		const result = new ConcatSource(source);
 | |
| 
 | |
| 		const exportsInfo = options.export
 | |
| 			? [
 | |
| 					moduleGraph.getExportInfo(
 | |
| 						module,
 | |
| 						Array.isArray(options.export) ? options.export[0] : options.export
 | |
| 					)
 | |
| 				]
 | |
| 			: moduleGraph.getExportsInfo(module).orderedExports;
 | |
| 		const definitions =
 | |
| 			inlined && !inlinedInIIFE
 | |
| 				? (module.buildMeta && module.buildMeta.exportsFinalName) || {}
 | |
| 				: {};
 | |
| 		/** @type {string[]} */
 | |
| 		const shortHandedExports = [];
 | |
| 		/** @type {[string, string][]} */
 | |
| 		const exports = [];
 | |
| 		const isAsync = moduleGraph.isAsync(module);
 | |
| 
 | |
| 		const treatAsCommonJs =
 | |
| 			module.buildMeta && module.buildMeta.treatAsCommonJs;
 | |
| 		const skipRenderDefaultExport = Boolean(treatAsCommonJs);
 | |
| 
 | |
| 		if (isAsync) {
 | |
| 			result.add(
 | |
| 				`${RuntimeGlobals.exports} = await ${RuntimeGlobals.exports};\n`
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		outer: for (const exportInfo of exportsInfo) {
 | |
| 			if (!exportInfo.provided) continue;
 | |
| 
 | |
| 			const originalName = exportInfo.name;
 | |
| 
 | |
| 			if (skipRenderDefaultExport && originalName === "default") continue;
 | |
| 
 | |
| 			const target = exportInfo.findTarget(moduleGraph, (_m) => true);
 | |
| 			if (target) {
 | |
| 				const reexportsInfo = moduleGraph.getExportsInfo(target.module);
 | |
| 
 | |
| 				for (const reexportInfo of reexportsInfo.orderedExports) {
 | |
| 					if (
 | |
| 						reexportInfo.provided === false &&
 | |
| 						reexportInfo.name !== "default" &&
 | |
| 						reexportInfo.name === /** @type {string[]} */ (target.export)[0]
 | |
| 					) {
 | |
| 						continue outer;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			const usedName =
 | |
| 				/** @type {string} */
 | |
| 				(exportInfo.getUsedName(originalName, chunk.runtime));
 | |
| 			/** @type {string | undefined} */
 | |
| 			const definition = definitions[usedName];
 | |
| 
 | |
| 			/** @type {string | undefined} */
 | |
| 			let finalName;
 | |
| 
 | |
| 			if (definition) {
 | |
| 				finalName = definition;
 | |
| 			} else {
 | |
| 				finalName = `${RuntimeGlobals.exports}${Template.toIdentifier(originalName)}`;
 | |
| 				result.add(
 | |
| 					`${runtimeTemplate.renderConst()} ${finalName} = ${RuntimeGlobals.exports}${propertyAccess(
 | |
| 						[usedName]
 | |
| 					)};\n`
 | |
| 				);
 | |
| 			}
 | |
| 
 | |
| 			if (
 | |
| 				finalName &&
 | |
| 				(finalName.includes(".") ||
 | |
| 					finalName.includes("[") ||
 | |
| 					finalName.includes("("))
 | |
| 			) {
 | |
| 				if (exportInfo.isReexport()) {
 | |
| 					const { data } = codeGenerationResults.get(module, chunk.runtime);
 | |
| 					const topLevelDeclarations =
 | |
| 						(data && data.get("topLevelDeclarations")) ||
 | |
| 						(module.buildInfo && module.buildInfo.topLevelDeclarations);
 | |
| 
 | |
| 					if (topLevelDeclarations && topLevelDeclarations.has(originalName)) {
 | |
| 						const name = `${RuntimeGlobals.exports}${Template.toIdentifier(originalName)}`;
 | |
| 						result.add(
 | |
| 							`${runtimeTemplate.renderConst()} ${name} = ${finalName};\n`
 | |
| 						);
 | |
| 						shortHandedExports.push(`${name} as ${originalName}`);
 | |
| 					} else {
 | |
| 						exports.push([originalName, finalName]);
 | |
| 					}
 | |
| 				} else {
 | |
| 					exports.push([originalName, finalName]);
 | |
| 				}
 | |
| 			} else {
 | |
| 				shortHandedExports.push(
 | |
| 					definition && finalName === originalName
 | |
| 						? finalName
 | |
| 						: `${finalName} as ${originalName}`
 | |
| 				);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (treatAsCommonJs) {
 | |
| 			shortHandedExports.push(`${RuntimeGlobals.exports} as default`);
 | |
| 		}
 | |
| 
 | |
| 		if (shortHandedExports.length > 0) {
 | |
| 			result.add(`export { ${shortHandedExports.join(", ")} };\n`);
 | |
| 		}
 | |
| 
 | |
| 		for (const [exportName, final] of exports) {
 | |
| 			result.add(
 | |
| 				`export ${runtimeTemplate.renderConst()} ${exportName} = ${final};\n`
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Source} source source
 | |
| 	 * @param {Module} module module
 | |
| 	 * @param {ModuleRenderContext} renderContext render context
 | |
| 	 * @param {Omit<LibraryContext<T>, 'options'>} libraryContext context
 | |
| 	 * @returns {Source} source with library export
 | |
| 	 */
 | |
| 	renderModuleContent(
 | |
| 		source,
 | |
| 		module,
 | |
| 		{ factory, inlinedInIIFE },
 | |
| 		libraryContext
 | |
| 	) {
 | |
| 		// Re-add `factoryExportsBinding` to the source
 | |
| 		// when the module is rendered as a factory or treated as an inlined (startup) module but wrapped in an IIFE
 | |
| 		if (
 | |
| 			(inlinedInIIFE || factory) &&
 | |
| 			module.buildMeta &&
 | |
| 			module.buildMeta.factoryExportsBinding
 | |
| 		) {
 | |
| 			return new ConcatSource(module.buildMeta.factoryExportsBinding, source);
 | |
| 		}
 | |
| 		return source;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| module.exports = ModuleLibraryPlugin;
 |