mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			308 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			308 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const path = require("path");
 | |
| const {
 | |
| 	evaluateToIdentifier,
 | |
| 	evaluateToString,
 | |
| 	expressionIsUnsupported,
 | |
| 	toConstantDependency
 | |
| } = require("./JavascriptParserHelpers");
 | |
| const RuntimeGlobals = require("./RuntimeGlobals");
 | |
| const RuntimeModule = require("./RuntimeModule");
 | |
| const Template = require("./Template");
 | |
| const CachedConstDependency = require("./dependencies/CachedConstDependency");
 | |
| const ModuleDecoratorDependency = require("./dependencies/ModuleDecoratorDependency");
 | |
| 
 | |
| /** @typedef {import("webpack-sources").ReplaceSource} ReplaceSource */
 | |
| /** @typedef {import("./Compiler")} Compiler */
 | |
| /** @typedef {import("./Dependency")} Dependency */
 | |
| /** @typedef {import("./DependencyTemplates")} DependencyTemplates */
 | |
| /** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
 | |
| 
 | |
| class NodeStuffPlugin {
 | |
| 	constructor(options) {
 | |
| 		this.options = options;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Apply the plugin
 | |
| 	 * @param {Compiler} compiler the compiler instance
 | |
| 	 * @returns {void}
 | |
| 	 */
 | |
