| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129 | 'use strict';const {findVariable} = require('@eslint-community/eslint-utils');const avoidCapture = require('./utils/avoid-capture.js');const {renameVariable} = require('./fix/index.js');const {matches, methodCallSelector} = require('./selectors/index.js');const MESSAGE_ID = 'catch-error-name';const messages = {	[MESSAGE_ID]: 'The catch parameter `{{originalName}}` should be named `{{fixedName}}`.',};const selector = matches([	// `try {} catch (foo) {}`	[		'CatchClause',		' > ',		'Identifier.param',	].join(''),	// - `promise.then(…, foo => {})`	// - `promise.then(…, function(foo) {})`	// - `promise.catch(foo => {})`	// - `promise.catch(function(foo) {})`	[		matches([			methodCallSelector({method: 'then', argumentsLength: 2}),			methodCallSelector({method: 'catch', argumentsLength: 1}),		]),		' > ',		':matches(FunctionExpression, ArrowFunctionExpression).arguments:last-child',		' > ',		'Identifier.params:first-child',	].join(''),]);/** @param {import('eslint').Rule.RuleContext} context */const create = context => {	const options = {		name: 'error',		ignore: [],		...context.options[0],	};	const {name: expectedName} = options;	const ignore = options.ignore.map(		pattern => pattern instanceof RegExp ? pattern : new RegExp(pattern, 'u'),	);	const isNameAllowed = name =>		name === expectedName		|| ignore.some(regexp => regexp.test(name))		|| name.endsWith(expectedName)		|| name.endsWith(expectedName.charAt(0).toUpperCase() + expectedName.slice(1));	return {		[selector](node) {			const originalName = node.name;			if (				isNameAllowed(originalName)				|| isNameAllowed(originalName.replace(/_+$/g, ''))			) {				return;			}			const scope = context.getScope();			const variable = findVariable(scope, node);			// This was reported https://github.com/sindresorhus/eslint-plugin-unicorn/issues/1075#issuecomment-768072967			// But can't reproduce, just ignore this case			/* c8 ignore next 3 */			if (!variable) {				return;			}			if (originalName === '_' && variable.references.length === 0) {				return;			}			const scopes = [				variable.scope,				...variable.references.map(({from}) => from),			];			const fixedName = avoidCapture(expectedName, scopes);			const problem = {				node,				messageId: MESSAGE_ID,				data: {					originalName,					fixedName: fixedName || expectedName,				},			};			if (fixedName) {				problem.fix = fixer => renameVariable(variable, fixedName, fixer);			}			return problem;		},	};};const schema = [	{		type: 'object',		additionalProperties: false,		properties: {			name: {				type: 'string',			},			ignore: {				type: 'array',				uniqueItems: true,			},		},	},];/** @type {import('eslint').Rule.RuleModule} */module.exports = {	create,	meta: {		type: 'suggestion',		docs: {			description: 'Enforce a specific parameter name in catch clauses.',		},		fixable: 'code',		schema,		messages,	},};
 |