package org.checkerframework.common.util.debug;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.MethodTree;
import com.sun.source.tree.VariableTree;
import com.sun.tools.javac.processing.JavacProcessingEnvironment;
import com.sun.tools.javac.util.Context;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.element.Element;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.util.Elements;
import org.checkerframework.common.basetype.BaseTypeChecker;
import org.checkerframework.common.basetype.BaseTypeVisitor;
import org.checkerframework.framework.type.AnnotatedTypeFactory;
import org.checkerframework.framework.type.AnnotatedTypeMirror;
import org.checkerframework.framework.type.AnnotatedTypeMirror.AnnotatedDeclaredType;
import org.checkerframework.framework.type.GeneralAnnotatedTypeFactory;
import org.checkerframework.framework.type.GenericAnnotatedTypeFactory;
import org.checkerframework.javacutil.TreeUtils;
/**
* A testing class that can be used to test {@link TypeElement}. In particular it tests that the
* types read from classfiles are the same to the ones from java files.
*
* <p>For testing, you need to do the following:
*
* <ol>
* <li>Run the Checker on the source file like any checker:
* <pre>{@code
* java -processor org.checkerframework.common.util.debug.TypeOutputtingChecker [source-file]
*
* }</pre>
* <li>Run the Checker on the bytecode, by simply running the main and passing the qualified name,
* e.g.
* <pre>{@code
* java org.checkerframework.common.util.debug.TypeOutputtingChecker [qualified-name]
*
* }</pre>
* <li>Apply a simple diff on the two outputs
* </ol>
*/
public class TypeOutputtingChecker extends BaseTypeChecker {
@Override
protected BaseTypeVisitor<?> createSourceVisitor() {
return new Visitor(this);
}
/** Prints the types of the class and all of its enclosing fields, methods, and inner classes */
public static class Visitor extends BaseTypeVisitor<GenericAnnotatedTypeFactory<?, ?, ?, ?>> {
String currentClass;
public Visitor(BaseTypeChecker checker) {
super(checker);
}
// Print types of classes, methods, and fields
@Override
public void processClassTree(ClassTree node) {
TypeElement element = TreeUtils.elementFromDeclaration(node);
currentClass = element.getSimpleName().toString();
AnnotatedDeclaredType type = atypeFactory.getAnnotatedType(node);
System.out.println(node.getSimpleName() + "\t" + type + "\t" + type.directSuperTypes());
super.processClassTree(node);
}
@Override
public Void visitMethod(MethodTree node, Void p) {
ExecutableElement elem = TreeUtils.elementFromDeclaration(node);
AnnotatedTypeMirror type = atypeFactory.getAnnotatedType(node);
System.out.println(currentClass + "." + elem + "\t\t" + type);
// Don't dig deeper
return null;
}
@Override
public Void visitVariable(VariableTree node, Void p) {
VariableElement elem = TreeUtils.elementFromDeclaration(node);
if (elem.getKind().isField()) {
AnnotatedTypeMirror type = atypeFactory.getAnnotatedType(node);
System.out.println(currentClass + "." + elem + "\t\t" + type);
}
// Don't dig deeper
return null;
}
}
public static void main(String[] args) {
new TypeOutputtingChecker().run(args);
}
public void run(String[] args) {
ProcessingEnvironment env = JavacProcessingEnvironment.instance(new Context());
Elements elements = env.getElementUtils();
AnnotatedTypeFactory atypeFactory = new GeneralAnnotatedTypeFactory(this);
for (String className : args) {
TypeElement typeElt = elements.getTypeElement(className);
printClassType(typeElt, atypeFactory);
}
}
/** Prints the types of the class and all of its enclosing fields, methods, and inner classes */
protected static void printClassType(TypeElement typeElt, AnnotatedTypeFactory atypeFactory) {
assert typeElt != null;
String simpleName = typeElt.getSimpleName().toString();
// Output class info
AnnotatedDeclaredType type = atypeFactory.fromElement(typeElt);
System.out.println(simpleName + "\t" + type + "\t" + type.directSuperTypes());
// output fields and methods
for (Element enclosedElt : typeElt.getEnclosedElements()) {
if (enclosedElt instanceof TypeElement) {
printClassType((TypeElement) enclosedElt, atypeFactory);
}
if (!enclosedElt.getKind().isField() && !(enclosedElt instanceof ExecutableElement))
continue;
AnnotatedTypeMirror memberType = atypeFactory.fromElement(enclosedElt);
System.out.println(simpleName + "." + enclosedElt + "\t\t" + memberType);
}
}
}