net.sourceforge.pmd.lang.plsql.symboltable.ClassScope Maven / Gradle / Ivy
/*
* BSD-style license; for more info see http://pmd.sourceforge.net/license.html
*/
package net.sourceforge.pmd.lang.plsql.symboltable;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import net.sourceforge.pmd.lang.ast.Node;
import net.sourceforge.pmd.lang.plsql.ast.ASTName;
import net.sourceforge.pmd.lang.plsql.ast.InternalApiBridge;
import net.sourceforge.pmd.lang.plsql.ast.PLSQLNode;
import net.sourceforge.pmd.lang.symboltable.AbstractScope;
import net.sourceforge.pmd.lang.symboltable.Applier;
import net.sourceforge.pmd.lang.symboltable.ImageFinderFunction;
import net.sourceforge.pmd.lang.symboltable.NameDeclaration;
import net.sourceforge.pmd.lang.symboltable.NameOccurrence;
public class ClassScope extends AbstractScope {
private static final Logger LOG = LoggerFactory.getLogger(ClassScope.class);
// FIXME - this breaks given sufficiently nested code
private static ThreadLocal anonymousInnerClassCounter = new ThreadLocal() {
@Override
protected Integer initialValue() {
return 1;
}
};
private String className;
public ClassScope(String className) {
this.className = PLSQLNode.getCanonicalImage(className);
anonymousInnerClassCounter.set(1);
}
/**
* This is only for anonymous inner classes.
*
* FIXME - should have name like Foo$1, not Anonymous$1 to get this working
* right, the parent scope needs to be passed in when instantiating a
* ClassScope
*/
public ClassScope() {
// this.className = getParent().getEnclosingClassScope().getClassName()
// + "$" + String.valueOf(anonymousInnerClassCounter);
int v = anonymousInnerClassCounter.get().intValue();
this.className = "Anonymous$" + v;
anonymousInnerClassCounter.set(v + 1);
}
@Override
public void addDeclaration(NameDeclaration declaration) {
if (declaration instanceof VariableNameDeclaration && getDeclarations().keySet().contains(declaration)) {
throw new RuntimeException(declaration + " is already in the symbol table");
}
super.addDeclaration(declaration);
}
@Override
public Set addNameOccurrence(NameOccurrence occ) {
PLSQLNameOccurrence occurrence = (PLSQLNameOccurrence) occ;
Set declarations = findVariableHere(occurrence);
Map> methodNames = getMethodDeclarations();
if (!declarations.isEmpty() && occurrence.isMethodOrConstructorInvocation()) {
for (NameDeclaration decl : declarations) {
List nameOccurrences = methodNames.get(decl);
if (nameOccurrences == null) {
// TODO may be a class name: Foo.this.super();
} else {
nameOccurrences.add(occurrence);
Node n = occurrence.getLocation();
if (n instanceof ASTName) {
InternalApiBridge.setNameDeclaration((ASTName) n, decl);
} // TODO what to do with PrimarySuffix case?
}
}
} else if (!declarations.isEmpty() && !occurrence.isThisOrSuper()) {
Map> variableNames = getVariableDeclarations();
for (NameDeclaration decl : declarations) {
List nameOccurrences = variableNames.get(decl);
if (nameOccurrences == null) {
// TODO may be a class name
} else {
nameOccurrences.add(occurrence);
Node n = occurrence.getLocation();
if (n instanceof ASTName) {
InternalApiBridge.setNameDeclaration((ASTName) n, decl);
} // TODO what to do with PrimarySuffix case?
}
}
}
return declarations;
}
public Map> getVariableDeclarations() {
return getDeclarations(VariableNameDeclaration.class);
}
public Map> getMethodDeclarations() {
return getDeclarations(MethodNameDeclaration.class);
}
public Map> getClassDeclarations() {
return getDeclarations(ClassNameDeclaration.class);
}
public ClassScope getEnclosingClassScope() {
return this;
}
public String getClassName() {
return this.className;
}
protected Set findVariableHere(PLSQLNameOccurrence occurrence) {
Set result = new HashSet<>();
Map> variableDeclarations = getVariableDeclarations();
Map> methodDeclarations = getMethodDeclarations();
if (occurrence.isThisOrSuper() || occurrence.getImage().equals(className)) {
if (variableDeclarations.isEmpty() && methodDeclarations.isEmpty()) {
// this could happen if you do this:
// public class Foo {
// private String x = super.toString();
// }
return result;
}
// return any name declaration, since all we really want is to get
// the scope
// for example, if there's a
// public class Foo {
// private static final int X = 2;
// private int y = Foo.X;
// }
// we'll look up Foo just to get a handle to the class scope
// and then we'll look up X.
if (!variableDeclarations.isEmpty()) {
result.add(variableDeclarations.keySet().iterator().next());
return result;
}
result.add(methodDeclarations.keySet().iterator().next());
return result;
}
if (occurrence.isMethodOrConstructorInvocation()) {
for (MethodNameDeclaration mnd : methodDeclarations.keySet()) {
if (mnd.getImage().equals(occurrence.getImage())) {
int args = occurrence.getArgumentCount();
if (args == mnd.getParameterCount() || mnd.isVarargs() && args >= mnd.getParameterCount() - 1) {
// FIXME if several methods have the same name
// and parameter count, only one will get caught here
// we need to make some attempt at type lookup and
// discrimination
// or, failing that, mark this as a usage of all those
// methods
result.add(mnd);
}
}
}
return result;
}
List images = new ArrayList<>();
images.add(occurrence.getImage());
if (null == occurrence.getImage()) {
LOG.trace("occurrence=={} with Argument Count == {} for className={}",
occurrence.toString(),
occurrence.getArgumentCount(),
className);
}
if (occurrence.getImage().startsWith(className)) {
images.add(clipClassName(occurrence.getImage()));
}
ImageFinderFunction finder = new ImageFinderFunction(images);
Applier.apply(finder, getVariableDeclarations().keySet().iterator());
if (finder.getDecl() != null) {
result.add(finder.getDecl());
}
return result;
}
@Override
public String toString() {
StringBuilder res = new StringBuilder("ClassScope (").append(className).append("): ");
Map> classNames = getClassDeclarations();
Map> methodNames = getMethodDeclarations();
Map> variableNames = getVariableDeclarations();
if (!classNames.isEmpty()) {
res.append('(').append(classNames.keySet()).append(')');
}
if (!methodNames.isEmpty()) {
for (MethodNameDeclaration mnd : methodNames.keySet()) {
res.append(mnd.toString());
int usages = methodNames.get(mnd).size();
res.append("(begins at line ").append(mnd.getNode().getBeginLine()).append(", ").append(usages).append(" usages)");
res.append(',');
}
}
if (!variableNames.isEmpty()) {
res.append('(').append(variableNames.keySet()).append(')');
}
return res.toString();
}
private String clipClassName(String s) {
return s.substring(s.indexOf('.') + 1);
}
}