Logo

dev-resources.site

for different kinds of informations.

Effective Project Structuring for Microservices with Quarkus

Published at
9/18/2024
Categories
java
quarkus
microservices
architecture
Author
yanev
Author
5 person written this
yanev
open
Effective Project Structuring for Microservices with Quarkus

In the ever-evolving landscape of software development, the adoption of microservices architecture is rapidly gaining traction due to its scalability, flexibility, and maintainability. Quarkus, as a Java-Native Kubernetes stack tailored for GraalVM and HotSpot, optimizes Java specifically for containers and enables it to become an effective platform for serverless, cloud, and Kubernetes environments.

Key to leveraging Quarkus effectively is understanding how to structure your project correctly. Proper project structuring not only enhances the manageability of the code but also plays a crucial role in the success of deploying microservices architecture.

Microservices architecture distinguishes itself by several defining characteristics that Quarkus enhances:

  1. Small and Focused - each service in a microservices architecture is designed to perform a single, well-defined task, promoting simplicity and focus. Quarkus supports this by facilitating lightweight, bootable jars and native compilations, which are ideal for specific tasks.

  2. Independent- independence allows services to be developed, deployed, and even scaled without reliance on other services. Quarkus's container-first philosophy ensures that each microservice can run in its isolated environment.

  3. Loosely Coupled- services communicate via well-defined APIs, reducing the dependency on internal implementations. Quarkus encourages decoupling through standards like MicroProfile and JAX-RS, making services interaction seamless and efficient.

  4. Scalable- efficiently managing increasing workloads is crucial. Quarkus's reactive programming capabilities enable services to be responsive and scalable under varying loads.

  5. Flexible- the ability to adapt to changing conditions is facilitated by the dynamic scaling and resilience features of Quarkus, supporting on-the-fly adjustments in a cloud-native environment.

  6. Resilient- Quarkus enhances the robustness of microservices with built-in fault tolerance and health-check capabilities, ensuring the services are resistant to failures and recover swiftly from interruptions.

Adopting Maven multi-module project structures has proven effective in managing large-scale Java applications, and Quarkus fully supports this modality. Let's detail the Maven multi-module structure:

A Maven multi-module project consists of an aggregator POM that orchestrates several submodules. This structure is beneficial for dividing a large application into manageable, independently deployable units that can be maintained and updated with ease. Each module typically has a focused responsibility and can be deployed independently from others.

Hereโ€™s a simplified layout based on a real-world application:



parent (aggregator pom.xml)
โ”œโ”€โ”€ api (API interfaces and DTOs)
โ”‚   โ”œโ”€โ”€ pom.xml
โ”‚   โ””โ”€โ”€ src
โ”œโ”€โ”€ core (Business logic)
โ”‚   โ”œโ”€โ”€ pom.xml
โ”‚   โ””โ”€โ”€ src
โ”œโ”€โ”€ data-access (Data layer integration)
โ”‚   โ”œโ”€โ”€ pom.xml
โ”‚   โ””โ”€โ”€ src


Enter fullscreen mode Exit fullscreen mode

I've tailored an optimized project structure specifically for Quarkus that enhances scalability, maintainability, and coherence across diverse development teams. The design, which I originally developed in 2018, has seen several refinements to better suit evolving technological trends. Initially, the structure wasn't specifically crafted with Quarkus in mind; however, throughout its application, it has proven exceptionally compatible with Quarkus's philosophy and coding practices. Let us examine this structure as implemented in an IDE like IntelliJ IDEA - though it could be effectively adapted to any other modern IDE.

Microservices Project Structure

You can find the full project structure in the GitHub repository

Starting from the top-level, the project is divided into several modules, each with a specific role:

  1. api-dtos- this module is crucial for de-coupling the internal representations of data from what is exposed externally through APIs. By separating Data Transfer Objects (DTOs) from domain models, the module ensures that any changes- be they in database schema or business logic- do not directly affect the clients who consume these APIs. This separation enhances API stability and versioning.

  2. commons- this utility module acts as a repository for shared logic, constants, and helpers that are used across multiple other modules in the project. Centralizing common functionalities in this way reduces duplication and fosters consistency throughout the application. Itโ€™s particularly helpful for storing utilities like data format converters, standard validators, or common business rules.

  3. configuration- configuration management is streamlined in this module, which consolidates all configuration settings for the entire application in one place, simplifying access and management. This central repository of configurations ensures that settings are consistently applied and managed, simplifying environment shifts (such as from development to production) and minimizing the potential for errors due to misconfiguration. Beyond simple configuration files, this module is also the ideal location for defining custom managed beans that are essential across the application, such as ObjectMapper for JSON serializing and deserializing, custom deserializers, and other specialized beans.

  4. db-entities- dedicated exclusively to database interactions, this module encapsulates all ORM classes or database schemas, ensuring they are separated from the business logic and API layers. Segregating entities in this manner helps maintain clean architecture principles by decoupling data access mechanisms from business rules processing, hence promoting a cleaner and more traceable codebase.

  5. services- here lies the core business logic, encapsulated in service classes that operate on domain models and use db-entities for data persistence. Each service focuses on a specific business task, enhancing modularity and reusability. This separation also aids in isolating business operations from direct client interactions, which are handled by controller classes in other modules, thus adhering to the single responsibility principle.

  6. resources- amid some debate, this module was named to reflect its alignment with JAX-RS, where classes annotated with JAX-RS are referred to as resources. This module handles the creation of REST endpoints, acting as the communication layer that interfaces with external clients. The name 'resources' was chosen over alternatives like 'rest' or 'controllers' to emphasize standardization and adherence to established RESTful principles as described in the Red Hat resource for building REST APIs with Quarkus. By aligning with standard terminology, the project aims to be intuitive for developers familiar with JAX-RS and REST conventions.

  7. application- serves as an umbrella module that brings all the pieces together, ensuring they are correctly wired and ready to deploy.

