RESTEasy Reactive - To block or not to block
In January 2021, the Quarkus team announced RESTEasy Reactive, a novel way to serve HTTP API in Quarkus. Since its introduction, RESTEasy Reactive adoption has been quite good, and we plan to make it the default approach to implement HTTP API shortly.
But, wait a minute, what does that mean for my imperative APIs? Do I need to learn reactive programming to use Quarkus now? Let’s be clear: no. This blog post will look at a few changes we made in RESTEasy reactive to make the transition smooth and transparent.
A brief history of HTTP APIs in Quarkus
Quarkus has, since its genesis, has been able to serve HTTP API. The
inclusion of RESTEasy has been a major
milestone of the first Quarkus beta releases. With RESTEasy classic, you
develop HTTP APIs using the well-known JAX-RS annotations such as @GET
,
@Path
, @POST
… The following snippet shows a short hello world
example:
package org.acme;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@Path("/hello")
public class GreetingResource {
@GET
public String hello() {
return "Hello";
}
}
RESTEasy classic invokes the HTTP endpoint (the hello
method in the
previous snippet) on a worker thread associated with the HTTP request. It
is a well-understood model, simple to understand. However, relying on
worker threads introduces a concurrency limit: the number of threads.
Even with the infusion of reactive at the core of Quarkus, RESTEasy classic kept this dispatching strategy. It was fragmenting the Quarkus ecosystem. On one side, we had the the imperative camp using RESTEasy classic, Hibernate ORM… On the other side, we had the reactive camp using Reactive Routes, Vert.x APIs and other reactive extensions. Both were using, under the hood, the reactive engine of Quarkus, but the reactive camp we using it in a more efficient way.
Following the unification of imperative and reactive idea, in Quarkus 1.11, we introduced RESTEasy reactive, a novel implementation of the JAX-RS model on top of the Quarkus reactive architecture. It offers a similar development model and much better throughput. I won’t detail the RESTEasy reactive architecture and benefits. Georgios covered them in two posts: RESTEasy Reactive introduction and Massive performance without headaches.
From the user point of view, the main difference between RESTEasy classic and reactive is how they call the HTTP endpoint methods:
-
classic - always on a worker thread,
-
reactive - on the I/O thread or on a worker thread (and you, as the developer, have the choice)
You may wonder why it’s so important. Threads are expensive, especially in containers or on the cloud where the resources are limited. Using the I/O threads avoids creating additional threads (improving memory consumption) and avoids context switches (improving response time). Emmanuel explained the benefits in the A IO thread and a worker thread walk into a bar: a microbenchmark story blog post.
To block or not to block, that is the question.
When we introduced RESTEasy reactive, we decided to use a non-blocking
approach by default: if not stated otherwise, it calls the HTTP endpoint
method on the I/O thread. This model resulted in outstanding performance
and was simple enough, thanks to the usage of the @Blocking
annotation.
In the last few months, the adoption of RESTEasy reactive has been incredible! We have received many questions and, obviously, bug reports. The central question is about the usage of Hibernate ORM.
As Hibernate ORM classic (we also have Hibernate reactive) is blocking,
you can’t use it with RESTEasy reactive without using the @Blocking
annotation. This annotation changes the dispatching strategy to use a
worker thread (instead of the I/O thread).
While the resulting model looked efficient and straightforward for us, non-aware users have seen a lot of:
You have attempted to perform a blocking operation on a IO thread. This is not allowed, as blocking the IO thread will cause major performance issues with your application. If you want to perform blocking EntityManager operations make sure you are doing it from a worker thread.: java.lang.IllegalStateException: You have attempted to perform a blocking operation on a IO thread. This is not allowed, as blocking the IO thread will cause major performance issues with your application. If you want to perform blocking EntityManager operations make sure you are doing it from a worker thread.
The error message is explicit. But, it rarely makes us happy when we have such a wall of text printed in our terminal.
You may say… “well, let’s do blocking by default.” It’s not that simple. It’s as dangerous to call reactive APIs expected to be called on an I/O thread on a worker thread than calling blocking APIs on the I/O thread.
New world, new rules!
In Quarkus 2.2.0, we introduced a new dispatching strategy based on the method signatures. The Quarkus build-time approach lets us be wise and deduce if a method should be called on the I/O thread or a worker thread at build time, reducing the runtime overhead.
The following table summarizes the new set of rules:
Method signature |
Dispatching strategy |
|
Worker thread |
|
I/O thread |
|
I/O thread |
|
I/O thread |
|
I/O thread |
|
Worker thread |
Basically: synchronous methods default to worker threads, and asynchronous
methods default to I/O threads, except if explicitly stated otherwise. Of
course, you can override the behavior using the @Blocking
and
@NonBlocking
annotations. The @Transactional
annotation is an exception
to the default rules as it often means you are accessing blocking resources
(such as an entity manager).
What does that change for you?
Let’s discuss a few examples explaining how this new strategy improves the user experience without limiting efficiency and flexibility.
Hello RESTEasy Reactive
Using RESTEasy reactive does not change the hello example from above:
package org.acme;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@Path("/hello")
public class GreetingResource {
@GET
public String hello() {
return "Hello";
}
}
That method is invoked on a worker thread because it has a synchronous
signature. Previously (before Quarkus 2.2), with RESTEasy reactive, it
would have been called on the I/O thread. To switch back to that behavior,
add @NonBlocking
:
package org.acme;
import io.smallrye.common.annotation.NonBlocking;
import javax.ws.rs.GET; import javax.ws.rs.Path;
@Path("/hello") public class GreetingResource {
@GET
@NonBlocking
public String hello() {
return "Hello";
}
}
Alternatively, you can return a Uni
:
package org.acme;
import io.smallrye.mutiny.Uni;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@Path("/hello")
public class GreetingResource {
@GET
public Uni<String> hello() {
return Uni.createFrom().item("Hello");
}
}
Integrating with Hibernate ORM
Following the feedback from users, let’s imagine you want to use Hibernate classic with RESTEasy reactive:
package org.acme;
import org.jboss.resteasy.reactive.RestQuery;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@Path("/fruit")
public class FruitResource {
@GET
public Fruit getFruit(@RestQuery String name) {
return Fruit.find("name", name).firstResult();
}
}
You don’t need to use @Blocking
as the signature is synchronous. No more
wall of text!
Integrating with Hibernate Reactive
If you use Hibernate reactive, you will use the Mutiny API, and so the resulting code will be:
package org.acme;
import io.smallrye.mutiny.Uni;
import org.jboss.resteasy.reactive.RestQuery;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@Path("/fruit")
public class FruitResource {
@GET
public Uni<Fruit> getFruit(@RestQuery String name) {
return Fruit.find("name", name).firstResult();
}
}
This method runs on the I/O thread, which is what Hibernate reactive expects.
Integrating with Kafka
If you combine HTTP and Kafka (using reactive messaging), you will use an
emitter. Depending on the emitter type (Emitter
or MutinyEmitter
), the
send
method returns a CompletionStage
or a Uni
. So, the following
HTTP method runs on the I/O thread:
package org.acme;
import io.smallrye.mutiny.Uni;
import io.smallrye.reactive.messaging.MutinyEmitter;
import org.eclipse.microprofile.reactive.messaging.Channel;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
@Path("/fruit")
public class FruitResource {
@Channel("kafka")
MutinyEmitter<Fruit> emitter;
@POST
public Uni<Void> writeToKafka(Fruit fruit) {
return emitter.send(fruit);
}
}
If you change it to a synchronous signature, it runs on a worker thread:
package org.acme;
import io.smallrye.reactive.messaging.MutinyEmitter;
import org.eclipse.microprofile.reactive.messaging.Channel;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import java.time.Duration;
@Path("/fruit")
public class FruitResource {
@Channel("kafka")
MutinyEmitter<Fruit> emitter;
@POST
public void writeToKafka(Fruit fruit) {
System.out.println(Thread.currentThread().getName());
emitter.send(fruit).await().atMost(Duration.ofSeconds(5));
}
}
Combining RESTEasy Reactive, Hibernate ORM and Kafka
Let’s now combine Resteasy reactive, Hibernate ORM classic and Kafka to persist an entity and write it to a Kafka topic:
package org.acme;
import io.smallrye.mutiny.Uni;
import io.smallrye.reactive.messaging.MutinyEmitter;
import org.eclipse.microprofile.reactive.messaging.Channel;
import javax.transaction.Transactional;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
@Path("/fruit")
public class FruitResource {
@Channel("kafka")
MutinyEmitter<Fruit> emitter;
@POST
@Transactional
public Uni<Void> persistAndWriteToKafka(Fruit fruit) {
System.out.println(Thread.currentThread().getName());
fruit.persist();
return emitter.send(fruit);
}
}
This method runs on a worker thread despite the signature. The
@Transactional
annotation configures the dispatching strategy to use a
worker thread.
Summary
With Quarkus 2.2, the dispatching strategy of RESTEasy reactive becomes smarter thus improving the developer experience.
-
You don’t need to learn the reactive way; you can keep using imperative code.
-
You don’t need to think about your threads; Quarkus does that for you.
-
You don’t lose in flexibility; you can override the decision.
Starting with Quarkus 2.3, the Quarkus team is thinking of making RESTEasy reactive the default way to implement HTTP APIs. It does not mean that the RESTEasy classic extension will be retired, just that we reach the point where RESTEasy reactive gives you more without burden.