Я пишу плагин компилятора Java, используя модуль jdk.compiler и Java 15 с включенными функциями предварительного просмотра. Модуль jdk.compiler, представленный в Java 9, согласно Oracle, содержит все API-интерфейсы, которые мы использовали в tools.jar java 8, за исключением пакета com.sun.tools.javac.tree. Что я хочу сделать, так это найти все классы, аннотированные моей аннотацией, и получить атрибут value(), в данном случае типа Class<?>.
Код плагина:
public class ExtensionMethodPlugin implements Plugin{
@Override
public String getName() {
return "ExtensionMethodPlugin";
}
@Override
public void init(JavacTask task, String... args) {
task.addTaskListener(new TaskListener() {
@Override
public void started(TaskEvent event) {
}
@Override
public void finished(TaskEvent event) {
if (event.getKind() != TaskEvent.Kind.PARSE) {
return;
}
event.getCompilationUnit().accept(new TreeScanner<Void, Void>() {
@Override
public Void visitClass(ClassTree node, Void aVoid) {
var trees = Trees.instance(task);
var targetClass = InstrumentationUtils.findExtensionTargetForClass(node, trees, event.getCompilationUnit());
if (targetClass.isEmpty()) {
return super.visitClass(node, aVoid);
}
var methods = node
.getMembers()
.stream()
.filter(e -> e instanceof MethodTree)
.map(e -> (MethodTree) e)
.filter(tree -> tree.getReturnType() != null)
.filter(e -> InstrumentationUtils.shouldInstrumentMethod(e, targetClass.get()))
.collect(Collectors.toList());
throw new RuntimeException("Not implemented");
}
}, null);
}
});
}
}
Код InstrumentationUtils:
@UtilityClass
public class InstrumentationUtils {
public Optional<Class<?>> findExtensionTargetForClass(ClassTree node, Trees trees, CompilationUnitTree tree) {
return node
.getModifiers()
.getAnnotations()
.stream()
.filter(a -> Extension.class.getSimpleName().equals(a.getAnnotationType().toString()))
.findFirst()
.map(e -> InstrumentationUtils.findConstantTargetClassInAnnotationTree(e, trees, tree));
}
private Class<?> findConstantTargetClassInAnnotationTree(AnnotationTree tree, Trees trees, CompilationUnitTree compilationUnitTree) {
return tree
.getArguments()
.stream()
.filter(entry -> entry.getKind() == Tree.Kind.ASSIGNMENT)
.findFirst()
.map(e -> (AssignmentTree) e)
.map(e -> classForName(e, trees, compilationUnitTree))
.orElseThrow(() -> new RuntimeException("Cannot compile: Missing constant class target in annotation!"));
}
private Class<?> classForName(AssignmentTree tree, Trees trees, CompilationUnitTree compilationUnitTree){
//Don't know how to move from here
}
}
Код аннотации:
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.SOURCE)
public @interface Extension {
Class<?> value();
}
Единственные решения, которые я нашел, связаны с использованием классов JCTree, но они, как я уже говорил ранее, больше не доступны. Я знаю, что могу довольно легко принудительно экспортировать этот пакет в модуль jdk.compiler, но я бы предпочел найти лучший способ. Я пытался использовать TreeScanner, но это ни к чему не привело:
tree.accept(new TreeScanner<Class<?>, Void>() {
// b.class
@Override
public Class<?> visitAssignment(AssignmentTree node, Void unused) {
return node.getExpression().accept(new TreeScanner<Class<?>, Void>(){
//.class
@Override
public Class<?> visitMemberSelect(MemberSelectTree node, Void unused) {
return node.getExpression().accept(new TreeScanner<Class<?>, Void>(){
// class ???
@Override
public Class<?> visitIdentifier(IdentifierTree node, Void unused) {
trees.printMessage(Diagnostic.Kind.WARNING, node.getName(), tree, compilationUnitTree);
return super.visitIdentifier(node, unused);
}
}, null);
}
}, null);
}
}, null);
Заранее спасибо!
Вы можете представить их как менее абстрактные реализации модуля jdk.compiler, они были плохо написаны, но раньше они многое упрощали.
JCLiteral#значение
Вы определенно хотите превратить спагетти-код в методы/классы.
Когда все, что вам нужно сделать, это «найти все классы, которые аннотированы моей аннотацией, и получить атрибут value()», тогда вам вообще не нужен API дерева из модуля jdk.compiler
, API процессора аннотаций содержит в модуле java.compiler
достаточно и намного проще в обращении.
Вот самодостаточный пример:
static final String EXAMPLE_CODE =
"import java.lang.annotation.*;\n" +
"\n" +
"@Target({ElementType.TYPE})\n" +
"@Retention(RetentionPolicy.SOURCE)\n" +
"@interface Extension {\n" +
" Class<?> value();\n" +
"}\n" +
"\n" +
"@Extension(Runnable.class)\n" +
"public class SomeClass {\n" +
" public static void aMethod() {\n" +
" }\n" +
"\n" +
" @Extension(SomeClass.class)\n" +
" class NestedClassToMakeItMoreExciting {\n" +
" \n" +
" }\n" +
"}";
public static void main(String[] args) throws IOException {
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fileManager=compiler.getStandardFileManager(null,null,null);
Path tmp = Files.createTempDirectory("compile-test-");
fileManager.setLocation(StandardLocation.CLASS_OUTPUT, Set.of(tmp.toFile()));
Set<JavaFileObject> srcObj = Set.of(new SimpleJavaFileObject(
URI.create("string:///SomeClass.java"), JavaFileObject.Kind.SOURCE) {
@Override
public CharSequence getCharContent(boolean ignoreEncodingErrors) {
return EXAMPLE_CODE;
}
});
JavaCompiler.CompilationTask task
= compiler.getTask(null, fileManager, null, null, null, srcObj);
task.setProcessors(Set.of(new MyProcessor()));
task.call();
}
@SupportedAnnotationTypes("Extension")
@SupportedSourceVersion(SourceVersion.RELEASE_14)
static class MyProcessor extends AbstractProcessor {
@Override
public boolean process(
Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
TypeElement ext = processingEnv.getElementUtils().getTypeElement("Extension");
for(Element annotatedElement: roundEnv.getElementsAnnotatedWith(ext)) {
System.out.println(annotatedElement);
for(AnnotationMirror am: annotatedElement.getAnnotationMirrors()) {
if (am.getAnnotationType().asElement() == ext) {
am.getElementValues().forEach((e,v) ->
System.out.println("\t"+e.getSimpleName()+" = "+v.getValue()));
}
}
}
return true;
}
}
SomeClass
value = java.lang.Runnable
SomeClass.NestedClassToMakeItMoreExciting
value = SomeClass
Конечно, при необходимости эту функциональность можно комбинировать с функциями расширения jdk.compiler
.
Что такое «классы JCTree» и чем они отличаются от тех древовидных классов модуля
jdk.compiler
?