| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  | // Adapted from work by jorge@jorgechamorro.com on 2010-11-25
 | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  | (function () { | 
					
						
							|  |  |  |   "use strict" | 
					
						
							| 
									
										
										
										
											2010-12-02 03:09:01 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |   // Array.prototype.forEachAsync(next, item, i, collection)
 | 
					
						
							| 
									
										
										
										
											2011-07-28 07:57:02 -06:00
										 |  |  |   require('Array.prototype.forEachAsync'); | 
					
						
							| 
									
										
										
										
											2010-12-02 03:09:01 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 23:57:11 -07:00
										 |  |  |   function noop() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |   var fs = require('fs'), | 
					
						
							|  |  |  |     EventEmitter = require('events').EventEmitter, | 
					
						
							|  |  |  |     TypeEmitter = require('./node-type-emitter'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |   function create(pathname, options, sync) { | 
					
						
							|  |  |  |     var emitter = new EventEmitter() | 
					
						
							|  |  |  |       , q = [] | 
					
						
							|  |  |  |       , queue = [q] | 
					
						
							|  |  |  |       , curpath; | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |     function readdirHandler(err, files) { | 
					
						
							|  |  |  |       var fnodeGroups = TypeEmitter.createNodeGroups(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       function filesHandler(cont, file) { | 
					
						
							|  |  |  |         var statPath; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         emitter.emit('name', curpath, file, noop); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         function lstatHandler(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); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-03 23:57:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |         statPath = curpath + '/' + file; | 
					
						
							|  |  |  |         if (sync) { | 
					
						
							|  |  |  |           try { | 
					
						
							|  |  |  |             lstatHandler(null, fs.lstatSync(statPath)); | 
					
						
							|  |  |  |           } catch(e) { | 
					
						
							|  |  |  |             lstatHandler(e); | 
					
						
							| 
									
										
										
										
											2011-02-03 23:57:11 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           fs.lstat(statPath, lstatHandler); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       function postFilesHandler() { | 
					
						
							|  |  |  |         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); | 
					
						
							| 
									
										
										
										
											2010-11-20 22:02:53 -07:00
										 |  |  |           }); | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |           dirs.forEach(fullPath); | 
					
						
							|  |  |  |           queue.push(q = dirs); | 
					
						
							|  |  |  |           next(); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (err) { | 
					
						
							|  |  |  |         emitter.emit('directoryError', curpath, { error: err }, noop); | 
					
						
							|  |  |  |         //emitter.emit('error', curpath, { error: err });
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!files || 0 == files.length) { | 
					
						
							|  |  |  |         return next(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // TODO could allow user to selectively stat
 | 
					
						
							|  |  |  |       // and don't stat if there are no stat listeners
 | 
					
						
							|  |  |  |       emitter.emit('names', curpath, files, noop); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (sync) { | 
					
						
							|  |  |  |         files.forEach(function (items) { | 
					
						
							|  |  |  |           filesHandler(noop, items); | 
					
						
							| 
									
										
										
										
											2010-12-12 01:58:43 -07:00
										 |  |  |         }); | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |         postFilesHandler(); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         files.forEachAsync(filesHandler).then(postFilesHandler); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function walkSync() { | 
					
						
							|  |  |  |       var err, files; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         files = fs.readdirSync(curpath); | 
					
						
							|  |  |  |       } catch(e) { | 
					
						
							|  |  |  |         err = e; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       readdirHandler(err, files); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function walk() {  | 
					
						
							|  |  |  |       if (sync) { | 
					
						
							|  |  |  |         walkSync(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       fs.readdir(curpath, readdirHandler); | 
					
						
							| 
									
										
										
										
											2010-11-20 22:02:53 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |      | 
					
						
							|  |  |  |     function next() { | 
					
						
							|  |  |  |       if (q.length) { | 
					
						
							|  |  |  |         curpath = q.pop(); | 
					
						
							|  |  |  |         return walk(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (queue.length -= 1) { | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |         q = queue[queue.length - 1]; | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |         return next(); | 
					
						
							| 
									
										
										
										
											2010-12-12 01:58:43 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2010-12-12 02:13:40 -07:00
										 |  |  |       emitter.emit('end'); | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |     function fullPath(v, i, o) { | 
					
						
							|  |  |  |       o[i] = [curpath, '/', v].join(''); | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     curpath = pathname; | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (sync) { | 
					
						
							|  |  |  |       process.nextTick(walk); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       walk(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-02 03:09:01 -07:00
										 |  |  |     return emitter; | 
					
						
							| 
									
										
										
										
											2010-11-20 22:02:53 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-02-03 23:31:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-02 21:11:03 -06:00
										 |  |  |   exports.walk = function (path, opts) { | 
					
						
							|  |  |  |     return create(path, opts, false); | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   exports.walkSync = function (path, opts) { | 
					
						
							|  |  |  |     return create(path, opts, true); | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2010-11-20 22:02:53 -07:00
										 |  |  | }()); |