One component (service) called Service Registry in the system, it enables dynamic routing between services. Essential for scaling apps.
The app registers themselves upon startup and sends Heartbeat (or check status using /health
endpoint) signals peridically to let the registry know they are up and running. Multiple instances of the same service can be registered.
The registry is not a single point of failure though, as there is a local copy of registry cached on every service which can be used for routing incase of outage of registry.
Types:
Client -> Load Balancer -> Registry -> Service
(a server component called Load Balancer finds service URL from registry and pings the service on behalf of the client)Client -> Registry -> Service
(client itself fetches the service URL from registry and pings the service themselves)Ex - Netflix Eureka.
A single entry point of traffic for all services, a reverse proxy, often registered as a service with the Service Registry.
We can place this as a Forward Proxy too. One use case for it is API response aggregation i.e. combine responses from User Service and Payment Service and send as a single response back to the user.
Manage traffic before it reaches services:
Ex - Spring Cloud API Gateway and Azure APIM.
Note: Just like other core services in a good distributed system, API Gateway also has fail-overs to ensure high availability such that there is no SPOF in the system.
Reference: API Gateway - YouTube
Helps scale horizontally by distributing traffic evenly across all service instances ensuring optimal performance, and preventing service overload which ensures high availability.
Uses an algorithm1 like Round-Robin to schedule incoming traffic to services.
Levels of Load Balancing:
Types of Load Balancing:
Types of Load Balancing strategies:
Ex - HAProxy and Nginx. All cloud service providers provide Load Balancers of their own too. We can also implement our own Load Balancing strategy using Spring Cloud Load Balancer.
A variation of the API Gateway pattern.
A separate API gateway for each kind of client. An API can be used by many clients like a web app, native mobile app, or other third-party services. Often times we may need slightly different responses and security policies for each of them.
Rate Limit - Set a maximum number of requests allowed within a specific time period.
// resilience4j annotation on a method that sends request to another service
@RateLimiter(name = "foobarRateLimit", fallbackMethod = "localCacheLookup")
Retry - Send request to a service, if we get a error response back or exception (or it times out), then retry a fixed number of times before falling back to a fallbackMethod
.
// resilience4j annotation on a method that sends request to another service
@Retry(name = "foobarRetry", fallbackMethod = "localCacheLookup")
It retries for all exceptions by default but we can configure a list of exceptions too. On a request taking too long to respond, we get back a TimeoutException
after a certain configured duration of time, we can specify retry for this exception to retry on timeouts.
Ex - Spring Retry’s @Retryable
, Resilience4j’s @Retry
.
Timeout - Don’t wait for a response indefinitely, use Timeout pattern (@TimeLimiter
in resilience4j) and if we don’t get any response back for a specific duration then trigger fallbackMethod
immediately.
// resilience4j annotation on a method that sends request to another service
@TimeLimiter(name = "foobarTimeout", fallbackMethod = "localCacheLookup")
If one service is down, we shouldn’t waste time sending it requests continually. Prevents requests from reaching a failing service, giving it time to recover.
When calls to a service fail or are slow beyond a certain threshold (a percentage of total calls made to it), we declare the circuit OPEN
and subseqent calls are rejected, we back-off for sometime and then make the circuit HALF_OPEN
and send a very limited number of requests to check if the service is up now.
highlight from the official docs
Don’t let one service’s resources be hogged up by another service.
Limits the number of concurrent calls to a service. Keep it below the maximum handling capacity so that not all resources are exhausted.
Consider this scenario, if we have 20 incoming requests to serviceA
from serviceB
but our max pool size is 15, then 5 requests have to wait (may timeout) and all resources of serviceA
are exhausted to serve only serviceB
.
If we limit requests to 10 using Bulkhead then while those 10 are processing, other 10 can be returned (error response) after a short wait time, and serviceA
can also serve requests coming from serviceC
simultaneously with the 10 pool connections that are unused still.
Reference: https://dzone.com/articles/resilient-microservices-pattern-bulkhead-pattern
Extend microservice functionality by attaching a sidecar service without altering code of the core service.
The “sidecar” handles tasks such as logging, monitoring, or security, allowing the main service to focus on its core functionality.
Often implemented in Service Mesh with Envoy Proxies.
Ex - Istio, linkerd.
/microservices/tools/#sidecar-proxy
Gradually replace the monolith with microservices. Incremental replacement of portions of monolith, minimizing downtime.
Event-driven architecture2 allows services to be decoupled from each other. Instead of calling another service directly, we can publish event messages to a centralized Event Bus and other subscribed services receive that event and update their state. Ex - MQs and Kafka Topics.
In Event Sourcing, microservices persist each stage change (event) in an event store, which is a datastore of all events in the system. The other services can then “react” to these state changes.
The event store also behaves like a message broker. It provides an API that enables services to subscribe to events. When a service saves an event in the event store, it is delivered to all interested subscribers, who then replay it to update their respective states.
Use Case: implementing Choreography Saga.
https://microservices.io/patterns/data/event-sourcing.html
https://learn.microsoft.com/en-us/azure/architecture/patterns/event-sourcing
Command and Query Responsibility Segragation
Problem - Read and write logic is different. In a highly normalized database, writes are simpler, whereas reads require complex JOINS. One repository doing both read and write can be split into two separating both of these concerns.
Create separate controller, service layer, DTOs inside a singular service to truly separate logic. This way we can also use a custom property and these beans annotated with @ConditionalOnProperty
and then we can start the application in read-only or write-only mode.
To further enhance this, instead of doing reads and writes on a single database, we can use 2 databases - one for read and one for write.
Pros:
Cons:
Further, we can totally have two different microservices too, one for reading and one for writing accessing their corresponding databases, allowing them to scale independently.
https://learn.microsoft.com/en-us/azure/architecture/patterns/cqrs
https://www.vinsguru.com/cqrs-pattern
Problem: How to make sure operations that span multiple services are atomic? Distributed Transactions.
One example of such transaction is when payment is deducted but inventory is out-of-stock, we then have to “rollback” transactions across inventory service as well as the payment service.
A saga is a sequence of local transactions carried out across diff microservices. Each local transaction updates the database (of that service only) and publishes a message/event/call to trigger the next transaction in the saga (in another service).
Every operation that is part of the Saga can be rolled back by a compensating transaction. Further, the Saga pattern guarantees that either all operations complete successfully or the corresponding compensation transactions are run to undo the work previously completed. So in a way, compensating transactions in Saga are equivalent of rollback in local transactions.
Two ways to implement Sagas:
Tools used to implement Saga pattern:
https://www.baeldung.com/cs/saga-pattern-microservices
Compose a single response by aggregating the responses of multiple independent microservices.
We can make REST calls to other services using WebClient and .zip
the results together in an AggregatorService
. It needs to be async otherwise we’ll just keep waiting for responses.
https://www.vinsguru.com/spring-webflux-aggregation
Three ways to implement:
order
and payment
services from the AggregatorService
order
service first and payment
later, and then combine in the AggregatorService
https://medium.com/geekculture/design-patterns-for-microservices-aggregation-pattern-1b8994516fa2
Health Check: expose a /health
API enpoint in actuator; it is polled by Service Mesh and Service Registry to check alive status of the microservice
Performance Metrics: with Micrometer, Prometheus, and Grafana
Log Aggregation: treat logs as streams and don’t store them in file like earlier times, aggregate (Logstash) and index them (Elasticsearch)
Distributed Tracing: trace a request across microservices with Zipkin [link]
Given below are deployment strategies that have zero downtime:
Big Bang Deployment: replace old application with new (involves downtime), and then after the deployment all traffic is received by the new version. Rollbacks are very expensive here.
Blue-Green Deployment: two identical production enviroments, only one is live at a given time and all traffic is routed to it, the other is on standby, helps in upgrading services to newer version with minimal downtime by gradually moving over traffic to newer version.
Canary Deployment: rollout features to a subset of users (early adopters) before making them available to all; redirect a part of traffic coming from users say 10% of the total users to the newer service (v1.1
), while the majority of traffic still goes to the stable service (v1.0
). If anything goes wrong, we can recover faster in this strategy.
Miners would bring canary birds in little cages into the mines because the birds were more sensitive to toxic gases like carbon monoxide. If the canary showed signs of distress, it was a warning to the miners to evacuate.
Rolling Deployment: slowly replace the pods containing old version with the pods containing new version of the code (in k8s). Also known as Phased Deployment.
A/B Testing: 50% of users are redirected to versionA
and the other half to versionB
to determine which one performs better.
com.example.app
├── controller (Presentation Layer)
│ └── OrderController.java
│ └── UserController.java
├── service (Service Layer)
│ └── OrderService.java
│ └── UserService.java
├── repository (Persistence Layer)
│ └── OrderRepository.java
│ └── UserRepository.java
├── model (Model Layer)
│ └── OrderEntity.java
│ └── UserEntity.java
├── dto (Optional - Data Transfer Objects)
│ └── OrderDTO.java
│ └── UserDTO.java
└── shared (Reusable Components)
├── exceptions
├── utils
└── config
com.example.app
├── orders
│ ├── controller
│ ├── service
│ ├── repository
│ ├── model
│ └── dto
├── customers
│ ├── controller
│ ├── service
│ ├── repository
│ ├── model
│ └── dto
└── shared
├── exceptions
├── utils
└── config
com.example.app
├── user-management
│ ├── UserController.java
│ ├── UserService.java
│ ├── UserRepository.java
│ └── User.java
├── product-catalog
│ ├── ProductController.java
│ ├── ProductService.java
│ ├── ProductRepository.java
│ └── Product.java
└── order-processing
├── OrderController.java
├── OrderService.java
├── OrderRepository.java
└── Order.java
com.example.app
├── modules
│ ├── user
│ │ ├── controller
│ │ ├── service
│ │ ├── repository
│ │ └── model
│ ├── product
│ │ ├── controller
│ │ ├── service
│ │ ├── repository
│ │ └── model
│ └── order
│ ├── controller
│ ├── service
│ ├── repository
│ └── model
└── shared
├── utils
├── config
└── exceptions
Highly decoupled architecture and suits very well for complex microservices.
Its also called “Ports and Adapters” because there is a clear separation between business rules and infrastructure code.
com.example.app
├── domain
│ ├── model
│ └── service
├── application
│ ├── usecase
│ └── dto
├── adapters
│ ├── web
│ │ └── controller
│ ├── persistence
│ │ └── repository
│ └── messaging
│ └── listener
└── config
com.example.app
├── users
│ ├── api
│ │ ├── UserController.java
│ │ └── UserDTO.java
│ ├── domain
│ │ └── User.java
│ ├── service
│ │ └── UserService.java
│ └── repository
│ └── UserRepository.java
├── orders
│ ├── api
│ │ ├── OrderController.java
│ │ └── OrderDTO.java
│ ├── domain
│ │ └── Order.java
│ ├── service
│ │ └── OrderService.java
│ └── repository
│ └── OrderRepository.java
└── shared
├── config
├── utils
└── exceptions
Use Maven or Gradle to create multiple modules. Each module can represent a domain, service, or technical layer.