The objective of the project is to re-engineer the ACME application adopting the approach of a decentralized/distributed system.
This project was developed for evaluation in the Systems Integration course of the Master in Informatics Engineering - Software Engineering at Instituto Superior de Engenharia do Porto - ISEP/IPP
This tutorial will guide you through the installation and execution of the services required to start your application locally with docker-compose.
- Docker Engine installed.
- Sufficient hardware and software resources to support the execution of all applications simultaneously without causing performance or stability issues.
- Applications:
- discovery-system
- load-balancer
- axonserver
- product-command
- product-command-bootstrapper
- product-query
- product-query-bootstrapper
- review-command
- review-command-bootstrapper
- review-query
- review-query-bootstrapper
- vote-command
- vote-command-bootstrapper
- vote-query
- vote-query-bootstrapper
- Applications:
-
Open and run the Docker desktop application.
-
Download the docker-compose.yml file required to start the services locally.
-
Open a terminal and navigate to the directory where the docker-compose.yml file is saved. Then, execute the following command to start the services in the background:
docker-compose up -d
-
Wait for all services to start. You can check the status of the services using the
docker-compose ps
command. To view the applications running on Docker, click on the "Containers/Apps" section of the dashboard. This will display a list of all the containers that are currently running. -
To access the services, use the following links:
- Discovery System (Eureka Server): http://localhost:8761
- Message Broker: http://localhost:15672 (Username: admin | Password: 123456)
- Axon Server (Event Sourcing): http://localhost:8024
If you want to create 5 new instances of the "product-command" service after starting it, use the command docker-compose scale product-command=5
. It is possible to change the name of the service and the number of instances as needed, but it is important to consider that the maximum number of instances is limited by the processing power and resources available on the machine.
Performing a test with HTTP requests using Postman
-
Create a POST request to http://localhost:8080/products, with the JSON {"sku": "123456789123", "designation": "Laptop", "description": "Black"} in the request body.
-
Create a POST request to http://localhost:8080/products/123456789123/reviews, with the JSON in the request body.
{"user": "John Doe", "reviewText": "This product is great!", "rating": 4.5}
-
Create a PATCH request to localhost:8080/reviews/{id_review}/acceptreject/APPROVED, where {id_review} is the ID of the review you want to accept.
-
Create a POST request to localhost:8080/votes, with the JSON in the request body.
{"user": "Jane Doe", "reviewId": "{id_review}", "voteType": "UP_VOTE"}
- Create a POST request to http://localhost:8080/products, with the JSON in the request body.
{"sku": "123456789156", "designation": "Mouse", "description": "Black"}
- Create a POST request to localhost:8080/votes/for-non-existing-review, with the JSON in the request body.
{"user": "Jonas", "voteType": "UP_VOTE", "review": {"sku": "123456789156", "user": "Maria", "reviewText": "Great!", "rating": 4.0}}
In all requests, select the content type "JSON (application/json)" and click "Send" to send the request.
- Open Postman and create a new request.
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/products".
- Click "Send" to send the request.
- A list of all registered products will be displayed.
To get information about a specific product:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/products/{sku}".
- Replace "{sku}" with the SKU of the product you want to query.
- Click "Send" to send the request.
- The information of the selected product will be displayed.
To get reviews of a specific product:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/products/{sku}/reviews".
- Replace "{sku}" with the SKU of the product you want to query.
- Click "Send" to send the request.
- The reviews of the selected product will be displayed.
To get information about all reviews:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/reviews".
- Click "Send" to send the request.
- All review will be displayed.
To get information about of products reviews by status:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/products/{sku}/reviews/{approvalStatus}".
- Replace "{sku}" with the sku of the product you want to query.
- Replace "{approvalStatus}" with one of the status (PENDING, ACCEPTED, REJECTED) to get the reviews with the desired status.
- Click "Send" to send the request.
- The information of the selected review will be displayed.
To get information about reviews of a specific user:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/reviews/{user}".
- Replace "{user}" with the name of the user you want to query.
- Click "Send" to send the request.
- The reviews of the selected user will be displayed.
To get the pending reviews:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/reviews/pending".
- Click "Send" to send the request.
- The reviews with pending status will be displayed.
To get the accepted or rejected reviews:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/reviews/{reviewId}/acceptreject/{approvalStatus}".
- Replace "{id_review}" with the ID of the review you want to desired status.
- Replace "{approvalStatus}" with "ACCEPTED" or "REJECTED" to get the reviews with the desired status.
- Click "Send" to send the request.
- The reviews with the selected status will be displayed.
To get information about all votes of a specific review:
- Select the HTTP method "GET" and insert the URL "http://localhost:8080/reviews/{id_review}/votes".
- Replace "{id_review}" with the ID of the review you want to query.
- Click "Send" to send the request.
- The votes of the selected review will be displayed.
You can stop the services at any time by using the docker-compose down
command in the directory where the docker-compose.yml
file is saved.
Now you can use the services locally with docker-compose. Make sure the services are stopped before running the application in another environment. If you have problems during installation or execution, make sure you followed the steps described in this tutorial correctly and that all prerequisites were met.
-
Business Domain Segregation: The monolithic application was segregated into three distinct but collaborative applications:
- Products
- Reviews
- Votes
-
Cloning: Multiple instances of each of the above applications are deployed in containers.
Non-functional requirements
- Deployment must be automated through CI/CD.
- Adoption of service component test pattern.
- Adoption of end-to-end test pattern.
- Two or more frameworks/programming languages must be adopted in implementing the services.
- AMQP Message Broker (e.g. RabbitMQ) must be adopted for communication between services.
Functional requirements
-
Endpoint in service Votes that allows creating a vote for a non-existing review. The review must be created for the specified product, and the vote is eventually associated with the review.
-
Develop a bootstrap process for the starting services. I.e. at any time, a new service can start and its data must be bootstrapped.
The following patterns must be adopted:
- Strangler fig
- Command-Query Responsibility Segregation (CQRS)
- Database-per-Service
- Polyglot persistence
- Messaging
- The Domain Events
- Event Sourcing
- Saga
- CI (Continuous Integration)
- Load Balance
- Discovery System
- Message Broker
- Event Sourcing
- Bootstrapper
- Applications: Product, Review e Vote (Command/Query)
- Java (JDK 17)
- Node.js
- Git/GitHub
- GitHub Actions (CI)
- Docker
- RabbitMQ (AMQP)
- Axon Server (Event Sourcing)
- Netflix Eureka
- H2 Database
- MongoDB
1. Layered architecture restructuring
We performed significant refactoring of the application by adopting the typical layered architecture of software development. This architecture includes the layers of controller, model, repository, and service, each with its well-defined responsibility.
- The controller is responsible for presentation logic;
- The model represents the entities and data of the application;
- The repository provides access to the system's data;
- The service performs the business logic of the application.
We isolated the innermost layers from the outer ones to ensure that the business logic remains separate from the presentation logic and data access. This resulted in greater security and scalability of the application, as well as enabling more organized development, facilitating maintenance and evolution of the application.
Before:
acme-monolithic-project/src/main/java/com/isep/acme/
.
├── bootstrapper
├── controllers
├── model
├── property
├── repositories
├── services
.
After:
acme-product-command/src/main/java/com/isep/acme/
.
├── bootstrapper
├── api
│ └── controllers
├── config
├── domain
│ ├── model
│ ├── repository
│ └── service
├── dto
│ ├── mapper
│ ├── message
│ ├── request
│ └── response
├── exception
│ ├── model
│ ├── repository
│ └── service
├── messaging
.
2. Changes in application logic
In addition to adopting the layered architecture, we made other significant improvements to optimize the application logic:
-
We used an Enum for
ApprovalStatus
, replacing the list of strings with the three status types. This change reduced code complexity and made the implementation more efficient. -
We transformed the
class Rating
into aDouble rate
attribute in theclass Review
, reducing code complexity and simplifying the application logic. -
We removed the
AggregatedRating
class to simplify the application logic and avoid data redundancy. -
We changed the logic in
review-command
so that theclass Review
now has aSet<>
with all the votes, positive and negative. This change reduced the complexity of theaddVote()
method in theclass Review
and allowed the application to obtain the number of votes and count positive and negative votes with just one database query, making the operation more efficient and scalable.
3. Changes in application code
-
We used the Lombok library to reduce the amount of boilerplate code in the application, eliminating the need to write getters, setters, constructors, and other common methods in Java classes.
-
We switched from
CrudRepository
toJPA
to simplify database access. This change resulted in faster and simplified development, as well as allowing the application to be easily integrated with other technologies and systems. -
We adopted
UUID
as the standard for the ids of each entity. This change brought a series of benefits, such as the guarantee of id uniqueness and the difficulty of predicting the ids, making the application more secure.
These changes were implemented with the aim of optimizing the application by reducing code complexity and improving the efficiency and scalability of the application logic.
1. Discovery System
We implemented a Discovery System to simplify the deployment and scalability of microservices. It allows services to find and communicate with each other without requiring coded service locations.
2. Load Balancer
We used a Load Balancer to distribute HTTP requests coming from clients among multiple server instances. This was done to optimize resource usage, maximize throughput, minimize response time, and ensure high availability and scalability of the application.
3. Message Broker
We used RabbitMQ to facilitate asynchronous communication between microservices via the AMQP protocol. It allows services to communicate in a decoupled and scalable manner.
4. Bootstrapper
We implemented a Bootstrapper to manage the initialization of microservices. It ensures that each service is started in the correct order and with the correct configuration, minimizing the risk of errors and reducing deployment time. It also sends events to the event sourcing.
5. Event Sourcing
We used Axon Server to maintain a complete audit trail of all changes to the application state. This allows events to be easily replayed and the application to be recovered from failures.
6. Command-Query Responsibility Segregation (CQRS)
We implemented the CQRS pattern to separate write and read operations into different services. This allows each service to be independently scaled and handle read and write operations separately. The Product, Review, and Vote services are examples of this approach.
7. GitHub Action
We use the Docker Image CI GitHub Action to implement continuous integration of our Docker images. This action automatically builds the Docker image and tags it with a specific version whenever we push changes to the main branch or create a pull request. The Docker Image CI Action also logs in to Docker Hub using secure credentials and pushes the image to the registry.
Using this GitHub Action ensures that our Docker images are always up-to-date and tested, reducing the risk of errors and inconsistencies during deployment. Additionally, it simplifies our deployment process by enabling easy deployment of Docker images to different environments.
David Gomes | Regiana Cruz | Nairon Carneiro |
Nuno Silva |