Compiling virtual thread applications into native executables

In another blog post, we have seen how you can implement a CRUD application with Quarkus to utilize virtual threads. This post will show how you can compile such an application into a native executable.

Installing GraalVM 21

To compile a Quarkus application leveraging virtual threads into a native executable, you need a GraalVM version supporting Java 21. You can download it from GitHub.

Alternatively, you can use the SDKMAN tool to install it:

> sdk install java 21-graalce
Downloading: java 21-graalce

In progress...

Repackaging Java 21-graalce...

Done repackaging...
Cleaning up residual files...

Installing: java 21-graalce
Done installing!

Do you want java 21-graalce to be set as default? (Y/n): n

Once installed, make sure the GRAALVM_HOME environment variable points to the GraalVM installation directory:

> export GRAALVM_HOME=$HOME/.sdkman/candidates/java/21-graalce

Compiling the application into a native executable

We will reuse the CRUD application developed in a previous blog post. The source code is located in the virtual-threads-demos GitHub repository. Note that while we are using the CRUD application, the same approach can be used with any Quarkus application leveraging virtual threads, including the other demos from the repository.

First make sure you use Java 21+ and that the GRAALVM_HOME environment variable points to the GraalVM installation directory.

Then, in the pom.xml file, add the native profile:

<profiles>
  <profile>
    <id>native</id>
      <activation>
        <property>
          <name>native</name>
        </property>
      </activation>
      <properties>
        <quarkus.package.type>native</quarkus.package.type>
      </properties>
  </profile>
</profiles>

The native profile is activated when the native property is set. So, compile the application with:

> mvn clean package -Dnative

The compilation takes a few minutes. Once done, you can run the application:

1) First, start the database:

> docker run --ulimit memlock=-1:-1 -d -it --rm=true --memory-swappiness=0 \
    --name postgres-quarkus-demo -e POSTGRES_USER=restcrud \
    -e POSTGRES_PASSWORD=restcrud -e POSTGRES_DB=rest-crud \
    -p 5432:5432 postgres:15-bullseye

2) Then, start the application:

> ./target/crud-example-1.0.0-SNAPSHOT-runner

You get:

> ./target/crud-example-1.0.0-SNAPSHOT-runner
__  ____  __  _____   ___  __ ____  ______
 --/ __ \/ / / / _ | / _ \/ //_/ / / / __/
 -/ /_/ / /_/ / __ |/ , _/ ,< / /_/ /\ \
--\___\_\____/_/ |_/_/|_/_/|_|\____/___/
2023-10-17 09:44:34,925 INFO  [io.quarkus] (main) crud-example 1.0.0-SNAPSHOT native (powered by Quarkus 3.4.1) started in 0.072s. Listening on: http://0.0.0.0:8080
2023-10-17 09:44:34,925 INFO  [io.quarkus] (main) Profile prod activated.
2023-10-17 09:44:34,925 INFO  [io.quarkus] (main) Installed features: [agroal, cdi, hibernate-orm, hibernate-orm-panache, hibernate-validator, jdbc-postgresql, narayana-jta, resteasy-reactive, resteasy-reactive-jackson, smallrye-context-propagation, vertx]

Then, open the application in a browser (http://localhost:8080) and start adding, updating, and completing tasks. You will see in the logs that the processing of these requests are executed on virtual threads:

2023-10-17 10:15:09,992 INFO  [org.acm.cru.TodoResource] (quarkus-virtual-thread-0) Called on VirtualThread[#78,quarkus-virtual-thread-0]/runnable@ForkJoinPool-5-worker-1
2023-10-17 10:15:13,136 INFO  [org.acm.cru.TodoResource] (quarkus-virtual-thread-1) Called on VirtualThread[#85,quarkus-virtual-thread-1]/runnable@ForkJoinPool-5-worker-1

Summary

This blog post explains how to compile a Quarkus application leveraging virtual threads into a native executable. First, make sure that you have a GraalVM installation supporting Java 21+. Then, add the native profile to the pom.xml file and compile the application using the -Dnative option. Finally, run it as any other native executable!