|
|
|
import bytes from 'bytes';
|
|
|
|
import flatten from 'arr-flatten';
|
|
|
|
import unique from 'array-unique';
|
|
|
|
import minimatch from 'minimatch';
|
|
|
|
import IGNORED from './ignored';
|
|
|
|
import { resolve } from 'path';
|
|
|
|
import { stat, readdir, readFile } from 'fs-promise';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of files in the given
|
|
|
|
* directory that are subject to be
|
|
|
|
* synchronized.
|
|
|
|
*
|
|
|
|
* @param {String} full path to directory
|
|
|
|
* @param {String} contents of `package.json` to avoid lookup
|
|
|
|
* @param {Object} options:
|
|
|
|
* - `limit` {Number|null} byte limit
|
|
|
|
* - `debug` {Boolean} warn upon ignore
|
|
|
|
* @return {Array} comprehensive list of paths to sync
|
|
|
|
*/
|
|
|
|
|
|
|
|
export default async function getFiles (path, pkg, { limit = null, debug = false }) {
|
|
|
|
if (!pkg) {
|
|
|
|
const pkgPath = resolve(path, 'package.json');
|
|
|
|
const pkgData = await readFile(pkgPath, 'utf8');
|
|
|
|
pkg = JSON.parse(pkgData);
|
|
|
|
}
|
|
|
|
|
|
|
|
let search = (pkg.files || ['.']).concat('package.json');
|
|
|
|
if (pkg.main) search = search.concat(pkg.main);
|
|
|
|
search = search.map((file) => asAbsolute(file, path));
|
|
|
|
|
|
|
|
// compile list of ignored patterns and files
|
|
|
|
const npmIgnore = await maybeRead(resolve(path, '.npmignore'));
|
|
|
|
const gitIgnore = npmIgnore
|
|
|
|
? ''
|
|
|
|
: (await maybeRead(resolve(path, '.gitignore')));
|
|
|
|
const ignored = unique(IGNORED
|
|
|
|
.concat(gitIgnore.split('\n').filter(invalidFilter))
|
|
|
|
.concat(npmIgnore.split('\n').filter(invalidFilter)))
|
|
|
|
.map((file) => resolve(path, file));
|
|
|
|
|
|
|
|
// get files
|
|
|
|
return unique((await explode(search, ignored, { limit, debug })));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a filter function that
|
|
|
|
* excludes ignored files in the path.
|
|
|
|
*
|
|
|
|
* @param {String} path
|
|
|
|
* @return {Function} filter fn
|
|
|
|
*/
|
|
|
|
|
|
|
|
const isIgnored = (file, ignored) => {
|
|
|
|
return ignored.some((test) => {
|
|
|
|
// test that the target file is not under
|
|
|
|
// an ignored directory
|
|
|
|
const dir = test + '/';
|
|
|
|
if (file.substr(0, dir.length) === dir) return true;
|
|
|
|
|
|
|
|
// if not match wildcards
|
|
|
|
return minimatch(file, test);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a filter function that
|
|
|
|
* excludes invalid rules for .*ignore files
|
|
|
|
*
|
|
|
|
* @param {String} path
|
|
|
|
* @return {Function} filter fn
|
|
|
|
*/
|
|
|
|
|
|
|
|
const invalidFilter = (path) => {
|
|
|
|
return !(
|
|
|
|
/* commments */
|
|
|
|
'#' === path[0] ||
|
|
|
|
|
|
|
|
/* empty lines or newlines */
|
|
|
|
!path.trim().length
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Transform relative paths into absolutes,
|
|
|
|
* and maintains absolutes as such.
|
|
|
|
*
|
|
|
|
* @param {String} maybe relative path
|
|
|
|
* @param {String} parent full path
|
|
|
|
*/
|
|
|
|
|
|
|
|
const asAbsolute = function (path, parent) {
|
|
|
|
if ('/' === path[0]) return path;
|
|
|
|
return resolve(parent, path);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Explodes directories into a full list of files.
|
|
|
|
* Eg:
|
|
|
|
* in: ['/a.js', '/b']
|
|
|
|
* out: ['/a.js', '/b/c.js', '/b/d.js']
|
|
|
|
*
|
|
|
|
* @param {Array} of {String}s representing paths
|
|
|
|
* @param {Array} of ignored {String}s.
|
|
|
|
* @param {Object} options:
|
|
|
|
* - `limit` {Number|null} byte limit
|
|
|
|
* - `debug` {Boolean} warn upon ignore
|
|
|
|
* @return {Array} of {String}s of full paths
|
|
|
|
*/
|
|
|
|
|
|
|
|
const explode = async function (paths, ignored, { limit, debug }) {
|
|
|
|
const many = async (all) => {
|
|
|
|
return await Promise.all(all.map(async (file) => {
|
|
|
|
return await list(file);
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
|
|
|
const list = async (file) => {
|
|
|
|
let path = file;
|
|
|
|
let s;
|
|
|
|
|
|
|
|
try {
|
|
|
|
s = await stat(path);
|
|
|
|
} catch (e) {
|
|
|
|
// in case the file comes from `files` or `main`
|
|
|
|
// and it wasn't specified with `.js` by the user
|
|
|
|
path = file + '.js';
|
|
|
|
|
|
|
|
try {
|
|
|
|
s = await stat(path);
|
|
|
|
} catch (e2) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isIgnored(file, ignored)) {
|
|
|
|
if (debug) console.log(`> [debug] Ignoring "${file}"`);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s.isDirectory()) {
|
|
|
|
const all = await readdir(file);
|
|
|
|
return many(all.map(subdir => asAbsolute(subdir, file)));
|
|
|
|
} else {
|
|
|
|
if (null != limit && s.size > limit) {
|
|
|
|
console.error('> \u001b[31mWarning!\u001b[39m Skipping file ' +
|
|
|
|
`over ${bytes(limit)}: ${path}`);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return flatten((await many(paths))).filter((v) => null != v);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the contents of a file if it exists.
|
|
|
|
*
|
|
|
|
* @return {String} results or `''`
|
|
|
|
*/
|
|
|
|
|
|
|
|
const maybeRead = async function (path) {
|
|
|
|
try {
|
|
|
|
return (await readFile(path, 'utf8'));
|
|
|
|
} catch (e) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
};
|