Each of these modules plays a strategic role tailored to foster ease of maintenance, scalability, and robustness in a Cloud-Native microservice environment using Quarkus.

The above structure is based on the following Architecture layers which is mentioned in the RedHat resources

Architecture layers: Resource, service and repository


Much Abstraction is Too Much

When discussing project structures, a common question arises about the need for the layered approach I previously described. Here, I evoke the words of David J. Wheeler:

All problems in computer science can be solved by another level of indirection, except for the problem of too many layers of indirection. - [David J. Wheeler from The C++ Programming Language 4th edition.]

quarkus Article's
30 articles in total
Favicon
Java Can Be Serverless Too: Using GraalVM for Fast Cold Starts
Favicon
Building Robust REST Client with Quarkus: A Comprehensive Guide
Favicon
Choosing the Right Java Microservices Framework: Spring Boot, Quarkus, Micronaut, and Beyond
Favicon
Agente de IA confiรกvel em prod com Java + Quarkus + Langchain4j - Parte 2 - Memรณria
Favicon
Agente de IA confiรกvel em prod com Java + Quarkus + Langchain4j - Parte 1 - AI as Service
Favicon
Calling Clojure from Java using a real example (Clojure + Quarkus)
Favicon
Turbocharge Java Microservices with Quarkus and GraalVM Native Image
Favicon
Introduction to Quarkus: Java Native for Kubernetes
Favicon
Effective Project Structuring for Microservices with Quarkus
Favicon
Unlock Lightning-Fast Web Services: Mastering Quarkus for Agile, Scalable, and Responsive RESTful APIs
Favicon
Harnessing Automatic Setup and Integration with Quarkus Dev Services for Efficient Development
Favicon
Why we discarded Reactive systems architecture from our code?
Favicon
Unveiling Challenges with @Named
Favicon
Exploring Synthetic Beans in Quarkus. A Powerful Extension Mechanism
Favicon
Registering Reflection in Quarkus Extensions
Favicon
Creating Custom Configuration in Quarkus Loaded from JSON File
Favicon
Extending Quarkus: When and How to Write Your Own Extensions
Favicon
Demystifying Quarkus Extension Development: Jandex vs. AdditionalBeanBuildItem
Favicon
๐— ๐—ถ๐—ฐ๐—ฟ๐—ผ๐˜€๐—ฒ๐—ฟ๐˜ƒ๐—ถ๐—ฐ๐—ฒ๐˜€ ๐—ถ๐—ป ๐—๐—ฎ๐˜ƒ๐—ฎ: ๐—”๐—ฟ๐—ฐ๐—ต๐—ถ๐˜๐—ฒ๐—ฐ๐˜๐˜‚๐—ฟ๐—ฒ, ๐—•๐—ฒ๐—ป๐—ฒ๐—ณ๐—ถ๐˜๐˜€, ๐—ฎ๐—ป๐—ฑ ๐—œ๐—บ๐—ฝ๐—น๐—ฒ๐—บ๐—ฒ๐—ป๐˜๐—ฎ๐˜๐—ถ๐—ผ๐—ป
Favicon
Spring Boot vs Quarkus: Pick one for Java!
Favicon
Deploying native Quarkus REST API's in AWS Lambda
Favicon
เน€เธฃเธดเนˆเธกเธ•เน‰เธ™ Quarkus 3 part 2.3 Renarde
Favicon
เน€เธฃเธดเนˆเธกเธ•เน‰เธ™ Quarkus 3 part 2.2 web bundler
Favicon
How to enable mongodb query logging in reactive java for quarkus with panache
Favicon
Beyond JWT: Unlocking PASETO for Secure Token Management
Favicon
เน€เธฃเธดเนˆเธกเธ•เน‰เธ™ Quarkus 3 part 1
Favicon
Exploring Quarkus vs Spring Boot
Favicon
Secure Quarkus application with ezto
Favicon
Spring Boot vs Quarkus: Pick one for Java
Favicon
Why Quarkus Native (probably) does not fit your project

Featured ones: