mirror of https://github.com/actions/toolkit
pull/287/head
parent
b6a6457adf
commit
f9f81cb92d
|
@ -38,7 +38,7 @@ describe('globber', () => {
|
||||||
const originalCwd = process.cwd()
|
const originalCwd = process.cwd()
|
||||||
try {
|
try {
|
||||||
process.chdir(path.join(root, 'first-cwd'))
|
process.chdir(path.join(root, 'first-cwd'))
|
||||||
const globber = await DefaultGlobber.parse('*')
|
const globber = await DefaultGlobber.create('*')
|
||||||
process.chdir(path.join(root, 'second-cwd'))
|
process.chdir(path.join(root, 'second-cwd'))
|
||||||
expect(globber.getSearchPaths()).toEqual([path.join(root, 'first-cwd')])
|
expect(globber.getSearchPaths()).toEqual([path.join(root, 'first-cwd')])
|
||||||
const itemPaths = await globber.glob()
|
const itemPaths = await globber.glob()
|
||||||
|
@ -175,7 +175,7 @@ describe('globber', () => {
|
||||||
// todo: ? expect(itemPaths[2]).toBe(path.join(root, 'symDir', 'symDir'));
|
// 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:
|
// Create the following layout:
|
||||||
// <root>
|
// <root>
|
||||||
// <root>/realDir
|
// <root>/realDir
|
||||||
|
@ -192,7 +192,7 @@ describe('globber', () => {
|
||||||
path.join(root, 'symDir')
|
path.join(root, 'symDir')
|
||||||
)
|
)
|
||||||
|
|
||||||
const itemPaths = await glob(root)
|
const itemPaths = await glob(root, {followSymbolicLinks: false})
|
||||||
expect(itemPaths).toEqual([
|
expect(itemPaths).toEqual([
|
||||||
root,
|
root,
|
||||||
path.join(root, 'realDir'),
|
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:
|
// Create the following layout:
|
||||||
// realDir
|
// realDir
|
||||||
// realDir/file
|
// realDir/file
|
||||||
|
@ -217,7 +217,9 @@ describe('globber', () => {
|
||||||
path.join(root, 'symDir')
|
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')])
|
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 () => {
|
it('returns broken symlink when followSymbolicLinks=false', async () => {
|
||||||
// Create the following layout:
|
// Create the following layout:
|
||||||
// <root>
|
// <root>
|
||||||
|
@ -443,7 +431,7 @@ describe('globber', () => {
|
||||||
path.join(root, 'symDir')
|
path.join(root, 'symDir')
|
||||||
)
|
)
|
||||||
|
|
||||||
const itemPaths = await glob(root)
|
const itemPaths = await glob(root, {followSymbolicLinks: false})
|
||||||
expect(itemPaths).toEqual([
|
expect(itemPaths).toEqual([
|
||||||
root,
|
root,
|
||||||
path.join(root, 'brokenSym'),
|
path.join(root, 'brokenSym'),
|
||||||
|
@ -465,7 +453,7 @@ describe('globber', () => {
|
||||||
const brokenSymPath = path.join(root, 'brokenSym')
|
const brokenSymPath = path.join(root, 'brokenSym')
|
||||||
await createSymlinkDir(path.join(root, 'noSuch'), brokenSymPath)
|
await createSymlinkDir(path.join(root, 'noSuch'), brokenSymPath)
|
||||||
|
|
||||||
const itemPaths = await glob(brokenSymPath)
|
const itemPaths = await glob(brokenSymPath, {followSymbolicLinks: false})
|
||||||
expect(itemPaths).toEqual([brokenSymPath])
|
expect(itemPaths).toEqual([brokenSymPath])
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -753,13 +741,15 @@ async function createSymlinkDir(real: string, link: string): Promise<void> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function getSearchPaths(input: string): Promise<string[]> {
|
async function getSearchPaths(patterns: string): Promise<string[]> {
|
||||||
const globber: Globber = await DefaultGlobber.parse(input)
|
const globber: Globber = await DefaultGlobber.create(patterns)
|
||||||
return globber.getSearchPaths()
|
return globber.getSearchPaths()
|
||||||
}
|
}
|
||||||
|
|
||||||
async function glob(input: string, options?: GlobOptions): Promise<string[]> {
|
async function glob(
|
||||||
const globber: Globber = await DefaultGlobber.parse(input)
|
patterns: string,
|
||||||
globber.options = {...(globber.options || {}), ...(options || {})}
|
options?: GlobOptions
|
||||||
|
): Promise<string[]> {
|
||||||
|
const globber: Globber = await DefaultGlobber.create(patterns, options)
|
||||||
return await globber.glob()
|
return await globber.glob()
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,11 @@ export {Globber, GlobOptions}
|
||||||
* Constructs a globber
|
* Constructs a globber
|
||||||
*
|
*
|
||||||
* @param patterns Patterns separated by newlines
|
* @param patterns Patterns separated by newlines
|
||||||
|
* @param options Glob options
|
||||||
*/
|
*/
|
||||||
export async function create(patterns: string): Promise<Globber> {
|
export async function create(
|
||||||
return await DefaultGlobber.parse(patterns)
|
patterns: string,
|
||||||
|
options?: GlobOptions
|
||||||
|
): Promise<Globber> {
|
||||||
|
return await DefaultGlobber.create(patterns, options)
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ import {GlobOptions} from './internal-glob-options'
|
||||||
*/
|
*/
|
||||||
export function getOptions(copy?: GlobOptions): GlobOptions {
|
export function getOptions(copy?: GlobOptions): GlobOptions {
|
||||||
const result: GlobOptions = {
|
const result: GlobOptions = {
|
||||||
followSymbolicLinks: false,
|
followSymbolicLinks: true,
|
||||||
implicitDescendants: true,
|
implicitDescendants: true,
|
||||||
omitBrokenSymbolicLinks: true
|
omitBrokenSymbolicLinks: true
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,10 @@
|
||||||
*/
|
*/
|
||||||
export interface GlobOptions {
|
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
|
followSymbolicLinks?: boolean
|
||||||
|
|
||||||
|
|
|
@ -16,11 +16,6 @@ export {GlobOptions}
|
||||||
* Used to match files and directories
|
* Used to match files and directories
|
||||||
*/
|
*/
|
||||||
export interface Globber {
|
export interface Globber {
|
||||||
/**
|
|
||||||
* Controls globbing behavior
|
|
||||||
*/
|
|
||||||
options: GlobOptions
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the search path preceeding the first glob segment, from each pattern.
|
* Returns the search path preceeding the first glob segment, from each pattern.
|
||||||
* Duplicates and descendants of other paths are filtered out.
|
* Duplicates and descendants of other paths are filtered out.
|
||||||
|
@ -47,11 +42,13 @@ export interface Globber {
|
||||||
}
|
}
|
||||||
|
|
||||||
export class DefaultGlobber implements Globber {
|
export class DefaultGlobber implements Globber {
|
||||||
options: GlobOptions = {}
|
private readonly options: GlobOptions
|
||||||
private readonly patterns: Pattern[] = []
|
private readonly patterns: Pattern[] = []
|
||||||
private readonly searchPaths: string[] = []
|
private readonly searchPaths: string[] = []
|
||||||
|
|
||||||
private constructor() {}
|
private constructor(options?: GlobOptions) {
|
||||||
|
this.options = globOptionsHelper.getOptions(options)
|
||||||
|
}
|
||||||
|
|
||||||
getSearchPaths(): string[] {
|
getSearchPaths(): string[] {
|
||||||
// Return a copy
|
// 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> {
|
static async create(
|
||||||
const result = new DefaultGlobber()
|
patterns: string,
|
||||||
|
options?: GlobOptions
|
||||||
|
): Promise<DefaultGlobber> {
|
||||||
|
const result = new DefaultGlobber(options)
|
||||||
|
|
||||||
if (IS_WINDOWS) {
|
if (IS_WINDOWS) {
|
||||||
input = input.replace(/\r\n/g, '\n')
|
patterns = patterns.replace(/\r\n/g, '\n')
|
||||||
input = input.replace(/\r/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) {
|
for (const line of lines) {
|
||||||
// Empty or comment
|
// Empty or comment
|
||||||
if (!line || line.startsWith('#')) {
|
if (!line || line.startsWith('#')) {
|
||||||
continue
|
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
|
// Pattern
|
||||||
else {
|
else {
|
||||||
result.patterns.push(new Pattern(line))
|
result.patterns.push(new Pattern(line))
|
||||||
|
|
Loading…
Reference in New Issue