UNPKG

eslint-plugin-sonarjs

Version:
106 lines 4.91 kB
"use strict"; /* * eslint-plugin-sonarjs * Copyright (C) 2018 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ // https://jira.sonarsource.com/browse/RSPEC-1488 const nodes_1 = require("../utils/nodes"); const rule = { meta: { type: "suggestion", fixable: "code", }, create(context) { return { BlockStatement(node) { processStatements(node.body); }, SwitchCase(node) { processStatements(node.consequent); }, }; function processStatements(statements) { if (statements.length > 1) { const last = statements[statements.length - 1]; const returnedIdentifier = getOnlyReturnedVariable(last); const lastButOne = statements[statements.length - 2]; const declaredIdentifier = getOnlyDeclaredVariable(lastButOne); if (returnedIdentifier && declaredIdentifier) { const sameVariable = getVariables(context).find(variable => { return (variable.references.find(ref => ref.identifier === returnedIdentifier) !== undefined && variable.references.find(ref => ref.identifier === declaredIdentifier.id) !== undefined); }); // there must be only one "read" - in `return` or `throw` if (sameVariable && sameVariable.references.filter(ref => ref.isRead()).length === 1) { context.report({ message: formatMessage(last, returnedIdentifier.name), node: declaredIdentifier.init, fix: fixer => fix(fixer, last, lastButOne, declaredIdentifier.init), }); } } } } function fix(fixer, last, lastButOne, expression) { const throwOrReturnKeyword = context.getSourceCode().getFirstToken(last); if (lastButOne.range && last.range && throwOrReturnKeyword) { const expressionText = context.getSourceCode().getText(expression); const fixedRangeStart = lastButOne.range[0]; const fixedRangeEnd = last.range[1]; const semicolonToken = context.getSourceCode().getLastToken(last); const semicolon = semicolonToken && semicolonToken.value === ";" ? ";" : ""; return [ fixer.removeRange([fixedRangeStart, fixedRangeEnd]), fixer.insertTextAfterRange([1, fixedRangeStart], `${throwOrReturnKeyword.value} ${expressionText}${semicolon}`), ]; } else { return null; } } function getOnlyReturnedVariable(node) { return (nodes_1.isReturnStatement(node) || nodes_1.isThrowStatement(node)) && node.argument && nodes_1.isIdentifier(node.argument) ? node.argument : undefined; } function getOnlyDeclaredVariable(node) { if (nodes_1.isVariableDeclaration(node) && node.declarations.length === 1) { const { id, init } = node.declarations[0]; if (nodes_1.isIdentifier(id) && init) { return { id, init }; } } return undefined; } function formatMessage(node, variable) { const action = nodes_1.isReturnStatement(node) ? "return" : "throw"; return `Immediately ${action} this expression instead of assigning it to the temporary variable "${variable}".`; } function getVariables(context) { const { variableScope, variables: currentScopeVariables } = context.getScope(); if (variableScope === context.getScope()) { return currentScopeVariables; } else { return currentScopeVariables.concat(variableScope.variables); } } }, }; module.exports = rule; //# sourceMappingURL=prefer-immediate-return.js.map