| 
									
										
										
										
											2013-06-19 19:49:57 +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-09 02:11:26 +08:00
										 |  |  | "use strict"; | 
					
						
							| 
									
										
										
										
											2018-03-22 19:05:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | const { SyncBailHook } = require("tapable"); | 
					
						
							|  |  |  | const { RawSource } = require("webpack-sources"); | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | const HotUpdateChunk = require("./HotUpdateChunk"); | 
					
						
							| 
									
										
										
										
											2018-07-09 14:41:59 +08:00
										 |  |  | const JavascriptParser = require("./JavascriptParser"); | 
					
						
							| 
									
										
										
										
											2018-07-03 16:24:29 +08:00
										 |  |  | const { | 
					
						
							|  |  |  | 	evaluateToIdentifier, | 
					
						
							|  |  |  | 	evaluateToString, | 
					
						
							|  |  |  | 	skipTraversal, | 
					
						
							|  |  |  | 	toConstantDependencyWithWebpackRequire | 
					
						
							| 
									
										
										
										
											2018-07-03 16:25:08 +08:00
										 |  |  | } = require("./JavascriptParserHelpers"); | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | const NullFactory = require("./NullFactory"); | 
					
						
							|  |  |  | const Template = require("./Template"); | 
					
						
							|  |  |  | const ConstDependency = require("./dependencies/ConstDependency"); | 
					
						
							|  |  |  | const ModuleHotAcceptDependency = require("./dependencies/ModuleHotAcceptDependency"); | 
					
						
							|  |  |  | const ModuleHotDeclineDependency = require("./dependencies/ModuleHotDeclineDependency"); | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | const { compareModulesById } = require("./util/comparators"); | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | /** @typedef {import("./Compiler")} Compiler */ | 
					
						
							|  |  |  | /** @typedef {import("./Module")} Module */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | const hotInitCode = Template.getFunctionContent( | 
					
						
							| 
									
										
										
										
											2018-07-11 17:13:47 +08:00
										 |  |  | 	require("./HotModuleReplacement.runtime") | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const parserHooksMap = new WeakMap(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | module.exports = class HotModuleReplacementPlugin { | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 	static getHooks(parser) { | 
					
						
							| 
									
										
										
										
											2018-07-09 14:41:59 +08:00
										 |  |  | 		if (!(parser instanceof JavascriptParser)) { | 
					
						
							|  |  |  | 			throw new TypeError( | 
					
						
							|  |  |  | 				"The 'parser' argument must be an instance of JavascriptParser" | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 		let hooks = parserHooksMap.get(parser); | 
					
						
							|  |  |  | 		if (hooks === undefined) { | 
					
						
							|  |  |  | 			hooks = { | 
					
						
							|  |  |  | 				hotAcceptCallback: new SyncBailHook(["expression", "requests"]), | 
					
						
							|  |  |  | 				hotAcceptWithoutCallback: new SyncBailHook(["expression", "requests"]) | 
					
						
							|  |  |  | 			}; | 
					
						
							|  |  |  | 			parserHooksMap.set(parser, hooks); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return hooks; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 	constructor(options) { | 
					
						
							|  |  |  | 		this.options = options || {}; | 
					
						
							|  |  |  | 		this.multiStep = this.options.multiStep; | 
					
						
							|  |  |  | 		this.fullBuildTimeout = this.options.fullBuildTimeout || 200; | 
					
						
							|  |  |  | 		this.requestTimeout = this.options.requestTimeout || 10000; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Compiler} compiler webpack compiler | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 	apply(compiler) { | 
					
						
							|  |  |  | 		const multiStep = this.multiStep; | 
					
						
							|  |  |  | 		const fullBuildTimeout = this.fullBuildTimeout; | 
					
						
							|  |  |  | 		const requestTimeout = this.requestTimeout; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const hotUpdateChunkFilename = | 
					
						
							|  |  |  | 			compiler.options.output.hotUpdateChunkFilename; | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 		const hotUpdateMainFilename = compiler.options.output.hotUpdateMainFilename; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		compiler.hooks.additionalPass.tapAsync( | 
					
						
							|  |  |  | 			"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 			callback => { | 
					
						
							|  |  |  | 				if (multiStep) return setTimeout(callback, fullBuildTimeout); | 
					
						
							|  |  |  | 				return callback(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		const addParserPlugins = (parser, parserOptions) => { | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 			const { | 
					
						
							|  |  |  | 				hotAcceptCallback, | 
					
						
							|  |  |  | 				hotAcceptWithoutCallback | 
					
						
							|  |  |  | 			} = HotModuleReplacementPlugin.getHooks(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 			parser.hooks.expression | 
					
						
							|  |  |  | 				.for("__webpack_hash__") | 
					
						
							|  |  |  | 				.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							| 
									
										
										
										
											2018-07-06 15:33:43 +08:00
										 |  |  | 					toConstantDependencyWithWebpackRequire( | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 						parser, | 
					
						
							|  |  |  | 						"__webpack_require__.h()" | 
					
						
							|  |  |  | 					) | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 			parser.hooks.evaluateTypeof | 
					
						
							|  |  |  | 				.for("__webpack_hash__") | 
					
						
							| 
									
										
										
										
											2018-07-06 15:33:43 +08:00
										 |  |  | 				.tap("HotModuleReplacementPlugin", evaluateToString("string")); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 			parser.hooks.evaluateIdentifier.for("module.hot").tap( | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					name: "HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					before: "NodeStuffPlugin" | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				expr => { | 
					
						
							| 
									
										
										
										
											2018-07-06 15:33:43 +08:00
										 |  |  | 					return evaluateToIdentifier( | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 						"module.hot", | 
					
						
							|  |  |  | 						!!parser.state.compilation.hotUpdateChunkTemplate | 
					
						
							|  |  |  | 					)(expr); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 			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) { | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 								hotAcceptCallback.call(expr.arguments[1], requests); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 								parser.walkExpression(expr.arguments[1]); // other args are ignored
 | 
					
						
							|  |  |  | 								return true; | 
					
						
							|  |  |  | 							} else { | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 								hotAcceptWithoutCallback.call(expr, requests); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 								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") | 
					
						
							| 
									
										
										
										
											2018-07-06 15:33:43 +08:00
										 |  |  | 				.tap("HotModuleReplacementPlugin", skipTraversal); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		compiler.hooks.compilation.tap( | 
					
						
							|  |  |  | 			"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 			(compilation, { normalModuleFactory }) => { | 
					
						
							| 
									
										
										
										
											2018-08-16 22:11:20 +08:00
										 |  |  | 				const moduleGraph = compilation.moduleGraph; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				const hotUpdateChunkTemplate = compilation.hotUpdateChunkTemplate; | 
					
						
							|  |  |  | 				if (!hotUpdateChunkTemplate) return; | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				compilation.dependencyFactories.set(ConstDependency, new NullFactory()); | 
					
						
							|  |  |  | 				compilation.dependencyTemplates.set( | 
					
						
							|  |  |  | 					ConstDependency, | 
					
						
							|  |  |  | 					new ConstDependency.Template() | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				compilation.dependencyFactories.set( | 
					
						
							|  |  |  | 					ModuleHotAcceptDependency, | 
					
						
							|  |  |  | 					normalModuleFactory | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 				compilation.dependencyTemplates.set( | 
					
						
							|  |  |  | 					ModuleHotAcceptDependency, | 
					
						
							|  |  |  | 					new ModuleHotAcceptDependency.Template() | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				compilation.dependencyFactories.set( | 
					
						
							|  |  |  | 					ModuleHotDeclineDependency, | 
					
						
							|  |  |  | 					normalModuleFactory | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 				compilation.dependencyTemplates.set( | 
					
						
							|  |  |  | 					ModuleHotDeclineDependency, | 
					
						
							|  |  |  | 					new ModuleHotDeclineDependency.Template() | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				compilation.hooks.record.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					(compilation, records) => { | 
					
						
							|  |  |  | 						if (records.hash === compilation.hash) return; | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 						const chunkGraph = compilation.chunkGraph; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						records.hash = compilation.hash; | 
					
						
							|  |  |  | 						records.moduleHashs = {}; | 
					
						
							|  |  |  | 						for (const module of compilation.modules) { | 
					
						
							|  |  |  | 							const identifier = module.identifier(); | 
					
						
							| 
									
										
										
										
											2018-08-23 01:23:48 +08:00
										 |  |  | 							records.moduleHashs[identifier] = chunkGraph.getModuleHash( | 
					
						
							|  |  |  | 								module | 
					
						
							|  |  |  | 							); | 
					
						
							| 
									
										
										
										
											2018-01-04 19:48:09 +08:00
										 |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						records.chunkHashs = {}; | 
					
						
							|  |  |  | 						for (const chunk of compilation.chunks) { | 
					
						
							|  |  |  | 							records.chunkHashs[chunk.id] = chunk.hash; | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						records.chunkModuleIds = {}; | 
					
						
							|  |  |  | 						for (const chunk of compilation.chunks) { | 
					
						
							|  |  |  | 							records.chunkModuleIds[chunk.id] = Array.from( | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 								chunkGraph.getOrderedChunkModulesIterable( | 
					
						
							|  |  |  | 									chunk, | 
					
						
							| 
									
										
										
										
											2018-08-28 17:50:33 +08:00
										 |  |  | 									compareModulesById(chunkGraph) | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 								), | 
					
						
							| 
									
										
										
										
											2018-08-28 17:56:48 +08:00
										 |  |  | 								m => chunkGraph.getModuleId(m) | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 				let initialPass = false; | 
					
						
							|  |  |  | 				let recompilation = false; | 
					
						
							|  |  |  | 				compilation.hooks.afterHash.tap("HotModuleReplacementPlugin", () => { | 
					
						
							|  |  |  | 					let records = compilation.records; | 
					
						
							|  |  |  | 					if (!records) { | 
					
						
							|  |  |  | 						initialPass = true; | 
					
						
							|  |  |  | 						return; | 
					
						
							| 
									
										
										
										
											2013-07-01 19:59:02 +08:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					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); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				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", | 
					
						
							|  |  |  | 					() => { | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 						const chunkGraph = compilation.chunkGraph; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						const records = compilation.records; | 
					
						
							|  |  |  | 						if (records.hash === compilation.hash) return; | 
					
						
							|  |  |  | 						if ( | 
					
						
							|  |  |  | 							!records.moduleHashs || | 
					
						
							|  |  |  | 							!records.chunkHashs || | 
					
						
							|  |  |  | 							!records.chunkModuleIds | 
					
						
							|  |  |  | 						) | 
					
						
							|  |  |  | 							return; | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | 						/** @type {Set<Module>} */ | 
					
						
							|  |  |  | 						const updatedModules = new Set(); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						for (const module of compilation.modules) { | 
					
						
							|  |  |  | 							const identifier = module.identifier(); | 
					
						
							| 
									
										
										
										
											2018-08-23 01:23:48 +08:00
										 |  |  | 							const hash = chunkGraph.getModuleHash(module); | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | 							if (records.moduleHashs[identifier] !== hash) { | 
					
						
							|  |  |  | 								updatedModules.add(module); | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						} | 
					
						
							|  |  |  | 						const hotUpdateMainContent = { | 
					
						
							|  |  |  | 							h: compilation.hash, | 
					
						
							|  |  |  | 							c: {} | 
					
						
							|  |  |  | 						}; | 
					
						
							| 
									
										
										
										
											2018-04-12 07:04:52 +08:00
										 |  |  | 						for (const key of Object.keys(records.chunkHashs)) { | 
					
						
							|  |  |  | 							const chunkId = isNaN(+key) ? key : +key; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							const currentChunk = compilation.chunks.find( | 
					
						
							|  |  |  | 								chunk => chunk.id === chunkId | 
					
						
							|  |  |  | 							); | 
					
						
							|  |  |  | 							if (currentChunk) { | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 								const newModules = chunkGraph | 
					
						
							|  |  |  | 									.getChunkModules(currentChunk) | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | 									.filter(module => updatedModules.has(module)); | 
					
						
							|  |  |  | 								/** @type {Set<number|string>} */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								const allModules = new Set(); | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 								for (const module of chunkGraph.getChunkModulesIterable( | 
					
						
							|  |  |  | 									currentChunk | 
					
						
							|  |  |  | 								)) { | 
					
						
							| 
									
										
										
										
											2018-08-28 17:56:48 +08:00
										 |  |  | 									allModules.add(chunkGraph.getModuleId(module)); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								} | 
					
						
							|  |  |  | 								const removedModules = records.chunkModuleIds[chunkId].filter( | 
					
						
							|  |  |  | 									id => !allModules.has(id) | 
					
						
							|  |  |  | 								); | 
					
						
							|  |  |  | 								if (newModules.length > 0 || removedModules.length > 0) { | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 									const hotUpdateChunk = new HotUpdateChunk(); | 
					
						
							|  |  |  | 									hotUpdateChunk.id = chunkId; | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 									chunkGraph.attachModules(hotUpdateChunk, newModules); | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 									hotUpdateChunk.removedModules = removedModules; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									const source = hotUpdateChunkTemplate.render( | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 										{ | 
					
						
							|  |  |  | 											chunk: hotUpdateChunk, | 
					
						
							|  |  |  | 											dependencyTemplates: compilation.dependencyTemplates, | 
					
						
							|  |  |  | 											runtimeTemplate: compilation.runtimeTemplate, | 
					
						
							| 
									
										
										
										
											2018-08-22 17:49:27 +08:00
										 |  |  | 											moduleGraph, | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 											chunkGraph: compilation.chunkGraph | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 										}, | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 										compilation.moduleTemplates.javascript, | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 										compilation.hash | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									); | 
					
						
							|  |  |  | 									const filename = compilation.getPath(hotUpdateChunkFilename, { | 
					
						
							|  |  |  | 										hash: records.hash, | 
					
						
							|  |  |  | 										chunk: currentChunk | 
					
						
							|  |  |  | 									}); | 
					
						
							|  |  |  | 									compilation.additionalChunkAssets.push(filename); | 
					
						
							|  |  |  | 									compilation.assets[filename] = source; | 
					
						
							|  |  |  | 									hotUpdateMainContent.c[chunkId] = true; | 
					
						
							|  |  |  | 									currentChunk.files.push(filename); | 
					
						
							| 
									
										
										
										
											2018-08-01 16:44:39 +08:00
										 |  |  | 									compilation.hooks.chunkAsset.call(currentChunk, filename); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} else { | 
					
						
							|  |  |  | 								hotUpdateMainContent.c[chunkId] = false; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						const source = new RawSource(JSON.stringify(hotUpdateMainContent)); | 
					
						
							|  |  |  | 						const filename = compilation.getPath(hotUpdateMainFilename, { | 
					
						
							|  |  |  | 							hash: records.hash | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 						compilation.assets[filename] = source; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				const mainTemplate = compilation.mainTemplate; | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.hash.tap("HotModuleReplacementPlugin", hash => { | 
					
						
							|  |  |  | 					hash.update("HotMainTemplateDecorator"); | 
					
						
							|  |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2014-08-25 15:50:26 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.moduleRequire.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					(_, chunk, hash, varModuleId) => { | 
					
						
							|  |  |  | 						return `hotCreateRequire(${varModuleId})`; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				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); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2018-01-20 00:06:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				const needChunkLoadingCode = chunk => { | 
					
						
							|  |  |  | 					for (const chunkGroup of chunk.groupsIterable) { | 
					
						
							|  |  |  | 						if (chunkGroup.chunks.length > 1) return true; | 
					
						
							|  |  |  | 						if (chunkGroup.getNumberOfChildren() > 0) return true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return false; | 
					
						
							|  |  |  | 				}; | 
					
						
							| 
									
										
										
										
											2014-04-20 03:35:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.bootstrap.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							| 
									
										
										
										
											2018-08-14 22:40:37 +08:00
										 |  |  | 					(source, moduleTemplate, { chunk, hash }) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						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)};` | 
					
						
							|  |  |  | 								) | 
					
						
							|  |  |  | 						]); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.globalHash.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					() => true | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.currentHash.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					(_, length) => { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 						if (isFinite(length)) { | 
					
						
							|  |  |  | 							return `hotCurrentHash.substr(0, ${length})`; | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							return "hotCurrentHash"; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				mainTemplate.hooks.moduleObj.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							| 
									
										
										
										
											2018-08-14 22:40:37 +08:00
										 |  |  | 					(source, varModuleId) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						return Template.asString([ | 
					
						
							|  |  |  | 							`${source},`, | 
					
						
							|  |  |  | 							`hot: hotCreateModule(${varModuleId}),`, | 
					
						
							|  |  |  | 							"parents: (hotCurrentParentsTemp = hotCurrentParents, hotCurrentParents = [], hotCurrentParentsTemp),", | 
					
						
							|  |  |  | 							"children: []" | 
					
						
							|  |  |  | 						]); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// TODO add HMR support for javascript/esm
 | 
					
						
							|  |  |  | 				normalModuleFactory.hooks.parser | 
					
						
							|  |  |  | 					.for("javascript/auto") | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 					.tap("HotModuleReplacementPlugin", addParserPlugins); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				normalModuleFactory.hooks.parser | 
					
						
							|  |  |  | 					.for("javascript/dynamic") | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 					.tap("HotModuleReplacementPlugin", addParserPlugins); | 
					
						
							| 
									
										
										
										
											2018-01-03 19:27:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				compilation.hooks.normalModuleLoader.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					context => { | 
					
						
							|  |  |  | 						context.hot = true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | }; |