/* * SonarQube Java * Copyright (C) 2012-2016 SonarSource SA * mailto:contact 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. */ package org.sonar.java.checks.naming; import com.google.common.collect.ImmutableList; import org.sonar.check.Rule; import org.sonar.check.RuleProperty; import org.sonar.plugins.java.api.IssuableSubscriptionVisitor; import org.sonar.plugins.java.api.JavaFileScannerContext; import org.sonar.plugins.java.api.tree.ClassTree; import org.sonar.plugins.java.api.tree.IdentifierTree; import org.sonar.plugins.java.api.tree.MethodTree; import org.sonar.plugins.java.api.tree.Tree; import javax.annotation.Nullable; import java.util.List; import java.util.regex.Pattern; @Rule(key = "S3577") public class BadTestClassNameCheck extends IssuableSubscriptionVisitor { private static final String DEFAULT_FORMAT = "^((Test|IT)[a-zA-Z0-9]+|[A-Z][a-zA-Z0-9]*(Test|IT|TestCase|ITCase))$"; @RuleProperty( key = "format", description = "Regular expression against which test class names are checked.", defaultValue = "" + DEFAULT_FORMAT) public String format = DEFAULT_FORMAT; private Pattern pattern = null; @Override public List<Tree.Kind> nodesToVisit() { return ImmutableList.of(Tree.Kind.CLASS); } @Override public void scanFile(JavaFileScannerContext context) { if (pattern == null) { pattern = Pattern.compile(format, Pattern.DOTALL); } super.scanFile(context); } @Override public void visitNode(Tree tree) { if (!hasSemantic()) { return; } ClassTree classTree = (ClassTree) tree; IdentifierTree simpleName = classTree.simpleName(); if (hasInvalidName(simpleName) && hasTestMethod(classTree.members())) { reportIssue(simpleName, "Rename class \"" + simpleName + "\" to match the regular expression: '" + format + "'"); } } private boolean hasInvalidName(@Nullable IdentifierTree className) { return className != null && !pattern.matcher(className.name()).matches(); } private static boolean hasTestMethod(List<Tree> members) { return members.stream() .filter(member -> member.is(Tree.Kind.METHOD)) .map(tree -> ((MethodTree) tree).symbol().metadata()) .anyMatch(metadata -> metadata.isAnnotatedWith("org.junit.Test") || metadata.isAnnotatedWith("org.testng.annotations.Test")); } }