mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			195 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const util = require("util");
 | |
| const Watchpack = require("watchpack");
 | |
| 
 | |
| /** @typedef {import("../../declarations/WebpackOptions").WatchOptions} WatchOptions */
 | |
| /** @typedef {import("../FileSystemInfo").FileSystemInfoEntry} FileSystemInfoEntry */
 | |
| /** @typedef {import("../util/fs").WatchFileSystem} WatchFileSystem */
 | |
| /** @typedef {import("../util/fs").WatchMethod} WatchMethod */
 | |
| /** @typedef {import("../util/fs").Watcher} Watcher */
 | |
| 
 | |
| class NodeWatchFileSystem {
 | |
| 	constructor(inputFileSystem) {
 | |
| 		this.inputFileSystem = inputFileSystem;
 | |
| 		this.watcherOptions = {
 | |
| 			aggregateTimeout: 0
 | |
| 		};
 | |
| 		this.watcher = new Watchpack(this.watcherOptions);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Iterable<string>} files watched files
 | |
| 	 * @param {Iterable<string>} directories watched directories
 | |
| 	 * @param {Iterable<string>} missing watched exitance entries
 | |
| 	 * @param {number} startTime timestamp of start time
 | |
| 	 * @param {WatchOptions} options options object
 | |
| 	 * @param {function(Error=, Map<string, FileSystemInfoEntry>, Map<string, FileSystemInfoEntry>, Set<string>, Set<string>): void} callback aggregated callback
 | |
| 	 * @param {function(string, number): void} callbackUndelayed callback when the first change was detected
 | |
| 	 * @returns {Watcher} a watcher
 | |
| 	 */
 | |
| 	watch(
 | |
| 		files,
 | |
| 		directories,
 | |
| 		missing,
 | |
| 		startTime,
 | |
| 		options,
 | |
| 		callback,
 | |
| 		callbackUndelayed
 | |
| 	) {
 | |
| 		if (!files || typeof files[Symbol.iterator] !== "function") {
 | |
| 			throw new Error("Invalid arguments: 'files'");
 | |
| 		}
 | |
| 		if (!directories || typeof directories[Symbol.iterator] !== "function") {
 | |
| 			throw new Error("Invalid arguments: 'directories'");
 | |
| 		}
 | |
| 		if (!missing || typeof missing[Symbol.iterator] !== "function") {
 | |
| 			throw new Error("Invalid arguments: 'missing'");
 | |
| 		}
 | |
| 		if (typeof callback !== "function") {
 | |
| 			throw new Error("Invalid arguments: 'callback'");
 | |
| 		}
 | |
| 		if (typeof startTime !== "number" && startTime) {
 | |
| 			throw new Error("Invalid arguments: 'startTime'");
 | |
| 		}
 | |
| 		if (typeof options !== "object") {
 | |
| 			throw new Error("Invalid arguments: 'options'");
 | |
| 		}
 | |
| 		if (typeof callbackUndelayed !== "function" && callbackUndelayed) {
 | |
| 			throw new Error("Invalid arguments: 'callbackUndelayed'");
 | |
| 		}
 | |
| 		const oldWatcher = this.watcher;
 | |
| 		this.watcher = new Watchpack(options);
 | |
| 
 | |
| 		if (callbackUndelayed) {
 | |
| 			this.watcher.once("change", callbackUndelayed);
 | |
| 		}
 | |
| 
 | |
| 		const fetchTimeInfo = () => {
 | |
| 			const fileTimeInfoEntries = new Map();
 | |
| 			const contextTimeInfoEntries = new Map();
 | |
| 			if (this.watcher) {
 | |
| 				this.watcher.collectTimeInfoEntries(
 | |
| 					fileTimeInfoEntries,
 | |
| 					contextTimeInfoEntries
 | |
| 				);
 | |
| 			}
 | |
| 			return { fileTimeInfoEntries, contextTimeInfoEntries };
 | |
| 		};
 | |
| 		this.watcher.once("aggregated", (changes, removals) => {
 | |
| 			// pause emitting events (avoids clearing aggregated changes and removals on timeout)
 | |
| 			this.watcher.pause();
 | |
| 
 | |
| 			if (this.inputFileSystem && this.inputFileSystem.purge) {
 | |
| 				const fs = this.inputFileSystem;
 | |
| 				for (const item of changes) {
 | |
| 					fs.purge(item);
 | |
| 				}
 | |
| 				for (const item of removals) {
 | |
| 					fs.purge(item);
 | |
| 				}
 | |
| 			}
 | |
| 			const { fileTimeInfoEntries, contextTimeInfoEntries } = fetchTimeInfo();
 | |
| 			callback(
 | |
| 				null,
 | |
| 				fileTimeInfoEntries,
 | |
| 				contextTimeInfoEntries,
 | |
| 				changes,
 | |
| 				removals
 | |
| 			);
 | |
| 		});
 | |
| 
 | |
| 		this.watcher.watch({ files, directories, missing, startTime });
 | |
| 
 | |
| 		if (oldWatcher) {
 | |
| 			oldWatcher.close();
 | |
| 		}
 | |
| 		return {
 | |
| 			close: () => {
 | |
| 				if (this.watcher) {
 | |
| 					this.watcher.close();
 | |
| 					this.watcher = null;
 | |
| 				}
 | |
| 			},
 | |
| 			pause: () => {
 | |
| 				if (this.watcher) {
 | |
| 					this.watcher.pause();
 | |
| 				}
 | |
| 			},
 | |
| 			getAggregatedRemovals: util.deprecate(
 | |
| 				() => {
 | |
| 					const items = this.watcher && this.watcher.aggregatedRemovals;
 | |
| 					if (items && this.inputFileSystem && this.inputFileSystem.purge) {
 | |
| 						const fs = this.inputFileSystem;
 | |
| 						for (const item of items) {
 | |
| 							fs.purge(item);
 | |
| 						}
 | |
| 					}
 | |
| 					return items;
 | |
| 				},
 | |
| 				"Watcher.getAggregatedRemovals is deprecated in favor of Watcher.getInfo since that's more performant.",
 | |
| 				"DEP_WEBPACK_WATCHER_GET_AGGREGATED_REMOVALS"
 | |
| 			),
 | |
| 			getAggregatedChanges: util.deprecate(
 | |
| 				() => {
 | |
| 					const items = this.watcher && this.watcher.aggregatedChanges;
 | |
| 					if (items && this.inputFileSystem && this.inputFileSystem.purge) {
 | |
| 						const fs = this.inputFileSystem;
 | |
| 						for (const item of items) {
 | |
| 							fs.purge(item);
 | |
| 						}
 | |
| 					}
 | |
| 					return items;
 | |
| 				},
 | |
| 				"Watcher.getAggregatedChanges is deprecated in favor of Watcher.getInfo since that's more performant.",
 | |
| 				"DEP_WEBPACK_WATCHER_GET_AGGREGATED_CHANGES"
 | |
| 			),
 | |
| 			getFileTimeInfoEntries: util.deprecate(
 | |
| 				() => {
 | |
| 					return fetchTimeInfo().fileTimeInfoEntries;
 | |
| 				},
 | |
| 				"Watcher.getFileTimeInfoEntries is deprecated in favor of Watcher.getInfo since that's more performant.",
 | |
| 				"DEP_WEBPACK_WATCHER_FILE_TIME_INFO_ENTRIES"
 | |
| 			),
 | |
| 			getContextTimeInfoEntries: util.deprecate(
 | |
| 				() => {
 | |
| 					return fetchTimeInfo().contextTimeInfoEntries;
 | |
| 				},
 | |
| 				"Watcher.getContextTimeInfoEntries is deprecated in favor of Watcher.getInfo since that's more performant.",
 | |
| 				"DEP_WEBPACK_WATCHER_CONTEXT_TIME_INFO_ENTRIES"
 | |
| 			),
 | |
| 			getInfo: () => {
 | |
| 				const removals = this.watcher && this.watcher.aggregatedRemovals;
 | |
| 				const changes = this.watcher && this.watcher.aggregatedChanges;
 | |
| 				if (this.inputFileSystem && this.inputFileSystem.purge) {
 | |
| 					const fs = this.inputFileSystem;
 | |
| 					if (removals) {
 | |
| 						for (const item of removals) {
 | |
| 							fs.purge(item);
 | |
| 						}
 | |
| 					}
 | |
| 					if (changes) {
 | |
| 						for (const item of changes) {
 | |
| 							fs.purge(item);
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				const { fileTimeInfoEntries, contextTimeInfoEntries } = fetchTimeInfo();
 | |
| 				return {
 | |
| 					changes,
 | |
| 					removals,
 | |
| 					fileTimeInfoEntries,
 | |
| 					contextTimeInfoEntries
 | |
| 				};
 | |
| 			}
 | |
| 		};
 | |
| 	}
 | |
| }
 | |
| 
 | |
| module.exports = NodeWatchFileSystem;
 |