Quarkus Superheroes to the Rescue!
Are you a developer building microservices? Do you struggle with developing and testing individual microservices that are part of a larger system? Do you want to learn about building REST-based and event-driven applications?
Introduction
Quarkus has excellent documentation and quickstarts to help developers become familiar with various features in the Quarkus ecosystem. However, what was missing was a fully-implemented sample set of real-world applications that use these features, patterns, and best practices while also reflecting problems Quarkus is trying to solve.
This article introduces the Quarkus Superheroes sample application, discusses some of the requirements for building it, and illustrates how to run it locally or deploy it to Kubernetes. You can also watch the Quarkus Insights episode where it was introduced.
Requirements
In addition to simply being fun, some other requirements were essential as well:
-
The overall architecture should be a multi-service microservices architecture.
-
It needs to be more than “hello world” but not the “kitchen sink,” picking a set of common capabilities, such as REST, WebSockets, database interaction, messaging, health checks, monitoring/metrics, Apache Avro, and OpenAPI integration.
-
The applications should be kept up-to-date with each Quarkus version and be part of the standard Quarkus ecosystem continuous integration and shown on the Quarkus status dashboard.
-
There should be both blocking and reactive examples as much as possible.
-
The system as a whole should
-
Be simple to build, run, deploy, and demo on a local developer machine or in Kubernetes.
-
Show how Quarkus can help facilitate local development of independent microservices.
-
Be contained in a single GitHub repository for simplicity, but NOT a multi-module project.
-
-
Each application within the system should
-
Be simple to build, run, deploy, and demo on a local developer machine or in Kubernetes.
-
Provide container images for Java versions 11 and 17, both JVM and native.
-
Be fully documented.
-
Have fully automated (and documented) CI/CD practices.
-
Contain a complete test suite of unit and integration tests.
-
Initial Release
The initial version of the Quarkus Superheroes application was released in early February 2022. Several microservices co-exist to form a larger system, communicating synchronously via REST or event-driven via Apache Kafka. Some of the applications are reactive, while others are more traditional, showcasing how to build each kind.
A web-based user interface, shown in Figure 1, allows superheroes to fight against supervillains. Previous fight outcomes are also shown on this page.
An additional user interface, shown in Figure 2, displays previous fight statistics, including the percentage of wins for heroes vs. villains and the top 10 winners and the number of wins for each.
Under the Cape!
Figure 3 shows the overall architecture of the application.
There are four Quarkus applications, each of which will be described further below, an Angular application, and some backing services: Prometheus, PostgreSQL, Apache Kafka, and Apicurio Schema Registry.
Hero Service
The Hero service is a fully reactive and non-blocking HTTP microservice
exposing CRUD operations on a Hero
data model. The REST layer is
implemented using RESTEasy
Reactive with reactive endpoints. Data is persisted into a
PostgreSQL database using the
repository
pattern from Quarkus
Hibernate Reactive with Panache. Additionally, the Hero service favors
constructor
injection of beans over
field
injection. The Hero service contains a full suite of unit and integration
tests, using
mocking,
spying,
and testing within
isolated
transactions.
To ensure fast startup times, only 100 random heroes from the
3,458
total available heroes are loaded into the database. If you’d like to load
more data, grab the
|
Villain Service
The Villain service is a blocking HTTP microservice exposing CRUD operations
on a Villain
data model. In fact, the Villain
data model is
identical to the Hero
data model in the Hero service
(other than the table’s name). It also implements all the same use cases as
the Hero service. The only difference is that the Villain service is
blocking, and it uses the
active
record pattern from
Quarkus Hibernate ORM with
Panache.
The REST layer is implemented using
RESTEasy Reactive with
blocking endpoints. Data is persisted into a
PostgreSQL database. The Villain service also
creates some custom configurations which show how to use
ConfigMapping
objects. Additionally, the Villain service favors
field
injection of beans over
constructor
injection. The Villain service contains a full suite of unit and
integration tests, using
mocking,
spying,
and testing within
isolated
transactions.
To ensure fast startup times, only 100 random villains from the
1,797
total available villains are loaded into the database. If you’d like to
load more data, grab the
|
Fight Service
The Fight service is a fully reactive and non-blocking HTTP microservice exposing a RESTful API for performing fights between heroes and villains. The REST layer is implemented using RESTEasy Reactive with reactive endpoints. Fight data is persisted into a MongoDB database using the active record pattern from Quarkus MongoDB Reactive with Panache. The Fight service also favors constructor injection of beans over field injection.
The Fight service makes outgoing REST calls to both the Hero and Villain services. Calls to the Hero service use the reactive rest client while calls to the Villain service use the RESTEasy Reactive client, a MicroProfile Rest Client implementation of the JAX-RS client API that is non-blocking at its core. All outgoing calls use resilience patterns such as retry, timeout, fallback, and circuit breaking.
Additionally, the Fight service emits fight events in Apache Avro format to Kafka. The fight schema is automatically registered in an Apicurio Schema Registry by the Quarkus Apicurio Avro extension.
The Fight service contains a full suite of unit and integration tests, using
mocking,
spying,
and testing within
isolated
transactions. It also makes heavy use of Wiremock
for mocking downstream REST endpoints, an
in-memory Kafka
connector for performing message verification within unit tests. It also
shows how to
inject
Dev Services information to configure test resources, such as a
KafkaConsumer
.
Event Statistics Service
The Event Statistics Service is an event-driven microservice exposing a WebSocket HTTP layer. It listens for fight events in Apache Avro format on a Kafka topic utilizing SmallRye Reactive Messaging. The fight schema is automatically registered in an Apicurio Schema Registry by the Quarkus Apicurio Avro extension.
Statistic information is presented via a browser-based user interface, shown in Figure 2 above. Team statistics are accumulated as a percentage of wins by heroes vs. villains. Winner statistics are accumulated by the number of wins for each hero or villain, retaining the top 10 winners. The statistics are pushed to the UI over WebSockets.
The Event Statistics service contains a full suite of unit and integration
tests using
mocking
and
spying.
It also shows how to
inject
Dev Services information to configure test resources, such as a
KafkaProducer
.
Super Hero UI
The Super Hero UI application, shown in Figure 1 above, is an Angular single-page application deployed via Node.js. It communicates to the Fight service via REST. The focus of the sample application is on the Quarkus applications and not the Angular UI.
Run the Application Locally
There are a few ways to run the application locally depending on what your goals are. Docker Compose can be used if you want to run the entire application, or a subset of it, using the pre-built container images. Alternatively, each application can be run from source.
Running Locally via Docker Compose
The entire application or a subset of it can be started with a single Docker Compose command.
Running the Entire Application
The
deploy/docker-compose
directory in the root of
the respository contains compose files for each of the four versions of
the application: JVM 11, JVM 17, native built with Java 11, and native built
with Java 17. Additionally, Prometheus monitoring
can be started with the
supplied
prometheus.yml
file.
From the quarkus-super-heroes directory, simply run the
command
based on the version of the application you want to run. For example, to
run the native Java 17 version, run docker-compose -f
deploy/docker-compose/native-java17.yml -f
deploy/docker-compose/prometheus.yml up
.
The
scripts
directory contains a
watch-services.sh
script that can be run in a separate terminal. This script will watch the
startup of all the services and report when they are all up and ready.
You may see errors as the applications start up. This may happen if one service completes startup before one of its required services (i.e. a database, Kafka broker, etc). Once everything completes startup things will work as intended. |
The following URLs will be available once everything is ready:
-
Super Heroes UI: http://localhost:8080
-
Event statistics UI: http://localhost:8085
-
Apicurio Schema Registry: http://localhost:8086
-
Prometheus: http://localhost:9090
-
Fight service: http://localhost:8082
-
Hero service: http://localhost:8083
-
Villain service: http://localhost:8084
Running a Subset of the Application
Each individual service can also be run via Docker Compose.
Inside each service’s subdirectory in the repository is a
deploy/docker-compose
directory containing the compose files for the
four versions of the application. Multiple compose files can be combined
into a single command.
This is useful if you wanted to, for example, work on the Fight service in
Quarkus Dev Mode and need
to have the Hero and Villain services available. From the
quarkus-super-heroes
directory, simply run the
command
based on the version of the application you want to run. For example, to
run the native Java 17 version, run docker-compose -f
rest-heroes/deploy/docker-compose/native-java17.yml -f
rest-villains/deploy/docker-compose/native-java17.yml up
.
Running Locally via Quarkus Dev Mode
Each of the services can also be run via
Quarkus Dev Mode
simultaneously. For each Quarkus service (event-statistics
,
rest-fights
, rest-heroes
, rest-villains
) open a terminal, cd
into the project directory, and run ./mvnw quarkus:dev
.
The Super Heroes UI can be
built
and run from source, but since the focus of this article is Quarkus, we
will run it as a container image. Open a new terminal and run docker run
-p 8080:8080 -e API_BASE_URL=http://localhost:8082
quay.io/quarkus-super-heroes/ui-super-heroes:latest
.
The following URLs will be available once everything is ready:
-
Super Heroes UI: http://localhost:8080
-
Event statistics UI: http://localhost:8085
-
Fight service: http://localhost:8082
-
Hero service: http://localhost:8083
-
Villain service: http://localhost:8084
Now you have all the benefits of Quarkus Dev Mode and Quarkus Continuous Testing across the entire application at once! All backing infrastructure (databases, Kafka, Apicurio) is provisioned for you automatically by Quarkus Dev Services.
Deploy the Application to Kubernetes
Kubernetes descriptors are provided for various flavors of Kubernetes: OpenShift, Minikube, KNative, and “vanilla” Kubernetes.
The only real difference between the Minikube and Kubernetes descriptors is
that all the application The KNative descriptors use KNative Serving for each of the five applications. |
Similar to Docker Compose, the
pre-built container images of the
entire application, or a subset of it, can be deployed to Kubernetes with a
single kubectl apply -f
(or oc apply -f
if using OpenShift).
The
deploy/k8s
directory in the root of
the respository contains Kubernetes descriptors for each of the four
versions of the application: JVM 11, JVM 17, native built with Java 11, and
native built with Java 17.
For example, to deploy the native Java 17 version to OpenShift, simply run
kubectl apply -f deploy/k8s/native-java17-openshift.yml
(or oc apply
-f deploy/k8s/native-java17-openshift.yml
).
To deploy Prometheus monitoring in addition to the application, kubectl
apply -f
(or oc apply -f
if using OpenShift) the appropriate
Prometheus Kubernetes descriptor based on the target Kubernetes environment:
prometheus-openshift.yml
,
prometheus-minikube.yml
,
or
prometheus-kubernetes.yml
.
On Minikube or Kubernetes only the Super Heroes UI and the Fights service is
exposed outside the cluster. If you want to access Prometheus or the Event
Statistics service then you will need to expose it, either by using an
Ingress or doing a kubectl port-forward of the Pod. On OpenShift,
all of the applications have Route s which expose them outside the
cluster.
|
Additionally, individual applications can be deployed in a similar
manner. Inside each service’s subdirectory in the repository is a
deploy/k8s
directory containing a similar matrix of Kubernetes
descriptors.
These descriptors are NOT considered to be production-ready. They are basic enough to deploy and run the system with as little configuration as possible. The databases, Kafka broker, and schema registry deployed are not highly-available and do not use any Kubernetes operators for management or monitoring. They also only use ephemeral storage. For production-ready Kafka brokers, please see the Strimzi documentation for how to properly deploy and configure production-ready Kafka brokers on Kubernetes. You can also try out a fully hosted and managed Kafka service! For a production-ready Apicurio Schema Registry, please see the Apicurio Registry Operator documentation. You can also try out a fully hosted and managed Schema Registry service! For production-ready Prometheus instances, please see the Prometheus Operator documentation for how to properly deploy and configure production-ready instances. |
The possibilities are endless once deployed to Kubernetes! Reusing the same example above (developing the Fight service), you could use Quarkus Remote Development Mode to connect the remote instance to your local machine and get an experience similar to Quarkus Dev Mode. No need to do any port forwarding!
Want to Get Involved?
We’re always looking for involvement from the community! There is always a list of enhancements and bugs to pick from if you’re interested in getting involved.