| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 | 
							- /*
 
- 	MIT License http://www.opensource.org/licenses/mit-license.php
 
- 	Author Tobias Koppers @sokra
 
- */
 
- "use strict";
 
- const { SyncBailHook } = require("tapable");
 
- const { RawSource } = require("webpack-sources");
 
- const Template = require("./Template");
 
- const ModuleHotAcceptDependency = require("./dependencies/ModuleHotAcceptDependency");
 
- const ModuleHotDeclineDependency = require("./dependencies/ModuleHotDeclineDependency");
 
- const ConstDependency = require("./dependencies/ConstDependency");
 
- const NullFactory = require("./NullFactory");
 
- const ParserHelpers = require("./ParserHelpers");
 
- module.exports = class HotModuleReplacementPlugin {
 
- 	constructor(options) {
 
- 		this.options = options || {};
 
- 		this.multiStep = this.options.multiStep;
 
- 		this.fullBuildTimeout = this.options.fullBuildTimeout || 200;
 
- 		this.requestTimeout = this.options.requestTimeout || 10000;
 
- 	}
 
- 	apply(compiler) {
 
- 		const multiStep = this.multiStep;
 
- 		const fullBuildTimeout = this.fullBuildTimeout;
 
- 		const requestTimeout = this.requestTimeout;
 
- 		const hotUpdateChunkFilename =
 
- 			compiler.options.output.hotUpdateChunkFilename;
 
- 		const hotUpdateMainFilename = compiler.options.output.hotUpdateMainFilename;
 
- 		compiler.hooks.additionalPass.tapAsync(
 
- 			"HotModuleReplacementPlugin",
 
- 			callback => {
 
- 				if (multiStep) return setTimeout(callback, fullBuildTimeout);
 
- 				return callback();
 
- 			}
 
- 		);
 
- 		const addParserPlugins = (parser, parserOptions) => {
 
- 			parser.hooks.expression
 
- 				.for("__webpack_hash__")
 
- 				.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					ParserHelpers.toConstantDependencyWithWebpackRequire(
 
- 						parser,
 
- 						"__webpack_require__.h()"
 
- 					)
 
- 				);
 
- 			parser.hooks.evaluateTypeof
 
- 				.for("__webpack_hash__")
 
- 				.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					ParserHelpers.evaluateToString("string")
 
- 				);
 
- 			parser.hooks.evaluateIdentifier.for("module.hot").tap(
 
- 				{
 
- 					name: "HotModuleReplacementPlugin",
 
- 					before: "NodeStuffPlugin"
 
- 				},
 
- 				expr => {
 
- 					return ParserHelpers.evaluateToIdentifier(
 
- 						"module.hot",
 
- 						!!parser.state.compilation.hotUpdateChunkTemplate
 
- 					)(expr);
 
- 				}
 
- 			);
 
- 			// TODO webpack 5: refactor this, no custom hooks
 
- 			if (!parser.hooks.hotAcceptCallback) {
 
- 				parser.hooks.hotAcceptCallback = new SyncBailHook([
 
- 					"expression",
 
- 					"requests"
 
- 				]);
 
- 			}
 
- 			if (!parser.hooks.hotAcceptWithoutCallback) {
 
- 				parser.hooks.hotAcceptWithoutCallback = new SyncBailHook([
 
- 					"expression",
 
- 					"requests"
 
- 				]);
 
- 			}
 
- 			parser.hooks.call
 
- 				.for("module.hot.accept")
 
- 				.tap("HotModuleReplacementPlugin", expr => {
 
- 					if (!parser.state.compilation.hotUpdateChunkTemplate) {
 
- 						return false;
 
- 					}
 
- 					if (expr.arguments.length >= 1) {
 
- 						const arg = parser.evaluateExpression(expr.arguments[0]);
 
- 						let params = [];
 
- 						let requests = [];
 
- 						if (arg.isString()) {
 
- 							params = [arg];
 
- 						} else if (arg.isArray()) {
 
- 							params = arg.items.filter(param => param.isString());
 
- 						}
 
- 						if (params.length > 0) {
 
- 							params.forEach((param, idx) => {
 
- 								const request = param.string;
 
- 								const dep = new ModuleHotAcceptDependency(request, param.range);
 
- 								dep.optional = true;
 
- 								dep.loc = Object.create(expr.loc);
 
- 								dep.loc.index = idx;
 
- 								parser.state.module.addDependency(dep);
 
- 								requests.push(request);
 
- 							});
 
- 							if (expr.arguments.length > 1) {
 
- 								parser.hooks.hotAcceptCallback.call(
 
- 									expr.arguments[1],
 
- 									requests
 
- 								);
 
- 								parser.walkExpression(expr.arguments[1]); // other args are ignored
 
- 								return true;
 
- 							} else {
 
- 								parser.hooks.hotAcceptWithoutCallback.call(expr, requests);
 
- 								return true;
 
- 							}
 
- 						}
 
- 					}
 
- 				});
 
- 			parser.hooks.call
 
- 				.for("module.hot.decline")
 
- 				.tap("HotModuleReplacementPlugin", expr => {
 
- 					if (!parser.state.compilation.hotUpdateChunkTemplate) {
 
- 						return false;
 
- 					}
 
- 					if (expr.arguments.length === 1) {
 
- 						const arg = parser.evaluateExpression(expr.arguments[0]);
 
- 						let params = [];
 
- 						if (arg.isString()) {
 
- 							params = [arg];
 
- 						} else if (arg.isArray()) {
 
- 							params = arg.items.filter(param => param.isString());
 
- 						}
 
- 						params.forEach((param, idx) => {
 
- 							const dep = new ModuleHotDeclineDependency(
 
- 								param.string,
 
- 								param.range
 
- 							);
 
- 							dep.optional = true;
 
- 							dep.loc = Object.create(expr.loc);
 
- 							dep.loc.index = idx;
 
- 							parser.state.module.addDependency(dep);
 
- 						});
 
- 					}
 
- 				});
 
- 			parser.hooks.expression
 
- 				.for("module.hot")
 
- 				.tap("HotModuleReplacementPlugin", ParserHelpers.skipTraversal);
 
- 		};
 
