
org.sonarsource.slang.checks.WrongAssignmentOperatorCheck Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of slang-checks Show documentation
Show all versions of slang-checks Show documentation
SonarSource Language analyzer
/*
* SonarSource SLang
* Copyright (C) 2018-2024 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 Sonar Source-Available License Version 1, as published by SonarSource SA.
*
* 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 Sonar Source-Available License for more details.
*
* You should have received a copy of the Sonar Source-Available License
* along with this program; if not, see https://sonarsource.com/license/ssal/
*/
package org.sonarsource.slang.checks;
import java.util.List;
import org.sonar.check.Rule;
import org.sonarsource.slang.api.AssignmentExpressionTree;
import org.sonarsource.slang.api.TextRange;
import org.sonarsource.slang.api.Token;
import org.sonarsource.slang.api.Tree;
import org.sonarsource.slang.api.UnaryExpressionTree;
import org.sonarsource.slang.api.UnaryExpressionTree.Operator;
import org.sonarsource.slang.checks.api.CheckContext;
import org.sonarsource.slang.checks.api.InitContext;
import org.sonarsource.slang.checks.api.SlangCheck;
import org.sonarsource.slang.impl.TextRanges;
import static java.util.Arrays.asList;
import static org.sonarsource.slang.api.AssignmentExpressionTree.Operator.EQUAL;
@Rule(key = "S2757")
public class WrongAssignmentOperatorCheck implements SlangCheck {
private static final List SUSPICIOUS_UNARY_OPERATORS = asList(Operator.NEGATE, Operator.PLUS, Operator.MINUS);
@Override
public void initialize(InitContext init) {
init.register(AssignmentExpressionTree.class, (ctx, assignment) -> {
Tree rightHandSide = assignment.statementOrExpression();
if (assignment.operator() != EQUAL || !isSuspiciousUnaryExpression(rightHandSide)) {
return;
}
List leftHandSideTokens = assignment.leftHandSide().metaData().tokens();
Token variableLastToken = leftHandSideTokens.get(leftHandSideTokens.size() - 1);
List allTokens = assignment.metaData().tokens();
Token operatorToken = allTokens.get(allTokens.indexOf(variableLastToken) + 1);
Token expressionFirstToken = rightHandSide.metaData().tokens().get(0);
if (!hasSpacingBetween(operatorToken, expressionFirstToken) && hasSpacingBetween(variableLastToken, operatorToken)) {
TextRange range = TextRanges.merge(asList(operatorToken.textRange(), expressionFirstToken.textRange()));
ctx.reportIssue(range, getMessage(expressionFirstToken, ctx));
}
});
}
private static String getMessage(Token expressionFirstToken, CheckContext aeTree) {
if (isSingleNegationAssignment(expressionFirstToken, aeTree)) {
// For expressions such as "a = b =! c" we want to display the other message
return "Add a space between \"=\" and \"!\" to avoid confusion.";
}
return "Was \"" + expressionFirstToken.text() + "=\" meant instead?";
}
private static boolean isSingleNegationAssignment(Token firstToken, CheckContext aeTree) {
return "!".equals(firstToken.text()) && !(aeTree.parent() instanceof AssignmentExpressionTree);
}
private static boolean hasSpacingBetween(Token firstToken, Token secondToken) {
return firstToken.textRange().end().line() != secondToken.textRange().start().line()
|| firstToken.textRange().end().lineOffset() != secondToken.textRange().start().lineOffset();
}
private static boolean isSuspiciousUnaryExpression(Tree tree) {
return tree instanceof UnaryExpressionTree && SUSPICIOUS_UNARY_OPERATORS.contains(((UnaryExpressionTree) tree).operator());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy