
org.sonar.java.checks.ForLoopCounterChangedCheck Maven / Gradle / Ivy
/*
* SonarQube Java
* Copyright (C) 2012 SonarSource
* [email protected]
*
* 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 02
*/
package org.sonar.java.checks;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import com.sonar.sslr.api.AstNode;
import com.sonar.sslr.api.Token;
import com.sonar.sslr.squid.checks.SquidCheck;
import org.sonar.check.BelongsToProfile;
import org.sonar.check.Priority;
import org.sonar.check.Rule;
import org.sonar.java.ast.api.JavaPunctuator;
import org.sonar.java.ast.api.JavaTokenType;
import org.sonar.java.ast.parser.JavaGrammar;
import org.sonar.sslr.ast.AstSelect;
import org.sonar.sslr.parser.LexerlessGrammar;
import java.util.Collections;
import java.util.Set;
@Rule(
key = "ForLoopCounterChangedCheck",
priority = Priority.MAJOR)
@BelongsToProfile(title = "Sonar way", priority = Priority.MAJOR)
public class ForLoopCounterChangedCheck extends SquidCheck {
private Set pendingLoopCounters = Collections.EMPTY_SET;
private final Set loopCounters = Sets.newHashSet();
@Override
public void init() {
subscribeTo(JavaGrammar.FOR_STATEMENT);
subscribeTo(JavaGrammar.ASSIGNMENT_EXPRESSION);
subscribeTo(JavaGrammar.UNARY_EXPRESSION);
subscribeTo(JavaGrammar.STATEMENT);
}
@Override
public void visitFile(AstNode astNode) {
loopCounters.clear();
}
@Override
public void visitNode(AstNode node) {
if (node.is(JavaGrammar.FOR_STATEMENT)) {
pendingLoopCounters = getLoopCounters(node);
} else if (node.is(JavaGrammar.STATEMENT) && node.getParent().is(JavaGrammar.FOR_STATEMENT)) {
loopCounters.addAll(pendingLoopCounters);
pendingLoopCounters = Collections.EMPTY_SET;
} else if (!loopCounters.isEmpty()) {
if (node.is(JavaGrammar.ASSIGNMENT_EXPRESSION)) {
for (int i = 0; i < node.getNumberOfChildren() - 1; i++) {
check(merge(node.getChild(i)), node.getChild(i).getTokenLine());
}
} else if (isIncrementOrDecrementExpression(node)) {
for (AstNode child : node.getChildren()) {
check(merge(child), child.getTokenLine());
}
}
}
}
@Override
public void leaveNode(AstNode node) {
if (node.is(JavaGrammar.FOR_STATEMENT)) {
loopCounters.removeAll(getLoopCounters(node));
}
}
private static boolean isIncrementOrDecrementExpression(AstNode node) {
return node.is(JavaGrammar.UNARY_EXPRESSION) &&
node.select()
.children(JavaGrammar.PREFIX_OP, JavaGrammar.POST_FIX_OP)
.children(JavaPunctuator.INC, JavaPunctuator.DEC)
.isNotEmpty();
}
private Set getLoopCounters(AstNode node) {
Set result;
AstSelect identifiers = node.select()
.children(JavaGrammar.FOR_INIT)
.children(JavaGrammar.VARIABLE_DECLARATORS)
.children(JavaGrammar.VARIABLE_DECLARATOR)
.children(JavaTokenType.IDENTIFIER);
if (identifiers.isNotEmpty()) {
ImmutableSet.Builder builder = ImmutableSet.builder();
for (AstNode identifier : identifiers) {
builder.add(identifier.getTokenOriginalValue());
}
result = builder.build();
} else {
result = Collections.EMPTY_SET;
}
return result;
}
private void check(String string, int line) {
if (loopCounters.contains(string)) {
getContext().createLineViolation(this, "Refactor the code in order to not assign to this loop counter from within the loop body.", line);
}
}
private static String merge(AstNode node) {
StringBuilder sb = new StringBuilder();
for (Token token : node.getTokens()) {
sb.append(token.getOriginalValue());
}
return sb.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy