1
0
Fork 0
pull/287/head
eric sciple 2020-01-08 16:07:06 -05:00
parent b6a6457adf
commit f9f81cb92d
5 changed files with 39 additions and 57 deletions

View File

@ -38,7 +38,7 @@ describe('globber', () => {
const originalCwd = process.cwd()
try {
process.chdir(path.join(root, 'first-cwd'))
const globber = await DefaultGlobber.parse('*')
const globber = await DefaultGlobber.create('*')
process.chdir(path.join(root, 'second-cwd'))
expect(globber.getSearchPaths()).toEqual([path.join(root, 'first-cwd')])
const itemPaths = await globber.glob()
@ -175,7 +175,7 @@ describe('globber', () => {
// todo: ? expect(itemPaths[2]).toBe(path.join(root, 'symDir', 'symDir'));
})
it('does not follow symlink when followSymbolicLink=false', async () => {
it('does not follow symlink when followSymbolicLinks=false', async () => {
// Create the following layout:
// <root>
// <root>/realDir
@ -192,7 +192,7 @@ describe('globber', () => {
path.join(root, 'symDir')
)
const itemPaths = await glob(root)
const itemPaths = await glob(root, {followSymbolicLinks: false})
expect(itemPaths).toEqual([
root,
path.join(root, 'realDir'),
@ -201,7 +201,7 @@ describe('globber', () => {
])
})
it('does not follow symlink when search path is symlink and followSymbolicLink=false', async () => {
it('does not follow symlink when search path is symlink and followSymbolicLinks=false', async () => {
// Create the following layout:
// realDir
// realDir/file
@ -217,7 +217,9 @@ describe('globber', () => {
path.join(root, 'symDir')
)
const itemPaths = await glob(path.join(root, 'symDir'))
const itemPaths = await glob(path.join(root, 'symDir'), {
followSymbolicLinks: false
})
expect(itemPaths).toEqual([path.join(root, 'symDir')])
})
@ -406,20 +408,6 @@ describe('globber', () => {
])
})
it('parses options', async () => {
let globber: Globber
globber = await DefaultGlobber.parse(
`--follow-symbolic-links${os.EOL}/foo/*`
)
expect(globber.options.followSymbolicLinks).toBeTruthy()
expect(globber.getSearchPaths()).toHaveLength(1)
globber = await DefaultGlobber.parse(`/foo/*`) // sanity check
expect(globber.options.implicitDescendants === undefined).toBeTruthy()
expect(globber.options.followSymbolicLinks === undefined).toBeTruthy()
expect(globber.options.omitBrokenSymbolicLinks === undefined).toBeTruthy()
expect(globber.getSearchPaths()).toHaveLength(1)
})
it('returns broken symlink when followSymbolicLinks=false', async () => {
// Create the following layout:
// <root>
@ -443,7 +431,7 @@ describe('globber', () => {
path.join(root, 'symDir')
)
const itemPaths = await glob(root)
const itemPaths = await glob(root, {followSymbolicLinks: false})
expect(itemPaths).toEqual([
root,
path.join(root, 'brokenSym'),
@ -465,7 +453,7 @@ describe('globber', () => {
const brokenSymPath = path.join(root, 'brokenSym')
await createSymlinkDir(path.join(root, 'noSuch'), brokenSymPath)
const itemPaths = await glob(brokenSymPath)
const itemPaths = await glob(brokenSymPath, {followSymbolicLinks: false})
expect(itemPaths).toEqual([brokenSymPath])
})
@ -753,13 +741,15 @@ async function createSymlinkDir(real: string, link: string): Promise<void> {
}
}
async function getSearchPaths(input: string): Promise<string[]> {
const globber: Globber = await DefaultGlobber.parse(input)
async function getSearchPaths(patterns: string): Promise<string[]> {
const globber: Globber = await DefaultGlobber.create(patterns)
return globber.getSearchPaths()
}
async function glob(input: string, options?: GlobOptions): Promise<string[]> {
const globber: Globber = await DefaultGlobber.parse(input)
globber.options = {...(globber.options || {}), ...(options || {})}
async function glob(
patterns: string,
options?: GlobOptions
): Promise<string[]> {
const globber: Globber = await DefaultGlobber.create(patterns, options)
return await globber.glob()
}

View File

@ -7,7 +7,11 @@ export {Globber, GlobOptions}
* Constructs a globber
*
* @param patterns Patterns separated by newlines
* @param options Glob options
*/
export async function create(patterns: string): Promise<Globber> {
return await DefaultGlobber.parse(patterns)
export async function create(
patterns: string,
options?: GlobOptions
): Promise<Globber> {
return await DefaultGlobber.create(patterns, options)
}

View File

@ -6,7 +6,7 @@ import {GlobOptions} from './internal-glob-options'
*/
export function getOptions(copy?: GlobOptions): GlobOptions {
const result: GlobOptions = {
followSymbolicLinks: false,
followSymbolicLinks: true,
implicitDescendants: true,
omitBrokenSymbolicLinks: true
}

View File

@ -3,9 +3,10 @@
*/
export interface GlobOptions {
/**
* Indicates whether to follow symbolic links
* Indicates whether to follow symbolic links. Generally should set to false
* when deleting files.
*
* @default false
* @default true
*/
followSymbolicLinks?: boolean

View File

@ -16,11 +16,6 @@ export {GlobOptions}
* Used to match files and directories
*/
export interface Globber {
/**
* Controls globbing behavior
*/
options: GlobOptions
/**
* Returns the search path preceeding the first glob segment, from each pattern.
* Duplicates and descendants of other paths are filtered out.
@ -47,11 +42,13 @@ export interface Globber {
}
export class DefaultGlobber implements Globber {
options: GlobOptions = {}
private readonly options: GlobOptions
private readonly patterns: Pattern[] = []
private readonly searchPaths: string[] = []
private constructor() {}
private constructor(options?: GlobOptions) {
this.options = globOptionsHelper.getOptions(options)
}
getSearchPaths(): string[] {
// Return a copy
@ -157,35 +154,25 @@ export class DefaultGlobber implements Globber {
}
/**
* Parses the input options and patterns to construct a new GlobberImpl instance.
* Constructs a DefaultGlobber
*/
static async parse(input: string): Promise<DefaultGlobber> {
const result = new DefaultGlobber()
static async create(
patterns: string,
options?: GlobOptions
): Promise<DefaultGlobber> {
const result = new DefaultGlobber(options)
if (IS_WINDOWS) {
input = input.replace(/\r\n/g, '\n')
input = input.replace(/\r/g, '\n')
patterns = patterns.replace(/\r\n/g, '\n')
patterns = patterns.replace(/\r/g, '\n')
}
const lines = input.split('\n').map(x => x.trim())
const lines = patterns.split('\n').map(x => x.trim())
for (const line of lines) {
// Empty or comment
if (!line || line.startsWith('#')) {
continue
}
// Options
else if (line.startsWith('-')) {
const options = line.split(/ +/g)
for (const option of options) {
switch (option) {
case '--follow-symbolic-links':
result.options.followSymbolicLinks = true
break
default:
throw new Error(`Unknown glob options '${option}'`)
}
}
}
// Pattern
else {
result.patterns.push(new Pattern(line))