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
|
|
|
|
2017-01-11 20:08:28 +08:00
|
|
|
"use strict";
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-01-11 20:08:28 +08:00
|
|
|
const Watchpack = require("watchpack");
|
2013-02-11 07:17:29 +08:00
|
|
|
|
2020-08-03 02:09:36 +08:00
|
|
|
/** @typedef {import("../../declarations/WebpackOptions").WatchOptions} WatchOptions */
|
2019-10-21 21:35:38 +08:00
|
|
|
/** @typedef {import("../FileSystemInfo").FileSystemInfoEntry} FileSystemInfoEntry */
|
2020-07-08 16:21:31 +08:00
|
|
|
/** @typedef {import("../util/fs").WatchFileSystem} WatchFileSystem */
|
|
|
|
/** @typedef {import("../util/fs").WatchMethod} WatchMethod */
|
|
|
|
/** @typedef {import("../util/fs").Watcher} Watcher */
|
2019-10-21 21:35:38 +08:00
|
|
|
|
2017-01-11 20:08:28 +08:00
|
|
|
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
|
|
|
|
2019-10-21 21:35:38 +08:00
|
|
|
/**
|
|
|
|
* @param {Iterable<string>} files watched files
|
2019-11-07 01:10:00 +08:00
|
|
|
* @param {Iterable<string>} directories watched directories
|
2019-10-21 21:35:38 +08:00
|
|
|
* @param {Iterable<string>} missing watched exitance entries
|
|
|
|
* @param {number} startTime timestamp of start time
|
2020-08-03 02:09:36 +08:00
|
|
|
* @param {WatchOptions} options options object
|
2019-10-21 21:35:38 +08:00
|
|
|
* @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
|
|
|
|
*/
|
2019-11-07 01:10:00 +08:00
|
|
|
watch(
|
|
|
|
files,
|
|
|
|
directories,
|
|
|
|
missing,
|
|
|
|
startTime,
|
|
|
|
options,
|
|
|
|
callback,
|
|
|
|
callbackUndelayed
|
|
|
|
) {
|
2019-10-21 21:35:38 +08:00
|
|
|
if (!files || typeof files[Symbol.iterator] !== "function") {
|
2018-05-29 20:50:40 +08:00
|
|
|
throw new Error("Invalid arguments: 'files'");
|
|
|
|
}
|
2019-11-07 01:10:00 +08:00
|
|
|
if (!directories || typeof directories[Symbol.iterator] !== "function") {
|
|
|
|
throw new Error("Invalid arguments: 'directories'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
2019-10-21 21:35:38 +08:00
|
|
|
if (!missing || typeof missing[Symbol.iterator] !== "function") {
|
2017-01-11 20:08:28 +08:00
|
|
|
throw new Error("Invalid arguments: 'missing'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
|
|
|
if (typeof callback !== "function") {
|
2017-01-11 20:08:28 +08:00
|
|
|
throw new Error("Invalid arguments: 'callback'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
|
|
|
if (typeof startTime !== "number" && startTime) {
|
2017-01-11 20:08:28 +08:00
|
|
|
throw new Error("Invalid arguments: 'startTime'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
|
|
|
if (typeof options !== "object") {
|
2017-01-11 20:08:28 +08:00
|
|
|
throw new Error("Invalid arguments: 'options'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
|
|
|
if (typeof callbackUndelayed !== "function" && callbackUndelayed) {
|
2017-01-11 20:08:28 +08:00
|
|
|
throw new Error("Invalid arguments: 'callbackUndelayed'");
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
2017-01-11 20:08:28 +08:00
|
|
|
const oldWatcher = this.watcher;
|
|
|
|
this.watcher = new Watchpack(options);
|
2013-02-01 01:00:22 +08:00
|
|
|
|
2018-05-29 20:50:40 +08:00
|
|
|
if (callbackUndelayed) {
|
|
|
|
this.watcher.once("change", callbackUndelayed);
|
|
|
|
}
|
2021-11-18 03:43:54 +08:00
|
|
|
|
|
|
|
let fileMap, directoryMap;
|
2017-01-18 05:26:38 +08:00
|
|
|
this.watcher.once("aggregated", (changes, removals) => {
|
2018-02-25 09:00:20 +08:00
|
|
|
if (this.inputFileSystem && this.inputFileSystem.purge) {
|
2021-05-18 19:31:02 +08:00
|
|
|
const fs = this.inputFileSystem;
|
2019-01-05 21:58:06 +08:00
|
|
|
for (const item of changes) {
|
2021-05-18 19:31:02 +08:00
|
|
|
fs.purge(item);
|
2019-01-05 21:58:06 +08:00
|
|
|
}
|
|
|
|
for (const item of removals) {
|
2021-05-18 19:31:02 +08:00
|
|
|
fs.purge(item);
|
2019-01-05 21:58:06 +08:00
|
|
|
}
|
2017-01-11 20:08:28 +08:00
|
|
|
}
|
2021-11-18 03:43:54 +08:00
|
|
|
fileMap = new Map();
|
|
|
|
directoryMap = new Map();
|
|
|
|
this.watcher.getTimeInfoEntries(fileMap, directoryMap);
|
|
|
|
callback(null, fileMap, directoryMap, changes, removals);
|
2017-01-26 06:39:32 +08:00
|
|
|
});
|
2015-01-18 04:55:44 +08:00
|
|
|
|
2019-11-07 01:10:00 +08:00
|
|
|
this.watcher.watch({ files, directories, missing, startTime });
|
2015-01-18 04:55:44 +08:00
|
|
|
|
2018-02-25 09:00:20 +08:00
|
|
|
if (oldWatcher) {
|
2017-01-11 20:08:28 +08:00
|
|
|
oldWatcher.close();
|
|
|
|
}
|
|
|
|
return {
|
2017-01-26 06:39:32 +08:00
|
|
|
close: () => {
|
2018-02-25 09:00:20 +08:00
|
|
|
if (this.watcher) {
|
2017-02-22 17:45:49 +08:00
|
|
|
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) {
|
2017-02-22 17:45:49 +08:00
|
|
|
this.watcher.pause();
|
|
|
|
}
|
2018-01-03 19:48:46 +08:00
|
|
|
},
|
2021-02-25 03:12:32 +08:00
|
|
|
getAggregatedRemovals: () => {
|
2021-05-18 19:31:02 +08:00
|
|
|
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;
|
2021-02-25 03:12:32 +08:00
|
|
|
},
|
|
|
|
getAggregatedChanges: () => {
|
2021-05-18 19:31:02 +08:00
|
|
|
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;
|
2021-02-25 03:12:32 +08:00
|
|
|
},
|
2019-01-05 21:58:06 +08:00
|
|
|
getFileTimeInfoEntries: () => {
|
2021-11-18 03:43:54 +08:00
|
|
|
if (fileMap) return fileMap;
|
2018-05-29 20:50:40 +08:00
|
|
|
if (this.watcher) {
|
2021-11-18 03:46:43 +08:00
|
|
|
this.watcher.getTimeInfoEntries(
|
|
|
|
(fileMap = new Map()),
|
|
|
|
(directoryMap = new Map())
|
|
|
|
);
|
2021-11-18 03:43:54 +08:00
|
|
|
return fileMap;
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
2021-11-18 03:43:54 +08:00
|
|
|
return new Map();
|
2018-01-03 19:48:46 +08:00
|
|
|
},
|
2019-10-29 17:03:37 +08:00
|
|
|
getContextTimeInfoEntries: () => {
|
2021-11-18 03:43:54 +08:00
|
|
|
if (directoryMap) return directoryMap;
|
2018-05-29 20:50:40 +08:00
|
|
|
if (this.watcher) {
|
2021-11-18 03:46:43 +08:00
|
|
|
this.watcher.getTimeInfoEntries(
|
|
|
|
(fileMap = new Map()),
|
|
|
|
(directoryMap = new Map())
|
|
|
|
);
|
2021-11-18 03:43:54 +08:00
|
|
|
return directoryMap;
|
2018-05-29 20:50:40 +08:00
|
|
|
}
|
2021-11-18 03:43:54 +08:00
|
|
|
return new Map();
|
2017-01-26 06:39:32 +08:00
|
|
|
}
|
2017-01-11 20:08:28 +08:00
|
|
|
};
|
2013-02-01 01:00:22 +08:00
|
|
|
}
|
2017-01-11 20:08:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = NodeWatchFileSystem;
|