UNPKG

2.15 kBJavaScriptView Raw
1const fs = require('fs');
2const globrex = require('globrex');
3const globalyzer = require('globalyzer');
4const { join, resolve, relative } = require('path');
5
6const isHidden = /(^|\/)\.[^\/\.]/g;
7const giveup = rgx => !rgx || rgx == '/^((?:[^\\/]*(?:\\/|$))*)$/';
8
9const CACHE = {};
10
11function walk(output, prefix, lexer, opts, dirname='', level=0) {
12 const rgx = lexer.segments[level];
13 const dir = join(opts.cwd, prefix, dirname);
14 const files = fs.readdirSync(dir);
15 const { dot, filesOnly } = opts;
16
17 let i=0, len=files.length, file;
18 let fullpath, relpath, stats, isMatch;
19
20 for (; i < len; i++) {
21 fullpath = join(dir, file=files[i]);
22 relpath = dirname ? join(dirname, file) : file;
23 if (!dot && isHidden.test(relpath)) continue;
24 isMatch = lexer.regex.test(relpath);
25
26 if ((stats=CACHE[relpath]) === void 0) {
27 CACHE[relpath] = stats = fs.lstatSync(fullpath);
28 }
29
30 if (!stats.isDirectory()) {
31 isMatch && output.push(relative(opts.cwd, fullpath));
32 continue;
33 }
34
35 if (rgx && !rgx.test(file)) continue;
36 !filesOnly && isMatch && output.push(join(prefix, relpath));
37
38 walk(output, prefix, lexer, opts, relpath, giveup(rgx) ? null : level + 1);
39 }
40}
41
42/**
43 * Find files using bash-like globbing.
44 * All paths are normalized compared to node-glob.
45 * @param {String} str Glob string
46 * @param {String} [options.cwd='.'] Current working directory
47 * @param {Boolean} [options.dot=false] Include dotfile matches
48 * @param {Boolean} [options.absolute=false] Return absolute paths
49 * @param {Boolean} [options.filesOnly=false] Do not include folders if true
50 * @param {Boolean} [options.flush=false] Reset cache object
51 * @returns {Array} array containing matching files
52 */
53module.exports = function (str, opts={}) {
54 let glob = globalyzer(str);
55
56 if (!glob.isGlob) return fs.existsSync(str) ? [str] : [];
57 if (opts.flush) CACHE = {};
58
59 let matches = [];
60 opts.cwd = opts.cwd || '.';
61 const patterns = globrex(glob.glob, { globstar:true, extended:true });
62
63 walk(matches, glob.base, patterns, opts, '.', 0);
64
65 return opts.absolute ? matches.map(x => resolve(opts.cwd, x)) : matches;
66};