| 
						 | 
						- 'use strict';
 - 
 - var path = require('path');
 - var minimatch = require('minimatch');
 - var createDebug = require('debug');
 - var objectSchema = require('@humanwhocodes/object-schema');
 - 
 - /**
 -  * @fileoverview ConfigSchema
 -  * @author Nicholas C. Zakas
 -  */
 - 
 - //------------------------------------------------------------------------------
 - // Helpers
 - //------------------------------------------------------------------------------
 - 
 - const NOOP_STRATEGY = {
 - 	required: false,
 - 	merge() {
 - 		return undefined;
 - 	},
 - 	validate() { }
 - };
 - 
 - //------------------------------------------------------------------------------
 - // Exports
 - //------------------------------------------------------------------------------
 - 
 - /**
 -  * The base schema that every ConfigArray uses.
 -  * @type Object
 -  */
 - const baseSchema = Object.freeze({
 - 	name: {
 - 		required: false,
 - 		merge() {
 - 			return undefined;
 - 		},
 - 		validate(value) {
 - 			if (typeof value !== 'string') {
 - 				throw new TypeError('Property must be a string.');
 - 			}
 - 		}
 - 	},
 - 	files: NOOP_STRATEGY,
 - 	ignores: NOOP_STRATEGY
 - });
 - 
 - /**
 -  * @fileoverview ConfigSchema
 -  * @author Nicholas C. Zakas
 -  */
 - 
 - //------------------------------------------------------------------------------
 - // Helpers
 - //------------------------------------------------------------------------------
 - 
 - /**
 -  * Asserts that a given value is an array.
 -  * @param {*} value The value to check.
 -  * @returns {void}
 -  * @throws {TypeError} When the value is not an array.
 -  */
 - function assertIsArray(value) {
 - 	if (!Array.isArray(value)) {
 - 		throw new TypeError('Expected value to be an array.');
 - 	}
 - }
 - 
 - /**
 -  * Asserts that a given value is an array containing only strings and functions.
 -  * @param {*} value The value to check.
 -  * @returns {void}
 -  * @throws {TypeError} When the value is not an array of strings and functions.
 -  */
 - function assertIsArrayOfStringsAndFunctions(value, name) {
 - 	assertIsArray(value);
 - 
 - 	if (value.some(item => typeof item !== 'string' && typeof item !== 'function')) {
 - 		throw new TypeError('Expected array to only contain strings and functions.');
 - 	}
 - }
 - 
 - /**
 -  * Asserts that a given value is a non-empty array.
 -  * @param {*} value The value to check.
 -  * @returns {void}
 -  * @throws {TypeError} When the value is not an array or an empty array.
 -  */
 - function assertIsNonEmptyArray(value) {
 - 	if (!Array.isArray(value) || value.length === 0) {
 - 		throw new TypeError('Expected value to be a non-empty array.');
 - 	}
 - }
 - 
 - //------------------------------------------------------------------------------
 - // Exports
 - //------------------------------------------------------------------------------
 - 
 - /**
 -  * The schema for `files` and `ignores` that every ConfigArray uses.
 -  * @type Object
 -  */
 - const filesAndIgnoresSchema = Object.freeze({
 - 	files: {
 - 		required: false,
 - 		merge() {
 - 			return undefined;
 - 		},
 - 		validate(value) {
 - 
 - 			// first check if it's an array
 - 			assertIsNonEmptyArray(value);
 - 
 - 			// then check each member
 - 			value.forEach(item => {
 - 				if (Array.isArray(item)) {
 - 					assertIsArrayOfStringsAndFunctions(item);
 - 				} else if (typeof item !== 'string' && typeof item !== 'function') {
 - 					throw new TypeError('Items must be a string, a function, or an array of strings and functions.');
 - 				}
 - 			});
 - 
 - 		}
 - 	},
 - 	ignores: {
 - 		required: false,
 - 		merge() {
 - 			return undefined;
 - 		},
 - 		validate: assertIsArrayOfStringsAndFunctions
 - 	}
 - });
 - 
 - /**
 -  * @fileoverview ConfigArray
 -  * @author Nicholas C. Zakas
 -  */
 - 
 - 
 - //------------------------------------------------------------------------------
 - // Helpers
 - //------------------------------------------------------------------------------
 - 
 - const Minimatch = minimatch.Minimatch;
 - const minimatchCache = new Map();
 - const negatedMinimatchCache = new Map();
 - const debug = createDebug('@hwc/config-array');
 - 
 - const MINIMATCH_OPTIONS = {
 - 	// matchBase: true,
 - 	dot: true
 - };
 - 
 - const CONFIG_TYPES = new Set(['array', 'function']);
 - 
 - /**
 -  * Fields that are considered metadata and not part of the config object.
 -  */
 - const META_FIELDS = new Set(['name']);
 - 
 - const FILES_AND_IGNORES_SCHEMA = new objectSchema.ObjectSchema(filesAndIgnoresSchema);
 - 
 - /**
 -  * Wrapper error for config validation errors that adds a name to the front of the
 -  * error message.
 -  */
 - class ConfigError extends Error {
 - 
 - 	/**
 - 	 * Creates a new instance.
 - 	 * @param {string} name The config object name causing the error.
 - 	 * @param {number} index The index of the config object in the array.
 - 	 * @param {Error} source The source error. 
 - 	 */
 - 	constructor(name, index, { cause, message }) {
 - 
 - 
 - 		const finalMessage = message || cause.message;
 - 
 - 		super(`Config ${name}: ${finalMessage}`, { cause });
 - 
 - 		// copy over custom properties that aren't represented
 - 		if (cause) {
 - 			for (const key of Object.keys(cause)) {
 - 				if (!(key in this)) {
 - 					this[key] = cause[key];
 - 				}
 - 			}
 - 		}
 - 
 - 		/**
 - 		 * The name of the error.
 - 		 * @type {string}
 - 		 * @readonly
 - 		 */
 - 		this.name = 'ConfigError';
 - 
 - 		/**
 - 		 * The index of the config object in the array.
 - 		 * @type {number}
 - 		 * @readonly
 - 		 */
 - 		this.index = index;
 - 	}
 - }
 - 
 - /**
 -  * Gets the name of a config object.
 -  * @param {object} config The config object to get the name of.
 -  * @returns {string} The name of the config object.
 -  */ 
 - function getConfigName(config) {
 - 	if (config && typeof config.name === 'string' && config.name) {
 - 		return `"${config.name}"`;
 - 	}
 - 
 - 	return '(unnamed)';
 - }
 - 
 - /**
 -  * Rethrows a config error with additional information about the config object.
 -  * @param {object} config The config object to get the name of. 
 -  * @param {number} index The index of the config object in the array.
 -  * @param {Error} error The error to rethrow.
 -  * @throws {ConfigError} When the error is rethrown for a config.
 -  */
 - function rethrowConfigError(config, index, error) {
 - 	const configName = getConfigName(config);
 - 	throw new ConfigError(configName, index, error);
 - }
 - 
 - /**
 -  * Shorthand for checking if a value is a string.
 -  * @param {any} value The value to check.
 -  * @returns {boolean} True if a string, false if not. 
 -  */
 - function isString(value) {
 - 	return typeof value === 'string';
 - }
 - 
 - /**
 -  * Creates a function that asserts that the config is valid
 -  * during normalization. This checks that the config is not nullish
 -  * and that files and ignores keys  of a config object are valid as per base schema.
 -  * @param {Object} config The config object to check.
 -  * @param {number} index The index of the config object in the array.
 -  * @returns {void}
 -  * @throws {ConfigError} If the files and ignores keys of a config object are not valid.
 -  */
 - function assertValidBaseConfig(config, index) {
 - 
 - 	if (config === null) {
 - 		throw new ConfigError(getConfigName(config), index, { message: 'Unexpected null config.' });
 - 	}
 - 
 - 	if (config === undefined) {
 - 		throw new ConfigError(getConfigName(config), index, { message: 'Unexpected undefined config.' });
 - 	}
 - 
 - 	if (typeof config !== 'object') {
 - 		throw new ConfigError(getConfigName(config), index, { message: 'Unexpected non-object config.' });
 - 	}
 - 
 - 	const validateConfig = { };
 - 	
 - 	if ('files' in config) {
 - 		validateConfig.files = config.files;
 - 	}
 - 	
 - 	if ('ignores' in config) {
 - 		validateConfig.ignores = config.ignores;
 - 	}
 - 
 - 	try {
 - 		FILES_AND_IGNORES_SCHEMA.validate(validateConfig);
 - 	} catch (validationError) {
 - 		rethrowConfigError(config, index, { cause: validationError });
 - 	}
 - }
 - 
 - /**
 -  * Wrapper around minimatch that caches minimatch patterns for
 -  * faster matching speed over multiple file path evaluations.
 -  * @param {string} filepath The file path to match.
 -  * @param {string} pattern The glob pattern to match against.
 -  * @param {object} options The minimatch options to use.
 -  * @returns 
 -  */
 - function doMatch(filepath, pattern, options = {}) {
 - 
 - 	let cache = minimatchCache;
 - 
 - 	if (options.flipNegate) {
 - 		cache = negatedMinimatchCache;
 - 	}
 - 
 - 	let matcher = cache.get(pattern);
 - 
 - 	if (!matcher) {
 - 		matcher = new Minimatch(pattern, Object.assign({}, MINIMATCH_OPTIONS, options));
 - 		cache.set(pattern, matcher);
 - 	}
 - 
 - 	return matcher.match(filepath);
 - }
 - 
 - /**
 -  * Normalizes a `ConfigArray` by flattening it and executing any functions
 -  * that are found inside.
 -  * @param {Array} items The items in a `ConfigArray`.
 -  * @param {Object} context The context object to pass into any function
 -  *      found.
 -  * @param {Array<string>} extraConfigTypes The config types to check.
 -  * @returns {Promise<Array>} A flattened array containing only config objects.
 -  * @throws {TypeError} When a config function returns a function.
 -  */
 - async function normalize(items, context, extraConfigTypes) {
 - 
 - 	const allowFunctions = extraConfigTypes.includes('function');
 - 	const allowArrays = extraConfigTypes.includes('array');
 - 
 - 	async function* flatTraverse(array) {
 - 		for (let item of array) {
 - 			if (typeof item === 'function') {
 - 				if (!allowFunctions) {
 - 					throw new TypeError('Unexpected function.');
 - 				}
 - 
 - 				item = item(context);
 - 				if (item.then) {
 - 					item = await item;
 - 				}
 - 			}
 - 
 - 			if (Array.isArray(item)) {
 - 				if (!allowArrays) {
 - 					throw new TypeError('Unexpected array.');
 - 				}
 - 				yield* flatTraverse(item);
 - 			} else if (typeof item === 'function') {
 - 				throw new TypeError('A config function can only return an object or array.');
 - 			} else {
 - 				yield item;
 - 			}
 - 		}
 - 	}
 - 
 - 	/*
 - 	 * Async iterables cannot be used with the spread operator, so we need to manually
 - 	 * create the array to return.
 - 	 */
 - 	const asyncIterable = await flatTraverse(items);
 - 	const configs = [];
 - 
 - 	for await (const config of asyncIterable) {
 - 		configs.push(config);
 - 	}
 - 
 - 	return configs;
 - }
 - 
 - /**
 -  * Normalizes a `ConfigArray` by flattening it and executing any functions
 -  * that are found inside.
 -  * @param {Array} items The items in a `ConfigArray`.
 -  * @param {Object} context The context object to pass into any function
 -  *      found.
 -  * @param {Array<string>} extraConfigTypes The config types to check.
 -  * @returns {Array} A flattened array containing only config objects.
 -  * @throws {TypeError} When a config function returns a function.
 -  */
 - function normalizeSync(items, context, extraConfigTypes) {
 - 
 - 	const allowFunctions = extraConfigTypes.includes('function');
 - 	const allowArrays = extraConfigTypes.includes('array');
 - 
 - 	function* flatTraverse(array) {
 - 		for (let item of array) {
 - 			if (typeof item === 'function') {
 - 
 - 				if (!allowFunctions) {
 - 					throw new TypeError('Unexpected function.');
 - 				}
 - 
 - 				item = item(context);
 - 				if (item.then) {
 - 					throw new TypeError('Async config functions are not supported.');
 - 				}
 - 			}
 - 
 - 			if (Array.isArray(item)) {
 - 
 - 				if (!allowArrays) {
 - 					throw new TypeError('Unexpected array.');
 - 				}
 - 
 - 				yield* flatTraverse(item);
 - 			} else if (typeof item === 'function') {
 - 				throw new TypeError('A config function can only return an object or array.');
 - 			} else {
 - 				yield item;
 - 			}
 - 		}
 - 	}
 - 
 - 	return [...flatTraverse(items)];
 - }
 - 
 - /**
 -  * Determines if a given file path should be ignored based on the given
 -  * matcher.
 -  * @param {Array<string|() => boolean>} ignores The ignore patterns to check. 
 -  * @param {string} filePath The absolute path of the file to check.
 -  * @param {string} relativeFilePath The relative path of the file to check.
 -  * @returns {boolean} True if the path should be ignored and false if not.
 -  */
 - function shouldIgnorePath(ignores, filePath, relativeFilePath) {
 - 
 - 	// all files outside of the basePath are ignored
 - 	if (relativeFilePath.startsWith('..')) {
 - 		return true;
 - 	}
 - 
 - 	return ignores.reduce((ignored, matcher) => {
 - 
 - 		if (!ignored) {
 - 
 - 			if (typeof matcher === 'function') {
 - 				return matcher(filePath);
 - 			}
 - 
 - 			// don't check negated patterns because we're not ignored yet
 - 			if (!matcher.startsWith('!')) {
 - 				return doMatch(relativeFilePath, matcher);
 - 			}
 - 
 - 			// otherwise we're still not ignored
 - 			return false;
 - 
 - 		}
 - 
 - 		// only need to check negated patterns because we're ignored
 - 		if (typeof matcher === 'string' && matcher.startsWith('!')) {
 - 			return !doMatch(relativeFilePath, matcher, {
 - 				flipNegate: true
 - 			});
 - 		}
 - 
 - 		return ignored;
 - 
 - 	}, false);
 - 
 - }
 - 
 - /**
 -  * Determines if a given file path is matched by a config based on
 -  * `ignores` only.
 -  * @param {string} filePath The absolute file path to check.
 -  * @param {string} basePath The base path for the config.
 -  * @param {Object} config The config object to check.
 -  * @returns {boolean} True if the file path is matched by the config,
 -  *      false if not.
 -  */
 - function pathMatchesIgnores(filePath, basePath, config) {
 - 
 - 	/*
 - 	 * For both files and ignores, functions are passed the absolute
 - 	 * file path while strings are compared against the relative
 - 	 * file path.
 - 	 */
 - 	const relativeFilePath = path.relative(basePath, filePath);
 - 
 - 	return Object.keys(config).filter(key => !META_FIELDS.has(key)).length > 1 &&
 - 		!shouldIgnorePath(config.ignores, filePath, relativeFilePath);
 - }
 - 
 - 
 - /**
 -  * Determines if a given file path is matched by a config. If the config
 -  * has no `files` field, then it matches; otherwise, if a `files` field
 -  * is present then we match the globs in `files` and exclude any globs in
 -  * `ignores`.
 -  * @param {string} filePath The absolute file path to check.
 -  * @param {string} basePath The base path for the config.
 -  * @param {Object} config The config object to check.
 -  * @returns {boolean} True if the file path is matched by the config,
 -  *      false if not.
 -  */
 - function pathMatches(filePath, basePath, config) {
 - 
 - 	/*
 - 	 * For both files and ignores, functions are passed the absolute
 - 	 * file path while strings are compared against the relative
 - 	 * file path.
 - 	 */
 - 	const relativeFilePath = path.relative(basePath, filePath);
 - 
 - 	// match both strings and functions
 - 	const match = pattern => {
 - 
 - 		if (isString(pattern)) {
 - 			return doMatch(relativeFilePath, pattern);
 - 		}
 - 
 - 		if (typeof pattern === 'function') {
 - 			return pattern(filePath);
 - 		}
 - 
 - 		throw new TypeError(`Unexpected matcher type ${pattern}.`);
 - 	};
 - 
 - 	// check for all matches to config.files
 - 	let filePathMatchesPattern = config.files.some(pattern => {
 - 		if (Array.isArray(pattern)) {
 - 			return pattern.every(match);
 - 		}
 - 
 - 		return match(pattern);
 - 	});
 - 
 - 	/*
 - 	 * If the file path matches the config.files patterns, then check to see
 - 	 * if there are any files to ignore.
 - 	 */
 - 	if (filePathMatchesPattern && config.ignores) {
 - 		filePathMatchesPattern = !shouldIgnorePath(config.ignores, filePath, relativeFilePath);
 - 	}
 - 
 - 	return filePathMatchesPattern;
 - }
 - 
 - /**
 -  * Ensures that a ConfigArray has been normalized.
 -  * @param {ConfigArray} configArray The ConfigArray to check. 
 -  * @returns {void}
 -  * @throws {Error} When the `ConfigArray` is not normalized.
 -  */
 - function assertNormalized(configArray) {
 - 	// TODO: Throw more verbose error
 - 	if (!configArray.isNormalized()) {
 - 		throw new Error('ConfigArray must be normalized to perform this operation.');
 - 	}
 - }
 - 
 - /**
 -  * Ensures that config types are valid.
 -  * @param {Array<string>} extraConfigTypes The config types to check.
 -  * @returns {void}
 -  * @throws {Error} When the config types array is invalid.
 -  */
 - function assertExtraConfigTypes(extraConfigTypes) {
 - 	if (extraConfigTypes.length > 2) {
 - 		throw new TypeError('configTypes must be an array with at most two items.');
 - 	}
 - 
 - 	for (const configType of extraConfigTypes) {
 - 		if (!CONFIG_TYPES.has(configType)) {
 - 			throw new TypeError(`Unexpected config type "${configType}" found. Expected one of: "object", "array", "function".`);
 - 		}
 - 	}
 - }
 - 
 - //------------------------------------------------------------------------------
 - // Public Interface
 - //------------------------------------------------------------------------------
 - 
 - const ConfigArraySymbol = {
 - 	isNormalized: Symbol('isNormalized'),
 - 	configCache: Symbol('configCache'),
 - 	schema: Symbol('schema'),
 - 	finalizeConfig: Symbol('finalizeConfig'),
 - 	preprocessConfig: Symbol('preprocessConfig')
 - };
 - 
 - // used to store calculate data for faster lookup
 - const dataCache = new WeakMap();
 - 
 - /**
 -  * Represents an array of config objects and provides method for working with
 -  * those config objects.
 -  */
 - class ConfigArray extends Array {
 - 
 - 	/**
 - 	 * Creates a new instance of ConfigArray.
 - 	 * @param {Iterable|Function|Object} configs An iterable yielding config
 - 	 *      objects, or a config function, or a config object.
 - 	 * @param {string} [options.basePath=""] The path of the config file
 - 	 * @param {boolean} [options.normalized=false] Flag indicating if the
 - 	 *      configs have already been normalized.
 - 	 * @param {Object} [options.schema] The additional schema 
 - 	 *      definitions to use for the ConfigArray schema.
 - 	 * @param {Array<string>} [options.configTypes] List of config types supported.
 - 	 */
 - 	constructor(configs, {
 - 		basePath = '',
 - 		normalized = false,
 - 		schema: customSchema,
 - 		extraConfigTypes = []
 - 	} = {}
 - 	) {
 - 		super();
 - 
 - 		/**
 - 		 * Tracks if the array has been normalized.
 - 		 * @property isNormalized
 - 		 * @type {boolean}
 - 		 * @private
 - 		 */
 - 		this[ConfigArraySymbol.isNormalized] = normalized;
 - 
 - 		/**
 - 		 * The schema used for validating and merging configs.
 - 		 * @property schema
 - 		 * @type ObjectSchema
 - 		 * @private
 - 		 */
 - 		this[ConfigArraySymbol.schema] = new objectSchema.ObjectSchema(
 - 			Object.assign({}, customSchema, baseSchema)
 - 		);
 - 
 - 		/**
 - 		 * The path of the config file that this array was loaded from.
 - 		 * This is used to calculate filename matches.
 - 		 * @property basePath
 - 		 * @type {string}
 - 		 */
 - 		this.basePath = basePath;
 - 
 - 		assertExtraConfigTypes(extraConfigTypes);
 - 
 - 		/**
 - 		 * The supported config types.
 - 		 * @property configTypes
 - 		 * @type {Array<string>}
 - 		 */
 - 		this.extraConfigTypes = Object.freeze([...extraConfigTypes]);
 - 
 - 		/**
 - 		 * A cache to store calculated configs for faster repeat lookup.
 - 		 * @property configCache
 - 		 * @type {Map<string, Object>}
 - 		 * @private
 - 		 */
 - 		this[ConfigArraySymbol.configCache] = new Map();
 - 
 - 		// init cache
 - 		dataCache.set(this, {
 - 			explicitMatches: new Map(),
 - 			directoryMatches: new Map(),
 - 			files: undefined,
 - 			ignores: undefined
 - 		});
 - 
 - 		// load the configs into this array
 - 		if (Array.isArray(configs)) {
 - 			this.push(...configs);
 - 		} else {
 - 			this.push(configs);
 - 		}
 - 
 - 	}
 - 
 - 	/**
 - 	 * Prevent normal array methods from creating a new `ConfigArray` instance.
 - 	 * This is to ensure that methods such as `slice()` won't try to create a 
 - 	 * new instance of `ConfigArray` behind the scenes as doing so may throw
 - 	 * an error due to the different constructor signature.
 - 	 * @returns {Function} The `Array` constructor.
 - 	 */
 - 	static get [Symbol.species]() {
 - 		return Array;
 - 	}
 - 
 - 	/**
 - 	 * Returns the `files` globs from every config object in the array.
 - 	 * This can be used to determine which files will be matched by a
 - 	 * config array or to use as a glob pattern when no patterns are provided
 - 	 * for a command line interface.
 - 	 * @returns {Array<string|Function>} An array of matchers.
 - 	 */
 - 	get files() {
 - 
 - 		assertNormalized(this);
 - 
 - 		// if this data has been cached, retrieve it
 - 		const cache = dataCache.get(this);
 - 
 - 		if (cache.files) {
 - 			return cache.files;
 - 		}
 - 
 - 		// otherwise calculate it
 - 
 - 		const result = [];
 - 
 - 		for (const config of this) {
 - 			if (config.files) {
 - 				config.files.forEach(filePattern => {
 - 					result.push(filePattern);
 - 				});
 - 			}
 - 		}
 - 
 - 		// store result
 - 		cache.files = result;
 - 		dataCache.set(this, cache);
 - 
 - 		return result;
 - 	}
 - 
 - 	/**
 - 	 * Returns ignore matchers that should always be ignored regardless of
 - 	 * the matching `files` fields in any configs. This is necessary to mimic
 - 	 * the behavior of things like .gitignore and .eslintignore, allowing a
 - 	 * globbing operation to be faster.
 - 	 * @returns {string[]} An array of string patterns and functions to be ignored.
 - 	 */
 - 	get ignores() {
 - 
 - 		assertNormalized(this);
 - 
 - 		// if this data has been cached, retrieve it
 - 		const cache = dataCache.get(this);
 - 
 - 		if (cache.ignores) {
 - 			return cache.ignores;
 - 		}
 - 
 - 		// otherwise calculate it
 - 
 - 		const result = [];
 - 
 - 		for (const config of this) {
 - 
 - 			/*
 - 			 * We only count ignores if there are no other keys in the object.
 - 			 * In this case, it acts list a globally ignored pattern. If there
 - 			 * are additional keys, then ignores act like exclusions.
 - 			 */
 - 			if (config.ignores && Object.keys(config).filter(key => !META_FIELDS.has(key)).length === 1) {
 - 				result.push(...config.ignores);
 - 			}
 - 		}
 - 
 - 		// store result
 - 		cache.ignores = result;
 - 		dataCache.set(this, cache);
 - 
 - 		return result;
 - 	}
 - 
 - 	/**
 - 	 * Indicates if the config array has been normalized.
 - 	 * @returns {boolean} True if the config array is normalized, false if not.
 - 	 */
 - 	isNormalized() {
 - 		return this[ConfigArraySymbol.isNormalized];
 - 	}
 - 
 - 	/**
 - 	 * Normalizes a config array by flattening embedded arrays and executing
 - 	 * config functions.
 - 	 * @param {ConfigContext} context The context object for config functions.
 - 	 * @returns {Promise<ConfigArray>} The current ConfigArray instance.
 - 	 */
 - 	async normalize(context = {}) {
 - 
 - 		if (!this.isNormalized()) {
 - 			const normalizedConfigs = await normalize(this, context, this.extraConfigTypes);
 - 			this.length = 0;
 - 			this.push(...normalizedConfigs.map(this[ConfigArraySymbol.preprocessConfig].bind(this)));
 - 			this.forEach(assertValidBaseConfig);
 - 			this[ConfigArraySymbol.isNormalized] = true;
 - 
 - 			// prevent further changes
 - 			Object.freeze(this);
 - 		}
 - 
 - 		return this;
 - 	}
 - 
 - 	/**
 - 	 * Normalizes a config array by flattening embedded arrays and executing
 - 	 * config functions.
 - 	 * @param {ConfigContext} context The context object for config functions.
 - 	 * @returns {ConfigArray} The current ConfigArray instance.
 - 	 */
 - 	normalizeSync(context = {}) {
 - 
 - 		if (!this.isNormalized()) {
 - 			const normalizedConfigs = normalizeSync(this, context, this.extraConfigTypes);
 - 			this.length = 0;
 - 			this.push(...normalizedConfigs.map(this[ConfigArraySymbol.preprocessConfig].bind(this)));
 - 			this.forEach(assertValidBaseConfig);
 - 			this[ConfigArraySymbol.isNormalized] = true;
 - 
 - 			// prevent further changes
 - 			Object.freeze(this);
 - 		}
 - 
 - 		return this;
 - 	}
 - 
 - 	/**
 - 	 * Finalizes the state of a config before being cached and returned by
 - 	 * `getConfig()`. Does nothing by default but is provided to be
 - 	 * overridden by subclasses as necessary.
 - 	 * @param {Object} config The config to finalize.
 - 	 * @returns {Object} The finalized config.
 - 	 */
 - 	[ConfigArraySymbol.finalizeConfig](config) {
 - 		return config;
 - 	}
 - 
 - 	/**
 - 	 * Preprocesses a config during the normalization process. This is the
 - 	 * method to override if you want to convert an array item before it is
 - 	 * validated for the first time. For example, if you want to replace a
 - 	 * string with an object, this is the method to override.
 - 	 * @param {Object} config The config to preprocess.
 - 	 * @returns {Object} The config to use in place of the argument.
 - 	 */
 - 	[ConfigArraySymbol.preprocessConfig](config) {
 - 		return config;
 - 	}
 - 
 - 	/**
 - 	 * Determines if a given file path explicitly matches a `files` entry
 - 	 * and also doesn't match an `ignores` entry. Configs that don't have
 - 	 * a `files` property are not considered an explicit match.
 - 	 * @param {string} filePath The complete path of a file to check.
 - 	 * @returns {boolean} True if the file path matches a `files` entry
 - 	 * 		or false if not.
 - 	 */
 - 	isExplicitMatch(filePath) {
 - 
 - 		assertNormalized(this);
 - 
 - 		const cache = dataCache.get(this);
 - 
 - 		// first check the cache to avoid duplicate work
 - 		let result = cache.explicitMatches.get(filePath);
 - 
 - 		if (typeof result == 'boolean') {
 - 			return result;
 - 		}
 - 
 - 		// TODO: Maybe move elsewhere? Maybe combine with getConfig() logic?
 - 		const relativeFilePath = path.relative(this.basePath, filePath);
 - 
 - 		if (shouldIgnorePath(this.ignores, filePath, relativeFilePath)) {
 - 			debug(`Ignoring ${filePath}`);
 - 
 - 			// cache and return result
 - 			cache.explicitMatches.set(filePath, false);
 - 			return false;
 - 		}
 - 
 - 		// filePath isn't automatically ignored, so try to find a match
 - 
 - 		for (const config of this) {
 - 
 - 			if (!config.files) {
 - 				continue;
 - 			}
 - 
 - 			if (pathMatches(filePath, this.basePath, config)) {
 - 				debug(`Matching config found for ${filePath}`);
 - 				cache.explicitMatches.set(filePath, true);
 - 				return true;
 - 			}
 - 		}
 - 
 - 		return false;
 - 	}
 - 
 - 	/**
 - 	 * Returns the config object for a given file path.
 - 	 * @param {string} filePath The complete path of a file to get a config for.
 - 	 * @returns {Object} The config object for this file.
 - 	 */
 - 	getConfig(filePath) {
 - 
 - 		assertNormalized(this);
 - 
 - 		const cache = this[ConfigArraySymbol.configCache];
 - 
 - 		// first check the cache for a filename match to avoid duplicate work
 - 		if (cache.has(filePath)) {
 - 			return cache.get(filePath);
 - 		}
 - 
 - 		let finalConfig;
 - 
 - 		// next check to see if the file should be ignored
 - 
 - 		// check if this should be ignored due to its directory
 - 		if (this.isDirectoryIgnored(path.dirname(filePath))) {
 - 			debug(`Ignoring ${filePath} based on directory pattern`);
 - 
 - 			// cache and return result - finalConfig is undefined at this point
 - 			cache.set(filePath, finalConfig);
 - 			return finalConfig;
 - 		}
 - 
 - 		// TODO: Maybe move elsewhere?
 - 		const relativeFilePath = path.relative(this.basePath, filePath);
 - 
 - 		if (shouldIgnorePath(this.ignores, filePath, relativeFilePath)) {
 - 			debug(`Ignoring ${filePath} based on file pattern`);
 - 
 - 			// cache and return result - finalConfig is undefined at this point
 - 			cache.set(filePath, finalConfig);
 - 			return finalConfig;
 - 		}
 - 
 - 		// filePath isn't automatically ignored, so try to construct config
 - 
 - 		const matchingConfigIndices = [];
 - 		let matchFound = false;
 - 		const universalPattern = /\/\*{1,2}$/;
 - 
 - 		this.forEach((config, index) => {
 - 
 - 			if (!config.files) {
 - 
 - 				if (!config.ignores) {
 - 					debug(`Anonymous universal config found for ${filePath}`);
 - 					matchingConfigIndices.push(index);
 - 					return;
 - 				}
 - 
 - 				if (pathMatchesIgnores(filePath, this.basePath, config)) {
 - 					debug(`Matching config found for ${filePath} (based on ignores: ${config.ignores})`);
 - 					matchingConfigIndices.push(index);
 - 					return;
 - 				}
 - 				
 - 				debug(`Skipped config found for ${filePath} (based on ignores: ${config.ignores})`);
 - 				return;
 - 			}
 - 
 - 			/*
 - 			 * If a config has a files pattern ending in /** or /*, and the
 - 			 * filePath only matches those patterns, then the config is only
 - 			 * applied if there is another config where the filePath matches
 - 			 * a file with a specific extensions such as *.js.
 - 			 */
 - 
 - 			const universalFiles = config.files.filter(
 - 				pattern => universalPattern.test(pattern)
 - 			);
 - 
 - 			// universal patterns were found so we need to check the config twice
 - 			if (universalFiles.length) {
 - 
 - 				debug('Universal files patterns found. Checking carefully.');
 - 
 - 				const nonUniversalFiles = config.files.filter(
 - 					pattern => !universalPattern.test(pattern)
 - 				);
 - 
 - 				// check that the config matches without the non-universal files first
 - 				if (
 - 					nonUniversalFiles.length && 
 - 					pathMatches(
 - 						filePath, this.basePath,
 - 						{ files: nonUniversalFiles, ignores: config.ignores }
 - 					)
 - 				) {
 - 					debug(`Matching config found for ${filePath}`);
 - 					matchingConfigIndices.push(index);
 - 					matchFound = true;
 - 					return;
 - 				}
 - 
 - 				// if there wasn't a match then check if it matches with universal files
 - 				if (
 - 					universalFiles.length &&
 - 					pathMatches(
 - 						filePath, this.basePath,
 - 						{ files: universalFiles, ignores: config.ignores }
 - 					)
 - 				) {
 - 					debug(`Matching config found for ${filePath}`);
 - 					matchingConfigIndices.push(index);
 - 					return;
 - 				}
 - 
 - 				// if we make here, then there was no match
 - 				return;
 - 			}
 - 
 - 			// the normal case
 - 			if (pathMatches(filePath, this.basePath, config)) {
 - 				debug(`Matching config found for ${filePath}`);
 - 				matchingConfigIndices.push(index);
 - 				matchFound = true;
 - 				return;
 - 			}
 - 
 - 		});
 - 
 - 		// if matching both files and ignores, there will be no config to create
 - 		if (!matchFound) {
 - 			debug(`No matching configs found for ${filePath}`);
 - 
 - 			// cache and return result - finalConfig is undefined at this point
 - 			cache.set(filePath, finalConfig);
 - 			return finalConfig;
 - 		}
 - 
 - 		// check to see if there is a config cached by indices
 - 		finalConfig = cache.get(matchingConfigIndices.toString());
 - 
 - 		if (finalConfig) {
 - 
 - 			// also store for filename for faster lookup next time
 - 			cache.set(filePath, finalConfig);
 - 
 - 			return finalConfig;
 - 		}
 - 
 - 		// otherwise construct the config
 - 
 - 		finalConfig = matchingConfigIndices.reduce((result, index) => {
 - 			try {
 - 				return this[ConfigArraySymbol.schema].merge(result, this[index]);
 - 			} catch (validationError) {
 - 				rethrowConfigError(this[index], index, { cause: validationError});
 - 			}
 - 		}, {}, this);
 - 
 - 		finalConfig = this[ConfigArraySymbol.finalizeConfig](finalConfig);
 - 
 - 		cache.set(filePath, finalConfig);
 - 		cache.set(matchingConfigIndices.toString(), finalConfig);
 - 
 - 		return finalConfig;
 - 	}
 - 
 - 	/**
 - 	 * Determines if the given filepath is ignored based on the configs.
 - 	 * @param {string} filePath The complete path of a file to check.
 - 	 * @returns {boolean} True if the path is ignored, false if not.
 - 	 * @deprecated Use `isFileIgnored` instead.
 - 	 */
 - 	isIgnored(filePath) {
 - 		return this.isFileIgnored(filePath);
 - 	}
 - 
 - 	/**
 - 	 * Determines if the given filepath is ignored based on the configs.
 - 	 * @param {string} filePath The complete path of a file to check.
 - 	 * @returns {boolean} True if the path is ignored, false if not.
 - 	 */
 - 	isFileIgnored(filePath) {
 - 		return this.getConfig(filePath) === undefined;
 - 	}
 - 
 - 	/**
 - 	 * Determines if the given directory is ignored based on the configs.
 - 	 * This checks only default `ignores` that don't have `files` in the 
 - 	 * same config. A pattern such as `/foo` be considered to ignore the directory
 - 	 * while a pattern such as `/foo/**` is not considered to ignore the
 - 	 * directory because it is matching files.
 - 	 * @param {string} directoryPath The complete path of a directory to check.
 - 	 * @returns {boolean} True if the directory is ignored, false if not. Will
 - 	 * 		return true for any directory that is not inside of `basePath`.
 - 	 * @throws {Error} When the `ConfigArray` is not normalized.
 - 	 */
 - 	isDirectoryIgnored(directoryPath) {
 - 
 - 		assertNormalized(this);
 - 
 - 		const relativeDirectoryPath = path.relative(this.basePath, directoryPath)
 - 			.replace(/\\/g, '/');
 - 
 - 		if (relativeDirectoryPath.startsWith('..')) {
 - 			return true;
 - 		}
 - 
 - 		// first check the cache
 - 		const cache = dataCache.get(this).directoryMatches;
 - 
 - 		if (cache.has(relativeDirectoryPath)) {
 - 			return cache.get(relativeDirectoryPath);
 - 		}
 - 
 - 		const directoryParts = relativeDirectoryPath.split('/');
 - 		let relativeDirectoryToCheck = '';
 - 		let result = false;
 - 
 - 		/*
 - 		 * In order to get the correct gitignore-style ignores, where an
 - 		 * ignored parent directory cannot have any descendants unignored,
 - 		 * we need to check every directory starting at the parent all
 - 		 * the way down to the actual requested directory.
 - 		 * 
 - 		 * We aggressively cache all of this info to make sure we don't
 - 		 * have to recalculate everything for every call.
 - 		 */
 - 		do {
 - 
 - 			relativeDirectoryToCheck += directoryParts.shift() + '/';
 - 
 - 			result = shouldIgnorePath(
 - 				this.ignores,
 - 				path.join(this.basePath, relativeDirectoryToCheck),
 - 				relativeDirectoryToCheck
 - 			);
 - 
 - 			cache.set(relativeDirectoryToCheck, result);
 - 
 - 		} while (!result && directoryParts.length);
 - 
 - 		// also cache the result for the requested path
 - 		cache.set(relativeDirectoryPath, result);
 - 
 - 		return result;
 - 	}
 - 
 - }
 - 
 - exports.ConfigArray = ConfigArray;
 - exports.ConfigArraySymbol = ConfigArraySymbol;
 
 
  |