Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.jooby.apt.JoobyProcessor Maven / Gradle / Ivy
/*
* Jooby https://jooby.io
* Apache License Version 2.0 https://jooby.io/LICENSE.txt
* Copyright 2014 Edgar Espina
*/
package io.jooby.apt;
import static io.jooby.apt.JoobyProcessor.Options.*;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Optional.ofNullable;
import static javax.tools.StandardLocation.SOURCE_OUTPUT;
import java.io.*;
import java.net.URI;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.*;
import java.util.function.BiConsumer;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import javax.annotation.processing.*;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.*;
import javax.lang.model.type.DeclaredType;
import javax.tools.Diagnostic;
import javax.tools.JavaFileObject;
import javax.tools.SimpleJavaFileObject;
import javax.tools.StandardLocation;
import io.jooby.internal.apt.*;
@SupportedOptions({
HANDLER,
DEBUG,
INCREMENTAL,
SERVICES,
MVC_METHOD,
RETURN_TYPE,
ROUTER_PREFIX,
ROUTER_SUFFIX,
SKIP_ATTRIBUTE_ANNOTATIONS
})
@SupportedSourceVersion(SourceVersion.RELEASE_17)
public class JoobyProcessor extends AbstractProcessor {
public interface Options {
String HANDLER = "jooby.handler";
String DEBUG = "jooby.debug";
String ROUTER_PREFIX = "jooby.routerPrefix";
String ROUTER_SUFFIX = "jooby.routerSuffix";
String INCREMENTAL = "jooby.incremental";
String RETURN_TYPE = "jooby.returnType";
String MVC_METHOD = "jooby.mvcMethod";
String SERVICES = "jooby.services";
String SKIP_ATTRIBUTE_ANNOTATIONS = "jooby.skipAttributeAnnotations";
static boolean boolOpt(ProcessingEnvironment environment, String option, boolean defaultValue) {
return Boolean.parseBoolean(
environment.getOptions().getOrDefault(option, String.valueOf(defaultValue)));
}
static List stringListOpt(ProcessingEnvironment environment, String option) {
String value = string(environment, option, null);
return value == null || value.isEmpty()
? List.of()
: Stream.of(value.split(",")).filter(it -> !it.isBlank()).map(String::trim).toList();
}
static String string(ProcessingEnvironment environment, String option, String defaultValue) {
String value = environment.getOptions().getOrDefault(option, defaultValue);
return value == null || value.isEmpty() ? defaultValue : value;
}
}
protected MvcContext context;
private BiConsumer output;
private final Set processed = new HashSet<>();
public JoobyProcessor(BiConsumer output) {
this.output = output;
}
public JoobyProcessor() {}
@Override
public synchronized void init(ProcessingEnvironment processingEnvironment) {
this.context =
new MvcContext(
processingEnvironment,
ofNullable(output)
.orElseGet(
() ->
(kind, message) ->
processingEnvironment.getMessager().printMessage(kind, message)));
super.init(processingEnvironment);
}
@Override
public boolean process(Set extends TypeElement> annotations, RoundEnvironment roundEnv) {
try {
if (roundEnv.processingOver()) {
context.debug("Output:");
context.getRouters().forEach(it -> context.debug(" %s.java", it.getGeneratedType()));
if (context.generateServices()) {
doServices(context.getProcessingEnvironment().getFiler(), context.getRouters());
}
return false;
} else {
var routeMap = buildRouteRegistry(annotations, roundEnv);
verifyBeanValidationDependency(routeMap.values());
for (var router : routeMap.values()) {
try {
context.add(router);
var sourceCode = router.toSourceCode(null);
var sourceLocation = router.getGeneratedFilename();
onGeneratedSource(toJavaFileObject(sourceLocation, sourceCode));
context.debug("router %s: %s", router.getTargetType(), router.getGeneratedType());
router.getRoutes().forEach(it -> context.debug(" %s", it));
writeSource(router, sourceLocation, sourceCode);
} catch (IOException cause) {
throw new RuntimeException("Unable to generate: " + router.getTargetType(), cause);
}
}
return true;
}
} catch (Exception cause) {
context.error(
Optional.ofNullable(cause.getMessage()).orElse("Unable to generate routes"), cause);
throw sneakyThrow0(cause);
}
}
private void writeSource(MvcRouter router, String sourceLocation, String sourceCode)
throws IOException {
var environment = context.getProcessingEnvironment();
var filer = environment.getFiler();
if (router.isKt()) {
var kapt = environment.getOptions().get("kapt.kotlin.generated");
if (kapt != null) {
var output = Paths.get(kapt, sourceLocation);
Files.createDirectories(output.getParent());
Files.writeString(output, sourceCode);
} else {
var ktFile =
filer.createResource(SOURCE_OUTPUT, "", sourceLocation, router.getTargetType());
try (var writer = ktFile.openWriter()) {
writer.write(sourceCode);
}
}
} else {
var javaFIle = filer.createSourceFile(router.getGeneratedType(), router.getTargetType());
try (var writer = javaFIle.openWriter()) {
writer.write(sourceCode);
}
}
}
private static JavaFileObject toJavaFileObject(String filename, String source) {
var uri = URI.create(filename);
return new SimpleJavaFileObject(uri, JavaFileObject.Kind.SOURCE) {
private final long lastModified = System.currentTimeMillis();
@Override
public String getCharContent(boolean ignoreEncodingErrors) {
return source;
}
@Override
public InputStream openInputStream() {
return new ByteArrayInputStream(getCharContent(true).getBytes(UTF_8));
}
@Override
public long getLastModified() {
return lastModified;
}
@Override
public String toString() {
return getCharContent(false);
}
};
}
protected void onGeneratedSource(JavaFileObject source) {}
private void doServices(Filer filer, List routers) {
try {
var location = "META-INF/services/io.jooby.MvcFactory";
context.debug(location);
var resource = filer.createResource(StandardLocation.CLASS_OUTPUT, "", location);
var content = new StringBuilder();
for (var router : routers) {
var classname = router.getGeneratedType();
context.debug(" %s", classname);
content.append(classname).append(System.lineSeparator());
}
try (var writer = new PrintWriter(resource.openOutputStream())) {
writer.println(content);
}
} catch (IOException cause) {
throw propagate(cause);
}
}
private Map buildRouteRegistry(
Set extends TypeElement> annotations, RoundEnvironment roundEnv) {
Map registry = new LinkedHashMap<>();
for (var annotation : annotations) {
context.debug("found annotation: %s", annotation);
var elements = roundEnv.getElementsAnnotatedWith(annotation);
// Element could be Class or Method, bc @Path can be applied to both of them
// Also we need to expand lookup to external jars see #2486
for (var element : elements) {
context.debug(" %s", element);
if (element instanceof TypeElement typeElement) {
buildRouteRegistry(registry, typeElement);
} else if (element instanceof ExecutableElement method) {
buildRouteRegistry(registry, (TypeElement) method.getEnclosingElement());
}
}
}
// Remove all abstract router
var abstractTypes =
registry.entrySet().stream()
.filter(it -> it.getValue().isAbstract())
.map(Map.Entry::getKey)
.collect(Collectors.toSet());
abstractTypes.forEach(registry::remove);
// Generate unique method name by router
for (var router : registry.values()) {
// Initialize with supports/create method from MvcFactory (avoid name collision)
var names = new HashSet<>();
for (var route : router.getRoutes()) {
if (!names.add(route.getMethodName())) {
var paramsString =
route.getRawParameterTypes(true).stream()
.map(it -> it.substring(Math.max(0, it.lastIndexOf(".") + 1)))
.map(it -> Character.toUpperCase(it.charAt(0)) + it.substring(1))
.collect(Collectors.joining());
route.setGeneratedName(route.getMethodName() + paramsString);
} else {
route.setGeneratedName(route.getMethodName());
}
}
}
return registry;
}
/**
* Scan routes from basType and any super class of it. It saves all route method found in current
* type or super (inherited). Routes method from super types are also saved.
*
* Abstract route method are ignored.
*
* @param registry Route registry.
* @param currentType Base type.
*/
private void buildRouteRegistry(Map registry, TypeElement currentType) {
for (TypeElement superType : context.superTypes(currentType)) {
if (processed.add(superType)) {
// collect all declared methods
superType.getEnclosedElements().stream()
.filter(ExecutableElement.class::isInstance)
.map(ExecutableElement.class::cast)
.forEach(
method -> {
if (method.getModifiers().contains(Modifier.ABSTRACT)) {
context.debug("ignoring abstract method: %s %s", superType, method);
} else {
method.getAnnotationMirrors().stream()
.map(AnnotationMirror::getAnnotationType)
.map(DeclaredType::asElement)
.filter(TypeElement.class::isInstance)
.map(TypeElement.class::cast)
.filter(HttpMethod::hasAnnotation)
.forEach(
annotation -> {
Stream.of(currentType, superType)
.distinct()
.forEach(
routerClass ->
registry
.computeIfAbsent(
routerClass, type -> new MvcRouter(context, type))
.put(annotation, method));
});
}
});
} else {
if (!currentType.equals(superType)) {
// edge case when controller has no method and extends another class which has.
registry.computeIfAbsent(currentType, key -> new MvcRouter(key, registry.get(superType)));
}
}
}
}
@Override
public Set getSupportedAnnotationTypes() {
var supportedTypes = new HashSet();
supportedTypes.addAll(HttpPath.PATH.getAnnotations());
supportedTypes.addAll(HttpMethod.annotations());
return supportedTypes;
}
@Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
}
@Override
public Set getSupportedOptions() {
var options = new HashSet<>(super.getSupportedOptions());
if (context.isIncremental()) {
// Enables incremental annotation processing support in Gradle.
// If service provider configuration is being generated,
// only 'aggregating' mode is supported since it's likely that
// more then one originating element is passed to the Filer
// API on writing the resource file - isolating mode does not
// allow this.
options.add(
String.format(
"org.gradle.annotation.processing.%s",
context.generateServices() ? "aggregating" : "isolating"));
}
return options;
}
/**
* Throws any throwable 'sneakily' - you don't need to catch it, nor declare that you throw it
* onwards. The exception is still thrown - javac will just stop whining about it.
*
* Example usage:
*
*
public void run() {
* throw sneakyThrow(new IOException("You don't need to catch me!"));
* }
*
* NB: The exception is not wrapped, ignored, swallowed, or redefined. The JVM actually does
* not know or care about the concept of a 'checked exception'. All this method does is hide the
* act of throwing a checked exception from the java compiler.
*
*
Note that this method has a return type of {@code RuntimeException}; it is advised you
* always call this method as argument to the {@code throw} statement to avoid compiler errors
* regarding no return statement and similar problems. This method won't of course return an
* actual {@code RuntimeException} - it never returns, it always throws the provided exception.
*
* @param x The throwable to throw without requiring you to catch its type.
* @return A dummy RuntimeException; this method never returns normally, it always throws
* an exception!
*/
public static RuntimeException propagate(final Throwable x) {
if (x == null) {
throw new NullPointerException("x");
}
return sneakyThrow0(x);
}
/**
* Make a checked exception un-checked and rethrow it.
*
* @param x Exception to throw.
* @param Exception type.
* @throws E Exception to throw.
*/
@SuppressWarnings("unchecked")
private static E sneakyThrow0(final Throwable x) throws E {
throw (E) x;
}
private void verifyBeanValidationDependency(Collection routers) {
var hasBeanValidation = routers.stream().anyMatch(MvcRouter::hasBeanValidation);
if (hasBeanValidation) {
var missingDependency =
Stream.of(
"io.jooby.hibernate.validator.HibernateValidatorModule",
"io.jooby.avaje.validator.AvajeValidatorModule",
"io.jooby.apt.validator.FakeValidatorModule")
.map(name -> processingEnv.getElementUtils().getTypeElement(name))
.filter(Objects::nonNull)
.findFirst()
.isEmpty();
if (missingDependency) {
processingEnv
.getMessager()
.printMessage(
Diagnostic.Kind.ERROR,
"Unable to load 'BeanValidator' class. Bean validation usage (@Valid) was detected,"
+ " but the appropriate dependency is missing. Please ensure that you have"
+ " added the corresponding validation dependency (e.g.,"
+ " jooby-hibernate-validator, jooby-avaje-validator).");
}
}
}
}