Just another example of CQRS and ES
Maybe you have heard about CQRS (Command Query Responsibility Segregation) design pattern together with ES (Event Sourcing) pattern so many times.
However, you may still confuse about it and I am here to give to you an example and also some personal feedback / view about it. (Yes, I am usually very critical instead of hype easily with new patterns, architectures or languages).
Before I start defining a bit CQRS - ES I prefer to share the GitHub repository and some instructions first.
You can find here https://github.com/DanielMerchan/cqrs-es-java-spring-plain a CQRS example with instructions to make it working
- There are two branches:
- initial: It is the skeleton project if you want to build it yourself by following every <Step>.md of the main branch
- main: Contains the CQRS example. Take into consideration the following
- Make sure 0.md is done in your machine and you have docker ready with the services running
- 14.md and 20.md has examples on how to test the application
- All the .md have a lot of information.
- I recommend to execute the programs in debug mode and slowly move around the code for understanding.
CQRS
Traditional vs CQRS |
- Command: They are instructions to perform an action that changes the state of the application. They represent intent and are typically initiated by the user or an external system.
- Query: They are requests to retrieve data from the system. They do not alter the state of the application but rather fetch and return the required information.
Benefits of CQRS
- The separation of each model allows to each own to be independently optimised.
- By isolating these concerns, this can lead to more efficient resource utilisation and evolution of complex systems.
- There are too many data stores, databases etc... which are optimal for massive write, or high performant query. It is important to select the appropiate data store for Command and for Query. Example could be to write into MongoDB and to read from Elastic Search for example.
- CQRS already enforces somehow to drive the solution with an event driven / asynchronous architecture which are "so fancy" nowadays and gets you ready for ensuring delivering the best scalable and performance.
Amazing right? But for a second, let's put or feet on the ground again and talk about what are the cons of using CQRS. It is not easy at all to implement and comes with a cost implication.
Careful... CQRS brings also not good things
- The obvious one is Complexity. Separating the read/write models into two microservices already implies and increase of complexity and the need for adding new data stores and somehow a communication system between those like Kafka, RabbitMQ, JMS or pigeon as a service (joke).
- Eventual Consistency!!!. Probably this is the most painful word about event driven asynchronous architectures, microservices and the nowadays world. Basically, when you write a transaction it will be a delay for the read microservice to replicate the data and make it available for the consumers. You will need to think on mechanisms to notify when data is ready for your UIs or other systems (callbacks).
- Learning curve: If you try also to apply DDD (domain driven design) and CQRS together you may notice you require some skill to implement it properly (even the example I am sharing later has flaws, but it works)
- Costs. Yeah, prepare your wallet. Instead a mono/modu-lith deployed in a single compute instance and making your life easier. Now you have 2 microservices in containers, deployed in an amazing Kubernetes cluster, communicated via Kakfa, Azure Service Bus or whatever and maybe a MongoDB and a Elastic Search (paying for enterprise licenses).
- Complex synchronisation: Yes, eventual consistency and also synchronising the data back to the read model can be challenging in case of failures (spoiler, Event Sourcing comes into picture to make this bullet point easier)
- For easy cases or systems I will avoid CQRS, it is an overkill.
- Do I really need all this scalability, performance, fine grain and complexity? Always make this question. If you have a system with a reasonable amount of transactions, concurrent users and you do not expect this will grow as fast as you think... then do not torture yourself, your customer or whoever. Why to pay for a formula 1 when you need a bycicle.
- am I skilled enough or my team is skilled enough to implement a robust CQRS pattern? We are humans after all (like the song)
ES
Event Sourcing |
- Events are immutable. They cannot be updated or deleted
- Consistency and reliability: Improves a lot these non-functional requirements as the source of truth is the log of events.
- Enables CQRS and scalability: They pair perfectly forcing you into an event driven architecture.
CQRS + ES
CQRS + ES example |
Comments
Post a Comment