Understanding concepts in Event Driven Architectures (EDA)
Event Driven Architecture has become a very popular choice in recent times for developing highly scalable distributed systems. Is a way of designing software where different parts of the system communicate by sending and reacting to events.
An event is something that happens, like:
- A user signs up
- A user uploads a photo
- A payment is completed
Instead of making direct requests to each other, different parts of the system listen for these events and react when needed(asynchronously).
For example:
- The Payment Service is completed and emits an event: "payment done".
- The Inventory Service listens for that event and updates the stock for the new product that has been bought.
- The Email Service listens and sends a confirmation email.
Let's see a very simple diagram of the components in Event Driven Architecture
Key Components
1. Event Producer/Publisher
The system or service that creates and emits events when something happens.
Example: A User Service sends an event "User Registered" when a new user signs up.
2. Message Broker
A message broker is a middleware that acts as an intermediary between event producers and event consumers. Its primary role is to receive, store, and route events efficiently, ensuring that the events are delivered to the correct consumer services.
In this context, there is a crucial concept inside message brokers called Channels. These channels act as communication routes that orchestrate the flow of events from producers to the right consumers.
- In Kafka, these channels are called Topics. It supports multi subscriber models, allowing multiple consumers to listen to the same publisher.
- In RabbitMQ, these channels are called Queues. RabbitMQ is typically point-to-point for queues, meaning that only one consumer can process a message from a queue.
Example technologies: Kafka, RabbitMQ, AWS SNS/SQS, Google Pub/Sub.
3. Event Consumer/Subscriber
The service that listens to events and reacts to them.
Example: A Notification Service listens for "User Registered" and sends a welcome email.
4. Event
The actual message that describes what happened. It usually contains structured data (JSON, XML).
Example event message:
{
"data": {
"id": "event-123456",
"type": "user.event.registered",
"timestamp": "2025-04-01T12:00",
"attributes": {
"id": "usr-98765",
"email": "[email protected]",
"surname": "example",
}
}
}
Broker Exchanges
In message brokers, an exchange is a routing mechanism that determines how messages are delivered to queues. Different types of exchanges provide different message distribution patterns.
Direct Exchange (1:1 Routing)
Routes messages to queues based on an exact match between a routing key and a queue binding key (RabbitMQ).
How it works?
- The producer sends a message with a routing key.
- The exchange forwards the message only to queues with a matching binding key.
Example:
A message with routing key = "order.created"
goes only to the queue bound with "order.created"
.
Fanout Exchange (Broadcast)
Sends messages to all bound queues, ignoring the routing key
(RabbitMQ-Fanout Exchange | Kafka-multiple consumers per topic).
How it works?
- The producer sends a message.
- The exchange copies the message to all queues bound to it.
Example:
A notification event is broadcasted to multiple services (email, SMS, push notifications).
Topic Exchange (Pattern Matching)
Routes messages based on wildcard patterns in routing keys. Supports *
, matches one word and #
, matches multiple words. (RabbitMQ-Topic Exchange | Kafka-topics with partition keys).
How it works?
- Uses
.
to separate words in routing keys ("order.payment.failed"
).
Example:
A queue bound to "order.*"
receives "order.created"
and "order.cancelled"
, but not "user.created"
.
Advantages of EDA - Reactive Manifesto
- Scalability: decoupled services can scale independently.
- Flexibility: easy to add new event consumers without modifying existing services.
- Real-time Processing: enables real-time data streaming and quick responses.
- Resilience: failures in one service don’t impact the entire system as long as events and subscriptions are managed properly
- Loose Coupling: components interact through events, reducing dependencies.
Disadvantages of EDA:
- Eventual consistency: different parts of the system might temporarily have outdated information until all events are processed.
- Complex Debugging: harder to trace issues across multiple async events.
- Increased Latency: some events may take longer to be processed due to async nature.
- Event Duplications: requires idempotency to avoid duplicate processing.
- Event Ordering Issues: some brokers don’t guarantee message order (RabbitMQ).
- Learning Curve: requires expertise in event-driven design patterns.
Conclusion
If you are thinking of implementing a distributed, real-time and highly scalable system, you should definitely consider implementing an event-driven architecture to take advantage of all the benefits it offers.