| 	apply(compiler) {
 | |
| 		const options = this.options;
 | |
| 		compiler.hooks.compilation.tap(
 | |
| 			"NodeStuffPlugin",
 | |
| 			(compilation, { normalModuleFactory }) => {
 | |
| 				compilation.dependencyFactories.set(
 | |
| 					ModuleDecoratorDependency,
 | |
| 					normalModuleFactory
 | |
| 				);
 | |
| 				compilation.dependencyTemplates.set(
 | |
| 					ModuleDecoratorDependency,
 | |
| 					new ModuleDecoratorDependency.Template()
 | |
| 				);
 | |
| 
 | |
| 				compilation.hooks.runtimeRequirementInModule
 | |
| 					.for(RuntimeGlobals.harmonyModuleDecorator)
 | |
| 					.tap("NodeStuffPlugin", (module, set) => {
 | |
| 						set.add(RuntimeGlobals.module);
 | |
| 						set.add(RuntimeGlobals.require);
 | |
| 					});
 | |
| 
 | |
| 				compilation.hooks.runtimeRequirementInModule
 | |
| 					.for(RuntimeGlobals.nodeModuleDecorator)
 | |
| 					.tap("NodeStuffPlugin", (module, set) => {
 | |
| 						set.add(RuntimeGlobals.module);
 | |
| 						set.add(RuntimeGlobals.require);
 | |
| 					});
 | |
| 
 | |
| 				compilation.hooks.runtimeRequirementInTree
 | |
| 					.for(RuntimeGlobals.harmonyModuleDecorator)
 | |
| 					.tap("NodeStuffPlugin", (chunk, set) => {
 | |
| 						compilation.addRuntimeModule(
 | |
| 							chunk,
 | |
| 							new HarmonyModuleDecoratorRuntimeModule()
 | |
| 						);
 | |
| 					});
 | |
| 
 | |
| 				compilation.hooks.runtimeRequirementInTree
 | |
| 					.for(RuntimeGlobals.nodeModuleDecorator)
 | |
| 					.tap("NodeStuffPlugin", (chunk, set) => {
 | |
| 						compilation.addRuntimeModule(
 | |
| 							chunk,
 | |
| 							new NodeModuleDecoratorRuntimeModule()
 | |
| 						);
 | |
| 					});
 | |
| 
 | |
| 				const handler = (parser, parserOptions) => {
 | |
| 					if (parserOptions.node === false) return;
 | |
| 
 | |
| 					let localOptions = options;
 | |
| 					if (parserOptions.node) {
 | |
| 						localOptions = Object.assign({}, localOptions, parserOptions.node);
 | |
| 					}
 | |
| 
 | |
| 					const setModuleConstant = (expressionName, fn) => {
 | |
| 						parser.hooks.expression
 | |
| 							.for(expressionName)
 | |
| 							.tap("NodeStuffPlugin", expr => {
 | |
| 								const dep = new CachedConstDependency(
 | |
| 									JSON.stringify(fn(parser.state.module)),
 | |
| 									expr.range,
 | |
| 									expressionName
 | |
| 								);
 | |
| 								dep.loc = expr.loc;
 | |
| 								parser.state.current.addDependency(dep);
 | |
| 								return true;
 | |
| 							});
 | |
| 					};
 | |
| 
 | |
| 					const setConstant = (expressionName, value) =>
 | |
| 						setModuleConstant(expressionName, () => value);
 | |
| 
 | |
| 					const context = compiler.context;
 | |
| 					if (localOptions.__filename === "mock") {
 | |
| 						setConstant("__filename", "/index.js");
 | |
| 					} else if (localOptions.__filename) {
 | |
| 						setModuleConstant("__filename", module =>
 | |
| 							path.relative(context, module.resource)
 | |
| 						);
 | |
| 					}
 | |
| 					parser.hooks.evaluateIdentifier
 | |
| 						.for("__filename")
 | |
| 						.tap("NodeStuffPlugin", expr => {
 | |
| 							if (!parser.state.module) return;
 | |
| 							const resource = parser.state.module.resource;
 | |
| 							const i = resource.indexOf("?");
 | |
| 							return evaluateToString(i < 0 ? resource : resource.substr(0, i))(
 | |
| 								expr
 | |
| 							);
 | |
| 						});
 | |
| 					if (localOptions.__dirname === "mock") {
 | |
| 						setConstant("__dirname", "/");
 | |
| 					} else if (localOptions.__dirname) {
 | |
| 						setModuleConstant("__dirname", module =>
 | |
| 							path.relative(context, module.context)
 | |
| 						);
 | |
| 					}
 | |
| 					parser.hooks.evaluateIdentifier
 | |
| 						.for("__dirname")
 | |
| 						.tap("NodeStuffPlugin", expr => {
 | |
| 							if (!parser.state.module) return;
 | |
| 							return evaluateToString(parser.state.module.context)(expr);
 | |
| 						});
 | |
| 					parser.hooks.expression
 | |
| 						.for("require.main")
 | |
| 						.tap(
 | |
| 							"NodeStuffPlugin",
 | |
| 							toConstantDependency(
 | |
| 								parser,
 | |
| 								`${RuntimeGlobals.moduleCache}[${
 | |
| 									RuntimeGlobals.entryModuleId
 | |
| 								}]`,
 | |
| 								[RuntimeGlobals.moduleCache, RuntimeGlobals.entryModuleId]
 | |
| 							)
 | |
| 						);
 | |
| 					parser.hooks.expression
 | |
| 						.for("require.extensions")
 | |
| 						.tap(
 | |
| 							"NodeStuffPlugin",
 | |
| 							expressionIsUnsupported(
 | |
| 								parser,
 | |
| 								"require.extensions is not supported by webpack. Use a loader instead."
 | |
| 							)
 | |
| 						);
 | |
| 					parser.hooks.expression
 | |
| 						.for("require.main.require")
 | |
| 						.tap(
 | |
| 							"NodeStuffPlugin",
 | |
| 							expressionIsUnsupported(
 | |
| 								parser,
 | |
| 								"require.main.require is not supported by webpack."
 | |
| 							)
 | |
| 						);
 | |
| 					parser.hooks.expression
 | |
| 						.for("module.parent.require")
 | |
| 						.tap(
 | |
| 							"NodeStuffPlugin",
 | |
| 							expressionIsUnsupported(
 | |
| 								parser,
 | |
| 								"module.parent.require is not supported by webpack."
 | |
| 							)
 | |
| 						);
 | |
| 					parser.hooks.expression
 | |
| 						.for("module.loaded")
 | |
| 						.tap("NodeStuffPlugin", expr => {
 | |
| 							parser.state.module.buildMeta.moduleConcatenationBailout =
 | |
| 								"module.loaded";
 | |
| 							return toConstantDependency(parser, "module.l", [
 | |
| 								RuntimeGlobals.module
 | |
| 							])(expr);
 | |
| 						});
 | |
| 					parser.hooks.expression
 | |
| 						.for("module.id")
 | |
| 						.tap("NodeStuffPlugin", expr => {
 | |
| 							parser.state.module.buildMeta.moduleConcatenationBailout =
 | |
| 								"module.id";
 | |
| 							return toConstantDependency(parser, "module.i", [
 | |
| 								RuntimeGlobals.module
 | |
| 							])(expr);
 | |
| 						});
 | |
| 					parser.hooks.expression
 | |
| 						.for("module.exports")
 | |
| 						.tap("NodeStuffPlugin", expr => {
 | |
| 							const module = parser.state.module;
 | |
| 							const isHarmony =
 | |
| 								module.buildMeta && module.buildMeta.exportsType;
 | |
| 							if (!isHarmony) {
 | |
| 								return toConstantDependency(parser, "module.exports", [
 | |
| 									RuntimeGlobals.module
 | |
| 								])(expr);
 | |
| 							}
 | |
| 						});
 | |
| 					parser.hooks.evaluateIdentifier
 | |
| 						.for("module.hot")
 | |
| 						.tap("NodeStuffPlugin", evaluateToIdentifier("module.hot", false));
 | |
| 					parser.hooks.expression.for("module").tap("NodeStuffPlugin", expr => {
 | |
| 						const isHarmony =
 | |
| 							parser.state.module.buildMeta &&
 | |
| 							parser.state.module.buildMeta.exportsType;
 | |
| 						const dep = new ModuleDecoratorDependency(
 | |
| 							isHarmony
 | |
| 								? RuntimeGlobals.harmonyModuleDecorator
 | |
| 								: RuntimeGlobals.nodeModuleDecorator
 | |
| 						);
 | |
| 						dep.loc = expr.loc;
 | |
| 						parser.state.module.addDependency(dep);
 | |
| 						return true;
 | |
| 					});
 | |
| 				};
 | |
| 
 | |
| 				normalModuleFactory.hooks.parser
 | |
| 					.for("javascript/auto")
 | |
| 					.tap("NodeStuffPlugin", handler);
 | |
| 				normalModuleFactory.hooks.parser
 | |
| 					.for("javascript/dynamic")
 | |
| 					.tap("NodeStuffPlugin", handler);
 | |
| 			}
 | |
| 		);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| class HarmonyModuleDecoratorRuntimeModule extends RuntimeModule {
 | |
| 	constructor() {
 | |
| 		super("harmony module decorator");
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @returns {string} runtime code
 | |
| 	 */
 | |
| 	generate() {
 | |
| 		return Template.asString([
 | |
| 			`${RuntimeGlobals.harmonyModuleDecorator} = function(module) {`,
 | |
| 			Template.indent([
 | |
| 				"module = Object.create(module);",
 | |
| 				"if (!module.children) module.children = [];",
 | |
| 				"Object.defineProperty(module, 'loaded', {",
 | |
| 				Template.indent([
 | |
| 					"enumerable: true,",
 | |
| 					"get: function () { return module.l; }"
 | |
| 				]),
 | |
| 				"});",
 | |
| 				"Object.defineProperty(module, 'id', {",
 | |
| 				Template.indent([
 | |
| 					"enumerable: true,",
 | |
| 					"get: function () { return module.i; }"
 | |
| 				]),
 | |
| 				"});",
 | |
| 				"Object.defineProperty(module, 'exports', {",
 | |
| 				Template.indent("enumerable: true"),
 | |
| 				"});",
 | |
| 				"return module;"
 | |
| 			]),
 | |
| 			"};"
 | |
| 		]);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| class NodeModuleDecoratorRuntimeModule extends RuntimeModule {
 | |
| 	constructor() {
 | |
| 		super("node module decorator");
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @returns {string} runtime code
 | |
| 	 */
 | |
| 	generate() {
 | |
| 		return Template.asString([
 | |
| 			`${RuntimeGlobals.nodeModuleDecorator} = function(module) {`,
 | |
| 			Template.indent([
 | |
| 				"module.paths = [];",
 | |
| 				"if (!module.children) module.children = [];",
 | |
| 				"Object.defineProperty(module, 'loaded', {",
 | |
| 				Template.indent([
 | |
| 					"enumerable: true,",
 | |
| 					"get: function() { return module.l; }"
 | |
| 				]),
 | |
| 				"});",
 | |
| 				"Object.defineProperty(module, 'id', {",
 | |
| 				Template.indent([
 | |
| 					"enumerable: true,",
 | |
| 					"get: function() { return module.i; }"
 | |
| 				]),
 | |
| 				"});",
 | |
| 				"return module;"
 | |
| 			]),
 | |
| 			"};"
 | |
| 		]);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| module.exports = NodeStuffPlugin;
 |