io.codemodder.ast.NameResolver Maven / Gradle / Ivy
package io.codemodder.ast;
import com.github.javaparser.ast.Node;
import com.github.javaparser.ast.body.*;
import com.github.javaparser.ast.expr.*;
import java.util.Optional;
import org.javatuples.Pair;
import org.javatuples.Triplet;
/** Find sources of names in JavaParser ASTs. */
final class NameResolver {
private NameResolver() {}
private static Optional isLocalNameSource(final Node n, final String name) {
final Optional maybe =
ASTs.isExpressionStmtDeclarationOf(n, name).map(Triplet::getValue2);
// Possible returns:
return maybe
.or(() -> ASTs.isResourceOf(n, name).map(Triplet::getValue2))
.or(() -> ASTs.isForVariableDeclarationOf(n, name).map(Triplet::getValue2))
.or(() -> ASTs.isForEachVariableDeclarationOf(n, name).map(Triplet::getValue2))
.or(() -> ASTs.isLambdaExprParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isExceptionParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isMethodFormalParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isMethodTypeParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isConstructorFormalParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isConstructorTypeParameterOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isLocalTypeDeclarationOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isLocalRecordDeclarationOf(n, name).map(Pair::getValue1))
.or(() -> ASTs.isPatternExprDeclarationOf(n, name).map(t -> t));
}
private static Optional findLocalNameSource(Node current, final String name) {
// Traverse the tree in reverse pre-order until it hits a declaration
final var it = ASTs.reversePreOrderIterator(current);
while (!(current instanceof TypeDeclaration) && it.hasNext()) {
current = it.next();
final var maybeFound = isLocalNameSource(current, name);
if (maybeFound.isPresent()) return maybeFound;
}
return Optional.empty();
}
private static Optional isFieldOfClass(
final ClassOrInterfaceDeclaration classDeclaration, final String name) {
return classDeclaration.getFields().stream()
.flatMap(field -> ASTs.isFieldDeclarationOf(field, name).stream())
.findAny()
.map(Pair::getValue1);
}
private static Optional isNamedMemberOfClass(
final ClassOrInterfaceDeclaration classDeclaration, final String name) {
return classDeclaration.getMembers().stream()
.flatMap(bodyDecl -> ASTs.isNamedMemberOf(bodyDecl, name).stream())
.findAny()
.map(n -> (Node) n);
}
private static Optional isNameOfClass(
final ClassOrInterfaceDeclaration classDeclaration, final String name) {
if (classDeclaration.getNameAsString().equals(name)) {
return Optional.of(classDeclaration);
}
return Optional.empty();
}
private static Optional findClassLevelNameSource(
final ClassOrInterfaceDeclaration classDeclaration, final String name) {
return isFieldOfClass(classDeclaration, name)
.or(() -> isNamedMemberOfClass(classDeclaration, name))
.or(() -> ASTs.isClassTypeParameterOf(classDeclaration, name).map(Pair::getValue1))
.or(() -> isNameOfClass(classDeclaration, name));
}
static Optional resolveSimpleName(final Node start, final String name) {
// Callable names need more context like signatures to be found. Also, can be overloaded
Node current = start;
// Alternate its search from local (i.e. method level) to class level. It may happen because of
// local type declarations.
while (current.hasParentNode()) {
current = current.getParentNode().get();
// try locally first
final Optional maybeDeclaration = findLocalNameSource(current, name);
if (maybeDeclaration.isPresent()) {
return maybeDeclaration;
}
// No local declaration. Either hit root or a TypeDeclaration after its search
// TypeDeclaration: ClassOrInterfaceDeclaration, EnumDeclaration, RecordDeclaration
if (current instanceof ClassOrInterfaceDeclaration) {
final var classDeclaration = (ClassOrInterfaceDeclaration) current;
final Optional maybeClassMember = findClassLevelNameSource(classDeclaration, name);
if (maybeClassMember.isPresent()) {
return maybeClassMember;
}
}
}
// reached CompilationUnit check for top level classes
final var topLevelTypes = current.findCompilationUnit().get().getTypes();
final var maybeDecl =
topLevelTypes.stream().filter(t -> t.getNameAsString().equals(name)).findFirst();
if (maybeDecl.isPresent()) return maybeDecl.map(n -> n);
// it's either wildcard imported, inherited, or in the package namespace
return Optional.empty();
}
/** Finds the {@link ClassOrInterfaceDeclaration} that is referenced by a {@link ThisExpr}. */
static ClassOrInterfaceDeclaration findThisDeclaration(final ThisExpr thisExpr) {
Node current = thisExpr;
while (current.hasParentNode() && !(current instanceof ClassOrInterfaceDeclaration)) {
current = current.getParentNode().get();
}
return (ClassOrInterfaceDeclaration) current;
}
static Optional findLocalDeclarationOf(final Node start, final String name) {
final var maybeSource = resolveSimpleName(start, name);
if (maybeSource.isPresent()) {
final var source = maybeSource.get();
if (source instanceof Parameter) {
return Optional.of(new ParameterDeclaration((Parameter) source));
}
if (source instanceof VariableDeclarator) {
final var vd = (VariableDeclarator) source;
return ASTs.isVariableOfLocalDeclarationStmt(vd)
.map(
t ->
(LocalDeclaration)
new ExpressionStmtVariableDeclaration(
t.getValue0(), t.getValue1(), t.getValue2()))
.or(
() ->
ASTs.isResource(vd)
.map(p -> new TryResourceDeclaration(p.getValue0(), p.getValue1(), vd)))
.or(
() ->
ASTs.isForInitVariable(vd)
.map(p -> new ForInitDeclaration(p.getValue0(), p.getValue1(), vd)))
.or(
() ->
ASTs.isForEachVariable(vd)
.map(p -> new ForEachDeclaration(p.getValue0(), p.getValue1(), vd)));
}
}
return Optional.empty();
}
static Optional findLocalVariableDeclarationOf(
final Node start, final String name) {
final var maybeSource =
resolveSimpleName(start, name)
.map(n -> n instanceof VariableDeclarator ? (VariableDeclarator) n : null);
if (maybeSource.isPresent()) {
final var vd = maybeSource.get();
return ASTs.isVariableOfLocalDeclarationStmt(vd)
.map(
t ->
(LocalVariableDeclaration)
new ExpressionStmtVariableDeclaration(
t.getValue0(), t.getValue1(), t.getValue2()))
.or(
() ->
ASTs.isResource(vd)
.map(p -> new TryResourceDeclaration(p.getValue0(), p.getValue1(), vd)))
.or(
() ->
ASTs.isForInitVariable(vd)
.map(p -> new ForInitDeclaration(p.getValue0(), p.getValue1(), vd)))
.or(
() ->
ASTs.isForEachVariable(vd)
.map(p -> new ForEachDeclaration(p.getValue0(), p.getValue1(), vd)));
}
return Optional.empty();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy