384 lines
9.5 KiB
JavaScript
Raw Normal View History

2014-01-05 20:07:11 +04:00
var fs = require('fs');
var glob = require('glob');
var mm = require('minimatch');
var q = require('q');
var helper = require('./helper');
var log = require('./logger').create('watcher');
var createWinGlob = function(realGlob) {
return function(pattern, options, done) {
realGlob(pattern, options, function(err, results) {
done(err, results.map(helper.normalizeWinPath));
});
};
};
if (process.platform === 'win32') {
glob = createWinGlob(glob);
}
var File = function(path, mtime) {
// used for serving (processed path, eg some/file.coffee -> some/file.coffee.js)
this.path = path;
// original absolute path, id of the file
this.originalPath = path;
// where the content is stored (processed)
this.contentPath = path;
this.mtime = mtime;
this.isUrl = false;
};
var Url = function(path) {
this.path = path;
this.isUrl = true;
};
Url.prototype.toString = File.prototype.toString = function() {
return this.path;
};
var GLOB_OPTS = {
// globDebug: true,
cwd: '/'
};
var byPath = function(a, b) {
if (a.path > b.path) {
return 1;
}
if (a.path < b.path) {
return -1;
}
return 0;
};
// TODO(vojta): ignore changes (add/change/remove) when in the middle of refresh
// TODO(vojta): do not glob patterns that are watched (both on init and refresh)
var List = function(patterns, excludes, emitter, preprocess, batchInterval) {
var self = this;
var pendingDeferred;
var pendingTimeout;
var resolveFiles = function(buckets) {
var uniqueMap = {};
var files = {
served: [],
included: []
};
buckets.forEach(function(bucket, idx) {
bucket.sort(byPath).forEach(function(file) {
if (!uniqueMap[file.path]) {
if (patterns[idx].served) {
files.served.push(file);
}
if (patterns[idx].included) {
files.included.push(file);
}
uniqueMap[file.path] = true;
}
});
});
return files;
};
var resolveDeferred = function(files) {
if (pendingTimeout) {
clearTimeout(pendingTimeout);
}
pendingDeferred.resolve(files || resolveFiles(self.buckets));
pendingDeferred = pendingTimeout = null;
};
var fireEventAndDefer = function() {
if (pendingTimeout) {
clearTimeout(pendingTimeout);
}
if (!pendingDeferred) {
pendingDeferred = q.defer();
emitter.emit('file_list_modified', pendingDeferred.promise);
}
pendingTimeout = setTimeout(resolveDeferred, batchInterval);
};
// re-glob all the patterns
this.refresh = function() {
// TODO(vojta): cancel refresh if another refresh starts
var buckets = self.buckets = new Array(patterns.length);
var complete = function() {
if (buckets !== self.buckets) {
return;
}
var files = resolveFiles(buckets);
resolveDeferred(files);
log.debug('Resolved files:\n\t' + files.served.join('\n\t'));
};
// TODO(vojta): use some async helper library for this
var pending = 0;
var finish = function() {
pending--;
if (!pending) {
complete();
}
};
if (!pendingDeferred) {
pendingDeferred = q.defer();
emitter.emit('file_list_modified', pendingDeferred.promise);
}
if (pendingTimeout) {
clearTimeout(pendingTimeout);
}
patterns.forEach(function(patternObject, i) {
var pattern = patternObject.pattern;
if (helper.isUrlAbsolute(pattern)) {
buckets[i] = [new Url(pattern)];
return;
}
pending++;
glob(pattern, GLOB_OPTS, function(err, resolvedFiles) {
var matchedAndNotIgnored = 0;
buckets[i] = [];
if (!resolvedFiles.length) {
log.warn('Pattern "%s" does not match any file.', pattern);
return finish();
}
// stat each file to get mtime and isDirectory
resolvedFiles.forEach(function(path) {
var matchExclude = function(excludePattern) {
return mm(path, excludePattern);
};
if (excludes.some(matchExclude)) {
log.debug('Excluded file "%s"', path);
return;
}
pending++;
matchedAndNotIgnored++;
fs.stat(path, function(error, stat) {
if (error) {
log.debug('An error occured while reading "%s"', path);
finish();
} else {
if (!stat.isDirectory()) {
// TODO(vojta): reuse file objects
var file = new File(path, stat.mtime);
preprocess(file, function() {
buckets[i].push(file);
finish();
});
} else {
log.debug('Ignored directory "%s"', path);
finish();
}
}
});
});
if (!matchedAndNotIgnored) {
log.warn('All files matched by "%s" were excluded.', pattern);
}
finish();
});
});
if (!pending) {
process.nextTick(complete);
}
return pendingDeferred.promise;
};
// set new patterns and excludes
// and re-glob
this.reload = function(newPatterns, newExcludes) {
patterns = newPatterns;
excludes = newExcludes;
return this.refresh();
};
/**
* Adds a new file into the list (called by watcher)
* - ignore excluded files
* - ignore files that are already in the list
* - get mtime (by stat)
* - fires "file_list_modified"
*/
this.addFile = function(path, done) {
var buckets = this.buckets;
var i, j;
// sorry, this callback is just for easier testing
done = done || function() {};
// check excludes
for (i = 0; i < excludes.length; i++) {
if (mm(path, excludes[i])) {
log.debug('Add file "%s" ignored. Excluded by "%s".', path, excludes[i]);
return done();
}
}
for (i = 0; i < patterns.length; i++) {
if (mm(path, patterns[i].pattern)) {
for (j = 0; j < buckets[i].length; j++) {
if (buckets[i][j].originalPath === path) {
log.debug('Add file "%s" ignored. Already in the list.', path);
return done();
}
}
break;
}
}
if (i >= patterns.length) {
log.debug('Add file "%s" ignored. Does not match any pattern.', path);
return done();
}
var addedFile = new File(path);
buckets[i].push(addedFile);
return fs.stat(path, function(err, stat) {
// in the case someone refresh() the list before stat callback
if (self.buckets === buckets) {
addedFile.mtime = stat.mtime;
return preprocess(addedFile, function() {
// TODO(vojta): ignore if refresh/reload happens
log.info('Added file "%s".', path);
fireEventAndDefer();
done();
});
}
return done();
});
};
/**
* Update mtime of a file (called by watcher)
* - ignore if file is not in the list
* - ignore if mtime has not changed
* - fire "file_list_modified"
*/
this.changeFile = function(path, done) {
var buckets = this.buckets;
var i, j;
// sorry, this callback is just for easier testing
done = done || function() {};
outer:
for (i = 0; i < buckets.length; i++) {
for (j = 0; j < buckets[i].length; j++) {
if (buckets[i][j].originalPath === path) {
break outer;
}
}
}
if (!buckets[i]) {
log.debug('Changed file "%s" ignored. Does not match any file in the list.', path);
return done();
}
var changedFile = buckets[i][j];
return fs.stat(path, function(err, stat) {
// https://github.com/paulmillr/chokidar/issues/11
if (err || !stat) {
return self.removeFile(path, done);
}
if (self.buckets === buckets && stat.mtime > changedFile.mtime) {
log.info('Changed file "%s".', path);
changedFile.mtime = stat.mtime;
// TODO(vojta): THIS CAN MAKE FILES INCONSISTENT
// if batched change is resolved before preprocessing is finished, the file can be in
// inconsistent state, when the promise is resolved.
// Solutions:
// 1/ the preprocessor should not change the object in place, but create a copy that would
// be eventually merged into the original file, here in the callback, synchronously.
// 2/ delay the promise resolution - wait for any changeFile operations to finish
return preprocess(changedFile, function() {
// TODO(vojta): ignore if refresh/reload happens
fireEventAndDefer();
done();
});
}
return done();
});
};
/**
* Remove a file from the list (called by watcher)
* - ignore if file is not in the list
* - fire "file_list_modified"
*/
this.removeFile = function(path, done) {
var buckets = this.buckets;
// sorry, this callback is just for easier testing
done = done || function() {};
for (var i = 0; i < buckets.length; i++) {
for (var j = 0; j < buckets[i].length; j++) {
if (buckets[i][j].originalPath === path) {
buckets[i].splice(j, 1);
log.info('Removed file "%s".', path);
fireEventAndDefer();
return done();
}
}
}
log.debug('Removed file "%s" ignored. Does not match any file in the list.', path);
return done();
};
};
List.$inject = ['config.files', 'config.exclude', 'emitter', 'preprocess',
'config.autoWatchBatchDelay'];
// PUBLIC
exports.List = List;
exports.File = File;
exports.Url = Url;