// 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)
							 | 
						|
								}
							 |