-
Notifications
You must be signed in to change notification settings - Fork 2.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Provider tighter integration of Qute into RESTEasy Reactive
This change uses the ServerRestHandler mechanism instead of a response filter to convert a template into a response. This also paves the way to implement returning template responses as chunks (using Qute's createMulti)
- Loading branch information
Showing
8 changed files
with
165 additions
and
98 deletions.
There are no files selected for viewing
64 changes: 54 additions & 10 deletions
64
...main/java/io/quarkus/resteasy/reactive/qute/deployment/ResteasyReactiveQuteProcessor.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,37 +1,81 @@ | ||
package io.quarkus.resteasy.reactive.qute.deployment; | ||
|
||
import static org.jboss.resteasy.reactive.common.processor.ResteasyReactiveDotNames.COMPLETION_STAGE; | ||
import static org.jboss.resteasy.reactive.common.processor.ResteasyReactiveDotNames.UNI; | ||
|
||
import java.util.Collections; | ||
import java.util.List; | ||
import java.util.Map; | ||
|
||
import org.jboss.jandex.ClassInfo; | ||
import org.jboss.jandex.DotName; | ||
import org.jboss.jandex.MethodInfo; | ||
import org.jboss.jandex.ParameterizedType; | ||
import org.jboss.jandex.Type; | ||
import org.jboss.resteasy.reactive.server.handlers.UniResponseHandler; | ||
import org.jboss.resteasy.reactive.server.model.FixedHandlersChainCustomizer; | ||
import org.jboss.resteasy.reactive.server.model.HandlerChainCustomizer; | ||
import org.jboss.resteasy.reactive.server.processor.scanning.MethodScanner; | ||
|
||
import io.quarkus.deployment.Feature; | ||
import io.quarkus.deployment.annotations.BuildStep; | ||
import io.quarkus.deployment.builditem.FeatureBuildItem; | ||
import io.quarkus.deployment.builditem.nativeimage.ReflectiveHierarchyIgnoreWarningBuildItem; | ||
import io.quarkus.qute.TemplateInstance; | ||
import io.quarkus.resteasy.reactive.qute.runtime.TemplateResponseFilter; | ||
import io.quarkus.resteasy.reactive.qute.runtime.TemplateResponseUniHandler; | ||
import io.quarkus.resteasy.reactive.server.spi.MethodScannerBuildItem; | ||
import io.quarkus.resteasy.reactive.server.spi.NonBlockingReturnTypeBuildItem; | ||
import io.quarkus.resteasy.reactive.spi.CustomContainerResponseFilterBuildItem; | ||
|
||
public class ResteasyReactiveQuteProcessor { | ||
|
||
private static final DotName TEMPLATE_INSTANCE = DotName.createSimple(TemplateInstance.class.getName()); | ||
|
||
@BuildStep | ||
FeatureBuildItem feature() { | ||
return new FeatureBuildItem(Feature.RESTEASY_REACTIVE_QUTE); | ||
} | ||
|
||
@BuildStep | ||
CustomContainerResponseFilterBuildItem registerProviders() { | ||
return new CustomContainerResponseFilterBuildItem(TemplateResponseFilter.class.getName()); | ||
} | ||
|
||
@BuildStep | ||
ReflectiveHierarchyIgnoreWarningBuildItem ignoreReflectiveWarning() { | ||
return new ReflectiveHierarchyIgnoreWarningBuildItem(new ReflectiveHierarchyIgnoreWarningBuildItem.DotNameExclusion( | ||
DotName.createSimple(TemplateInstance.class.getName()))); | ||
return new ReflectiveHierarchyIgnoreWarningBuildItem( | ||
new ReflectiveHierarchyIgnoreWarningBuildItem.DotNameExclusion(TEMPLATE_INSTANCE)); | ||
} | ||
|
||
@BuildStep | ||
NonBlockingReturnTypeBuildItem nonBlockingTemplateInstance() { | ||
return new NonBlockingReturnTypeBuildItem(DotName.createSimple(TemplateInstance.class.getName())); | ||
return new NonBlockingReturnTypeBuildItem(TEMPLATE_INSTANCE); | ||
} | ||
|
||
@BuildStep | ||
public MethodScannerBuildItem configureHandler() { | ||
return new MethodScannerBuildItem(new MethodScanner() { | ||
@Override | ||
public List<HandlerChainCustomizer> scan(MethodInfo method, ClassInfo actualEndpointClass, | ||
Map<String, Object> methodContext) { | ||
if (method.returnType().name().equals(TEMPLATE_INSTANCE) || isAsyncTemplateInstance(method.returnType())) { | ||
// TemplateResponseUniHandler creates a Uni, so we also need to introduce another Uni handler | ||
// so RR actually gets the result | ||
// the reason why we use AFTER_METHOD_INVOKE_SECOND_ROUND is to be able to properly support Uni<TemplateInstance> | ||
return Collections.singletonList( | ||
new FixedHandlersChainCustomizer( | ||
List.of(new TemplateResponseUniHandler(), new UniResponseHandler()), | ||
HandlerChainCustomizer.Phase.AFTER_METHOD_INVOKE_SECOND_ROUND)); | ||
} | ||
return Collections.emptyList(); | ||
} | ||
|
||
private boolean isAsyncTemplateInstance(Type type) { | ||
boolean isAsyncTemplateInstance = false; | ||
if (type.kind() == Type.Kind.PARAMETERIZED_TYPE) { | ||
ParameterizedType parameterizedType = type.asParameterizedType(); | ||
if ((parameterizedType.name().equals(UNI) || parameterizedType.name().equals(COMPLETION_STAGE)) | ||
&& (parameterizedType.arguments().size() == 1)) { | ||
DotName firstParameterType = parameterizedType.arguments().get(0).name(); | ||
isAsyncTemplateInstance = firstParameterType.equals(TEMPLATE_INSTANCE); | ||
} | ||
} | ||
return isAsyncTemplateInstance; | ||
} | ||
}); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
80 changes: 0 additions & 80 deletions
80
...ntime/src/main/java/io/quarkus/resteasy/reactive/qute/runtime/TemplateResponseFilter.java
This file was deleted.
Oops, something went wrong.
78 changes: 78 additions & 0 deletions
78
...e/src/main/java/io/quarkus/resteasy/reactive/qute/runtime/TemplateResponseUniHandler.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
package io.quarkus.resteasy.reactive.qute.runtime; | ||
|
||
import java.time.Duration; | ||
import java.util.ArrayList; | ||
import java.util.List; | ||
import java.util.Locale; | ||
|
||
import javax.ws.rs.core.MediaType; | ||
import javax.ws.rs.core.Variant; | ||
|
||
import org.jboss.resteasy.reactive.server.core.ResteasyReactiveRequestContext; | ||
import org.jboss.resteasy.reactive.server.spi.ServerRestHandler; | ||
|
||
import io.quarkus.arc.Arc; | ||
import io.quarkus.qute.Engine; | ||
import io.quarkus.qute.TemplateException; | ||
import io.quarkus.qute.TemplateInstance; | ||
import io.smallrye.mutiny.Uni; | ||
|
||
public class TemplateResponseUniHandler implements ServerRestHandler { | ||
|
||
private volatile Engine engine; | ||
|
||
@Override | ||
public void handle(ResteasyReactiveRequestContext requestContext) { | ||
Object result = requestContext.getResult(); | ||
if (!(result instanceof TemplateInstance)) { | ||
return; | ||
} | ||
|
||
if (engine == null) { | ||
synchronized (this) { | ||
if (engine == null) { | ||
engine = Arc.container().instance(Engine.class).get(); | ||
} | ||
} | ||
} | ||
requestContext.setResult(createUni(requestContext, (TemplateInstance) result)); | ||
} | ||
|
||
@SuppressWarnings("unchecked") | ||
private Uni<String> createUni(ResteasyReactiveRequestContext requestContext, TemplateInstance result) { | ||
Object variantsAttr = result.getAttribute(TemplateInstance.VARIANTS); | ||
if (variantsAttr != null) { | ||
List<io.quarkus.qute.Variant> quteVariants = (List<io.quarkus.qute.Variant>) variantsAttr; | ||
List<Variant> jaxRsVariants = new ArrayList<>(quteVariants.size()); | ||
for (io.quarkus.qute.Variant variant : quteVariants) { | ||
jaxRsVariants.add(new Variant(MediaType.valueOf(variant.getMediaType()), variant.getLocale(), | ||
variant.getEncoding())); | ||
} | ||
Variant selected = requestContext.getRequest() | ||
.selectVariant(jaxRsVariants); | ||
|
||
if (selected != null) { | ||
Locale selectedLocale = selected.getLanguage(); | ||
if (selectedLocale == null) { | ||
List<Locale> acceptableLocales = requestContext.getHttpHeaders().getAcceptableLanguages(); | ||
if (!acceptableLocales.isEmpty()) { | ||
selectedLocale = acceptableLocales.get(0); | ||
} | ||
} | ||
result.setAttribute(TemplateInstance.SELECTED_VARIANT, | ||
new io.quarkus.qute.Variant(selectedLocale, selected.getMediaType().toString(), | ||
selected.getEncoding())); | ||
} | ||
} | ||
|
||
Uni<String> uni = result.createUni(); | ||
if (!engine.useAsyncTimeout()) { | ||
// Make sure the timeout is always used | ||
long timeout = result.getTimeout(); | ||
uni = uni.ifNoItem().after(Duration.ofMillis(timeout)) | ||
.failWith(() -> new TemplateException(result + " rendering timeout [" + timeout + "ms] occured")); | ||
} | ||
return uni; | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters