Kritim Yantra
Apr 02, 2025
In modern software development, systems must be scalable, resilient, and responsive. Event-Driven Architecture (EDA) is a design paradigm that enables applications to react to events (changes in state) in real-time, improving flexibility and decoupling components.
This blog explores three key concepts in EDA:
We’ll break down each concept, compare them, and discuss real-world use cases.
EDA is a software architecture where components communicate via events—notifications that something has happened (e.g., "Order Placed," "Payment Processed"). Instead of direct API calls, systems emit and react to events asynchronously.
A message queue is a middleware that stores messages until they are processed by a consumer. It follows a First-In-First-Out (FIFO) model.
# Producer (Order Service)
channel.queue_declare(queue='orders')
channel.basic_publish(exchange='', routing_key='orders', body='Order123')
# Consumer (Payment Service)
def callback(ch, method, properties, body):
print(f"Processing order: {body}")
channel.basic_consume(queue='orders', on_message_callback=callback, auto_ack=True)
channel.start_consuming()
Pros | Cons |
---|---|
Guaranteed delivery | Only one consumer per message |
Ordered processing | Requires manual scaling |
Decouples producers & consumers | Can become a bottleneck |
In Pub/Sub, messages are broadcasted to multiple consumers via topics (channels). Unlike queues, multiple subscribers can receive the same message.
# Publisher (Notification Service)
producer = KafkaProducer(bootstrap_servers='localhost:9092')
producer.send('notifications', value='New user registered')
# Subscriber (Email Service)
consumer = KafkaConsumer('notifications', bootstrap_servers='localhost:9092')
for message in consumer:
print(f"Sending email for: {message.value}")
Pros | Cons |
---|---|
Multiple consumers per message | No guaranteed order (unless partitioned) |
Highly scalable | Complex setup (e.g., Kafka) |
Real-time broadcasting | Can lead to duplicate processing |
Event Sourcing is a pattern where state changes are stored as an immutable sequence of events. Instead of updating a database record, we append events (e.g., "UserCreated," "AddressUpdated").
# Event Store
events = [
{"type": "AccountCreated", "data": {"id": "acc1", "balance": 0}},
{"type": "Deposited", "data": {"id": "acc1", "amount": 100}},
{"type": "Withdrawn", "data": {"id": "acc1", "amount": 50}},
]
# Reconstructing state
balance = 0
for event in events:
if event["type"] == "Deposited":
balance += event["data"]["amount"]
elif event["type"] == "Withdrawn":
balance -= event["data"]["amount"]
print(f"Current balance: {balance}") # Output: 50
Pros | Cons |
---|---|
Complete audit trail | Event replay can be slow |
Time-travel debugging | Storage can grow large |
Works well with CQRS | Requires careful schema design |
Feature | Message Queues | Pub/Sub | Event Sourcing |
---|---|---|---|
Communication | Point-to-point | Broadcast | Append-only log |
Consumers | Single consumer per message | Multiple consumers | Replayable events |
Ordering | FIFO (usually) | Depends on system | Strictly ordered |
Use Case | Task processing | Real-time notifications | State reconstruction |
Event-Driven Architecture enables scalable, decoupled, and resilient systems.
By combining these patterns, you can build highly responsive and maintainable systems.
Would you like a deeper dive into any of these topics? Let me know in the comments! 🚀
No comments yet. Be the first to comment!
Please log in to post a comment:
Continue with Google