637 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			637 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
// A simple implementation of make-array
 | 
						|
function makeArray (subject) {
 | 
						|
  return Array.isArray(subject)
 | 
						|
    ? subject
 | 
						|
    : [subject]
 | 
						|
}
 | 
						|
 | 
						|
const EMPTY = ''
 | 
						|
const SPACE = ' '
 | 
						|
const ESCAPE = '\\'
 | 
						|
const REGEX_TEST_BLANK_LINE = /^\s+$/
 | 
						|
const REGEX_INVALID_TRAILING_BACKSLASH = /(?:[^\\]|^)\\$/
 | 
						|
const REGEX_REPLACE_LEADING_EXCAPED_EXCLAMATION = /^\\!/
 | 
						|
const REGEX_REPLACE_LEADING_EXCAPED_HASH = /^\\#/
 | 
						|
const REGEX_SPLITALL_CRLF = /\r?\n/g
 | 
						|
// /foo,
 | 
						|
// ./foo,
 | 
						|
// ../foo,
 | 
						|
// .
 | 
						|
// ..
 | 
						|
const REGEX_TEST_INVALID_PATH = /^\.*\/|^\.+$/
 | 
						|
 | 
						|
const SLASH = '/'
 | 
						|
 | 
						|
// Do not use ternary expression here, since "istanbul ignore next" is buggy
 | 
						|
let TMP_KEY_IGNORE = 'node-ignore'
 | 
						|
/* istanbul ignore else */
 | 
						|
if (typeof Symbol !== 'undefined') {
 | 
						|
  TMP_KEY_IGNORE = Symbol.for('node-ignore')
 | 
						|
}
 | 
						|
const KEY_IGNORE = TMP_KEY_IGNORE
 | 
						|
 | 
						|
const define = (object, key, value) =>
 | 
						|
  Object.defineProperty(object, key, {value})
 | 
						|
 | 
						|
const REGEX_REGEXP_RANGE = /([0-z])-([0-z])/g
 | 
						|
 | 
						|
const RETURN_FALSE = () => false
 | 
						|
 | 
						|
// Sanitize the range of a regular expression
 | 
						|
// The cases are complicated, see test cases for details
 | 
						|
const sanitizeRange = range => range.replace(
 | 
						|
  REGEX_REGEXP_RANGE,
 | 
						|
  (match, from, to) => from.charCodeAt(0) <= to.charCodeAt(0)
 | 
						|
    ? match
 | 
						|
    // Invalid range (out of order) which is ok for gitignore rules but
 | 
						|
    //   fatal for JavaScript regular expression, so eliminate it.
 | 
						|
    : EMPTY
 | 
						|
)
 | 
						|
 | 
						|
// See fixtures #59
 | 
						|
const cleanRangeBackSlash = slashes => {
 | 
						|
  const {length} = slashes
 | 
						|
  return slashes.slice(0, length - length % 2)
 | 
						|
}
 | 
						|
 | 
						|
// > If the pattern ends with a slash,
 | 
						|
// > it is removed for the purpose of the following description,
 | 
						|
// > but it would only find a match with a directory.
 | 
						|
// > In other words, foo/ will match a directory foo and paths underneath it,
 | 
						|
// > but will not match a regular file or a symbolic link foo
 | 
						|
// >  (this is consistent with the way how pathspec works in general in Git).
 | 
						|
// '`foo/`' will not match regular file '`foo`' or symbolic link '`foo`'
 | 
						|
// -> ignore-rules will not deal with it, because it costs extra `fs.stat` call
 | 
						|
//      you could use option `mark: true` with `glob`
 | 
						|
 | 
						|
// '`foo/`' should not continue with the '`..`'
 | 
						|