- 		compiler.hooks.compilation.tap(
 
- 			"HotModuleReplacementPlugin",
 
- 			(compilation, { normalModuleFactory }) => {
 
- 				// This applies the HMR plugin only to the targeted compiler
 
- 				// It should not affect child compilations
 
- 				if (compilation.compiler !== compiler) return;
 
- 				const hotUpdateChunkTemplate = compilation.hotUpdateChunkTemplate;
 
- 				if (!hotUpdateChunkTemplate) return;
 
- 				compilation.dependencyFactories.set(ConstDependency, new NullFactory());
 
- 				compilation.dependencyTemplates.set(
 
- 					ConstDependency,
 
- 					new ConstDependency.Template()
 
- 				);
 
- 				compilation.dependencyFactories.set(
 
- 					ModuleHotAcceptDependency,
 
- 					normalModuleFactory
 
- 				);
 
- 				compilation.dependencyTemplates.set(
 
- 					ModuleHotAcceptDependency,
 
- 					new ModuleHotAcceptDependency.Template()
 
- 				);
 
- 				compilation.dependencyFactories.set(
 
- 					ModuleHotDeclineDependency,
 
- 					normalModuleFactory
 
- 				);
 
- 				compilation.dependencyTemplates.set(
 
- 					ModuleHotDeclineDependency,
 
- 					new ModuleHotDeclineDependency.Template()
 
- 				);
 
- 				compilation.hooks.record.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					(compilation, records) => {
 
- 						if (records.hash === compilation.hash) return;
 
- 						records.hash = compilation.hash;
 
- 						records.moduleHashs = {};
 
- 						for (const module of compilation.modules) {
 
- 							const identifier = module.identifier();
 
- 							records.moduleHashs[identifier] = module.hash;
 
- 						}
 
- 						records.chunkHashs = {};
 
- 						for (const chunk of compilation.chunks) {
 
- 							records.chunkHashs[chunk.id] = chunk.hash;
 
- 						}
 
- 						records.chunkModuleIds = {};
 
- 						for (const chunk of compilation.chunks) {
 
- 							records.chunkModuleIds[chunk.id] = Array.from(
 
- 								chunk.modulesIterable,
 
- 								m => m.id
 
- 							);
 
- 						}
 
- 					}
 
- 				);
 
- 				let initialPass = false;
 
- 				let recompilation = false;
 
- 				compilation.hooks.afterHash.tap("HotModuleReplacementPlugin", () => {
 
- 					let records = compilation.records;
 
- 					if (!records) {
 
- 						initialPass = true;
 
- 						return;
 
- 					}
 
- 					if (!records.hash) initialPass = true;
 
- 					const preHash = records.preHash || "x";
 
- 					const prepreHash = records.prepreHash || "x";
 
- 					if (preHash === compilation.hash) {
 
- 						recompilation = true;
 
- 						compilation.modifyHash(prepreHash);
 
- 						return;
 
- 					}
 
- 					records.prepreHash = records.hash || "x";
 
- 					records.preHash = compilation.hash;
 
- 					compilation.modifyHash(records.prepreHash);
 
- 				});
 
- 				compilation.hooks.shouldGenerateChunkAssets.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					() => {
 
- 						if (multiStep && !recompilation && !initialPass) return false;
 
- 					}
 
- 				);
 
