1
Fork 0

initial copy of code

This commit is contained in:
Conduitry 2018-04-02 16:33:41 -04:00
parent f1214ae8f4
commit 44ec2257ed
6 changed files with 173 additions and 0 deletions

7
.eslintrc.yaml Normal file
View file

@ -0,0 +1,7 @@
env:
es6: true
node: true
extends: 'eslint:recommended'
parserOptions:
ecmaVersion: 2017
sourceType: module

1
.gitignore vendored Normal file
View file

@ -0,0 +1 @@
/dist/

3
.prettierrc Normal file
View file

@ -0,0 +1,3 @@
useTabs: true
singleQuote: true
trailingComma: all

128
CheapWatch.js Normal file
View file

@ -0,0 +1,128 @@
import EventEmitter from 'events';
import { readdir, stat, watch } from 'fs';
import { promisify } from 'util';
const readdirAsync = promisify(readdir);
const statAsync = promisify(stat);
const _watchers = Symbol('_watchers');
const _stats = Symbol('_stats');
const _timeouts = Symbol('_timeouts');
const _queue = Symbol('_queue');
const _isProcessing = Symbol('_isProcessing');
const _recurse = Symbol('_recurse');
const _handle = Symbol('_handle');
const _enqueue = Symbol('_enqueue');
export default class CheapWatch extends EventEmitter {
constructor({ dir, filter, watch = true, debounce = 10 }) {
super();
this.dir = dir;
this.filter = filter;
this.watch = watch;
this.debounce = debounce;
this[_watchers] = new Map(); // paths of all directories -> FSWatcher instances
this[_stats] = new Map(); // paths of all files -> file stats
this[_timeouts] = new Map(); // paths of files with pending debounced events -> setTimeout timer ids
this[_queue] = []; // queue of pending FSWatcher events to handle
this[_isProcessing] = false; // whether some FSWatcher event is currently already in the process of being handled
}
// recurse directroy, get stats, set up FSWatcher instances
// returns array of { path, stats }
async init() {
await this[_recurse](this.dir);
return [...this[_stats].entries()].map(([path, stats]) => ({
path,
stats,
}));
}
// recurse a given directory
async [_recurse](full) {
const path = full.slice(this.dir.length + 1);
const stats = await statAsync(full);
if (this.filter && !await this.filter({ path, stats })) {
return;
}
if (stats.isFile()) {
this[_stats].set(path, stats);
} else if (stats.isDirectory()) {
if (this.watch) {
this[_watchers].set(path, watch(full, this[_handle].bind(this, full)));
}
await Promise.all(
(await readdirAsync(full)).map(sub => this[_recurse](full + '/' + sub)),
);
}
}
// handle FSWatcher event for given directory
[_handle](dir, event, file) {
const full = dir + '/' + file;
if (this[_timeouts].has(full)) {
clearTimeout(this[_timeouts].get(full));
}
this[_timeouts].set(
full,
setTimeout(() => {
this[_timeouts].delete(full);
this[_enqueue](full);
}, this.debounce),
);
}
// add an FSWatcher event to the queue, and handle queued events
async [_enqueue](full) {
this[_queue].push(full);
if (this[_isProcessing]) {
return;
}
this[_isProcessing] = true;
while (this[_queue].length) {
const full = this[_queue].shift();
const path = full.slice(this.dir.length + 1);
try {
const stats = await statAsync(full);
if (this.filter && !await this.filter({ path, stats })) {
continue;
}
if (stats.isFile()) {
// note the new/changed file
this[_stats].set(path, stats);
this.emit('+', { path, stats });
} else if (stats.isDirectory() && !this[_watchers].has(path)) {
// note the new directory: start watching it, and report any files in it
await this[_recurse](full);
for (const [newPath, stats] of this[_stats].entries()) {
if (newPath.startsWith(path + '/')) {
this.emit('+', { path: newPath, stats });
}
}
}
} catch (e) {
// probably this was a deleted file/directory
if (this[_stats].has(path)) {
// note the deleted file
this[_stats].delete(path);
this.emit('-', { path });
} else if (this[_watchers].has(path)) {
// note the deleted directory: stop watching it, and report any files that were in it
for (const old of this[_watchers].keys()) {
if (old === path || old.startsWith(path + '/')) {
this[_watchers].get(old).close();
this[_watchers].delete(old);
}
}
for (const old of this[_stats].keys()) {
if (old.startsWith(path + '/')) {
this[_stats].delete(old);
this.emit('-', { path: old });
}
}
}
}
}
this[_isProcessing] = false;
}
}

22
package.json Normal file
View file

@ -0,0 +1,22 @@
{
"name": "cheap-watch",
"version": "1.0.0",
"description": "If it works, why use something else?",
"keywords": ["async", "asynchronous", "file", "watch", "watcher"],
"main": "./dist/CheapWatch.cjs.js",
"module": "./CheapWatch.js",
"files": ["dist"],
"engines": {
"node": ">=8"
},
"repository": {
"type": "git",
"url": "https://github.com/Conduitry/cheap-watch.git"
},
"author": "Conduitry",
"license": "MIT",
"bugs": {
"url": "https://github.com/Conduitry/cheap-watch/issues"
},
"homepage": "https://cndtr.io/cheap-watch/"
}

12
rollup.config.js Normal file
View file

@ -0,0 +1,12 @@
export default {
input: './CheapWatch.js',
external: name => /^[-a-z]+$/.test(name),
output: [
{
file: './dist/CheapWatch.cjs.js',
format: 'cjs',
sourcemap: true,
interop: false,
},
],
};