const REPLACERS = [
 | 
						|
 | 
						|
  [
 | 
						|
    // remove BOM
 | 
						|
    // TODO:
 | 
						|
    // Other similar zero-width characters?
 | 
						|
    /^\uFEFF/,
 | 
						|
    () => EMPTY
 | 
						|
  ],
 | 
						|
 | 
						|
  // > Trailing spaces are ignored unless they are quoted with backslash ("\")
 | 
						|
  [
 | 
						|
    // (a\ ) -> (a )
 | 
						|
    // (a  ) -> (a)
 | 
						|
    // (a ) -> (a)
 | 
						|
    // (a \ ) -> (a  )
 | 
						|
    /((?:\\\\)*?)(\\?\s+)$/,
 | 
						|
    (_, m1, m2) => m1 + (
 | 
						|
      m2.indexOf('\\') === 0
 | 
						|
        ? SPACE
 | 
						|
        : EMPTY
 | 
						|
    )
 | 
						|
  ],
 | 
						|
 | 
						|
  // replace (\ ) with ' '
 | 
						|
  // (\ ) -> ' '
 | 
						|
  // (\\ ) -> '\\ '
 | 
						|
  // (\\\ ) -> '\\ '
 | 
						|
  [
 | 
						|
    /(\\+?)\s/g,
 | 
						|
    (_, m1) => {
 | 
						|
      const {length} = m1
 | 
						|
      return m1.slice(0, length - length % 2) + SPACE
 | 
						|
    }
 | 
						|
  ],
 | 
						|
 | 
						|
  // Escape metacharacters
 | 
						|
  // which is written down by users but means special for regular expressions.
 | 
						|
 | 
						|
  // > There are 12 characters with special meanings:
 | 
						|
  // > - the backslash \,
 | 
						|
  // > - the caret ^,
 | 
						|
  // > - the dollar sign $,
 | 
						|
  // > - the period or dot .,
 | 
						|
  // > - the vertical bar or pipe symbol |,
 | 
						|
  // > - the question mark ?,
 | 
						|
  // > - the asterisk or star *,
 | 
						|
  // > - the plus sign +,
 | 
						|
  // > - the opening parenthesis (,
 | 
						|
  // > - the closing parenthesis ),
 | 
						|
  // > - and the opening square bracket [,
 | 
						|
  // > - the opening curly brace {,
 | 
						|
  // > These special characters are often called "metacharacters".
 | 
						|
  [
 | 
						|
    /[\\$.|*+(){^]/g,
 | 
						|
    match => `\\${match}`
 | 
						|
  ],
 | 
						|
 | 
						|
  [
 | 
						|
    // > a question mark (?) matches a single character
 | 
						|
    /(?!\\)\?/g,
 | 
						|
    () => '[^/]'
 | 
						|
  ],
 | 
						|
 | 
						|
  // leading slash
 | 
						|
  [
 | 
						|
 | 
						|
    // > A leading slash matches the beginning of the pathname.
 | 
						|
    // > For example, "/*.c" matches "cat-file.c" but not "mozilla-sha1/sha1.c".
 | 
						|
    // A leading slash matches the beginning of the pathname
 | 
						|
    /^\//,
 | 
						|
    () => '^'
 | 
						|
  ],
 | 
						|
 | 
						|
  // replace special metacharacter slash after the leading slash
 | 
						|
  [
 | 
						|
    /\//g,
 | 
						|
    () => '\\/'
 | 
						|
  ],
 | 
						|
 | 
						|
  [
 | 
						|
    // > A leading "**" followed by a slash means match in all directories.
 | 
						|
    // > For example, "**/foo" matches file or directory "foo" anywhere,
 | 
						|
    // > the same as pattern "foo".
 | 
						|
    // > "**/foo/bar" matches file or directory "bar" anywhere that is directly
 | 
						|
    // >   under directory "foo".
 | 
						|
    // Notice that the '*'s have been replaced as '\\*'
 | 
						|
    /^\^*\\\*\\\*\\\//,
 | 
						|
 | 
						|
    // '**/foo' <-> 'foo'
 | 
						|
    () => '^(?:.*\\/)?'
 | 
						|
  ],
 | 
						|
 | 
						|
  // starting
 | 
						|
  [
 | 
						|
    // there will be no leading '/'
 | 
						|
    //   (which has been replaced by section "leading slash")
 | 
						|
    // If starts with '**', adding a '^' to the regular expression also works
 | 
						|
    /^(?=[^^])/,
 | 
						|
    function startingReplacer () {
 | 
						|
      // If has a slash `/` at the beginning or middle
 | 
						|
      return !/\/(?!$)/.test(this)
 | 
						|
        // > Prior to 2.22.1
 | 
						|
        // > If the pattern does not contain a slash /,
 | 
						|
        // >   Git treats it as a shell glob pattern
 | 
						|
        // Actually, if there is only a trailing slash,
 | 
						|
        //   git also treats it as a shell glob pattern
 | 
						|
 | 
						|
        // After 2.22.1 (compatible but clearer)
 | 
						|
        // > If there is a separator at the beginning or middle (or both)
 | 
						|
        // > of the pattern, then the pattern is relative to the directory
 | 
						|
        // > level of the particular .gitignore file itself.
 | 
						|
        // > Otherwise the pattern may also match at any level below
 | 
						|
        // > the .gitignore level.
 | 
						|
        ? '(?:^|\\/)'
 | 
						|
 | 
						|
        // > Otherwise, Git treats the pattern as a shell glob suitable for
 | 
						|
        // >   consumption by fnmatch(3)
 | 
						|
        : '^'
 | 
						|
    }
 | 
						|
  ],
 | 
						|
 | 
						|
  // two globstars
 | 
						|
  [
 | 
						|
    // Use lookahead assertions so that we could match more than one `'/**'`
 | 
						|
    /\\\/\\\*\\\*(?=\\\/|$)/g,
 | 
						|
 | 
						|
    // Zero, one or several directories
 | 
						|
    // should not use '*', or it will be replaced by the next replacer
 | 
						|
 | 
						|
    // Check if it is not the last `'/**'`
 | 
						|
    (_, index, str) => index + 6 < str.length
 | 
						|
 | 
						|
      // case: /**/
 | 
						|
      // > A slash followed by two consecutive asterisks then a slash matches
 | 
						|
      // >   zero or more directories.
 | 
						|
      // > For example, "a/**/b" matches "a/b", "a/x/b", "a/x/y/b" and so on.
 | 
						|
      // '/**/'
 | 
						|
      ? '(?:\\/[^\\/]+)*'
 | 
						|
 | 
						|
      // case: /**
 | 
						|
      // > A trailing `"/**"` matches everything inside.
 | 
						|
 | 
						|
      // #21: everything inside but it should not include the current folder
 | 
						|
      : '\\/.+'
 | 
						|
  ],
 | 
						|
 | 
						|
  // normal intermediate wildcards
 | 
						|
  [
 | 
						|
    // Never replace escaped '*'
 | 
						|
    // ignore rule '\*' will match the path '*'
 | 
						|
 | 
						|
    // 'abc.*/' -> go
 | 
						|
    // 'abc.*'  -> skip this rule,
 | 
						|
    //    coz trailing single wildcard will be handed by [trailing wildcard]
 | 
						|
    /(^|[^\\]+)(\\\*)+(?=.+)/g,
 | 
						|
 | 
						|
    // '*.js' matches '.js'
 | 
						|
    // '*.js' doesn't match 'abc'
 | 
						|
    (_, p1, p2) => {
 | 
						|
      // 1.
 | 
						|
      // > An asterisk "*" matches anything except a slash.
 | 
						|
      // 2.
 | 
						|
      // > Other consecutive asterisks are considered regular asterisks
 | 
						|
      // > and will match according to the previous rules.
 | 
						|
      const unescaped = p2.replace(/\\\*/g, '[^\\/]*')
 | 
						|
      return p1 + unescaped
 | 
						|
    }
 | 
						|
  ],
 | 
						|
 | 
						|
  [
 | 
						|
    // unescape, revert step 3 except for back slash
 | 
						|
    // For example, if a user escape a '\\*',
 | 
						|
    // after step 3, the result will be '\\\\\\*'
 | 
						|
    /\\\\\\(?=[$.|*+(){^])/g,
 | 
						|
    () => ESCAPE
 | 
						|
  ],
 | 
						|
 | 
						|
  [
 | 
						|
    // '\\\\' -> '\\'
 | 
						|
    /\\\\/g,
 | 
						|
    () => ESCAPE
 | 
						|
  ],
 | 
						|
 | 
						|
  [
 | 
						|
    // > The range notation, e.g. [a-zA-Z],
 | 
						|
    // > can be used to match one of the characters in a range.
 | 
						|
 | 
						|
    // `\` is escaped by step 3
 | 
						|
    /(\\)?\[([^\]/]*?)(\\*)($|\])/g,
 | 
						|
    (match, leadEscape, range, endEscape, close) => leadEscape === ESCAPE
 | 
						|
      // '\\[bar]' -> '\\\\[bar\\]'
 | 
						|
      ? `\\[${range}${cleanRangeBackSlash(endEscape)}${close}`
 | 
						|
      : close === ']'
 | 
						|
        ? endEscape.length % 2 === 0
 | 
						|
          // A normal case, and it is a range notation
 | 
						|
          // '[bar]'
 | 
						|
          // '[bar\\\\]'
 | 
						|
          ? `[${sanitizeRange(range)}${endEscape}]`
 | 
						|
          // Invalid range notaton
 | 
						|
          // '[bar\\]' -> '[bar\\\\]'
 | 
						|
          : '[]'
 | 
						|
        : '[]'
 | 
						|
  ],
 | 
						|
 | 
						|
  // ending
 | 
						|
  [
 | 
						|
    // 'js' will not match 'js.'
 | 
						|
    // 'ab' will not match 'abc'
 | 
						|
    /(?:[^*])$/,
 | 
						|
 | 
						|
    // WTF!
 | 
						|
    // https://git-scm.com/docs/gitignore
 | 
						|
    // changes in [2.22.1](https://git-scm.com/docs/gitignore/2.22.1)
 | 
						|
    // which re-fixes #24, #38
 | 
						|
 | 
						|
    // > If there is a separator at the end of the pattern then the pattern
 | 
						|
    // > will only match directories, otherwise the pattern can match both
 | 
						|
    // > files and directories.
 | 
						|
 | 
						|
    // 'js*' will not match 'a.js'
 | 
						|
    // 'js/' will not match 'a.js'
 | 
						|
    // 'js' will match 'a.js' and 'a.js/'
 | 
						|
    match => /\/$/.test(match)
 | 
						|
      // foo/ will not match 'foo'
 | 
						|
      ? `${match}$`
 | 
						|
      // foo matches 'foo' and 'foo/'
 | 
						|
      : `${match}(?=$|\\/$)`
 | 
						|
  ],
 | 
						|
 | 
						|
  // trailing wildcard
 | 
						|
  [
 | 
						|
    /(\^|\\\/)?\\\*$/,
 | 
						|
    (_, p1) => {
 | 
						|
      const prefix = p1
 | 
						|
        // '\^':
 | 
						|
        // '/*' does not match EMPTY
 | 
						|
        // '/*' does not match everything
 | 
						|
 | 
						|
        // '\\\/':
 | 
						|
        // 'abc/*' does not match 'abc/'
 | 
						|
        ? `${p1}[^/]+`
 | 
						|
 | 
						|
        // 'a*' matches 'a'
 | 
						|
        // 'a*' matches 'aa'
 | 
						|
        : '[^/]*'
 | 
						|
 | 
						|
      return `${prefix}(?=$|\\/$)`
 | 
						|
    }
 | 
						|
  ],
 | 
						|
]
 | 
						|
 | 
						|
// A simple cache, because an ignore rule only has only one certain meaning
 | 
						|
const regexCache = Object.create(null)
 | 
						|
 | 
						|
// @param {pattern}
 | 
						|
const makeRegex = (pattern, ignoreCase) => {
 | 
						|
  let source = regexCache[pattern]
 | 
						|
 | 
						|
  if (!source) {
 | 
						|
    source = REPLACERS.reduce(
 | 
						|
      (prev, [matcher, replacer]) =>
 | 
						|
        prev.replace(matcher, replacer.bind(pattern)),
 | 
						|
      pattern
 | 
						|
    )
 | 
						|
    regexCache[pattern] = source
 | 
						|
  }
 | 
						|
 | 
						|
  return ignoreCase
 | 
						|
    ? new RegExp(source, 'i')
 | 
						|
    : new RegExp(source)
 | 
						|
}
 | 
						|
 | 
						|
const isString = subject => typeof subject === 'string'
 | 
						|
 | 
						|
// > A blank line matches no files, so it can serve as a separator for readability.
 | 
						|
const checkPattern = pattern => pattern
 | 
						|
  && isString(pattern)
 | 
						|
  && !REGEX_TEST_BLANK_LINE.test(pattern)
 | 
						|
  && !REGEX_INVALID_TRAILING_BACKSLASH.test(pattern)
 | 
						|
 | 
						|
  // > A line starting with # serves as a comment.
 | 
						|
  && pattern.indexOf('#') !== 0
 | 
						|
 | 
						|
const splitPattern = pattern => pattern.split(REGEX_SPLITALL_CRLF)
 | 
						|
 | 
						|
class IgnoreRule {
 | 
						|
  constructor (
 | 
						|
    origin,
 | 
						|
    pattern,
 | 
						|
    negative,
 | 
						|
    regex
 | 
						|
  ) {
 | 
						|
    this.origin = origin
 | 
						|
    this.pattern = pattern
 | 
						|
    this.negative = negative
 | 
						|
    this.regex = regex
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const createRule = (pattern, ignoreCase) => {
 | 
						|
  const origin = pattern
 | 
						|
  let negative = false
 | 
						|
 | 
						|
  // > An optional prefix "!" which negates the pattern;
 | 
						|
  if (pattern.indexOf('!') === 0) {
 | 
						|
    negative = true
 | 
						|
    pattern = pattern.substr(1)
 | 
						|
  }
 | 
						|
 | 
						|
  pattern = pattern
 | 
						|
  // > Put a backslash ("\") in front of the first "!" for patterns that
 | 
						|
  // >   begin with a literal "!", for example, `"\!important!.txt"`.
 | 
						|
  .replace(REGEX_REPLACE_LEADING_EXCAPED_EXCLAMATION, '!')
 | 
						|
  // > Put a backslash ("\") in front of the first hash for patterns that
 | 
						|
  // >   begin with a hash.
 | 
						|
  .replace(REGEX_REPLACE_LEADING_EXCAPED_HASH, '#')
 | 
						|
 | 
						|
  const regex = makeRegex(pattern, ignoreCase)
 | 
						|
 | 
						|
  return new IgnoreRule(
 | 
						|
    origin,
 | 
						|
    pattern,
 | 
						|
    negative,
 | 
						|
    regex
 | 
						|
  )
 | 
						|
}
 | 
						|
 | 
						|
const throwError = (message, Ctor) => {
 | 
						|
  throw new Ctor(message)
 | 
						|
}
 | 
						|
 | 
						|
const checkPath = (path, originalPath, doThrow) => {
 | 
						|
  if (!isString(path)) {
 | 
						|
    return doThrow(
 | 
						|
      `path must be a string, but got \`${originalPath}\``,
 | 
						|
      TypeError
 | 
						|
    )
 | 
						|
  }
 | 
						|
 | 
						|
  // We don't know if we should ignore EMPTY, so throw
 | 
						|
  if (!path) {
 | 
						|
    return doThrow(`path must not be empty`, TypeError)
 | 
						|
  }
 | 
						|
 | 
						|
  // Check if it is a relative path
 | 
						|
  if (checkPath.isNotRelative(path)) {
 | 
						|
    const r = '`path.relative()`d'
 | 
						|
    return doThrow(
 | 
						|
      `path should be a ${r} string, but got "${originalPath}"`,
 | 
						|
      RangeError
 | 
						|
    )
 | 
						|
  }
 | 
						|
 | 
						|
  return true
 | 
						|
}
 | 
						|
 | 
						|
const isNotRelative = path => REGEX_TEST_INVALID_PATH.test(path)
 | 
						|
 | 
						|
checkPath.isNotRelative = isNotRelative
 | 
						|
checkPath.convert = p => p
 | 
						|
 | 
						|
class Ignore {
 | 
						|
  constructor ({
 | 
						|
    ignorecase = true,
 | 
						|
    ignoreCase = ignorecase,
 | 
						|
    allowRelativePaths = false
 | 
						|
  } = {}) {
 | 
						|
    define(this, KEY_IGNORE, true)
 | 
						|
 | 
						|
    this._rules = []
 | 
						|
    this._ignoreCase = ignoreCase
 | 
						|
    this._allowRelativePaths = allowRelativePaths
 | 
						|
    this._initCache()
 | 
						|
  }
 | 
						|
 | 
						|
  _initCache () {
 | 
						|
    this._ignoreCache = Object.create(null)
 | 
						|
    this._testCache = Object.create(null)
 | 
						|
  }
 | 
						|
 | 
						|
  _addPattern (pattern) {
 | 
						|
    // #32
 | 
						|
    if (pattern && pattern[KEY_IGNORE]) {
 | 
						|
      this._rules = this._rules.concat(pattern._rules)
 | 
						|
      this._added = true
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    if (checkPattern(pattern)) {
 | 
						|
      const rule = createRule(pattern, this._ignoreCase)
 | 
						|
      this._added = true
 | 
						|
      this._rules.push(rule)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // @param {Array<string> | string | Ignore} pattern
 | 
						|
  add (pattern) {
 | 
						|
    this._added = false
 | 
						|
 | 
						|
    makeArray(
 | 
						|
      isString(pattern)
 | 
						|
        ? splitPattern(pattern)
 | 
						|
        : pattern
 | 
						|
    ).forEach(this._addPattern, this)
 | 
						|
 | 
						|
    // Some rules have just added to the ignore,
 | 
						|
    // making the behavior changed.
 | 
						|
    if (this._added) {
 | 
						|
      this._initCache()
 | 
						|
    }
 | 
						|
 | 
						|
    return this
 | 
						|
  }
 | 
						|
 | 
						|
  // legacy
 | 
						|
  addPattern (pattern) {
 | 
						|
    return this.add(pattern)
 | 
						|
  }
 | 
						|
 | 
						|
  //          |           ignored : unignored
 | 
						|
  // negative |   0:0   |   0:1   |   1:0   |   1:1
 | 
						|
  // -------- | ------- | ------- | ------- | --------
 | 
						|
  //     0    |  TEST   |  TEST   |  SKIP   |    X
 | 
						|
  //     1    |  TESTIF |  SKIP   |  TEST   |    X
 | 
						|
 | 
						|
  // - SKIP: always skip
 | 
						|
  // - TEST: always test
 | 
						|
  // - TESTIF: only test if checkUnignored
 | 
						|
  // - X: that never happen
 | 
						|
 | 
						|
  // @param {boolean} whether should check if the path is unignored,
 | 
						|
  //   setting `checkUnignored` to `false` could reduce additional
 | 
						|
  //   path matching.
 | 
						|
 | 
						|
  // @returns {TestResult} true if a file is ignored
 | 
						|
  _testOne (path, checkUnignored) {
 | 
						|
    let ignored = false
 | 
						|
    let unignored = false
 | 
						|
 | 
						|
    this._rules.forEach(rule => {
 | 
						|
      const {negative} = rule
 | 
						|
      if (
 | 
						|
        unignored === negative && ignored !== unignored
 | 
						|
        || negative && !ignored && !unignored && !checkUnignored
 | 
						|
      ) {
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      const matched = rule.regex.test(path)
 | 
						|
 | 
						|
      if (matched) {
 | 
						|
        ignored = !negative
 | 
						|
        unignored = negative
 | 
						|
      }
 | 
						|
    })
 | 
						|
 | 
						|
    return {
 | 
						|
      ignored,
 | 
						|
      unignored
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // @returns {TestResult}
 | 
						|
  _test (originalPath, cache, checkUnignored, slices) {
 | 
						|
    const path = originalPath
 | 
						|
      // Supports nullable path
 | 
						|
      && checkPath.convert(originalPath)
 | 
						|
 | 
						|
    checkPath(
 | 
						|
      path,
 | 
						|
      originalPath,
 | 
						|
      this._allowRelativePaths
 | 
						|
        ? RETURN_FALSE
 | 
						|
        : throwError
 | 
						|
    )
 | 
						|
 | 
						|
    return this._t(path, cache, checkUnignored, slices)
 | 
						|
  }
 | 
						|
 | 
						|
  _t (path, cache, checkUnignored, slices) {
 | 
						|
    if (path in cache) {
 | 
						|
      return cache[path]
 | 
						|
    }
 | 
						|
 | 
						|
    if (!slices) {
 | 
						|
      // path/to/a.js
 | 
						|
      // ['path', 'to', 'a.js']
 | 
						|
      slices = path.split(SLASH)
 | 
						|
    }
 | 
						|
 | 
						|
    slices.pop()
 | 
						|
 | 
						|
    // If the path has no parent directory, just test it
 | 
						|
    if (!slices.length) {
 | 
						|
      return cache[path] = this._testOne(path, checkUnignored)
 | 
						|
    }
 | 
						|
 | 
						|
    const parent = this._t(
 | 
						|
      slices.join(SLASH) + SLASH,
 | 
						|
      cache,
 | 
						|
      checkUnignored,
 | 
						|
      slices
 | 
						|
    )
 | 
						|
 | 
						|
    // If the path contains a parent directory, check the parent first
 | 
						|
    return cache[path] = parent.ignored
 | 
						|
      // > It is not possible to re-include a file if a parent directory of
 | 
						|
      // >   that file is excluded.
 | 
						|
      ? parent
 | 
						|
      : this._testOne(path, checkUnignored)
 | 
						|
  }
 | 
						|
 | 
						|
  ignores (path) {
 | 
						|
    return this._test(path, this._ignoreCache, false).ignored
 | 
						|
  }
 | 
						|
 | 
						|
  createFilter () {
 | 
						|
    return path => !this.ignores(path)
 | 
						|
  }
 | 
						|
 | 
						|
  filter (paths) {
 | 
						|
    return makeArray(paths).filter(this.createFilter())
 | 
						|
  }
 | 
						|
 | 
						|
  // @returns {TestResult}
 | 
						|
  test (path) {
 | 
						|
    return this._test(path, this._testCache, true)
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const factory = options => new Ignore(options)
 | 
						|
 | 
						|
const isPathValid = path =>
 | 
						|
  checkPath(path && checkPath.convert(path), path, RETURN_FALSE)
 | 
						|
 | 
						|
factory.isPathValid = isPathValid
 | 
						|
 | 
						|
// Fixes typescript
 | 
						|
factory.default = factory
 | 
						|
 | 
						|
module.exports = factory
 | 
						|
 | 
						|
// Windows
 | 
						|
// --------------------------------------------------------------
 | 
						|
/* istanbul ignore if */
 | 
						|
if (
 | 
						|
  // Detect `process` so that it can run in browsers.
 | 
						|
  typeof process !== 'undefined'
 | 
						|
  && (
 | 
						|
    process.env && process.env.IGNORE_TEST_WIN32
 | 
						|
    || process.platform === 'win32'
 | 
						|
  )
 | 
						|
) {
 | 
						|
  /* eslint no-control-regex: "off" */
 | 
						|
  const makePosix = str => /^\\\\\?\\/.test(str)
 | 
						|
  || /["<>|\u0000-\u001F]+/u.test(str)
 | 
						|
    ? str
 | 
						|
    : str.replace(/\\/g, '/')
 | 
						|
 | 
						|
  checkPath.convert = makePosix
 | 
						|
 | 
						|
  // 'C:\\foo'     <- 'C:\\foo' has been converted to 'C:/'
 | 
						|
  // 'd:\\foo'
 | 
						|
  const REGIX_IS_WINDOWS_PATH_ABSOLUTE = /^[a-z]:\//i
 | 
						|
  checkPath.isNotRelative = path =>
 | 
						|
    REGIX_IS_WINDOWS_PATH_ABSOLUTE.test(path)
 | 
						|
    || isNotRelative(path)
 | 
						|
}
 |