Fixed bug 2404700 - UseSingleton should not act on enums

git-svn-id: https://pmd.svn.sourceforge.net/svnroot/pmd/branches/pmd/4.2.x@6746 51baf565-9d33-0410-a72c-fc3788e3496d
This commit is contained in:
Xavier Le Vourch
2008-12-11 06:32:25 +00:00
parent e300a2630b
commit 0fb697a50b
3 changed files with 81 additions and 55 deletions

View File

@ -6,6 +6,7 @@ Fixed bug 2230809 - False +: ClassWithOnlyPrivateConstructorsShouldBeFinal
Fixed bug 2338341 - ArrayIndexOutOfBoundsException in CPD (on Ruby)
Fixed bug 2315599 - False +: UseSingleton with class containing constructor
Fixed bug 1955852 - false positives for UnusedPrivateMethod & UnusedLocalVariable
Fixed bug 2404700 - UseSingleton should not act on enums
October 12, 2008 - 4.2.4:

View File

@ -134,6 +134,36 @@ public class UseSingleton extends Exception {
}
]]></code>
</test-code>
</test-data>
<test-code>
<description><![CDATA[
inner should be singleton since all static, public constructor
]]></description>
<expected-problems>1</expected-problems>
<code><![CDATA[
public class Foo {
static class Bar {
public Bar() { }
public static void doSomething() {}
}
public void doSomething() {}
}
]]></code>
</test-code>
<test-code>
<description><![CDATA[
[ 2404700 ] UseSingleton should not act on enums
]]></description>
<expected-problems>0</expected-problems>
<code><![CDATA[
public enum EnumTest {
A, B;
EnumTest() { }
public static void main(String[] args) {
System.out.println(EnumTest.A);
}
}
]]></code>
</test-code>
</test-data>

View File

@ -4,71 +4,66 @@
package net.sourceforge.pmd.rules.design;
import net.sourceforge.pmd.AbstractRule;
import net.sourceforge.pmd.ast.ASTClassOrInterfaceBody;
import net.sourceforge.pmd.ast.ASTClassOrInterfaceDeclaration;
import net.sourceforge.pmd.ast.ASTClassOrInterfaceType;
import net.sourceforge.pmd.ast.ASTCompilationUnit;
import net.sourceforge.pmd.ast.ASTConstructorDeclaration;
import net.sourceforge.pmd.ast.ASTFieldDeclaration;
import net.sourceforge.pmd.ast.ASTMethodDeclaration;
import net.sourceforge.pmd.ast.ASTResultType;
import net.sourceforge.pmd.ast.Node;
public class UseSingleton extends AbstractRule {
private boolean isOK;
private int methodCount;
@Override
public Object visit(ASTClassOrInterfaceBody decl, Object data) {
if (decl.jjtGetParent() instanceof ASTClassOrInterfaceDeclaration) {
ASTClassOrInterfaceDeclaration parent = (ASTClassOrInterfaceDeclaration) decl.jjtGetParent();
if (parent.isAbstract() || parent.isInterface()) {
return super.visit(decl, data);
}
int i = decl.jjtGetNumChildren();
int methodCount = 0;
boolean isOK = false;
while (i > 0) {
Node n = decl.jjtGetChild(--i).jjtGetChild(0);
if (n instanceof ASTFieldDeclaration) {
if (!((ASTFieldDeclaration) n).isStatic()) {
isOK = true;
break;
}
} else if (n instanceof ASTConstructorDeclaration) {
if (((ASTConstructorDeclaration) n).isPrivate()) {
isOK = true;
break;
}
} else if (n instanceof ASTMethodDeclaration) {
ASTMethodDeclaration m = (ASTMethodDeclaration) n;
if (!m.isPrivate()) {
methodCount++;
}
if (!m.isStatic()) {
isOK = true;
break;
}
public Object visit(ASTCompilationUnit cu, Object data) {
methodCount = 0;
isOK = false;
Object result = cu.childrenAccept(this, data);
if (!isOK && methodCount > 0) {
addViolation(data, cu);
}
// TODO use symbol table
if (m.getMethodName().equals("suite")) {
ASTResultType res = m.getResultType();
ASTClassOrInterfaceType c = res.getFirstChildOfType(ASTClassOrInterfaceType.class);
if (c != null && c.hasImageEqualTo("Test")) {
isOK = true;
break;
}
}
return result;
}
public Object visit(ASTFieldDeclaration decl, Object data) {
if (!decl.isStatic()) {
isOK = true;
}
return data;
}
public Object visit(ASTConstructorDeclaration decl, Object data) {
if (decl.isPrivate()) {
isOK = true;
}
return data;
}
public Object visit(ASTClassOrInterfaceDeclaration decl, Object data) {
if (decl.isAbstract()) {
isOK = true;
}
}
if (!isOK && methodCount > 0) {
addViolation(data, decl);
}
}
return super.visit(decl, data);
}
public Object visit(ASTMethodDeclaration decl, Object data) {
// Private method does no count
if ( ! decl.isPrivate() ) {
methodCount++;
}
if (!isOK && !decl.isStatic() ) {
isOK = true;
}
// TODO use symbol table
if (decl.getMethodName().equals("suite")) {
ASTResultType res = decl.getFirstChildOfType(ASTResultType.class);
ASTClassOrInterfaceType c = res.getFirstChildOfType(ASTClassOrInterfaceType.class);
if (c != null && c.hasImageEqualTo("Test")) {
isOK = true;
}
}
return data;
}
}