| 
									
										
										
										
											2017-12-31 23:19:18 +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-12-31 23:19:18 +08:00
										 |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Stats = require("./Stats"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | /** @typedef {import("./Compiler")} Compiler */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-28 03:28:07 +08:00
										 |  |  | // TODO refactor watchpack to report timestamps in the correct format
 | 
					
						
							|  |  |  | const toFileSystemInfoEntryMap = timestamps => { | 
					
						
							|  |  |  | 	const map = new Map(); | 
					
						
							|  |  |  | 	for (const [key, ts] of timestamps) { | 
					
						
							|  |  |  | 		map.set(key, { safeTime: ts }); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return map; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | class Watching { | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Compiler} compiler the compiler | 
					
						
							|  |  |  | 	 * @param {TODO} watchOptions options | 
					
						
							|  |  |  | 	 * @param {TODO} handler TODO | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 	constructor(compiler, watchOptions, handler) { | 
					
						
							|  |  |  | 		this.startTime = null; | 
					
						
							|  |  |  | 		this.invalid = false; | 
					
						
							|  |  |  | 		this.handler = handler; | 
					
						
							|  |  |  | 		this.callbacks = []; | 
					
						
							|  |  |  | 		this.closed = false; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (typeof watchOptions === "number") { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.watchOptions = { | 
					
						
							|  |  |  | 				aggregateTimeout: watchOptions | 
					
						
							|  |  |  | 			}; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		} else if (watchOptions && typeof watchOptions === "object") { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.watchOptions = Object.assign({}, watchOptions); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			this.watchOptions = {}; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		this.watchOptions.aggregateTimeout = | 
					
						
							|  |  |  | 			this.watchOptions.aggregateTimeout || 200; | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 		this.compiler = compiler; | 
					
						
							|  |  |  | 		this.running = true; | 
					
						
							|  |  |  | 		this.compiler.readRecords(err => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			this._go(); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_go() { | 
					
						
							|  |  |  | 		this.startTime = Date.now(); | 
					
						
							|  |  |  | 		this.running = true; | 
					
						
							|  |  |  | 		this.invalid = false; | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 		this.compiler.cache.endIdle(err => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 			this.compiler.hooks.watchRun.callAsync(this.compiler, err => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 				const onCompiled = (err, compilation) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					if (err) return this._done(err); | 
					
						
							|  |  |  | 					if (this.invalid) return this._done(); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 					if (this.compiler.hooks.shouldEmit.call(compilation) === false) { | 
					
						
							|  |  |  | 						return this._done(null, compilation); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					this.compiler.emitAssets(compilation, err => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 						if (this.invalid) return this._done(); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 						this.compiler.emitRecords(err => { | 
					
						
							|  |  |  | 							if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 							if (compilation.hooks.needAdditionalPass.call()) { | 
					
						
							|  |  |  | 								compilation.needAdditionalPass = true; | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 								const stats = new Stats(compilation); | 
					
						
							|  |  |  | 								stats.startTime = this.startTime; | 
					
						
							|  |  |  | 								stats.endTime = Date.now(); | 
					
						
							|  |  |  | 								this.compiler.hooks.done.callAsync(stats, err => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									if (err) return this._done(err); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 									this.compiler.hooks.additionalPass.callAsync(err => { | 
					
						
							|  |  |  | 										if (err) return this._done(err); | 
					
						
							|  |  |  | 										this.compiler.compile(onCompiled); | 
					
						
							|  |  |  | 									}); | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 								}); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 								return; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							return this._done(null, compilation); | 
					
						
							|  |  |  | 						}); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 					}); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 				}; | 
					
						
							|  |  |  | 				this.compiler.compile(onCompiled); | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_getStats(compilation) { | 
					
						
							|  |  |  | 		const stats = new Stats(compilation); | 
					
						
							|  |  |  | 		stats.startTime = this.startTime; | 
					
						
							|  |  |  | 		stats.endTime = Date.now(); | 
					
						
							|  |  |  | 		return stats; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_done(err, compilation) { | 
					
						
							|  |  |  | 		this.running = false; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.invalid) return this._go(); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		const stats = compilation ? this._getStats(compilation) : null; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (err) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.compiler.hooks.failed.call(err); | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 			this.compiler.cache.beginIdle(); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.handler(err, stats); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 		this.compiler.hooks.done.callAsync(stats, () => { | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 			this.compiler.cache.beginIdle(); | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 			this.handler(null, stats); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			if (!this.closed) { | 
					
						
							|  |  |  | 				this.watch( | 
					
						
							|  |  |  | 					Array.from(compilation.fileDependencies), | 
					
						
							|  |  |  | 					Array.from(compilation.contextDependencies), | 
					
						
							|  |  |  | 					Array.from(compilation.missingDependencies) | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			for (const cb of this.callbacks) cb(); | 
					
						
							| 
									
										
										
										
											2018-01-25 20:56:50 +08:00
										 |  |  | 			this.callbacks.length = 0; | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	watch(files, dirs, missing) { | 
					
						
							|  |  |  | 		this.pausedWatcher = null; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		this.watcher = this.compiler.watchFileSystem.watch( | 
					
						
							|  |  |  | 			files, | 
					
						
							|  |  |  | 			dirs, | 
					
						
							|  |  |  | 			missing, | 
					
						
							|  |  |  | 			this.startTime, | 
					
						
							|  |  |  | 			this.watchOptions, | 
					
						
							|  |  |  | 			( | 
					
						
							|  |  |  | 				err, | 
					
						
							|  |  |  | 				filesModified, | 
					
						
							|  |  |  | 				contextModified, | 
					
						
							|  |  |  | 				missingModified, | 
					
						
							|  |  |  | 				fileTimestamps, | 
					
						
							| 
									
										
										
										
											2018-10-19 06:58:58 +08:00
										 |  |  | 				contextTimestamps, | 
					
						
							|  |  |  | 				removedFiles | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			) => { | 
					
						
							|  |  |  | 				this.pausedWatcher = this.watcher; | 
					
						
							|  |  |  | 				this.watcher = null; | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 				if (err) { | 
					
						
							|  |  |  | 					return this.handler(err); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-09-28 03:28:07 +08:00
										 |  |  | 				this.compiler.fileTimestamps = toFileSystemInfoEntryMap(fileTimestamps); | 
					
						
							|  |  |  | 				this.compiler.contextTimestamps = toFileSystemInfoEntryMap( | 
					
						
							|  |  |  | 					contextTimestamps | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2018-10-19 06:58:58 +08:00
										 |  |  | 				this.compiler.removedFiles = removedFiles; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				this._invalidate(); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			(fileName, changeTime) => { | 
					
						
							|  |  |  | 				this.compiler.hooks.invalid.call(fileName, changeTime); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	invalidate(callback) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (callback) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.callbacks.push(callback); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.watcher) { | 
					
						
							| 
									
										
										
										
											2018-09-28 03:28:07 +08:00
										 |  |  | 			this.compiler.fileTimestamps = toFileSystemInfoEntryMap( | 
					
						
							|  |  |  | 				this.watcher.getFileTimestamps() | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 			this.compiler.contextTimestamps = toFileSystemInfoEntryMap( | 
					
						
							|  |  |  | 				this.watcher.getContextTimestamps() | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2018-01-03 19:48:46 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return this._invalidate(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_invalidate() { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.watcher) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.pausedWatcher = this.watcher; | 
					
						
							|  |  |  | 			this.watcher.pause(); | 
					
						
							|  |  |  | 			this.watcher = null; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.running) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.invalid = true; | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			this._go(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	close(callback) { | 
					
						
							| 
									
										
										
										
											2018-03-07 17:15:46 +08:00
										 |  |  | 		const finalCallback = () => { | 
					
						
							|  |  |  | 			this.compiler.running = false; | 
					
						
							| 
									
										
										
										
											2018-10-24 01:52:32 +08:00
										 |  |  | 			this.compiler.watchMode = false; | 
					
						
							| 
									
										
										
										
											2018-09-28 03:28:07 +08:00
										 |  |  | 			this.compiler.fileTimestamps = undefined; | 
					
						
							|  |  |  | 			this.compiler.contextTimestamps = undefined; | 
					
						
							| 
									
										
										
										
											2018-10-25 02:10:30 +08:00
										 |  |  | 			this.compiler.removedFiles = undefined; | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 			this.compiler.cache.shutdown(err => { | 
					
						
							|  |  |  | 				this.compiler.hooks.watchClose.call(); | 
					
						
							|  |  |  | 				if (callback !== undefined) callback(err); | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2018-03-07 17:15:46 +08:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		this.closed = true; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.watcher) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.watcher.close(); | 
					
						
							|  |  |  | 			this.watcher = null; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.pausedWatcher) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.pausedWatcher.close(); | 
					
						
							|  |  |  | 			this.pausedWatcher = null; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.running) { | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 			this.invalid = true; | 
					
						
							| 
									
										
										
										
											2018-03-07 17:15:46 +08:00
										 |  |  | 			this._done = finalCallback; | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2018-03-07 17:15:46 +08:00
										 |  |  | 			finalCallback(); | 
					
						
							| 
									
										
										
										
											2017-12-31 23:19:18 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = Watching; |