mirror of
https://github.com/denoland/deno.git
synced 2025-01-24 08:00:10 -05:00
a472b6732d
Original: 17a214bbd5
143 lines
3.8 KiB
TypeScript
143 lines
3.8 KiB
TypeScript
import { globrex } from "./globrex.ts";
|
|
import { isAbsolute, join } from "./path/mod.ts";
|
|
import { WalkInfo, walk, walkSync } from "./walk.ts";
|
|
const { cwd } = Deno;
|
|
|
|
export interface GlobOptions {
|
|
// Allow ExtGlob features
|
|
extended?: boolean;
|
|
// When globstar is true, '/foo/**' is equivelant
|
|
// to '/foo/*' when globstar is false.
|
|
// Having globstar set to true is the same usage as
|
|
// using wildcards in bash
|
|
globstar?: boolean;
|
|
// be laissez faire about mutiple slashes
|
|
strict?: boolean;
|
|
// Parse as filepath for extra path related features
|
|
filepath?: boolean;
|
|
// Flag to use in the generated RegExp
|
|
flags?: string;
|
|
}
|
|
|
|
/**
|
|
* Generate a regex based on glob pattern and options
|
|
* This was meant to be using the the `fs.walk` function
|
|
* but can be used anywhere else.
|
|
* Examples:
|
|
*
|
|
* Looking for all the `ts` files:
|
|
* walkSync(".", {
|
|
* match: [glob("*.ts")]
|
|
* })
|
|
*
|
|
* Looking for all the `.json` files in any subfolder:
|
|
* walkSync(".", {
|
|
* match: [glob(join("a", "**", "*.json"),{
|
|
* flags: "g",
|
|
* extended: true,
|
|
* globstar: true
|
|
* })]
|
|
* })
|
|
*
|
|
* @param glob - Glob pattern to be used
|
|
* @param options - Specific options for the glob pattern
|
|
* @returns A RegExp for the glob pattern
|
|
*/
|
|
export function glob(glob: string, options: GlobOptions = {}): RegExp {
|
|
const result = globrex(glob, options);
|
|
if (options.filepath) {
|
|
return result.path!.regex;
|
|
}
|
|
return result.regex;
|
|
}
|
|
|
|
/** Test whether the given string is a glob */
|
|
export function isGlob(str: string): boolean {
|
|
const chars: Record<string, string> = { "{": "}", "(": ")", "[": "]" };
|
|
/* eslint-disable-next-line max-len */
|
|
const regex = /\\(.)|(^!|\*|[\].+)]\?|\[[^\\\]]+\]|\{[^\\}]+\}|\(\?[:!=][^\\)]+\)|\([^|]+\|[^\\)]+\))/;
|
|
|
|
if (str === "") {
|
|
return false;
|
|
}
|
|
|
|
let match: RegExpExecArray | null;
|
|
|
|
while ((match = regex.exec(str))) {
|
|
if (match[2]) return true;
|
|
let idx = match.index + match[0].length;
|
|
|
|
// if an open bracket/brace/paren is escaped,
|
|
// set the index to the next closing character
|
|
const open = match[1];
|
|
const close = open ? chars[open] : null;
|
|
if (open && close) {
|
|
const n = str.indexOf(close, idx);
|
|
if (n !== -1) {
|
|
idx = n + 1;
|
|
}
|
|
}
|
|
|
|
str = str.slice(idx);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export interface ExpandGlobOptions extends GlobOptions {
|
|
root?: string;
|
|
includeDirs?: boolean;
|
|
}
|
|
|
|
/**
|
|
* Expand the glob string from the specified `root` directory and yield each
|
|
* result as a `WalkInfo` object.
|
|
*/
|
|
// TODO: Use a proper glob expansion algorithm.
|
|
// This is a very incomplete solution. The whole directory tree from `root` is
|
|
// walked and parent paths are not supported.
|
|
export async function* expandGlob(
|
|
globString: string,
|
|
{
|
|
root = cwd(),
|
|
includeDirs = true,
|
|
extended = false,
|
|
globstar = false,
|
|
strict = false,
|
|
filepath = true,
|
|
flags = ""
|
|
}: ExpandGlobOptions = {}
|
|
): AsyncIterableIterator<WalkInfo> {
|
|
const absoluteGlob = isAbsolute(globString)
|
|
? globString
|
|
: join(root, globString);
|
|
const globOptions = { extended, globstar, strict, filepath, flags };
|
|
yield* walk(root, {
|
|
match: [glob(absoluteGlob, globOptions)],
|
|
includeDirs
|
|
});
|
|
}
|
|
|
|
/** Synchronous version of `expandGlob()`. */
|
|
// TODO: As `expandGlob()`.
|
|
export function* expandGlobSync(
|
|
globString: string,
|
|
{
|
|
root = cwd(),
|
|
includeDirs = true,
|
|
extended = false,
|
|
globstar = false,
|
|
strict = false,
|
|
filepath = true,
|
|
flags = ""
|
|
}: ExpandGlobOptions = {}
|
|
): IterableIterator<WalkInfo> {
|
|
const absoluteGlob = isAbsolute(globString)
|
|
? globString
|
|
: join(root, globString);
|
|
const globOptions = { extended, globstar, strict, filepath, flags };
|
|
yield* walkSync(root, {
|
|
match: [glob(absoluteGlob, globOptions)],
|
|
includeDirs
|
|
});
|
|
}
|