Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ttulka/ddd-example-ecommerce-microservices
Domain-driven design microservices example
https://github.com/ttulka/ddd-example-ecommerce-microservices
architecture ddd docker domain-driven-design event-driven example kubernetes microservices oop soa spring-boot
Last synced: 3 months ago
JSON representation
Domain-driven design microservices example
- Host: GitHub
- URL: https://github.com/ttulka/ddd-example-ecommerce-microservices
- Owner: ttulka
- License: mit
- Created: 2020-11-02T15:32:01.000Z (about 4 years ago)
- Default Branch: main
- Last Pushed: 2023-12-11T15:11:31.000Z (about 1 year ago)
- Last Synced: 2024-10-13T04:13:27.683Z (3 months ago)
- Topics: architecture, ddd, docker, domain-driven-design, event-driven, example, kubernetes, microservices, oop, soa, spring-boot
- Language: Java
- Homepage:
- Size: 595 KB
- Stars: 53
- Watchers: 6
- Forks: 21
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# DDD Microservices Example Project in Java: eCommerce
The purpose of this project is to provide a sample implementation of an e-commerce product following **Domain-Driven Design (DDD)** and **Service-Oriented Architecture (SOA)** principles.
Programming language is Java with heavy use of Spring Boot, Docker and Kubernetes.
## Purpose of the Project
This repository focuses mostly on cross-cutting, infrastructure and deployment concerns.
For the domain and application concepts see the [original repository](https://github.com/ttulka/ddd-example-ecommerce).
## Monolith vs Microservices
Both monolithic and microservices deployments are implemented.
To run the monolithic application:
```sh
./gradlew :application:bootRun
```To set up and run microservices, see the [Docker](#docker-containers) and [Kubernetes](#kubernetes) sections.
Read more about monoliths vs microservices at https://blog.ttulka.com/good-and-bad-monolith
## Message Broker
As the message broker a simple **Redis** instance could be used with Spring profile `redis`:
```sh
docker run --rm --name redis-broker -p 6379:6379 -d redis:6 redis-server./gradlew :application:bootRun --args='--spring.profiles.active=redis'
```Alternatively, **RabbitMq** could be used as the message broker with Spring profile `rabbitmq`:
```sh
docker run --rm --name rabbitmq-broker -p 5672:5672 -d rabbitmq:3./gradlew :application:bootRun --args='--spring.profiles.active=rabbitmq'
```When neither `redis` not `rabbitmq` profiles are active, the system will fall-back to use of Spring application events as the default messaging mechanism.
### Messaging Integration
To make the code independent of a concrete messaging implementation and easy to use, Spring application events are used for the internal communication.
In practice, this means that messages are published via `EventPublisher` abstraction and consumed via Spring's `@EventListener`.
To make this work, the external messages are re-sent as Spring application events under the hood.
## Database
The whole system uses one externalized database with particular tables owned exclusively by services.
In a real-world system this separation would be further implemented by separate schemas/namespaces.
As the database **PostgreSQL** instance could be used:
```sh
docker run --rm --name postgres -p 5432:5432 -e POSTGRES_PASSWORD=secret -d postgres:13
```Start the application with Spring profile `postgres`:
```sh
./gradlew :application:bootRun --args='--spring.profiles.active=postgres'
```When the `postgres` profile is not active, the system will fall-back to use H2 as the default database.
## Gradle Build
The project is a Gradle multi-project, all sub-projects can be built in a single command:
```sh
./gradlew clean build
```## Docker Containers
Build an image per microservice via Gradle Spring Boot plugin:
```sh
./gradlew bootBuildImage
```To run the containers:
```sh
docker container run --rm -p 8080:8001 ttulka/ecommerce-catalog-service
docker container run --rm -p 8080:8002 ttulka/ecommerce-order-service
docker container run --rm -p 8080:8003 ttulka/ecommerce-cart-service
docker container run --rm -p 8080:8004 ttulka/ecommerce-payment-service
docker container run --rm -p 8080:8005 ttulka/ecommerce-delivery-service
docker container run --rm -p 8080:8006 ttulka/ecommerce-dispatching-service
docker container run --rm -p 8080:8007 ttulka/ecommerce-warehouse-service
docker container run --rm -p 8080:8000 ttulka/ecommerce-portal-service
```Active profiles can be set as follows:
```sh
docker container run --rm -e "SPRING_PROFILES_ACTIVE=redis,postgres" -p 8080:8001 ttulka/ecommerce-catalog-service
```### Docker-Compose
Build NGINX reverse proxy image:
```sh
docker build -t ttulka/ecommerce-reverseproxy reverseproxy
```Start the entire microservices stack:
```sh
docker-compose up
```Access the Postgres database and init some data:
```sh
docker exec -it psql -U postgres postgres
``````sql
INSERT INTO categories VALUES
('C1', 'books', 'Books'),
('C2', 'games', 'Games');INSERT INTO products VALUES
('P1', 'Domain-Driven Design', 'by Eric Evans', 45.00),
('P2', 'Object Thinking', 'by David West', 35.00),
('P3', 'Chess', 'Classic game.', 3.20);INSERT INTO products_in_categories VALUES
('P1', 'C1'),
('P2', 'C1'),
('P3', 'C2');INSERT INTO products_in_stock VALUES
('P1', 5),
('P2', 0),
('P3', 1);
```The NGINX reverse proxy serves as a simple API gateway:
```sh
curl localhost:8080/catalog/products
curl localhost:8080/warehouse/stock/5
```## Kubernetes
To use local images for development with Minikube, run the following command to use local Docker images registry:
```sh
minikube start
eval $(minikube docker-env)
```Afterwards, build the docker images again for the Minikube's Docker daemon:
```sh
./gradlew bootBuildImage
```Create deployments:
```sh
kubectl apply -f 1-infrastructure.k8s.yml
kubectl apply -f 2-backend-services.k8s.yml
kubectl apply -f 3-frontend-portal.k8s.yml
kubectl apply -f 4-api-gateway.k8s.yml
```Set up ports forwarding to access the cluster from your local network:
```sh
kubectl port-forward service/reverseproxy 8080:8080
```Alternatively, you can create an Ingress:
```sh
minikube addons enable ingresskubectl apply -f 5-ingress.k8s.yml
# get the ingress address
kubectl get ingress ecommerce-ingress# add the address into hosts
sudo cp /etc/hosts hosts.bak
sudo echo -e '\n ecommerce.local' >> /etc/hosts# access the application in browser: http://ecommerce.local
```