| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-30 00:14:40 +08:00
										 |  |  | const { SyncBailHook } = require("tapable"); | 
					
						
							| 
									
										
										
										
											2018-03-22 19:05:58 +08:00
										 |  |  | const { RawSource } = require("webpack-sources"); | 
					
						
							| 
									
										
										
										
											2019-12-16 20:21:39 +08:00
										 |  |  | const ChunkGraph = require("./ChunkGraph"); | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | const HotUpdateChunk = require("./HotUpdateChunk"); | 
					
						
							| 
									
										
										
										
											2018-11-12 21:13:55 +08:00
										 |  |  | const NormalModule = require("./NormalModule"); | 
					
						
							| 
									
										
										
										
											2018-11-05 21:36:15 +08:00
										 |  |  | const RuntimeGlobals = require("./RuntimeGlobals"); | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | const ModuleHotAcceptDependency = require("./dependencies/ModuleHotAcceptDependency"); | 
					
						
							|  |  |  | const ModuleHotDeclineDependency = require("./dependencies/ModuleHotDeclineDependency"); | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | const ModuleHotDependency = require("./dependencies/ModuleHotDependency"); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | const HotModuleReplacementRuntimeModule = require("./hmr/HotModuleReplacementRuntimeModule"); | 
					
						
							| 
									
										
										
										
											2019-10-11 21:46:57 +08:00
										 |  |  | const JavascriptParser = require("./javascript/JavascriptParser"); | 
					
						
							| 
									
										
										
										
											2019-10-22 15:27:52 +08:00
										 |  |  | const { | 
					
						
							|  |  |  | 	evaluateToIdentifier, | 
					
						
							|  |  |  | 	evaluateToString, | 
					
						
							|  |  |  | 	toConstantDependency | 
					
						
							|  |  |  | } = require("./javascript/JavascriptParserHelpers"); | 
					
						
							| 
									
										
										
										
											2018-09-06 22:59:11 +08:00
										 |  |  | const { find } = require("./util/SetHelpers"); | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | const { compareModulesById } = require("./util/comparators"); | 
					
						
							| 
									
										
										
										
											2013-06-19 19:49:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 21:26:06 +08:00
										 |  |  | /** @typedef {import("./Chunk")} Chunk */ | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | /** @typedef {import("./Compiler")} Compiler */ | 
					
						
							|  |  |  | /** @typedef {import("./Module")} Module */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 21:26:06 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @typedef {Object} HMRJavascriptParserHooks | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  |  * @property {SyncBailHook<[TODO, string[]], void>} hotAcceptCallback | 
					
						
							|  |  |  |  * @property {SyncBailHook<[TODO, string[]], void>} hotAcceptWithoutCallback | 
					
						
							| 
									
										
										
										
											2018-11-12 21:26:06 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** @type {WeakMap<JavascriptParser, HMRJavascriptParserHooks>} */ | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | const parserHooksMap = new WeakMap(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 00:54:54 +08:00
										 |  |  | class HotModuleReplacementPlugin { | 
					
						
							| 
									
										
										
										
											2018-11-12 21:26:06 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {JavascriptParser} parser the parser | 
					
						
							|  |  |  | 	 * @returns {HMRJavascriptParserHooks} the attached hooks | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2018-07-31 00:54:54 +08:00
										 |  |  | 	static getParserHooks(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; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2018-07-31 00:54:54 +08:00
										 |  |  | 			} = HotModuleReplacementPlugin.getParserHooks(parser); | 
					
						
							| 
									
										
										
										
											2018-07-06 18:51:14 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 			parser.hooks.expression | 
					
						
							|  |  |  | 				.for("__webpack_hash__") | 
					
						
							|  |  |  | 				.tap( | 
					
						
							|  |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							| 
									
										
										
										
											2018-11-06 02:03:12 +08:00
										 |  |  | 					toConstantDependency(parser, `${RuntimeGlobals.getFullHash}()`, [ | 
					
						
							|  |  |  | 						RuntimeGlobals.getFullHash | 
					
						
							|  |  |  | 					]) | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 				); | 
					
						
							|  |  |  | 			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 => { | 
					
						
							| 
									
										
										
										
											2019-08-29 21:28:19 +08:00
										 |  |  | 					return evaluateToIdentifier( | 
					
						
							|  |  |  | 						"module.hot", | 
					
						
							|  |  |  | 						"module", | 
					
						
							|  |  |  | 						() => ["hot"], | 
					
						
							|  |  |  | 						true | 
					
						
							|  |  |  | 					)(expr); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 			parser.hooks.call | 
					
						
							|  |  |  | 				.for("module.hot.accept") | 
					
						
							|  |  |  | 				.tap("HotModuleReplacementPlugin", expr => { | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | 					const dep = new ModuleHotDependency(expr.callee.range, "accept"); | 
					
						
							|  |  |  | 					dep.loc = expr.loc; | 
					
						
							|  |  |  | 					parser.state.module.addDependency(dep); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 					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; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | 					return true; | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 				}); | 
					
						
							|  |  |  | 			parser.hooks.call | 
					
						
							|  |  |  | 				.for("module.hot.decline") | 
					
						
							|  |  |  | 				.tap("HotModuleReplacementPlugin", expr => { | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | 					const dep = new ModuleHotDependency(expr.callee.range, "decline"); | 
					
						
							|  |  |  | 					dep.loc = expr.loc; | 
					
						
							|  |  |  | 					parser.state.module.addDependency(dep); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 					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); | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | 					return true; | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 				}); | 
					
						
							|  |  |  | 			parser.hooks.expression | 
					
						
							|  |  |  | 				.for("module.hot") | 
					
						
							| 
									
										
										
										
											2018-11-18 19:25:13 +08:00
										 |  |  | 				.tap("HotModuleReplacementPlugin", expr => { | 
					
						
							|  |  |  | 					const dep = new ModuleHotDependency(expr.range); | 
					
						
							|  |  |  | 					dep.loc = expr.loc; | 
					
						
							|  |  |  | 					parser.state.module.addDependency(dep); | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2018-07-05 17:25:34 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		compiler.hooks.compilation.tap( | 
					
						
							|  |  |  | 			"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 			(compilation, { normalModuleFactory }) => { | 
					
						
							| 
									
										
										
										
											2019-09-23 17:42:48 +08:00
										 |  |  | 				// This applies the HMR plugin only to the targeted compiler
 | 
					
						
							|  |  |  | 				// It should not affect child compilations
 | 
					
						
							|  |  |  | 				if (compilation.compiler !== compiler) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-11-18 19:25:13 +08:00
										 |  |  | 				compilation.dependencyTemplates.set( | 
					
						
							|  |  |  | 					ModuleHotDependency, | 
					
						
							|  |  |  | 					new ModuleHotDependency.Template() | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2019-11-15 17:20:45 +08:00
										 |  |  | 							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"; | 
					
						
							| 
									
										
										
										
											2020-03-13 17:13:55 +08:00
										 |  |  | 					const prePreHash = records.prePreHash || "x"; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					if (preHash === compilation.hash) { | 
					
						
							|  |  |  | 						recompilation = true; | 
					
						
							| 
									
										
										
										
											2020-03-13 17:13:55 +08:00
										 |  |  | 						compilation.modifyHash(prePreHash); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						return; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2020-03-13 17:13:55 +08:00
										 |  |  | 					records.prePreHash = records.hash || "x"; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					records.preHash = compilation.hash; | 
					
						
							| 
									
										
										
										
											2020-03-13 17:13:55 +08:00
										 |  |  | 					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, | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 							c: [], | 
					
						
							|  |  |  | 							r: [], | 
					
						
							|  |  |  | 							m: undefined | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						}; | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 						const allRemovedModules = new Set(); | 
					
						
							| 
									
										
										
										
											2018-04-12 07:04:52 +08:00
										 |  |  | 						for (const key of Object.keys(records.chunkHashs)) { | 
					
						
							| 
									
										
										
										
											2018-09-06 22:59:11 +08:00
										 |  |  | 							const currentChunk = find( | 
					
						
							|  |  |  | 								compilation.chunks, | 
					
						
							| 
									
										
										
										
											2018-10-30 15:49:48 +08:00
										 |  |  | 								chunk => `${chunk.id}` === key | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							); | 
					
						
							|  |  |  | 							if (currentChunk) { | 
					
						
							| 
									
										
										
										
											2020-01-15 17:43:58 +08:00
										 |  |  | 								const chunkId = currentChunk.id; | 
					
						
							| 
									
										
										
										
											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)); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 								const newRuntimeModules = Array.from( | 
					
						
							|  |  |  | 									chunkGraph.getChunkRuntimeModulesIterable(currentChunk) | 
					
						
							|  |  |  | 								).filter(module => updatedModules.has(module)); | 
					
						
							| 
									
										
										
										
											2018-07-20 22:24:35 +08:00
										 |  |  | 								/** @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(); | 
					
						
							| 
									
										
										
										
											2019-12-16 20:21:39 +08:00
										 |  |  | 									ChunkGraph.setChunkGraphForChunk(hotUpdateChunk, chunkGraph); | 
					
						
							| 
									
										
										
										
											2018-07-24 23:35:36 +08:00
										 |  |  | 									hotUpdateChunk.id = chunkId; | 
					
						
							| 
									
										
										
										
											2018-08-14 17:18:22 +08:00
										 |  |  | 									chunkGraph.attachModules(hotUpdateChunk, newModules); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 									chunkGraph.attachRuntimeModules( | 
					
						
							|  |  |  | 										hotUpdateChunk, | 
					
						
							|  |  |  | 										newRuntimeModules | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 									hotUpdateChunk.removedModules = removedModules; | 
					
						
							| 
									
										
										
										
											2019-10-02 14:54:21 +08:00
										 |  |  | 									const renderManifest = compilation.getRenderManifest({ | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 										chunk: hotUpdateChunk, | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 										hash: records.hash, | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 										fullHash: records.hash, | 
					
						
							| 
									
										
										
										
											2019-10-04 18:24:52 +08:00
										 |  |  | 										outputOptions: compilation.outputOptions, | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 										moduleTemplates: compilation.moduleTemplates, | 
					
						
							|  |  |  | 										dependencyTemplates: compilation.dependencyTemplates, | 
					
						
							| 
									
										
										
										
											2019-10-09 04:29:46 +08:00
										 |  |  | 										codeGenerationResults: compilation.codeGenerationResults, | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 										runtimeTemplate: compilation.runtimeTemplate, | 
					
						
							|  |  |  | 										moduleGraph: compilation.moduleGraph, | 
					
						
							|  |  |  | 										chunkGraph | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									}); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 									for (const entry of renderManifest) { | 
					
						
							| 
									
										
										
										
											2019-09-13 17:12:26 +08:00
										 |  |  | 										const { | 
					
						
							|  |  |  | 											path: filename, | 
					
						
							|  |  |  | 											info: assetInfo | 
					
						
							|  |  |  | 										} = compilation.getPathWithInfo( | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 											entry.filenameTemplate, | 
					
						
							|  |  |  | 											entry.pathOptions | 
					
						
							|  |  |  | 										); | 
					
						
							|  |  |  | 										const source = entry.render(); | 
					
						
							|  |  |  | 										compilation.additionalChunkAssets.push(filename); | 
					
						
							| 
									
										
										
										
											2019-09-13 17:12:26 +08:00
										 |  |  | 										compilation.emitAsset(filename, source, { | 
					
						
							|  |  |  | 											hotModuleReplacement: true, | 
					
						
							|  |  |  | 											...assetInfo | 
					
						
							|  |  |  | 										}); | 
					
						
							| 
									
										
										
										
											2019-06-14 16:45:56 +08:00
										 |  |  | 										currentChunk.files.add(filename); | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 										compilation.hooks.chunkAsset.call(currentChunk, filename); | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 									hotUpdateMainContent.c.push(chunkId); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} else { | 
					
						
							| 
									
										
										
										
											2020-01-15 17:43:58 +08:00
										 |  |  | 								const chunkId = `${+key}` === key ? +key : key; | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 								hotUpdateMainContent.r.push(chunkId); | 
					
						
							|  |  |  | 								for (const id of records.chunkModuleIds[chunkId]) | 
					
						
							|  |  |  | 									allRemovedModules.add(id); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 						hotUpdateMainContent.m = Array.from(allRemovedModules); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						const source = new RawSource(JSON.stringify(hotUpdateMainContent)); | 
					
						
							| 
									
										
										
										
											2019-09-11 17:13:46 +08:00
										 |  |  | 						const { | 
					
						
							|  |  |  | 							path: filename, | 
					
						
							|  |  |  | 							info: assetInfo | 
					
						
							| 
									
										
										
										
											2020-02-17 17:27:46 +08:00
										 |  |  | 						} = compilation.getPathWithInfo( | 
					
						
							|  |  |  | 							compilation.outputOptions.hotUpdateMainFilename, | 
					
						
							|  |  |  | 							{ | 
					
						
							|  |  |  | 								hash: records.hash | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						); | 
					
						
							| 
									
										
										
										
											2019-09-13 17:12:26 +08:00
										 |  |  | 						compilation.emitAsset(filename, source, { | 
					
						
							|  |  |  | 							hotModuleReplacement: true, | 
					
						
							|  |  |  | 							...assetInfo | 
					
						
							|  |  |  | 						}); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 				compilation.hooks.additionalTreeRuntimeRequirements.tap( | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							| 
									
										
										
										
											2018-11-28 20:07:40 +08:00
										 |  |  | 					(chunk, runtimeRequirements) => { | 
					
						
							|  |  |  | 						runtimeRequirements.add(RuntimeGlobals.hmrDownloadManifest); | 
					
						
							|  |  |  | 						runtimeRequirements.add(RuntimeGlobals.hmrDownloadUpdateHandlers); | 
					
						
							|  |  |  | 						runtimeRequirements.add(RuntimeGlobals.getFullHash); | 
					
						
							|  |  |  | 						runtimeRequirements.add(RuntimeGlobals.interceptModuleExecution); | 
					
						
							|  |  |  | 						runtimeRequirements.add(RuntimeGlobals.moduleCache); | 
					
						
							|  |  |  | 						compilation.addRuntimeModule( | 
					
						
							|  |  |  | 							chunk, | 
					
						
							|  |  |  | 							new HotModuleReplacementRuntimeModule() | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						); | 
					
						
							| 
									
										
										
										
											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-11-12 21:13:55 +08:00
										 |  |  | 				NormalModule.getCompilationHooks(compilation).loader.tap( | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					"HotModuleReplacementPlugin", | 
					
						
							|  |  |  | 					context => { | 
					
						
							|  |  |  | 						context.hot = true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2017-06-18 02:57:32 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-31 00:54:54 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = HotModuleReplacementPlugin; |