2017-09-22 14:40:37 -07:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
var assert = require('assert'),
|
|
|
|
|
fs = require('fs'),
|
|
|
|
|
path = require('path'),
|
|
|
|
|
paths = require('./paths.js'),
|
|
|
|
|
safe = require('safetydance');
|
|
|
|
|
|
|
|
|
|
exports = module.exports = {
|
|
|
|
|
sync: sync
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function readCache(cacheFile) {
|
|
|
|
|
assert.strictEqual(typeof cacheFile, 'string');
|
|
|
|
|
|
|
|
|
|
var cache = safe.fs.readFileSync(cacheFile, 'utf8');
|
|
|
|
|
if (!cache) return [ ];
|
|
|
|
|
var result = cache.split('\n').map(JSON.parse);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function readTree(dir) {
|
|
|
|
|
assert.strictEqual(typeof dir, 'string');
|
|
|
|
|
|
|
|
|
|
var list = safe.fs.readdirSync(dir).sort();
|
|
|
|
|
if (!list) return [ ];
|
|
|
|
|
|
|
|
|
|
// TODO: handle lstat errors
|
|
|
|
|
return list.map(function (e) { return { stat: fs.lstatSync(path.join(dir, e)), name: e }; });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: concurrency
|
|
|
|
|
// TODO: if dir became a file, remove the dir first
|
|
|
|
|
function sync(dir, taskProcessor, callback) {
|
|
|
|
|
assert.strictEqual(typeof dir, 'string');
|
|
|
|
|
assert.strictEqual(typeof taskProcessor, 'function');
|
|
|
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
|
|
2017-09-26 07:57:20 -07:00
|
|
|
var curCacheIndex = 0;
|
|
|
|
|
var cacheFile = path.join(paths.SNAPSHOT_DIR, path.basename(dir) + '.cache'),
|
|
|
|
|
newCacheFile = path.join(paths.SNAPSHOT_DIR, path.basename(dir) + '.cache.new');
|
|
|
|
|
|
|
|
|
|
var cache = readCache(cacheFile);
|
|
|
|
|
|
|
|
|
|
var newCacheFd = fs.openSync(newCacheFile, 'w'); // truncates any existing file
|
2017-09-22 14:40:37 -07:00
|
|
|
|
|
|
|
|
var dummyCallback = function() { };
|
|
|
|
|
|
|
|
|
|
function advanceCache(entryPath) {
|
|
|
|
|
for (; curCacheIndex !== cache.length && (entryPath === '' || cache[curCacheIndex].path < entryPath); ++curCacheIndex) {
|
|
|
|
|
taskProcessor({ operation: 'remove', path: cache[curCacheIndex].path }, dummyCallback);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function traverse(relpath) {
|
|
|
|
|
var entries = readTree(path.join(dir, relpath));
|
|
|
|
|
|
|
|
|
|
for (var i = 0; i < entries.length; i++) {
|
|
|
|
|
var entryPath = path.join(relpath, entries[i].name);
|
|
|
|
|
|
|
|
|
|
if (entries[i].stat.isSymbolicLink()) continue;
|
|
|
|
|
|
|
|
|
|
if (entries[i].stat.isDirectory()) {
|
|
|
|
|
traverse(entryPath);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-26 07:57:20 -07:00
|
|
|
fs.appendFileSync(newCacheFd, JSON.stringify({ path: entryPath, mtime: entries[i].stat.mtime.getTime() }) + '\n');
|
2017-09-22 14:40:37 -07:00
|
|
|
|
|
|
|
|
advanceCache(entryPath);
|
|
|
|
|
|
|
|
|
|
if (curCacheIndex !== cache.length && cache[curCacheIndex].path === entryPath) {
|
|
|
|
|
if (entries[i].stat.mtime.getTime() !== cache[curCacheIndex].mtime) {
|
|
|
|
|
taskProcessor({ operation: 'add', path: entryPath }, dummyCallback);
|
|
|
|
|
}
|
|
|
|
|
++curCacheIndex;
|
|
|
|
|
} else {
|
|
|
|
|
taskProcessor({ operation: 'add', path: entryPath }, dummyCallback);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
traverse('');
|
|
|
|
|
advanceCache(''); // remove rest of the cache entries
|
|
|
|
|
|
2017-09-26 07:57:20 -07:00
|
|
|
// move the new cache file
|
|
|
|
|
fs.closeSync(newCacheFd);
|
|
|
|
|
fs.unlinkSync(cacheFile);
|
|
|
|
|
fs.renameSync(cacheFile, newCacheFd);
|
2017-09-22 14:40:37 -07:00
|
|
|
|
|
|
|
|
callback();
|
|
|
|
|
}
|