123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221 |
- // @ts-self-types="./index.d.ts"
- import path from 'node:path';
- import minimatch from 'minimatch';
- import createDebug from 'debug';
- import { ObjectSchema } from '@eslint/object-schema';
- export { ObjectSchema } from '@eslint/object-schema';
- /**
- * @fileoverview ConfigSchema
- * @author Nicholas C. Zakas
- */
- //------------------------------------------------------------------------------
- // Types
- //------------------------------------------------------------------------------
- /** @typedef {import("@eslint/object-schema").PropertyDefinition} PropertyDefinition */
- /** @typedef {import("@eslint/object-schema").ObjectDefinition} ObjectDefinition */
- //------------------------------------------------------------------------------
- // Helpers
- //------------------------------------------------------------------------------
- /**
- * A strategy that does nothing.
- * @type {PropertyDefinition}
- */
- const NOOP_STRATEGY = {
- required: false,
- merge() {
- return undefined;
- },
- validate() {},
- };
- //------------------------------------------------------------------------------
- // Exports
- //------------------------------------------------------------------------------
- /**
- * The base schema that every ConfigArray uses.
- * @type {ObjectDefinition}
- */
- 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
- */
- //------------------------------------------------------------------------------
- // Types
- //------------------------------------------------------------------------------
- //------------------------------------------------------------------------------
- // 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) {
- 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 {ObjectDefinition}
- */
- 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
- */
- //------------------------------------------------------------------------------
- // Types
- //------------------------------------------------------------------------------
- /** @typedef {import("./types.ts").ConfigObject} ConfigObject */
- /** @typedef {import("minimatch").IMinimatchStatic} IMinimatchStatic */
- /** @typedef {import("minimatch").IMinimatch} IMinimatch */
- /*
- * This is a bit of a hack to make TypeScript happy with the Rollup-created
- * CommonJS file. Rollup doesn't do object destructuring for imported files
- * and instead imports the default via `require()`. This messes up type checking
- * for `ObjectSchema`. To work around that, we just import the type manually
- * and give it a different name to use in the JSDoc comments.
- */
- /** @typedef {import("@eslint/object-schema").ObjectSchema} ObjectSchemaInstance */
- //------------------------------------------------------------------------------
- // Helpers
- //------------------------------------------------------------------------------
- const Minimatch = minimatch.Minimatch;
- const debug = createDebug("@eslint/config-array");
- /**
- * A cache for minimatch instances.
- * @type {Map<string, IMinimatch>}
- */
- const minimatchCache = new Map();
- /**
- * A cache for negated minimatch instances.
- * @type {Map<string, IMinimatch>}
- */
- const negatedMinimatchCache = new Map();
- /**
- * Options to use with minimatch.
- * @type {Object}
- */
- const MINIMATCH_OPTIONS = {
- // matchBase: true,
- dot: true,
- allowWindowsEscape: true,
- };
- /**
- * The types of config objects that are supported.
- * @type {Set<string>}
- */
- const CONFIG_TYPES = new Set(["array", "function"]);
- /**
- * Fields that are considered metadata and not part of the config object.
- * @type {Set<string>}
- */
- const META_FIELDS = new Set(["name"]);
- /**
- * A schema containing just files and ignores for early validation.
- * @type {ObjectSchemaInstance}
- */
- const FILES_AND_IGNORES_SCHEMA = new ObjectSchema(filesAndIgnoresSchema);
- // Precomputed constant objects returned by `ConfigArray.getConfigWithStatus`.
- const CONFIG_WITH_STATUS_EXTERNAL = Object.freeze({ status: "external" });
- const CONFIG_WITH_STATUS_IGNORED = Object.freeze({ status: "ignored" });
- const CONFIG_WITH_STATUS_UNCONFIGURED = Object.freeze({
- status: "unconfigured",
- });
- /**
- * 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 {Object} options The options for the error.
- * @param {Error} [options.cause] The error that caused this error.
- * @param {string} [options.message] The message to use for the 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 {ConfigObject} 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, { cause: 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, 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|((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
- function 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 {Object} options The options for the ConfigArray.
- * @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.extraConfigTypes] 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 {ObjectSchemaInstance}
- * @private
- */
- this[ConfigArraySymbol.schema] = new 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.
- * @type {Array<string>}
- */
- this.extraConfigTypes = [...extraConfigTypes];
- Object.freeze(this.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.
- * @type {ArrayConstructor} 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 {Object} [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 {Object} [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;
- }
- /* eslint-disable class-methods-use-this -- Desired as instance methods */
- /**
- * 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;
- }
- /* eslint-enable class-methods-use-this -- Desired as instance methods */
- /**
- * Returns the config object for a given file path and a status that can be used to determine why a file has no config.
- * @param {string} filePath The complete path of a file to get a config for.
- * @returns {{ config?: Object, status: "ignored"|"external"|"unconfigured"|"matched" }}
- * An object with an optional property `config` and property `status`.
- * `config` is the config object for the specified file as returned by {@linkcode ConfigArray.getConfig},
- * `status` a is one of the constants returned by {@linkcode ConfigArray.getConfigStatus}.
- */
- getConfigWithStatus(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);
- }
- // check to see if the file is outside the base path
- const relativeFilePath = path.relative(this.basePath, filePath);
- if (relativeFilePath.startsWith("..")) {
- debug(`No config for file ${filePath} outside of base path`);
- // cache and return result
- cache.set(filePath, CONFIG_WITH_STATUS_EXTERNAL);
- return CONFIG_WITH_STATUS_EXTERNAL;
- }
- // 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
- cache.set(filePath, CONFIG_WITH_STATUS_IGNORED);
- return CONFIG_WITH_STATUS_IGNORED;
- }
- if (shouldIgnorePath(this.ignores, filePath, relativeFilePath)) {
- debug(`Ignoring ${filePath} based on file pattern`);
- // cache and return result
- cache.set(filePath, CONFIG_WITH_STATUS_IGNORED);
- return CONFIG_WITH_STATUS_IGNORED;
- }
- // filePath isn't automatically ignored, so try to construct config
- const matchingConfigIndices = [];
- let matchFound = false;
- const universalPattern = /^\*$|\/\*{1,2}$/u;
- 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 * or patterns 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;
- }
- });
- // 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
- cache.set(filePath, CONFIG_WITH_STATUS_UNCONFIGURED);
- return CONFIG_WITH_STATUS_UNCONFIGURED;
- }
- // check to see if there is a config cached by indices
- const indicesKey = matchingConfigIndices.toString();
- let configWithStatus = cache.get(indicesKey);
- if (configWithStatus) {
- // also store for filename for faster lookup next time
- cache.set(filePath, configWithStatus);
- return configWithStatus;
- }
- // otherwise construct the config
- // eslint-disable-next-line array-callback-return, consistent-return -- rethrowConfigError always throws an error
- let finalConfig = matchingConfigIndices.reduce((result, index) => {
- try {
- return this[ConfigArraySymbol.schema].merge(
- result,
- this[index],
- );
- } catch (validationError) {
- rethrowConfigError(this[index], index, validationError);
- }
- }, {});
- finalConfig = this[ConfigArraySymbol.finalizeConfig](finalConfig);
- configWithStatus = Object.freeze({
- config: finalConfig,
- status: "matched",
- });
- cache.set(filePath, configWithStatus);
- cache.set(indicesKey, configWithStatus);
- return configWithStatus;
- }
- /**
- * 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|undefined} The config object for this file or `undefined`.
- */
- getConfig(filePath) {
- return this.getConfigWithStatus(filePath).config;
- }
- /**
- * Determines whether a file has a config or why it doesn't.
- * @param {string} filePath The complete path of the file to check.
- * @returns {"ignored"|"external"|"unconfigured"|"matched"} One of the following values:
- * * `"ignored"`: the file is ignored
- * * `"external"`: the file is outside the base path
- * * `"unconfigured"`: the file is not matched by any config
- * * `"matched"`: the file has a matching config
- */
- getConfigStatus(filePath) {
- return this.getConfigWithStatus(filePath).status;
- }
- /**
- * 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.getConfigStatus(filePath) === "ignored";
- }
- /**
- * 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(/\\/gu, "/");
- // basePath directory can never be ignored
- if (relativeDirectoryPath === "") {
- return false;
- }
- 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;
- /*
- * 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;
- }
- }
- export { ConfigArray, ConfigArraySymbol };
|