Vinci is a microservice shopping webapp. The name is inspired by my favourite artist - Leonardo da Vinci. 🧑🏻🎨
Vinci chooses API First approach using Open API 3.0 and Open API Maven Generator to boost API development and allow foreseeing how the product looks like. The generated code can be overridden via Mustache templates such as data transfer object.
Whenever an order is placed, a corresponding payment record is asynchronouly created for the user to purchase later on. Behind the scene, order service produces Kafka messages and payment service consumes them.
As Order's Kafka messages tend to evolve by development's needs, Confluent Avro is used to version schemas of Kafka messages. Schemas are stored in Kafka's log files and indices are stored in Avro's in-memory storage. For example, OrderMessage's schema:
{
"type": "record",
"name": "OrderMessage",
"namespace": "com.emeraldhieu.vinci.payment",
"fields":
[
{
"name": "orderId",
"type": "string"
}
]
}
Liquibase supports revisioning, deploying and rolling back database changes. On top of that, it allows initializing data from CSV for demonstrative purpose.
Flyway is similar. It's used to define database structure, bootstrap initial data via SQL statements, and manage database migrations.
gRPC is said to be faster and more secured than traditional REST API because it transmits binary data instead of JSON.
Module grpc-interface
contains protobuf3 files to define services, request, and response messages then use protobuf-maven-plugin to generate Java service stubs. The gRPC server shipping
implements those stubs to follow the contracts defined in protobuf files.
You can test the gRPC using grpcurl.
grpcurl --plaintext -d '{"id": "d707ada36e6644ddaec63a52e7a40d56"}' localhost:50013 com.emeraldhieu.vinci.shipping.grpc.ShippingService/GetShipping
GraphQL is a query language that allows users to retrieve only necessary data in their own way in a single request. GraphQL for Spring uses annotations to map handler methods to queries and fields in a GraphQL schema.
Vinci incorporates gRPC into GraphQL. For example, GraphQL ShippingDetailController#shipping
calls gRPC server to retrieve a Shipping
in gRPC manner.
This is a simple GraphQL to test.
curl --location 'http://localhost:50003/graphql' \
--header 'Content-Type: application/json' \
--data '{"query":"{\r\n shippingDetails(offset: 0, limit: 10, sortOrders: []) {\r\n id\r\n amount\r\n shipping {\r\n id\r\n status\r\n }\r\n }\r\n}","variables":{}}'
Response
{
"data": {
"shippingDetails": [
{
"id": "ae61181973fd4896a99ecb4089005197",
"amount": 2.0,
"shipping": {
"id": "d707ada36e6644ddaec63a52e7a40d56",
"status": "IN_PROGRESS"
}
}
]
}
}
GraphQL DGS is a framework built on top of GraphQL with more improvements.
This is a sample request.
curl --location 'http://localhost:50001/graphql' \
--header 'Content-Type: application/json' \
--data '{"query":"query GetOrder($id : String!) {\n getOrder(id: $id) {\n id\n products\n createdBy\n createdAt\n updatedBy\n updatedAt\n }\n}\n","variables":{"id":"0a5eb04756f54776ac7752d3c8fae45b"}}'
Response
{
"data":
{
"getOrder":
{
"id": "0a5eb04756f54776ac7752d3c8fae45b",
"products":
[
"car",
"bike",
"house"
],
"createdBy": "20825389f950461b8766c051b9182dd4",
"createdAt": "2022-11-27T00:00:00Z",
"updatedBy": "cca4806536fe4b218c12cdcde4d173df",
"updatedAt": "2022-11-28T00:00:00Z"
}
}
}
Vinci uses Spring 6's Problem Details to keep error responses consistent across microservices.
{
"type": "http://localhost:50001/vinci/types",
"title": "Unprocessable Entity",
"status": 422,
"detail": "Invalid sort order",
"instance": "/orders"
}
Like Lombok, Mapstruct is a code generator library that supports mapping between entities and DTOs without writing boilerplate code. A significant benefit is that mappers don't need unit tests because there's no code to test!
JDK 17, Maven, Docker Desktop.
At the module directories of bom
, grpc-interface
, order
, payment
, and shipping
, respectively, run this command
mvn clean install
It will take a while. Be patient. :)
NOTE: If you're using Apple Chip, as for grpc-interface
, run this instead
mvn clean install -Papple-chip
At the project directory, run this command
docker compose up -d
NOTE: If you're using Apple Chip, uncomment all the lines below in docker-compse.yml
.
platform: linux/x86_64
curl --location 'http://localhost:50001/orders' \
--header 'Content-Type: application/json' \
--data '{
"products": [
"coke",
"juice",
"cider"
]
}'
If it returns 201 with a JSON response, the app "order" is working. Check Order API for other endpoints.
K3d is a lightweight Kubernetes distribution that supports creating a K8s cluster. This guide assumes you've had K3d installed.
Create a local registry to store docker images of our apps
k3d registry create registry42 --port 5050
At the project directory, run this
k3d cluster create cluster42 -p "8080:50001@loadbalancer" --registry-use k3d-registry42:5050 --registry-config k8s/registries.yaml
What it does
- Create a K8s cluster
- Use an existing docker registry
- Map host machine's port 8080 to K3d's loadbalancer's port 50001
Helm is a Kubernetes package manager that allows reusing sets of K8s manifests (called "charts"). To make a long story short, Helm chart is similar to Docker image but used for K8s. This guide assumes you've had Helm installed.
At the project directory, run this
helm install local -f k8s/schema-registry/values.yaml oci://registry-1.docker.io/bitnamicharts/schema-registry --version 10.0.0
What it does
- Install a Helm chart release named
local
based on Confluent Schema Registry packaged by Bitnami - Create K8s resources for Kafka and Schema Registry
3.1) At the directory order
, build the app
mvn clean package
3.2) Create docker image order
docker build -t localhost:5050/order:1.0-SNAPSHOT .
3.3) Push image to the registry
docker push localhost:5050/order:1.0-SNAPSHOT
Repeat all steps 3.x for payment
and shipping
.
Open another terminal, create k8s resources
kubectl apply -f deployment.yaml
Listen on port 5432, forward data to a pod selected by the service "postgres"
kubectl port-forward svc/postgres 5432
Connect to postgres from the host machine. Enter password "postgres".
psql -h 127.0.0.1 -d postgres -U postgres -W
List databases. If you see databases order
, payment
, and shipping
, the setup is working.
postgres=# \l
Since K3d Load balancer has routed requests to app services by K8s ingress rules, you won't need port-forwarding. Mind that the context paths /order
is mandatory.
Create an order on your host machine
curl --location 'http://localhost:8080/order/orders' \
--header 'Content-Type: application/json' \
--data '{
"products": [
"coke",
"juice",
"cider"
]
}'
If it returns a JSON response with an ID, it's working.
Listen on port 8081, forward data to a pod selected by the service "schema-registry"
kubectl port-forward svc/local-schema-registry 8081
Ask for the very first schema of "order"
curl http://localhost:8081/schemas/ids/1
Response
{
"schema": "{\"type\":\"record\",\"name\":\"OrderMessage\",\"namespace\":\"com.emeraldhieu.vinci.order\",\"fields\":[{\"name\":\"orderId\",\"type\":\"string\"}]}"
}
Note that the endpoint depends on how you deploy the stack
- Profile
local
: The endpoint starts withhttp://localhost:50001/orders
- Profile
k8s
: The endpoint starts withhttp://localhost:8080/order/orders
GET /orders
Parameters | Description | Format |
---|---|---|
sortOrders |
Sort orders | column1,direction|column2,direction |
Some examples of sortOrders
:
createdAt,desc
updatedAt,desc|createdBy,asc
curl --location --request GET 'http://localhost:50001/orders?sortOrders=updatedAt,desc|createdBy,asc'
[
{
"id": "0a5eb04756f54776ac7752d3c8fae45b",
"products":
[
"car",
"bike",
"house"
],
"createdBy": "20825389f950461b8766c051b9182dd4",
"createdAt": "2022-11-27T00:00:00",
"updatedBy": "cca4806536fe4b218c12cdcde4d173df",
"updatedAt": "2022-11-28T00:00:00"
}
]
POST /orders
Required parameters
Parameters | Type | Description |
---|---|---|
products |
List | List of products |
curl --location --request POST 'http://localhost:50001/orders' \
--header 'Content-Type: application/json' \
--data-raw '{
"products": [
"coke",
"juice",
"cider"
]
}'
{
"id": "ac621782642942e3b9cd239f2ce28575",
"products":
[
"coke",
"juice",
"cider"
],
"createdBy": "7f64bc8819ef464aa5dfd704d85a35ef",
"createdAt": "2023-05-03T09:18:03.589068545",
"updatedBy": "b5920f1ec6234d199de627bca0e297a2",
"updatedAt": "2023-05-03T09:18:03.589068545"
}
GET /orders/<id>
Parameters | Description | Type |
---|---|---|
id |
Order ID | String |
curl --location 'http://localhost:50001/orders/0a5eb04756f54776ac7752d3c8fae45b'
{
"id": "0a5eb04756f54776ac7752d3c8fae45b",
"products":
[
"car",
"bike",
"house"
],
"createdBy": "20825389f950461b8766c051b9182dd4",
"createdAt": "2022-11-27T00:00:00",
"updatedBy": "cca4806536fe4b218c12cdcde4d173df",
"updatedAt": "2022-11-28T00:00:00"
}
DELETE /orders/<id>
Parameters | Description | Type |
---|---|---|
id |
Order ID | String |
curl --location 'http://localhost:50001/orders/0a5eb04756f54776ac7752d3c8fae45b'
Response status is 204 with no content
- Infrastructure
Deploy to K8sScale app services by ingress and loadbalancerScale Kafka by ingress and loadbalancerRemove Zookeeper since KDraft was introduced
- App
- Implement OAuth2
- Improve database modeling
- Improve field validation
- Update README.md for other endpoints