webpack/lib/node/NodeWatchFileSystem.js

119 lines
3.4 KiB
JavaScript
Raw Normal View History

2013-01-31 01:49:25 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2018-07-30 23:08:51 +08:00
"use strict";
2013-01-31 01:49:25 +08:00
const Watchpack = require("watchpack");
2013-02-11 07:17:29 +08:00
/** @typedef {import("../FileSystemInfo").FileSystemInfoEntry} FileSystemInfoEntry */
/**
* @typedef {Object} Watcher
* @property {function(): void} close
* @property {function(): void} pause
* @property {function(): Map<string, FileSystemInfoEntry>} getFileTimeInfoEntries
* @property {function(): Map<string, FileSystemInfoEntry>} getContextTimeInfoEntries
*/
class NodeWatchFileSystem {
constructor(inputFileSystem) {
this.inputFileSystem = inputFileSystem;
this.watcherOptions = {
aggregateTimeout: 0
};
this.watcher = new Watchpack(this.watcherOptions);
}
2013-02-11 07:17:29 +08:00
/**
* @param {Iterable<string>} files watched files
* @param {Iterable<string>} dirs watched directories
* @param {Iterable<string>} missing watched exitance entries
* @param {number} startTime timestamp of start time
* @param {TODO} 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, dirs, missing, startTime, options, callback, callbackUndelayed) {
if (!files || typeof files[Symbol.iterator] !== "function") {
throw new Error("Invalid arguments: 'files'");
}
if (!dirs || typeof dirs[Symbol.iterator] !== "function") {
throw new Error("Invalid arguments: 'dirs'");
}
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);
2013-02-01 01:00:22 +08:00
if (callbackUndelayed) {
this.watcher.once("change", callbackUndelayed);
}
this.watcher.once("aggregated", (changes, removals) => {
2018-02-25 09:00:20 +08:00
if (this.inputFileSystem && this.inputFileSystem.purge) {
for (const item of changes) {
this.inputFileSystem.purge(item);
}
for (const item of removals) {
this.inputFileSystem.purge(item);
}
}
const times = this.watcher.getTimeInfoEntries();
callback(null, times, times, changes, removals);
2017-01-26 06:39:32 +08:00
});
const filesSet = new Set(files);
for (const item of missing) filesSet.add(item);
this.watcher.watch(filesSet, dirs, startTime);
2018-02-25 09:00:20 +08:00
if (oldWatcher) {
oldWatcher.close();
}
return {
2017-01-26 06:39:32 +08:00
close: () => {
2018-02-25 09:00:20 +08:00
if (this.watcher) {
this.watcher.close();
this.watcher = null;
}
2017-01-26 06:39:32 +08:00
},
pause: () => {
2018-02-25 09:00:20 +08:00
if (this.watcher) {
this.watcher.pause();
}
},
getFileTimeInfoEntries: () => {
if (this.watcher) {
return this.watcher.getTimeInfoEntries();
} else {
return new Map();
}
},
2019-10-29 17:03:37 +08:00
getContextTimeInfoEntries: () => {
if (this.watcher) {
return this.watcher.getTimeInfoEntries();
} else {
return new Map();
}
2017-01-26 06:39:32 +08:00
}
};
2013-02-01 01:00:22 +08:00
}
}
module.exports = NodeWatchFileSystem;