149 lines
3.7 KiB
JavaScript
149 lines
3.7 KiB
JavaScript
|
'use strict';
|
||
|
const fs = require('fs');
|
||
|
const arrayUnion = require('array-union');
|
||
|
const glob = require('glob');
|
||
|
const fastGlob = require('fast-glob');
|
||
|
const dirGlob = require('dir-glob');
|
||
|
const gitignore = require('./gitignore');
|
||
|
|
||
|
const DEFAULT_FILTER = () => false;
|
||
|
|
||
|
const isNegative = pattern => pattern[0] === '!';
|
||
|
|
||
|
const assertPatternsInput = patterns => {
|
||
|
if (!patterns.every(x => typeof x === 'string')) {
|
||
|
throw new TypeError('Patterns must be a string or an array of strings');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
const checkCwdOption = options => {
|
||
|
if (options && options.cwd && !fs.statSync(options.cwd).isDirectory()) {
|
||
|
throw new Error('The `cwd` option must be a path to a directory');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
const generateGlobTasks = (patterns, taskOptions) => {
|
||
|
patterns = arrayUnion([].concat(patterns));
|
||
|
assertPatternsInput(patterns);
|
||
|
checkCwdOption(taskOptions);
|
||
|
|
||
|
const globTasks = [];
|
||
|
|
||
|
taskOptions = Object.assign({
|
||
|
ignore: [],
|
||
|
expandDirectories: true
|
||
|
}, taskOptions);
|
||
|
|
||
|
patterns.forEach((pattern, i) => {
|
||
|
if (isNegative(pattern)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const ignore = patterns
|
||
|
.slice(i)
|
||
|
.filter(isNegative)
|
||
|
.map(pattern => pattern.slice(1));
|
||
|
|
||
|
const options = Object.assign({}, taskOptions, {
|
||
|
ignore: taskOptions.ignore.concat(ignore)
|
||
|
});
|
||
|
|
||
|
globTasks.push({pattern, options});
|
||
|
});
|
||
|
|
||
|
return globTasks;
|
||
|
};
|
||
|
|
||
|
const globDirs = (task, fn) => {
|
||
|
let options = {};
|
||
|
if (task.options.cwd) {
|
||
|
options.cwd = task.options.cwd;
|
||
|
}
|
||
|
|
||
|
if (Array.isArray(task.options.expandDirectories)) {
|
||
|
options = Object.assign(options, {files: task.options.expandDirectories});
|
||
|
} else if (typeof task.options.expandDirectories === 'object') {
|
||
|
options = Object.assign(options, task.options.expandDirectories);
|
||
|
}
|
||
|
|
||
|
return fn(task.pattern, options);
|
||
|
};
|
||
|
|
||
|
const getPattern = (task, fn) => task.options.expandDirectories ? globDirs(task, fn) : [task.pattern];
|
||
|
|
||
|
const globToTask = task => glob => {
|
||
|
const {options} = task;
|
||
|
if (options.ignore && Array.isArray(options.ignore) && options.expandDirectories) {
|
||
|
options.ignore = dirGlob.sync(options.ignore);
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
pattern: glob,
|
||
|
options
|
||
|
};
|
||
|
};
|
||
|
|
||
|
const globby = (patterns, options) => {
|
||
|
let globTasks;
|
||
|
|
||
|
try {
|
||
|
globTasks = generateGlobTasks(patterns, options);
|
||
|
} catch (error) {
|
||
|
return Promise.reject(error);
|
||
|
}
|
||
|
|
||
|
const getTasks = Promise.all(globTasks.map(task => Promise.resolve(getPattern(task, dirGlob))
|
||
|
.then(globs => Promise.all(globs.map(globToTask(task))))
|
||
|
))
|
||
|
.then(tasks => arrayUnion(...tasks));
|
||
|
|
||
|
const getFilter = () => {
|
||
|
return Promise.resolve(
|
||
|
options && options.gitignore ?
|
||
|
gitignore({cwd: options.cwd, ignore: options.ignore}) :
|
||
|
DEFAULT_FILTER
|
||
|
);
|
||
|
};
|
||
|
|
||
|
return getFilter()
|
||
|
.then(filter => {
|
||
|
return getTasks
|
||
|
.then(tasks => Promise.all(tasks.map(task => fastGlob(task.pattern, task.options))))
|
||
|
.then(paths => arrayUnion(...paths))
|
||
|
.then(paths => paths.filter(p => !filter(p)));
|
||
|
});
|
||
|
};
|
||
|
|
||
|
module.exports = globby;
|
||
|
// TODO: Remove this for the next major release
|
||
|
module.exports.default = globby;
|
||
|
|
||
|
module.exports.sync = (patterns, options) => {
|
||
|
const globTasks = generateGlobTasks(patterns, options);
|
||
|
|
||
|
const getFilter = () => {
|
||
|
return options && options.gitignore ?
|
||
|
gitignore.sync({cwd: options.cwd, ignore: options.ignore}) :
|
||
|
DEFAULT_FILTER;
|
||
|
};
|
||
|
|
||
|
const tasks = globTasks.reduce((tasks, task) => {
|
||
|
const newTask = getPattern(task, dirGlob.sync).map(globToTask(task));
|
||
|
return tasks.concat(newTask);
|
||
|
}, []);
|
||
|
|
||
|
const filter = getFilter();
|
||
|
return tasks.reduce(
|
||
|
(matches, task) => arrayUnion(matches, fastGlob.sync(task.pattern, task.options)),
|
||
|
[]
|
||
|
).filter(p => !filter(p));
|
||
|
};
|
||
|
|
||
|
module.exports.generateGlobTasks = generateGlobTasks;
|
||
|
|
||
|
module.exports.hasMagic = (patterns, options) => []
|
||
|
.concat(patterns)
|
||
|
.some(pattern => glob.hasMagic(pattern, options));
|
||
|
|
||
|
module.exports.gitignore = gitignore;
|