'use strict';
const path = require('path');
const {defaultsDeep, upperFirst, lowerFirst} = require('lodash');
const avoidCapture = require('./utils/avoid-capture.js');
const cartesianProductSamples = require('./utils/cartesian-product-samples.js');
const isShorthandPropertyValue = require('./utils/is-shorthand-property-value.js');
const isShorthandImportLocal = require('./utils/is-shorthand-import-local.js');
const getVariableIdentifiers = require('./utils/get-variable-identifiers.js');
const isStaticRequire = require('./utils/is-static-require.js');
const {defaultReplacements, defaultAllowList, defaultIgnore} = require('./shared/abbreviations.js');
const {renameVariable} = require('./fix/index.js');
const getScopes = require('./utils/get-scopes.js');
const MESSAGE_ID_REPLACE = 'replace';
const MESSAGE_ID_SUGGESTION = 'suggestion';
const anotherNameMessage = 'A more descriptive name will do too.';
const messages = {
[MESSAGE_ID_REPLACE]: `The {{nameTypeText}} \`{{discouragedName}}\` should be named \`{{replacement}}\`. ${anotherNameMessage}`,
[MESSAGE_ID_SUGGESTION]: `Please rename the {{nameTypeText}} \`{{discouragedName}}\`. Suggested names are: {{replacementsText}}. ${anotherNameMessage}`,
};
const isUpperCase = string => string === string.toUpperCase();
const isUpperFirst = string => isUpperCase(string[0]);
const prepareOptions = ({
checkProperties = false,
checkVariables = true,
checkDefaultAndNamespaceImports = 'internal',
checkShorthandImports = 'internal',
checkShorthandProperties = false,
checkFilenames = true,
extendDefaultReplacements = true,
replacements = {},
extendDefaultAllowList = true,
allowList = {},
ignore = [],
} = {}) => {
const mergedReplacements = extendDefaultReplacements
? defaultsDeep({}, replacements, defaultReplacements)
: replacements;
const mergedAllowList = extendDefaultAllowList
? defaultsDeep({}, allowList, defaultAllowList)
: allowList;
ignore = [...defaultIgnore, ...ignore];
ignore = ignore.map(
pattern => pattern instanceof RegExp ? pattern : new RegExp(pattern, 'u'),
);
return {
checkProperties,
checkVariables,
checkDefaultAndNamespaceImports,
checkShorthandImports,
checkShorthandProperties,
checkFilenames,
replacements: new Map(
Object.entries(mergedReplacements).map(
([discouragedName, replacements]) =>
[discouragedName, new Map(Object.entries(replacements))],
),
),
allowList: new Map(Object.entries(mergedAllowList)),
ignore,
};
};
const getWordReplacements = (word, {replacements, allowList}) => {
// Skip constants and allowList
if (isUpperCase(word) || allowList.get(word)) {
return [];
}
const replacement = replacements.get(lowerFirst(word))
|| replacements.get(word)
|| replacements.get(upperFirst(word));
let wordReplacement = [];
if (replacement) {
const transform = isUpperFirst(word) ? upperFirst : lowerFirst;
wordReplacement = [...replacement.keys()]
.filter(name => replacement.get(name))
.map(name => transform(name));
}
return wordReplacement.length > 0 ? wordReplacement.sort() : [];
};
const getNameReplacements = (name, options, limit = 3) => {
const {allowList, ignore} = options;
// Skip constants and allowList
if (isUpperCase(name) || allowList.get(name) || ignore.some(regexp => regexp.test(name))) {
return {total: 0};
}
// Find exact replacements
const exactReplacements = getWordReplacements(name, options);
if (exactReplacements.length > 0) {
return {
total: exactReplacements.length,
samples: exactReplacements.slice(0, limit),
};
}
// Split words
const words = name.split(/(?=[^a-z])|(?<=[^A-Za-z])/).filter(Boolean);
let hasReplacements = false;
const combinations = words.map(word => {
const wordReplacements = getWordReplacements(word, options);
if (wordReplacements.length > 0) {
hasReplacements = true;
return wordReplacements;
}
return [word];
});
// No replacements for any word
if (!hasReplacements) {
return {total: 0};
}
const {
total,
samples,
} = cartesianProductSamples(combinations, limit);
return {
total,
samples: samples.map(words => words.join('')),
};
};
const getMessage = (discouragedName, replacements, nameTypeText) => {
const {total, samples = []} = replacements;
if (total === 1) {
return {
messageId: MESSAGE_ID_REPLACE,
data: {
nameTypeText,
discouragedName,
replacement: samples[0],
},
};
}
let replacementsText = samples
.map(replacement => `\`${replacement}\``)
.join(', ');
const omittedReplacementsCount = total - samples.length;
if (omittedReplacementsCount > 0) {
replacementsText += `, ... (${omittedReplacementsCount > 99 ? '99+' : omittedReplacementsCount} more omitted)`;
}
return {
messageId: MESSAGE_ID_SUGGESTION,
data: {
nameTypeText,
discouragedName,
replacementsText,
},
};
};
const isExportedIdentifier = identifier => {
if (
identifier.parent.type === 'VariableDeclarator'
&& identifier.parent.id === identifier
) {
return (
identifier.parent.parent.type === 'VariableDeclaration'
&& identifier.parent.parent.parent.type === 'ExportNamedDeclaration'
);
}
if (
identifier.parent.type === 'FunctionDeclaration'
&& identifier.parent.id === identifier
) {
return identifier.parent.parent.type === 'ExportNamedDeclaration';
}
if (
identifier.parent.type === 'ClassDeclaration'
&& identifier.parent.id === identifier
) {
return identifier.parent.parent.type === 'ExportNamedDeclaration';
}
if (
identifier.parent.type === 'TSTypeAliasDeclaration'
&& identifier.parent.id === identifier
) {
return identifier.parent.parent.type === 'ExportNamedDeclaration';
}
return false;
};
const shouldFix = variable => !getVariableIdentifiers(variable).some(identifier => isExportedIdentifier(identifier));
const isDefaultOrNamespaceImportName = identifier => {
if (
identifier.parent.type === 'ImportDefaultSpecifier'
&& identifier.parent.local === identifier
) {
return true;
}
if (
identifier.parent.type === 'ImportNamespaceSpecifier'
&& identifier.parent.local === identifier
) {
return true;
}
if (
identifier.parent.type === 'ImportSpecifier'
&& identifier.parent.local === identifier
&& identifier.parent.imported.type === 'Identifier'
&& identifier.parent.imported.name === 'default'
) {
return true;
}
if (
identifier.parent.type === 'VariableDeclarator'
&& identifier.parent.id === identifier
&& isStaticRequire(identifier.parent.init)
) {
return true;
}
return false;
};
const isClassVariable = variable => {
if (variable.defs.length !== 1) {
return false;
}
const [definition] = variable.defs;
return definition.type === 'ClassName';
};
const shouldReportIdentifierAsProperty = identifier => {
if (
identifier.parent.type === 'MemberExpression'
&& identifier.parent.property === identifier
&& !identifier.parent.computed
&& identifier.parent.parent.type === 'AssignmentExpression'
&& identifier.parent.parent.left === identifier.parent
) {
return true;
}
if (
identifier.parent.type === 'Property'
&& identifier.parent.key === identifier
&& !identifier.parent.computed
&& !identifier.parent.shorthand // Shorthand properties are reported and fixed as variables
&& identifier.parent.parent.type === 'ObjectExpression'
) {
return true;
}
if (
identifier.parent.type === 'ExportSpecifier'
&& identifier.parent.exported === identifier
&& identifier.parent.local !== identifier // Same as shorthand properties above
) {
return true;
}
if (
(
identifier.parent.type === 'MethodDefinition'
|| identifier.parent.type === 'PropertyDefinition'
)
&& identifier.parent.key === identifier
&& !identifier.parent.computed
) {
return true;
}
return false;
};
const isInternalImport = node => {
let source = '';
if (node.type === 'Variable') {
source = node.node.init.arguments[0].value;
} else if (node.type === 'ImportBinding') {
source = node.parent.source.value;
}
return (
!source.includes('node_modules')
&& (source.startsWith('.') || source.startsWith('/'))
);
};
/** @param {import('eslint').Rule.RuleContext} context */
const create = context => {
const options = prepareOptions(context.options[0]);
const filenameWithExtension = context.getPhysicalFilename();
// A `class` declaration produces two variables in two scopes:
// the inner class scope, and the outer one (wherever the class is declared).
// This map holds the outer ones to be later processed when the inner one is encountered.
// For why this is not a eslint issue see https://github.com/eslint/eslint-scope/issues/48#issuecomment-464358754
const identifierToOuterClassVariable = new WeakMap();
const checkPossiblyWeirdClassVariable = variable => {
if (isClassVariable(variable)) {
if (variable.scope.type === 'class') { // The inner class variable
const [definition] = variable.defs;
const outerClassVariable = identifierToOuterClassVariable.get(definition.name);
if (!outerClassVariable) {
return checkVariable(variable);
}
// Create a normal-looking variable (like a `var` or a `function`)
// For which a single `variable` holds all references, unlike with a `class`
const combinedReferencesVariable = {
name: variable.name,
scope: variable.scope,
defs: variable.defs,
identifiers: variable.identifiers,
references: [...variable.references, ...outerClassVariable.references],
};
// Call the common checker with the newly forged normalized class variable
return checkVariable(combinedReferencesVariable);
}
// The outer class variable, we save it for later, when it's inner counterpart is encountered
const [definition] = variable.defs;
identifierToOuterClassVariable.set(definition.name, variable);
return;
}
return checkVariable(variable);
};
// Holds a map from a `Scope` to a `Set` of new variable names generated by our fixer.
// Used to avoid generating duplicate names, see for instance `let errCb, errorCb` test.
const scopeToNamesGeneratedByFixer = new WeakMap();
const isSafeName = (name, scopes) => scopes.every(scope => {
const generatedNames = scopeToNamesGeneratedByFixer.get(scope);
return !generatedNames || !generatedNames.has(name);
});
const checkVariable = variable => {
if (variable.defs.length === 0) {
return;
}
const [definition] = variable.defs;
if (isDefaultOrNamespaceImportName(definition.name)) {
if (!options.checkDefaultAndNamespaceImports) {
return;
}
if (
options.checkDefaultAndNamespaceImports === 'internal'
&& !isInternalImport(definition)
) {
return;
}
}
if (isShorthandImportLocal(definition.name)) {
if (!options.checkShorthandImports) {
return;
}
if (
options.checkShorthandImports === 'internal'
&& !isInternalImport(definition)
) {
return;
}
}
if (
!options.checkShorthandProperties
&& isShorthandPropertyValue(definition.name)
) {
return;
}
const variableReplacements = getNameReplacements(variable.name, options);
if (variableReplacements.total === 0) {
return;
}
const scopes = [
...variable.references.map(reference => reference.from),
variable.scope,
];
variableReplacements.samples = variableReplacements.samples.map(
name => avoidCapture(name, scopes, isSafeName),
);
const problem = {
...getMessage(definition.name.name, variableReplacements, 'variable'),
node: definition.name,
};
if (variableReplacements.total === 1 && shouldFix(variable) && variableReplacements.samples[0]) {
const [replacement] = variableReplacements.samples;
for (const scope of scopes) {
if (!scopeToNamesGeneratedByFixer.has(scope)) {
scopeToNamesGeneratedByFixer.set(scope, new Set());
}
const generatedNames = scopeToNamesGeneratedByFixer.get(scope);
generatedNames.add(replacement);
}
problem.fix = fixer => renameVariable(variable, replacement, fixer);
}
context.report(problem);
};
const checkVariables = scope => {
for (const variable of scope.variables) {
checkPossiblyWeirdClassVariable(variable);
}
};
const checkScope = scope => {
const scopes = getScopes(scope);
for (const scope of scopes) {
checkVariables(scope);
}
};
return {
Identifier(node) {
if (!options.checkProperties) {
return;
}
if (node.name === '__proto__') {
return;
}
const identifierReplacements = getNameReplacements(node.name, options);
if (identifierReplacements.total === 0) {
return;
}
if (!shouldReportIdentifierAsProperty(node)) {
return;
}
const problem = {
...getMessage(node.name, identifierReplacements, 'property'),
node,
};
context.report(problem);
},
Program(node) {
if (!options.checkFilenames) {
return;
}
if (
filenameWithExtension === ''
|| filenameWithExtension === ''
) {
return;
}
const filename = path.basename(filenameWithExtension);
const extension = path.extname(filename);
const filenameReplacements = getNameReplacements(path.basename(filename, extension), options);
if (filenameReplacements.total === 0) {
return;
}
filenameReplacements.samples = filenameReplacements.samples.map(replacement => `${replacement}${extension}`);
context.report({
...getMessage(filename, filenameReplacements, 'filename'),
node,
});
},
'Program:exit'() {
if (!options.checkVariables) {
return;
}
checkScope(context.getScope());
},
};
};
const schema = {
type: 'array',
additionalItems: false,
items: [
{
type: 'object',
additionalProperties: false,
properties: {
checkProperties: {
type: 'boolean',
},
checkVariables: {
type: 'boolean',
},
checkDefaultAndNamespaceImports: {
type: [
'boolean',
'string',
],
pattern: 'internal',
},
checkShorthandImports: {
type: [
'boolean',
'string',
],
pattern: 'internal',
},
checkShorthandProperties: {
type: 'boolean',
},
checkFilenames: {
type: 'boolean',
},
extendDefaultReplacements: {
type: 'boolean',
},
replacements: {
$ref: '#/definitions/abbreviations',
},
extendDefaultAllowList: {
type: 'boolean',
},
allowList: {
$ref: '#/definitions/booleanObject',
},
ignore: {
type: 'array',
uniqueItems: true,
},
},
},
],
definitions: {
abbreviations: {
type: 'object',
additionalProperties: {
$ref: '#/definitions/replacements',
},
},
replacements: {
anyOf: [
{
enum: [
false,
],
},
{
$ref: '#/definitions/booleanObject',
},
],
},
booleanObject: {
type: 'object',
additionalProperties: {
type: 'boolean',
},
},
},
};
/** @type {import('eslint').Rule.RuleModule} */
module.exports = {
create,
meta: {
type: 'suggestion',
docs: {
description: 'Prevent abbreviations.',
},
fixable: 'code',
schema,
messages,
},
};