98 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			98 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
"use strict";
 | 
						|
Object.defineProperty(exports, "__esModule", { value: true });
 | 
						|
const events_1 = require("events");
 | 
						|
const fsScandir = require("@nodelib/fs.scandir");
 | 
						|
const fastq = require("fastq");
 | 
						|
const common = require("./common");
 | 
						|
const reader_1 = require("./reader");
 | 
						|
class AsyncReader extends reader_1.default {
 | 
						|
    constructor(_root, _settings) {
 | 
						|
        super(_root, _settings);
 | 
						|
        this._settings = _settings;
 | 
						|
        this._scandir = fsScandir.scandir;
 | 
						|
        this._emitter = new events_1.EventEmitter();
 | 
						|
        this._queue = fastq(this._worker.bind(this), this._settings.concurrency);
 | 
						|
        this._isFatalError = false;
 | 
						|
        this._isDestroyed = false;
 | 
						|
        this._queue.drain = () => {
 | 
						|
            if (!this._isFatalError) {
 | 
						|
                this._emitter.emit('end');
 | 
						|
            }
 | 
						|
        };
 | 
						|
    }
 | 
						|
    read() {
 | 
						|
        this._isFatalError = false;
 | 
						|
        this._isDestroyed = false;
 | 
						|
        setImmediate(() => {
 | 
						|
            this._pushToQueue(this._root, this._settings.basePath);
 | 
						|
        });
 | 
						|
        return this._emitter;
 | 
						|
    }
 | 
						|
    get isDestroyed() {
 | 
						|
        return this._isDestroyed;
 | 
						|
    }
 | 
						|
    destroy() {
 | 
						|
        if (this._isDestroyed) {
 | 
						|
            throw new Error('The reader is already destroyed');
 | 
						|
        }
 | 
						|
        this._isDestroyed = true;
 | 
						|
        this._queue.killAndDrain();
 | 
						|
    }
 | 
						|
    onEntry(callback) {
 | 
						|
        this._emitter.on('entry', callback);
 | 
						|
    }
 | 
						|
    onError(callback) {
 | 
						|
        this._emitter.once('error', callback);
 | 
						|
    }
 | 
						|
    onEnd(callback) {
 | 
						|
        this._emitter.once('end', callback);
 | 
						|
    }
 | 
						|
    _pushToQueue(directory, base) {
 | 
						|
        const queueItem = { directory, base };
 | 
						|
        this._queue.push(queueItem, (error) => {
 | 
						|
            if (error !== null) {
 | 
						|
                this._handleError(error);
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
    _worker(item, done) {
 | 
						|
        this._scandir(item.directory, this._settings.fsScandirSettings, (error, entries) => {
 | 
						|
            if (error !== null) {
 | 
						|
                done(error, undefined);
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            for (const entry of entries) {
 | 
						|
                this._handleEntry(entry, item.base);
 | 
						|
            }
 | 
						|
            done(null, undefined);
 | 
						|
        });
 | 
						|
    }
 | 
						|
    _handleError(error) {
 | 
						|
        if (this._isDestroyed || !common.isFatalError(this._settings, error)) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        this._isFatalError = true;
 | 
						|
        this._isDestroyed = true;
 | 
						|
        this._emitter.emit('error', error);
 | 
						|
    }
 | 
						|
    _handleEntry(entry, base) {
 | 
						|
        if (this._isDestroyed || this._isFatalError) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        const fullpath = entry.path;
 | 
						|
        if (base !== undefined) {
 | 
						|
            entry.path = common.joinPathSegments(base, entry.name, this._settings.pathSegmentSeparator);
 | 
						|
        }
 | 
						|
        if (common.isAppliedFilter(this._settings.entryFilter, entry)) {
 | 
						|
            this._emitEntry(entry);
 | 
						|
        }
 | 
						|
        if (entry.dirent.isDirectory() && common.isAppliedFilter(this._settings.deepFilter, entry)) {
 | 
						|
            this._pushToQueue(fullpath, base === undefined ? undefined : entry.path);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    _emitEntry(entry) {
 | 
						|
        this._emitter.emit('entry', entry);
 | 
						|
    }
 | 
						|
}
 | 
						|
exports.default = AsyncReader;
 |