lix-website/themes/lix/assets/bootstrap/node_modules/stylelint/lib/augmentConfig.js
2024-04-26 22:49:34 -06:00

359 lines
11 KiB
JavaScript

'use strict';
const _ = require('lodash');
const configurationError = require('./utils/configurationError');
const getModulePath = require('./utils/getModulePath');
const globjoin = require('globjoin');
const normalizeAllRuleSettings = require('./normalizeAllRuleSettings');
const path = require('path');
/** @typedef {import('stylelint').StylelintConfigPlugins} StylelintConfigPlugins */
/** @typedef {import('stylelint').StylelintConfigProcessor} StylelintConfigProcessor */
/** @typedef {import('stylelint').StylelintConfigProcessors} StylelintConfigProcessors */
/** @typedef {import('stylelint').StylelintConfigRules} StylelintConfigRules */
/** @typedef {import('stylelint').StylelintInternalApi} StylelintInternalApi */
/** @typedef {import('stylelint').StylelintConfig} StylelintConfig */
/** @typedef {import('stylelint').CosmiconfigResult} CosmiconfigResult */
/**
* - Merges config and configOverrides
* - Makes all paths absolute
* - Merges extends
* @param {StylelintInternalApi} stylelint
* @param {StylelintConfig} config
* @param {string} configDir
* @param {boolean} [allowOverrides]
* @returns {Promise<StylelintConfig>}
*/
function augmentConfigBasic(stylelint, config, configDir, allowOverrides) {
return Promise.resolve()
.then(() => {
if (!allowOverrides) return config;
return _.merge(config, stylelint._options.configOverrides);
})
.then((augmentedConfig) => {
return extendConfig(stylelint, augmentedConfig, configDir);
})
.then((augmentedConfig) => {
return absolutizePaths(augmentedConfig, configDir);
});
}
/**
* Extended configs need to be run through augmentConfigBasic
* but do not need the full treatment. Things like pluginFunctions
* will be resolved and added by the parent config.
* @param {StylelintInternalApi} stylelint
* @param {CosmiconfigResult} [cosmiconfigResult]
* @returns {Promise<CosmiconfigResult | null>}
*/
function augmentConfigExtended(stylelint, cosmiconfigResult) {
if (!cosmiconfigResult) return Promise.resolve(null);
const configDir = path.dirname(cosmiconfigResult.filepath || '');
const { ignoreFiles, ...cleanedConfig } = cosmiconfigResult.config;
return augmentConfigBasic(stylelint, cleanedConfig, configDir).then((augmentedConfig) => {
return {
config: augmentedConfig,
filepath: cosmiconfigResult.filepath,
};
});
}
/**
* @param {StylelintInternalApi} stylelint
* @param {CosmiconfigResult} [cosmiconfigResult]
* @returns {Promise<CosmiconfigResult | null>}
*/
function augmentConfigFull(stylelint, cosmiconfigResult) {
if (!cosmiconfigResult) return Promise.resolve(null);
const config = cosmiconfigResult.config;
const filepath = cosmiconfigResult.filepath;
const configDir = stylelint._options.configBasedir || path.dirname(filepath || '');
return augmentConfigBasic(stylelint, config, configDir, true)
.then((augmentedConfig) => {
return addPluginFunctions(augmentedConfig);
})
.then((augmentedConfig) => {
return addProcessorFunctions(augmentedConfig);
})
.then((augmentedConfig) => {
if (!augmentedConfig.rules) {
throw configurationError(
'No rules found within configuration. Have you provided a "rules" property?',
);
}
return normalizeAllRuleSettings(augmentedConfig);
})
.then((augmentedConfig) => {
return {
config: augmentedConfig,
filepath: cosmiconfigResult.filepath,
};
});
}
/**
* Make all paths in the config absolute:
* - ignoreFiles
* - plugins
* - processors
* (extends handled elsewhere)
* @param {StylelintConfig} config
* @param {string} configDir
* @returns {StylelintConfig}
*/
function absolutizePaths(config, configDir) {
if (config.ignoreFiles) {
config.ignoreFiles = /** @type {string[]} */ ([]).concat(config.ignoreFiles).map((glob) => {
if (path.isAbsolute(glob.replace(/^!/, ''))) return glob;
return globjoin(configDir, glob);
});
}
if (config.plugins) {
config.plugins = /** @type {string[]} */ ([]).concat(config.plugins).map((lookup) => {
return getModulePath(configDir, lookup);
});
}
if (config.processors) {
config.processors = absolutizeProcessors(config.processors, configDir);
}
return config;
}
/**
* Processors are absolutized in their own way because
* they can be and return a string or an array
* @param {StylelintConfigProcessors} processors
* @param {string} configDir
* @return {StylelintConfigProcessors}
*/
function absolutizeProcessors(processors, configDir) {
const normalizedProcessors = Array.isArray(processors) ? processors : [processors];
return normalizedProcessors.map((item) => {
if (typeof item === 'string') {
return getModulePath(configDir, item);
}
return [getModulePath(configDir, item[0]), item[1]];
});
}
/**
* @param {StylelintInternalApi} stylelint
* @param {StylelintConfig} config
* @param {string} configDir
* @return {Promise<StylelintConfig>}
*/
function extendConfig(stylelint, config, configDir) {
if (config.extends === undefined) return Promise.resolve(config);
const normalizedExtends = Array.isArray(config.extends) ? config.extends : [config.extends];
const { extends: configExtends, ...originalWithoutExtends } = config;
const loadExtends = normalizedExtends.reduce((resultPromise, extendLookup) => {
return resultPromise.then((resultConfig) => {
return loadExtendedConfig(stylelint, resultConfig, configDir, extendLookup).then(
(extendResult) => {
if (!extendResult) return resultConfig;
return mergeConfigs(resultConfig, extendResult.config);
},
);
});
}, Promise.resolve(originalWithoutExtends));
return loadExtends.then((resultConfig) => {
return mergeConfigs(resultConfig, originalWithoutExtends);
});
}
/**
* @param {StylelintInternalApi} stylelint
* @param {StylelintConfig} config
* @param {string} configDir
* @param {string} extendLookup
* @return {Promise<CosmiconfigResult | null>}
*/
function loadExtendedConfig(stylelint, config, configDir, extendLookup) {
const extendPath = getModulePath(configDir, extendLookup);
return stylelint._extendExplorer.load(extendPath);
}
/**
* When merging configs (via extends)
* - plugin and processor arrays are joined
* - rules are merged via Object.assign, so there is no attempt made to
* merge any given rule's settings. If b contains the same rule as a,
* b's rule settings will override a's rule settings entirely.
* - Everything else is merged via Object.assign
* @param {StylelintConfig} a
* @param {StylelintConfig} b
* @returns {StylelintConfig}
*/
function mergeConfigs(a, b) {
/** @type {{plugins: StylelintConfigPlugins}} */
const pluginMerger = {};
if (a.plugins || b.plugins) {
pluginMerger.plugins = [];
if (a.plugins) {
pluginMerger.plugins = pluginMerger.plugins.concat(a.plugins);
}
if (b.plugins) {
pluginMerger.plugins = [...new Set(pluginMerger.plugins.concat(b.plugins))];
}
}
/** @type {{processors: StylelintConfigProcessors}} */
const processorMerger = {};
if (a.processors || b.processors) {
processorMerger.processors = [];
if (a.processors) {
processorMerger.processors = processorMerger.processors.concat(a.processors);
}
if (b.processors) {
processorMerger.processors = [...new Set(processorMerger.processors.concat(b.processors))];
}
}
const rulesMerger = {};
if (a.rules || b.rules) {
rulesMerger.rules = { ...a.rules, ...b.rules };
}
const result = { ...a, ...b, ...processorMerger, ...pluginMerger, ...rulesMerger };
return result;
}
/**
* @param {StylelintConfig} config
* @returns {StylelintConfig}
*/
function addPluginFunctions(config) {
if (!config.plugins) return config;
const normalizedPlugins = Array.isArray(config.plugins) ? config.plugins : [config.plugins];
const pluginFunctions = normalizedPlugins.reduce((result, pluginLookup) => {
let pluginImport = require(pluginLookup);
// Handle either ES6 or CommonJS modules
pluginImport = pluginImport.default || pluginImport;
// A plugin can export either a single rule definition
// or an array of them
const normalizedPluginImport = Array.isArray(pluginImport) ? pluginImport : [pluginImport];
normalizedPluginImport.forEach((pluginRuleDefinition) => {
if (!pluginRuleDefinition.ruleName) {
throw configurationError(
'stylelint v3+ requires plugins to expose a ruleName. ' +
`The plugin "${pluginLookup}" is not doing this, so will not work ` +
'with stylelint v3+. Please file an issue with the plugin.',
);
}
if (!pluginRuleDefinition.ruleName.includes('/')) {
throw configurationError(
'stylelint v7+ requires plugin rules to be namespaced, ' +
'i.e. only `plugin-namespace/plugin-rule-name` plugin rule names are supported. ' +
`The plugin rule "${pluginRuleDefinition.ruleName}" does not do this, so will not work. ` +
'Please file an issue with the plugin.',
);
}
result[pluginRuleDefinition.ruleName] = pluginRuleDefinition.rule;
});
return result;
}, /** @type {{[k: string]: Function}} */ ({}));
config.pluginFunctions = pluginFunctions;
return config;
}
/**
* Given an array of processors strings, we want to add two
* properties to the augmented config:
* - codeProcessors: functions that will run on code as it comes in
* - resultProcessors: functions that will run on results as they go out
*
* To create these properties, we need to:
* - Find the processor module
* - Initialize the processor module by calling its functions with any
* provided options
* - Push the processor's code and result processors to their respective arrays
* @type {Map<string, string | Object>}
*/
const processorCache = new Map();
/**
* @param {StylelintConfig} config
* @return {StylelintConfig}
*/
function addProcessorFunctions(config) {
if (!config.processors) return config;
/** @type {Array<Function>} */
const codeProcessors = [];
/** @type {Array<Function>} */
const resultProcessors = [];
/** @type {Array<StylelintConfigProcessor>} */ ([])
.concat(config.processors)
.forEach((processorConfig) => {
const processorKey = JSON.stringify(processorConfig);
let initializedProcessor;
if (processorCache.has(processorKey)) {
initializedProcessor = processorCache.get(processorKey);
} else {
const processorLookup =
typeof processorConfig === 'string' ? processorConfig : processorConfig[0];
const processorOptions =
typeof processorConfig === 'string' ? undefined : processorConfig[1];
let processor = require(processorLookup);
processor = processor.default || processor;
initializedProcessor = processor(processorOptions);
processorCache.set(processorKey, initializedProcessor);
}
if (initializedProcessor && initializedProcessor.code) {
codeProcessors.push(initializedProcessor.code);
}
if (initializedProcessor && initializedProcessor.result) {
resultProcessors.push(initializedProcessor.result);
}
});
config.codeProcessors = codeProcessors;
config.resultProcessors = resultProcessors;
return config;
}
module.exports = { augmentConfigExtended, augmentConfigFull };