2020-04-07 16:12:08 +02:00
|
|
|
'use strict';
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
function _fs() {
|
|
|
|
const data = _interopRequireDefault(require('fs'));
|
2020-04-07 16:12:08 +02:00
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_fs = function _fs() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
function _path() {
|
|
|
|
const data = _interopRequireDefault(require('path'));
|
2020-04-07 16:12:08 +02:00
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_path = function _path() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _events() {
|
|
|
|
const data = require('events');
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_events = function _events() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _anymatch() {
|
|
|
|
const data = _interopRequireDefault(require('anymatch'));
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_anymatch = function _anymatch() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _micromatch() {
|
|
|
|
const data = _interopRequireDefault(require('micromatch'));
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_micromatch = function _micromatch() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _walker() {
|
|
|
|
const data = _interopRequireDefault(require('walker'));
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_walker = function _walker() {
|
2020-04-07 16:12:08 +02:00
|
|
|
return data;
|
|
|
|
};
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _interopRequireDefault(obj) {
|
|
|
|
return obj && obj.__esModule ? obj : {default: obj};
|
|
|
|
}
|
|
|
|
|
|
|
|
function _defineProperty(obj, key, value) {
|
|
|
|
if (key in obj) {
|
|
|
|
Object.defineProperty(obj, key, {
|
|
|
|
value: value,
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
writable: true
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
obj[key] = value;
|
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
let fsevents;
|
2020-04-07 16:12:08 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
fsevents = require('fsevents');
|
|
|
|
} catch (e) {
|
|
|
|
// Optional dependency, only supported on Darwin.
|
|
|
|
}
|
|
|
|
|
|
|
|
const CHANGE_EVENT = 'change';
|
|
|
|
const DELETE_EVENT = 'delete';
|
|
|
|
const ADD_EVENT = 'add';
|
|
|
|
const ALL_EVENT = 'all';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Export `FSEventsWatcher` class.
|
|
|
|
* Watches `dir`.
|
|
|
|
*/
|
|
|
|
class FSEventsWatcher extends _events().EventEmitter {
|
|
|
|
static isSupported() {
|
2020-04-20 17:17:11 +02:00
|
|
|
return fsevents !== undefined;
|
2020-04-07 16:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static normalizeProxy(callback) {
|
2020-04-20 17:17:11 +02:00
|
|
|
return (filepath, stats) =>
|
|
|
|
callback(_path().default.normalize(filepath), stats);
|
2020-04-07 16:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static recReaddir(
|
|
|
|
dir,
|
|
|
|
dirCallback,
|
|
|
|
fileCallback,
|
|
|
|
endCallback,
|
|
|
|
errorCallback,
|
|
|
|
ignored
|
|
|
|
) {
|
|
|
|
(0, _walker().default)(dir)
|
|
|
|
.filterDir(
|
|
|
|
currentDir => !ignored || !(0, _anymatch().default)(ignored, currentDir)
|
|
|
|
)
|
|
|
|
.on('dir', FSEventsWatcher.normalizeProxy(dirCallback))
|
|
|
|
.on('file', FSEventsWatcher.normalizeProxy(fileCallback))
|
|
|
|
.on('error', errorCallback)
|
|
|
|
.on('end', () => {
|
|
|
|
endCallback();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(dir, opts) {
|
|
|
|
if (!fsevents) {
|
|
|
|
throw new Error(
|
|
|
|
'`fsevents` unavailable (this watcher can only be used on Darwin)'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
super();
|
|
|
|
|
|
|
|
_defineProperty(this, 'root', void 0);
|
|
|
|
|
|
|
|
_defineProperty(this, 'ignored', void 0);
|
|
|
|
|
|
|
|
_defineProperty(this, 'glob', void 0);
|
|
|
|
|
|
|
|
_defineProperty(this, 'dot', void 0);
|
|
|
|
|
|
|
|
_defineProperty(this, 'hasIgnore', void 0);
|
|
|
|
|
|
|
|
_defineProperty(this, 'doIgnore', void 0);
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_defineProperty(this, 'watcher', void 0);
|
2020-04-07 16:12:08 +02:00
|
|
|
|
|
|
|
_defineProperty(this, '_tracked', void 0);
|
|
|
|
|
|
|
|
this.dot = opts.dot || false;
|
|
|
|
this.ignored = opts.ignored;
|
|
|
|
this.glob = Array.isArray(opts.glob) ? opts.glob : [opts.glob];
|
|
|
|
this.hasIgnore =
|
|
|
|
Boolean(opts.ignored) && !(Array.isArray(opts) && opts.length > 0);
|
|
|
|
this.doIgnore = opts.ignored
|
|
|
|
? (0, _anymatch().default)(opts.ignored)
|
|
|
|
: () => false;
|
2020-04-20 17:17:11 +02:00
|
|
|
this.root = _path().default.resolve(dir);
|
|
|
|
this.watcher = fsevents(this.root);
|
|
|
|
this.watcher.start().on('change', this.handleEvent.bind(this));
|
2020-04-07 16:12:08 +02:00
|
|
|
this._tracked = new Set();
|
|
|
|
FSEventsWatcher.recReaddir(
|
|
|
|
this.root,
|
|
|
|
filepath => {
|
|
|
|
this._tracked.add(filepath);
|
|
|
|
},
|
|
|
|
filepath => {
|
|
|
|
this._tracked.add(filepath);
|
|
|
|
},
|
|
|
|
this.emit.bind(this, 'ready'),
|
|
|
|
this.emit.bind(this, 'error'),
|
|
|
|
this.ignored
|
|
|
|
);
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* End watching.
|
|
|
|
*/
|
|
|
|
|
|
|
|
close(callback) {
|
2020-04-20 17:17:11 +02:00
|
|
|
this.watcher.stop();
|
|
|
|
this.removeAllListeners();
|
2020-04-07 16:12:08 +02:00
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
if (typeof callback === 'function') {
|
|
|
|
process.nextTick(callback.bind(null, null, true));
|
|
|
|
}
|
2020-04-07 16:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
isFileIncluded(relativePath) {
|
|
|
|
if (this.doIgnore(relativePath)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.glob.length
|
2020-04-20 17:17:11 +02:00
|
|
|
? _micromatch().default.some(relativePath, this.glob, {
|
2020-04-07 16:12:08 +02:00
|
|
|
dot: this.dot
|
2020-04-20 17:17:11 +02:00
|
|
|
})
|
|
|
|
: this.dot || _micromatch().default.some(relativePath, '**/*');
|
2020-04-07 16:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
handleEvent(filepath) {
|
2020-04-20 17:17:11 +02:00
|
|
|
const relativePath = _path().default.relative(this.root, filepath);
|
2020-04-07 16:12:08 +02:00
|
|
|
|
|
|
|
if (!this.isFileIncluded(relativePath)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-20 17:17:11 +02:00
|
|
|
_fs().default.lstat(filepath, (error, stat) => {
|
2020-04-07 16:12:08 +02:00
|
|
|
if (error && error.code !== 'ENOENT') {
|
|
|
|
this.emit('error', error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
// Ignore files that aren't tracked and don't exist.
|
|
|
|
if (!this._tracked.has(filepath)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._emit(DELETE_EVENT, relativePath);
|
|
|
|
|
|
|
|
this._tracked.delete(filepath);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this._tracked.has(filepath)) {
|
|
|
|
this._emit(CHANGE_EVENT, relativePath, stat);
|
|
|
|
} else {
|
|
|
|
this._tracked.add(filepath);
|
|
|
|
|
|
|
|
this._emit(ADD_EVENT, relativePath, stat);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Emit events.
|
|
|
|
*/
|
|
|
|
|
|
|
|
_emit(type, file, stat) {
|
|
|
|
this.emit(type, file, this.root, stat);
|
|
|
|
this.emit(ALL_EVENT, type, file, this.root, stat);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = FSEventsWatcher;
|