'use strict'; const {findVariable} = require('eslint-utils'); const MESSAGE_ID = 'preferDefaultParameters'; const MESSAGE_ID_SUGGEST = 'preferDefaultParametersSuggest'; const assignmentSelector = [ 'ExpressionStatement', '[expression.type="AssignmentExpression"]', ].join(''); const declarationSelector = [ 'VariableDeclaration', '[declarations.0.type="VariableDeclarator"]', ].join(''); const isDefaultExpression = (left, right) => left && right && left.type === 'Identifier' && right.type === 'LogicalExpression' && (right.operator === '||' || right.operator === '??') && right.left.type === 'Identifier' && right.right.type === 'Literal'; const containsCallExpression = (sourceCode, node) => { if (!node) { return false; } if (node.type === 'CallExpression') { return true; } const keys = sourceCode.visitorKeys[node.type]; for (const key of keys) { const value = node[key]; if (Array.isArray(value)) { for (const element of value) { if (containsCallExpression(sourceCode, element)) { return true; } } } else if (containsCallExpression(sourceCode, value)) { return true; } } return false; }; const hasSideEffects = (sourceCode, function_, node) => { for (const element of function_.body.body) { if (element === node) { break; } // Function call before default-assignment if (containsCallExpression(sourceCode, element)) { return true; } } return false; }; const hasExtraReferences = (assignment, references, left) => { // Parameter is referenced prior to default-assignment if (assignment && references[0].identifier !== left) { return true; } // Old parameter is still referenced somewhere else if (!assignment && references.length > 1) { return true; } return false; }; const isLastParameter = (parameters, parameter) => { const lastParameter = parameters[parameters.length - 1]; // See 'default-param-last' rule return parameter && parameter === lastParameter; }; const needsParentheses = (sourceCode, function_) => { if (function_.type !== 'ArrowFunctionExpression' || function_.params.length > 1) { return false; } const [parameter] = function_.params; const before = sourceCode.getTokenBefore(parameter); const after = sourceCode.getTokenAfter(parameter); return !after || !before || before.value !== '(' || after.value !== ')'; }; /** @param {import('eslint').Rule.RuleFixer} fixer */ const fixDefaultExpression = (fixer, sourceCode, node) => { const {line} = node.loc.start; const {column} = node.loc.end; const nodeText = sourceCode.getText(node); const lineText = sourceCode.lines[line - 1]; const isOnlyNodeOnLine = lineText.trim() === nodeText; const endsWithWhitespace = lineText[column] === ' '; if (isOnlyNodeOnLine) { return fixer.removeRange([ sourceCode.getIndexFromLoc({line, column: 0}), sourceCode.getIndexFromLoc({line: line + 1, column: 0}), ]); } if (endsWithWhitespace) { return fixer.removeRange([ node.range[0], node.range[1] + 1, ]); } return fixer.remove(node); }; /** @param {import('eslint').Rule.RuleContext} context */ const create = context => { const sourceCode = context.getSourceCode(); const functionStack = []; const checkExpression = (node, left, right, assignment) => { const currentFunction = functionStack[functionStack.length - 1]; if (!currentFunction || !isDefaultExpression(left, right)) { return; } const {name: firstId} = left; const { left: {name: secondId}, right: {raw: literal}, } = right; // Parameter is reassigned to a different identifier if (assignment && firstId !== secondId) { return; } const variable = findVariable(context.getScope(), secondId); // This was reported https://github.com/sindresorhus/eslint-plugin-unicorn/issues/1122 // But can't reproduce, just ignore this case /* istanbul ignore next */ if (!variable) { return; } const {references} = variable; const {params} = currentFunction; const parameter = params.find(parameter => parameter.type === 'Identifier' && parameter.name === secondId, ); if ( hasSideEffects(sourceCode, currentFunction, node) || hasExtraReferences(assignment, references, left) || !isLastParameter(params, parameter) ) { return; } const replacement = needsParentheses(sourceCode, currentFunction) ? `(${firstId} = ${literal})` : `${firstId} = ${literal}`; return { node, messageId: MESSAGE_ID, suggest: [{ messageId: MESSAGE_ID_SUGGEST, fix: fixer => [ fixer.replaceText(parameter, replacement), fixDefaultExpression(fixer, sourceCode, node), ], }], }; }; return { ':function': node => { functionStack.push(node); }, ':function:exit': () => { functionStack.pop(); }, [assignmentSelector]: node => { const {left, right} = node.expression; return checkExpression(node, left, right, true); }, [declarationSelector]: node => { const {id, init} = node.declarations[0]; return checkExpression(node, id, init, false); }, }; }; /** @type {import('eslint').Rule.RuleModule} */ module.exports = { create, meta: { type: 'suggestion', docs: { description: 'Prefer default parameters over reassignment.', }, fixable: 'code', hasSuggestions: true, messages: { [MESSAGE_ID]: 'Prefer default parameters over reassignment.', [MESSAGE_ID_SUGGEST]: 'Replace reassignment with default parameter.', }, }, };