93 lines
2.5 KiB
JavaScript
93 lines
2.5 KiB
JavaScript
(function () {
|
|
"use strict"
|
|
|
|
// Array.prototype.forEachAsync(next, item, i, collection)
|
|
require('futures/forEachAsync');
|
|
|
|
function noop() {}
|
|
|
|
var fs = require('fs'),
|
|
EventEmitter = require('events').EventEmitter,
|
|
TypeEmitter = require('./node-type-emitter');
|
|
|
|
// 2010-11-25 jorge@jorgechamorro.com
|
|
function create(pathname, cb) {
|
|
var emitter = new EventEmitter(),
|
|
q = [],
|
|
queue = [q],
|
|
curpath;
|
|
|
|
function walk() {
|
|
fs.readdir(curpath, function(err, files) {
|
|
if (err) {
|
|
emitter.emit('directoryError', curpath, { error: err }, noop);
|
|
//emitter.emit('error', curpath, { error: err });
|
|
}
|
|
// XXX bug was here. next() was omitted
|
|
if (!files || 0 == files.length) {
|
|
return next();
|
|
}
|
|
|
|
var fnodeGroups = TypeEmitter.createNodeGroups();
|
|
|
|
// TODO could allow user to selectively stat
|
|
// and don't stat if there are no stat listeners
|
|
emitter.emit('names', curpath, files, noop);
|
|
files.forEachAsync(function (cont, file) {
|
|
emitter.emit('name', curpath, file, noop);
|
|
fs.lstat(curpath + '/' + file, function (err, stat) {
|
|
stat = stat || {};
|
|
stat.name = file;
|
|
if (err) {
|
|
stat.error = err;
|
|
//emitter.emit('error', curpath, stat);
|
|
emitter.emit('nodeError', curpath, stat, noop);
|
|
fnodeGroups.errors.push(stat);
|
|
cont();
|
|
} else {
|
|
TypeEmitter.sortFnodesByType(stat, fnodeGroups);
|
|
TypeEmitter.emitNodeType(emitter, curpath, stat, cont);
|
|
}
|
|
});
|
|
}).then(function () {
|
|
if (fnodeGroups.errors.length) {
|
|
emitter.emit('errors', curpath, fnodeGroups.errors, noop);
|
|
}
|
|
TypeEmitter.emitNodeTypeGroups(emitter, curpath, fnodeGroups, function () {
|
|
var dirs = [];
|
|
fnodeGroups.directories.forEach(function (stat) {
|
|
dirs.push(stat.name);
|
|
});
|
|
dirs.forEach(fullPath);
|
|
queue.push(q = dirs);
|
|
next();
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
function next() {
|
|
if (q.length) {
|
|
curpath = q.pop();
|
|
return walk();
|
|
}
|
|
if (queue.length -= 1) {
|
|
q = queue[queue.length-1];
|
|
return next();
|
|
}
|
|
emitter.emit('end');
|
|
}
|
|
|
|
function fullPath(v,i,o) {
|
|
o[i]= [curpath, '/', v].join('');
|
|
}
|
|
|
|
curpath = pathname;
|
|
walk();
|
|
|
|
return emitter;
|
|
}
|
|
|
|
module.exports = create;
|
|
}());
|