globby
Version:
User-friendly glob matching
338 lines (279 loc) • 11.9 kB
JavaScript
import process from 'node:process';
import fs from 'node:fs';
import fsPromises from 'node:fs/promises';
import path from 'node:path';
import fastGlob from 'fast-glob';
import gitIgnore from 'ignore';
import isPathInside from 'is-path-inside';
import slash from 'slash';
import {toPath} from 'unicorn-magic/node';
import {
isNegativePattern,
bindFsMethod,
promisifyFsMethod,
findGitRoot,
findGitRootSync,
getParentGitignorePaths,
} from './utilities.js';
const defaultIgnoredDirectories = [
'**/node_modules',
'**/flow-typed',
'**/coverage',
'**/.git',
];
const ignoreFilesGlobOptions = {
absolute: true,
dot: true,
};
export const GITIGNORE_FILES_PATTERN = '**/.gitignore';
const getReadFileMethod = fsImplementation =>
bindFsMethod(fsImplementation?.promises, 'readFile')
?? bindFsMethod(fsPromises, 'readFile')
?? promisifyFsMethod(fsImplementation, 'readFile');
const getReadFileSyncMethod = fsImplementation =>
bindFsMethod(fsImplementation, 'readFileSync')
?? bindFsMethod(fs, 'readFileSync');
const shouldSkipIgnoreFileError = (error, suppressErrors) => {
if (!error) {
return Boolean(suppressErrors);
}
if (error.code === 'ENOENT' || error.code === 'ENOTDIR') {
return true;
}
return Boolean(suppressErrors);
};
const createIgnoreFileReadError = (filePath, error) => {
if (error instanceof Error) {
error.message = `Failed to read ignore file at ${filePath}: ${error.message}`;
return error;
}
return new Error(`Failed to read ignore file at ${filePath}: ${String(error)}`);
};
const processIgnoreFileCore = (filePath, readMethod, suppressErrors) => {
try {
const content = readMethod(filePath, 'utf8');
return {filePath, content};
} catch (error) {
if (shouldSkipIgnoreFileError(error, suppressErrors)) {
return undefined;
}
throw createIgnoreFileReadError(filePath, error);
}
};
const readIgnoreFilesSafely = async (paths, readFileMethod, suppressErrors) => {
const fileResults = await Promise.all(paths.map(async filePath => {
try {
const content = await readFileMethod(filePath, 'utf8');
return {filePath, content};
} catch (error) {
if (shouldSkipIgnoreFileError(error, suppressErrors)) {
return undefined;
}
throw createIgnoreFileReadError(filePath, error);
}
}));
return fileResults.filter(Boolean);
};
const readIgnoreFilesSafelySync = (paths, readFileSyncMethod, suppressErrors) => paths
.map(filePath => processIgnoreFileCore(filePath, readFileSyncMethod, suppressErrors))
.filter(Boolean);
const dedupePaths = paths => {
const seen = new Set();
return paths.filter(filePath => {
if (seen.has(filePath)) {
return false;
}
seen.add(filePath);
return true;
});
};
const globIgnoreFiles = (globFunction, patterns, normalizedOptions) => globFunction(patterns, {
...normalizedOptions,
...ignoreFilesGlobOptions, // Must be last to ensure absolute/dot flags stick
});
const getParentIgnorePaths = (gitRoot, normalizedOptions) => gitRoot
? getParentGitignorePaths(gitRoot, normalizedOptions.cwd)
: [];
const combineIgnoreFilePaths = (gitRoot, normalizedOptions, childPaths) => dedupePaths([
...getParentIgnorePaths(gitRoot, normalizedOptions),
...childPaths,
]);
const buildIgnoreResult = (files, normalizedOptions, gitRoot) => {
const baseDir = gitRoot || normalizedOptions.cwd;
const patterns = getPatternsFromIgnoreFiles(files, baseDir);
return {
patterns,
predicate: createIgnorePredicate(patterns, normalizedOptions.cwd, baseDir),
usingGitRoot: Boolean(gitRoot && gitRoot !== normalizedOptions.cwd),
};
};
// Apply base path to gitignore patterns based on .gitignore spec 2.22.1
// https://git-scm.com/docs/gitignore#_pattern_format
// See also https://github.com/sindresorhus/globby/issues/146
const applyBaseToPattern = (pattern, base) => {
if (!base) {
return pattern;
}
const isNegative = isNegativePattern(pattern);
const cleanPattern = isNegative ? pattern.slice(1) : pattern;
// Check if pattern has non-trailing slashes
const slashIndex = cleanPattern.indexOf('/');
const hasNonTrailingSlash = slashIndex !== -1 && slashIndex !== cleanPattern.length - 1;
let result;
if (!hasNonTrailingSlash) {
// "If there is no separator at the beginning or middle of the pattern,
// then the pattern may also match at any level below the .gitignore level."
// So patterns like '*.log' or 'temp' or 'build/' (trailing slash) match recursively.
result = path.posix.join(base, '**', cleanPattern);
} else if (cleanPattern.startsWith('/')) {
// "If there is a separator at the beginning [...] of the pattern,
// then the pattern is relative to the directory level of the particular .gitignore file itself."
// Leading slash anchors the pattern to the .gitignore's directory.
result = path.posix.join(base, cleanPattern.slice(1));
} else {
// "If there is a separator [...] middle [...] of the pattern,
// then the pattern is relative to the directory level of the particular .gitignore file itself."
// Patterns like 'src/foo' are relative to the .gitignore's directory.
result = path.posix.join(base, cleanPattern);
}
return isNegative ? '!' + result : result;
};
const parseIgnoreFile = (file, cwd) => {
const base = slash(path.relative(cwd, path.dirname(file.filePath)));
return file.content
.split(/\r?\n/)
.filter(line => line && !line.startsWith('#'))
.map(pattern => applyBaseToPattern(pattern, base));
};
const toRelativePath = (fileOrDirectory, cwd) => {
if (path.isAbsolute(fileOrDirectory)) {
// When paths are equal, path.relative returns empty string which is valid
// isPathInside returns false for equal paths, so check this case first
const relativePath = path.relative(cwd, fileOrDirectory);
if (relativePath && !isPathInside(fileOrDirectory, cwd)) {
// Path is outside cwd - it cannot be ignored by patterns in cwd
// Return undefined to indicate this path is outside scope
return undefined;
}
return relativePath;
}
// Normalize relative paths:
// - Git treats './foo' as 'foo' when checking against patterns
// - Patterns starting with './' in .gitignore are invalid and don't match anything
// - The ignore library expects normalized paths without './' prefix
if (fileOrDirectory.startsWith('./')) {
return fileOrDirectory.slice(2);
}
// Paths with ../ point outside cwd and cannot match patterns from this directory
// Return undefined to indicate this path is outside scope
if (fileOrDirectory.startsWith('../')) {
return undefined;
}
return fileOrDirectory;
};
const createIgnorePredicate = (patterns, cwd, baseDir) => {
const ignores = gitIgnore().add(patterns);
// Normalize to handle path separator and . / .. components consistently
const resolvedCwd = path.normalize(path.resolve(cwd));
const resolvedBaseDir = path.normalize(path.resolve(baseDir));
return fileOrDirectory => {
fileOrDirectory = toPath(fileOrDirectory);
// Never ignore the cwd itself - use normalized comparison
const normalizedPath = path.normalize(path.resolve(fileOrDirectory));
if (normalizedPath === resolvedCwd) {
return false;
}
// Convert to relative path from baseDir (use normalized baseDir)
const relativePath = toRelativePath(fileOrDirectory, resolvedBaseDir);
// If path is outside baseDir (undefined), it can't be ignored by patterns
if (relativePath === undefined) {
return false;
}
return relativePath ? ignores.ignores(slash(relativePath)) : false;
};
};
const normalizeOptions = (options = {}) => {
const ignoreOption = options.ignore
? (Array.isArray(options.ignore) ? options.ignore : [options.ignore])
: [];
const cwd = toPath(options.cwd) ?? process.cwd();
// Adjust deep option for fast-glob: fast-glob's deep counts differently than expected
// User's deep: 0 = root only -> fast-glob needs: 1
// User's deep: 1 = root + 1 level -> fast-glob needs: 2
const deep = typeof options.deep === 'number' ? Math.max(0, options.deep) + 1 : Number.POSITIVE_INFINITY;
// Only pass through specific fast-glob options that make sense for finding ignore files
return {
cwd,
suppressErrors: options.suppressErrors ?? false,
deep,
ignore: [...ignoreOption, ...defaultIgnoredDirectories],
followSymbolicLinks: options.followSymbolicLinks ?? true,
concurrency: options.concurrency,
throwErrorOnBrokenSymbolicLink: options.throwErrorOnBrokenSymbolicLink ?? false,
fs: options.fs,
};
};
const collectIgnoreFileArtifactsAsync = async (patterns, options, includeParentIgnoreFiles) => {
const normalizedOptions = normalizeOptions(options);
const childPaths = await globIgnoreFiles(fastGlob, patterns, normalizedOptions);
const gitRoot = includeParentIgnoreFiles
? await findGitRoot(normalizedOptions.cwd, normalizedOptions.fs)
: undefined;
const allPaths = combineIgnoreFilePaths(gitRoot, normalizedOptions, childPaths);
const readFileMethod = getReadFileMethod(normalizedOptions.fs);
const files = await readIgnoreFilesSafely(allPaths, readFileMethod, normalizedOptions.suppressErrors);
return {files, normalizedOptions, gitRoot};
};
const collectIgnoreFileArtifactsSync = (patterns, options, includeParentIgnoreFiles) => {
const normalizedOptions = normalizeOptions(options);
const childPaths = globIgnoreFiles(fastGlob.sync, patterns, normalizedOptions);
const gitRoot = includeParentIgnoreFiles
? findGitRootSync(normalizedOptions.cwd, normalizedOptions.fs)
: undefined;
const allPaths = combineIgnoreFilePaths(gitRoot, normalizedOptions, childPaths);
const readFileSyncMethod = getReadFileSyncMethod(normalizedOptions.fs);
const files = readIgnoreFilesSafelySync(allPaths, readFileSyncMethod, normalizedOptions.suppressErrors);
return {files, normalizedOptions, gitRoot};
};
export const isIgnoredByIgnoreFiles = async (patterns, options) => {
const {files, normalizedOptions, gitRoot} = await collectIgnoreFileArtifactsAsync(patterns, options, false);
return buildIgnoreResult(files, normalizedOptions, gitRoot).predicate;
};
export const isIgnoredByIgnoreFilesSync = (patterns, options) => {
const {files, normalizedOptions, gitRoot} = collectIgnoreFileArtifactsSync(patterns, options, false);
return buildIgnoreResult(files, normalizedOptions, gitRoot).predicate;
};
const getPatternsFromIgnoreFiles = (files, baseDir) => files.flatMap(file => parseIgnoreFile(file, baseDir));
/**
Read ignore files and return both patterns and predicate.
This avoids reading the same files twice (once for patterns, once for filtering).
@param {string[]} patterns - Patterns to find ignore files
@param {Object} options - Options object
@param {boolean} [includeParentIgnoreFiles=false] - Whether to search for parent .gitignore files
@returns {Promise<{patterns: string[], predicate: Function, usingGitRoot: boolean}>}
*/
export const getIgnorePatternsAndPredicate = async (patterns, options, includeParentIgnoreFiles = false) => {
const {files, normalizedOptions, gitRoot} = await collectIgnoreFileArtifactsAsync(
patterns,
options,
includeParentIgnoreFiles,
);
return buildIgnoreResult(files, normalizedOptions, gitRoot);
};
/**
Read ignore files and return both patterns and predicate (sync version).
@param {string[]} patterns - Patterns to find ignore files
@param {Object} options - Options object
@param {boolean} [includeParentIgnoreFiles=false] - Whether to search for parent .gitignore files
@returns {{patterns: string[], predicate: Function, usingGitRoot: boolean}}
*/
export const getIgnorePatternsAndPredicateSync = (patterns, options, includeParentIgnoreFiles = false) => {
const {files, normalizedOptions, gitRoot} = collectIgnoreFileArtifactsSync(
patterns,
options,
includeParentIgnoreFiles,
);
return buildIgnoreResult(files, normalizedOptions, gitRoot);
};
export const isGitIgnored = options => isIgnoredByIgnoreFiles(GITIGNORE_FILES_PATTERN, options);
export const isGitIgnoredSync = options => isIgnoredByIgnoreFilesSync(GITIGNORE_FILES_PATTERN, options);