Skip to content

Latest commit

 

History

History
136 lines (106 loc) · 6.27 KB

README.md

File metadata and controls

136 lines (106 loc) · 6.27 KB

gRPC Load Balancing with ETCD

License

The objective of hereby project is to enable service discovery and client-side load balancing with ETCD cluster acting as lookaside load balancer (gRPC Blog).

All servers implementing given contract will create unique keys (with common prefix) in ETCD. Assume service which fully qualified name equals helloworld.Greeter, two available instances may create keys helloworld.Greeter/server1:3333 and helloworld.Greeter/server2:4444. Clients connected to ETCD watch for updates to common prefix and act accordingly, when new entry is added or existing removed.

Quickstart

Server Bootstrapping

Correct propagation of endpoint availability to ETCD requires notification about gRPC server lifecycle events like startup or shutdown. Below listing shows how to decorate well-known server initialization with ListenableServerState. Supply EtcdServiceRegistry that automatically handles instance registration and deregistration with service discovery.

// initialize ETCD client
Client etcdClient = Client.builder().endpoints( "http://localhost:2379" ).build();

// choose service discovery options
Configuration options = Configuration.builder().withKeyPrefix( "services" ).build()

// create ETCD service registry client
EtcdServiceRegistry serviceDiscovery = new EtcdServiceRegistry( etcdClient, options );

// bootstrap gRPC server
Server server = ListenableServerStateImpl.decorate(
        ServerBuilder.forPort( 8080 )
                .addService( new GreeterImpl() )
                .build()
).addStateListener( serviceDiscovery );
server.start();

Client Initialization

Bootstrapping gRPC client is simpler and requires only providing correct NameResolverProvider.

// initialize ETCD client
Client etcdClient = Client.builder().endpoints( "http://localhost:2379" ).build();

// choose service discovery options
Configuration options = Configuration.builder().withKeyPrefix( "services" ).build()

// register ETCD name resolver
NameResolverRegistry.getDefaultRegistry().register(
    new EtcdNameResolverProvider( etcdClient, options )
);

// create gRPC client
ManagedChannel channel = ManagedChannelBuilder.forTarget( "etcd://helloworld.Greeter" ).build();

Documentation

Configuration Options

Table describes client-side and server-side configuration options:

Property Name Default Description
keyPrefix Prefix added to ETCD which allows to distinguish service registry keys from other users of the cluster.
keySeparator / Character used as a separator between prefix, fully qualified service name and endpoint addresses.
leaseTtlSec 10 Time-to-live (in seconds) of ETCD lease associated with server key. If gRPC server does not send keep-alive on time, it will be considered dead and removed from service registry.

Propagating Service Configuration

Service configuration is a mechanism allowing service owners to publish parameters that will be automatically propagated and used by all clients of their service. ETCD clients listen not only to updates of service key children, but also to the key itself. Value stored under the key can contain gRPC service configuration in JSON format. Users need to leverage external tools like etcdctl or any ETCD client (e.g. jetcd) to manually update service configuration.

$ export SERVICE_CONFIG="
{
   \"loadBalancingPolicy\": \"round_robin\",
   \"methodConfig\": [
      {
         \"name\": [
            {
               \"service\": \"helloworld.Greeter\",
               \"method\": \"SayHello\"
            }
         ],
         \"waitForReady\": false,
         \"retryPolicy\": {
            \"maxAttempts\": 3,
            \"initialBackoff\": \"2.1s\",
            \"maxBackoff\": \"2.2s\",
            \"backoffMultiplier\": 3,
            \"retryableStatusCodes\": [
               \"UNAVAILABLE\",
               \"RESOURCE_EXHAUSTED\"
            ]
         }
      }
   ],
   \"retryThrottling\": {
      \"maxTokens\": 10,
      \"tokenRatio\": 0.1
   }
}
"
$ etcdctl put helloworld.Greeter "$( echo $SERVICE_CONFIG )"

Example

Assume we have deployed gRPC service helloworld.Greeter with two instances across servers machine1 and machine2. Property keyPrefix has been configured to services. The following ETCD resources will be created.

Key Value Description
services/helloworld.Greeter ${service config JSON} Manually inserted service configuration in JSON format.
services/helloworld.Greeter/machine1:1234 {"dataCenter": "DC1"} Custom JSON representing endpoint which can be used by load balancing policy.
services/helloworld.Greeter/machine2:2345 {"dataCenter": "DC2"}

gRPC clients listen to all updates to keys with services/helloworld.Greeter prefix. They act accordingly when service configuration is updated or service endpoints join or leave the cluster.

Build from Source

Unit tests take advantage of test containers, so developers need to install and start Docker environment first.

$ mvn clean install -DskipTests
$ mvn clean install