- 				compilation.hooks.needAdditionalPass.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					() => {
 
- 						if (multiStep && !recompilation && !initialPass) return true;
 
- 					}
 
- 				);
 
- 				compilation.hooks.additionalChunkAssets.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					() => {
 
- 						const records = compilation.records;
 
- 						if (records.hash === compilation.hash) return;
 
- 						if (
 
- 							!records.moduleHashs ||
 
- 							!records.chunkHashs ||
 
- 							!records.chunkModuleIds
 
- 						)
 
- 							return;
 
- 						for (const module of compilation.modules) {
 
- 							const identifier = module.identifier();
 
- 							let hash = module.hash;
 
- 							module.hotUpdate = records.moduleHashs[identifier] !== hash;
 
- 						}
 
- 						const hotUpdateMainContent = {
 
- 							h: compilation.hash,
 
- 							c: {}
 
- 						};
 
- 						for (const key of Object.keys(records.chunkHashs)) {
 
- 							const chunkId = isNaN(+key) ? key : +key;
 
- 							const currentChunk = compilation.chunks.find(
 
- 								chunk => `${chunk.id}` === key
 
- 							);
 
- 							if (currentChunk) {
 
- 								const newModules = currentChunk
 
- 									.getModules()
 
- 									.filter(module => module.hotUpdate);
 
- 								const allModules = new Set();
 
- 								for (const module of currentChunk.modulesIterable) {
 
- 									allModules.add(module.id);
 
- 								}
 
- 								const removedModules = records.chunkModuleIds[chunkId].filter(
 
- 									id => !allModules.has(id)
 
- 								);
 
- 								if (newModules.length > 0 || removedModules.length > 0) {
 
- 									const source = hotUpdateChunkTemplate.render(
 
- 										chunkId,
 
- 										newModules,
 
- 										removedModules,
 
- 										compilation.hash,
 
- 										compilation.moduleTemplates.javascript,
 
- 										compilation.dependencyTemplates
 
- 									);
 
- 									const {
 
- 										path: filename,
 
- 										info: assetInfo
 
- 									} = compilation.getPathWithInfo(hotUpdateChunkFilename, {
 
- 										hash: records.hash,
 
- 										chunk: currentChunk
 
- 									});
 
- 									compilation.additionalChunkAssets.push(filename);
 
- 									compilation.emitAsset(
 
- 										filename,
 
- 										source,
 
- 										Object.assign({ hotModuleReplacement: true }, assetInfo)
 
- 									);
 
- 									hotUpdateMainContent.c[chunkId] = true;
 
- 									currentChunk.files.push(filename);
 
- 									compilation.hooks.chunkAsset.call(currentChunk, filename);
 
- 								}
 
- 							} else {
 
- 								hotUpdateMainContent.c[chunkId] = false;
 
- 							}
 
- 						}
 
- 						const source = new RawSource(JSON.stringify(hotUpdateMainContent));
 
- 						const {
 
- 							path: filename,
 
- 							info: assetInfo
 
- 						} = compilation.getPathWithInfo(hotUpdateMainFilename, {
 
- 							hash: records.hash
 
- 						});
 
- 						compilation.emitAsset(
 
- 							filename,
 
- 							source,
 
- 							Object.assign({ hotModuleReplacement: true }, assetInfo)
 
- 						);
 
- 					}
 
- 				);
 
- 				const mainTemplate = compilation.mainTemplate;
 
- 				mainTemplate.hooks.hash.tap("HotModuleReplacementPlugin", hash => {
 
- 					hash.update("HotMainTemplateDecorator");
 
- 				});
 
- 				mainTemplate.hooks.moduleRequire.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					(_, chunk, hash, varModuleId) => {
 
- 						return `hotCreateRequire(${varModuleId})`;
 
- 					}
 
- 				);
 
- 				mainTemplate.hooks.requireExtensions.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					source => {
 
- 						const buf = [source];
 
- 						buf.push("");
 
- 						buf.push("// __webpack_hash__");
 
- 						buf.push(
 
- 							mainTemplate.requireFn +
 
- 								".h = function() { return hotCurrentHash; };"
 
- 						);
 
- 						return Template.asString(buf);
 
- 					}
 
- 				);
 
- 				const needChunkLoadingCode = chunk => {
 
- 					for (const chunkGroup of chunk.groupsIterable) {
 
- 						if (chunkGroup.chunks.length > 1) return true;
 
- 						if (chunkGroup.getNumberOfChildren() > 0) return true;
 
- 					}
 
- 					return false;
 
- 				};
 
- 				mainTemplate.hooks.bootstrap.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					(source, chunk, hash) => {
 
- 						source = mainTemplate.hooks.hotBootstrap.call(source, chunk, hash);
 
- 						return Template.asString([
 
- 							source,
 
- 							"",
 
- 							hotInitCode
 
- 								.replace(/\$require\$/g, mainTemplate.requireFn)
 
- 								.replace(/\$hash\$/g, JSON.stringify(hash))
 
- 								.replace(/\$requestTimeout\$/g, requestTimeout)
 
- 								.replace(
 
- 									/\/\*foreachInstalledChunks\*\//g,
 
- 									needChunkLoadingCode(chunk)
 
- 										? "for(var chunkId in installedChunks)"
 
- 										: `var chunkId = ${JSON.stringify(chunk.id)};`
 
- 								)
 
- 						]);
 
- 					}
 
- 				);
 
- 				mainTemplate.hooks.globalHash.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					() => true
 
- 				);
 
- 				mainTemplate.hooks.currentHash.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					(_, length) => {
 
- 						if (isFinite(length)) {
 
- 							return `hotCurrentHash.substr(0, ${length})`;
 
- 						} else {
 
- 							return "hotCurrentHash";
 
- 						}
 
- 					}
 
- 				);
 
- 				mainTemplate.hooks.moduleObj.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					(source, chunk, hash, varModuleId) => {
 
- 						return Template.asString([
 
- 							`${source},`,
 
- 							`hot: hotCreateModule(${varModuleId}),`,
 
- 							"parents: (hotCurrentParentsTemp = hotCurrentParents, hotCurrentParents = [], hotCurrentParentsTemp),",
 
- 							"children: []"
 
- 						]);
 
- 					}
 
- 				);
 
- 				// TODO add HMR support for javascript/esm
 
- 				normalModuleFactory.hooks.parser
 
- 					.for("javascript/auto")
 
- 					.tap("HotModuleReplacementPlugin", addParserPlugins);
 
- 				normalModuleFactory.hooks.parser
 
- 					.for("javascript/dynamic")
 
- 					.tap("HotModuleReplacementPlugin", addParserPlugins);
 
- 				compilation.hooks.normalModuleLoader.tap(
 
- 					"HotModuleReplacementPlugin",
 
- 					context => {
 
- 						context.hot = true;
 
- 					}
 
- 				);
 
- 			}
 
- 		);
 
- 	}
 
- };
 
- const hotInitCode = Template.getFunctionContent(
 
- 	require("./HotModuleReplacement.runtime")
 
- );
 
 
  |