What is load balancing and how to do it on client side
“Load balancing” can be explained as maintaining any workload. For example if you have to serve 1000 breakfast in the morning you can divide the work among 2-3 or more caterers to lower the delivery/preparation time.
In the computer world, same logic applies, if you want to deliver fast, you can divide the work, for example for a website we can have 5-10 webserver, this way website will be delivered faster(especially during high traffic), this is server side.
I want to talk about client side load balancing and how this works, for example you are querying backend for every request, and you have multiple servers doing the backend work, for this I will use WHOIS service
whois1.service.com whois2.service.com
We can load balance every request in these two servers, most used algorithm and default round robin meaning every request will go to one than another, like who1, who2 who1 and so on, this can be done on the client side with “Spring Cloud LoadBalancer”. we need to configure 3 classes
- ServiceInstanceListSupplier for defining the backend addresses and ports
- Bean ServiceInstanceListSupplier for supplying the list of end points
- @LoadBalancerClient which is the webclient will be used for backend calls
Now, we have the whole setup for the backend
public class RestCaller implements ServiceInstanceListSupplier {
private final String serviceId;
public RestCaller(String serviceId) {
this.serviceId = serviceId;
}
@Override
public String getServiceId() {
return serviceId;
}
@Override
public Flux<List<ServiceInstance>> get() {
return Flux.just(Arrays.asList(
new DefaultServiceInstance(serviceId, serviceId, "abc.cyclic.app", 443, true),
new DefaultServiceInstance(serviceId, serviceId, "someting.koyeb.app", 443, true),
new DefaultServiceInstance(serviceId, serviceId, "127.0.0.1", 8080, false)
));
}
}
@Configuration
public class RestCallerConfiguration {
@Autowired
@Qualifier("webClientBuilder")
WebClient.Builder webClientBuilder;
@Bean
@Primary
ServiceInstanceListSupplier serviceInstanceListSupplier(ConfigurableApplicationContext ctx) {
return ServiceInstanceListSupplier
.builder()
.withRetryAwareness()
.withHealthChecks(webClientBuilder.build())
.withBase(new RestCaller("whoisService"))
.build(ctx);
}
}
@Configuration
@LoadBalancerClient(name = "whoisService", configuration = RestCallerConfiguration.class)
public class WebClientConfig {
@LoadBalanced
@Bean
@Qualifier("loadBalancedWebClientBuilder")
WebClient.Builder loadBalancedWebClientBuilder() {
return WebClient.builder();
}
@Bean
@Qualifier("webClientBuilder")
WebClient.Builder webClientBuilder() {
return WebClient.builder();
}
}
This can now be used in any @Component
or @Service
.
@Component
public class WhoisCaller {
@Autowired
@Qualifier("loadBalancedWebClientBuilder")
WebClient.Builder webClientBuilder;
public String getWhois(String ip, String source) {
String url = "http://whoisService/whois?ip=" + ip + "&source=" + source;
return webClientBuilder.build()
.get()
.uri(url)
.retrieve()
.bodyToMono(String.class)
.timeout(Duration.ofMinutes(1))
.doOnError(e -> {
WebClientResponseException e1 = (WebClientResponseException) e;
log.error(e1.getRequest().getURI().getHost());
})
.block();
}
}
References: