alternatorlb

package module
v2.0.0-...-bc1159d Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Dec 24, 2024 License: Apache-2.0 Imports: 14 Imported by: 0

README

Alternator - Client-side load balancing - Go

Introduction

As explained in the toplevel README, DynamoDB applications are usually aware of a single endpoint, a single URL to which they connect - e.g., http://dynamodb.us-east-1.amazonaws.com. But Alternator is distributed over a cluster of nodes and we would like the application to send requests to all these nodes - not just to one. This is important for two reasons: high availability (the failure of a single Alternator node should not prevent the client from proceeding) and load balancing over all Alternator nodes.

One of the ways to do this is to provide a modified library, which will allow a mostly-unmodified application which is only aware of one "enpoint URL" to send its requests to many different Alternator nodes.

Our intention is not to fork the existing AWS client library for Go. Rather, our intention is to provide a small library which tacks on to the existing "aws-sdk-go" library which the application is already using, and makes it do the right thing for Alternator.

The alternator_lb.go library

The AlternatorNodes class defined in alternator_lb.go can be used to easily change any application using aws-sdk-go-v2 from using Amazon DynamoDB to use Alternator: While DynamoDB only has one "endpoint", this class helps us balance the requests between all the nodes in the Alternator cluster.

Using the library

To use this class, simply replace the code which creates a AWS SDK aws.Config Instead of the usual way of creating a Config, like:

cfg, err := config.LoadDefaultConfig(ctx)

Use an AlternatorNodes object, which keeps track of the live Alternator nodes, to create a session with the following commands:

alternatorNodes := alternatorlb.NewAlternatorNodes("http", 8000, "127.0.0.1")
alternatorNodes.Start(ctx, 1*time.Second)
defer alternatorNodes.Stop()

cfg := alternatorNodes.Config("dog.scylladb.com", "alternator", "secret_pass")

Then, the rest of the applicaton can use this config normally - call db := dynamodb.NewFromConfig(cfg)and then send DynamoDB requests to db; As usual, thisdb` object is thread-safe and can be used from multiple threads.

The parameters to NewAlternatorNodes() indicate a list of known Alternator nodes, and their common scheme (http or https) and port. This list can contain one or more nodes - we then periodically contact these nodes to fetch the full list of nodes using Alternator's /localnodes request. In the Config() method, one needs to pick a "fake domain" which doesn't really mean anything (except it will be used as the Host header, and be returned by the DescribeEndpoints request), and the key and secret key for authentication to Alternator.

Every request performed on this new session will pick a different live Alternator node to send it to. Despite us sending different requests to different nodes, Go will keep these connections cached and reuse them when we send another request to the same node.

(TODO: figure out the limitations of this caching. Where is it documented?).

Example

This directory also contains one trivial example of using alternator_lb.go, example.go. This example opens a config using NewAlternatorNodes(), as described above, and then uses it 20 times in a loop - and we'll see that every request will be sent to a different node.

Documentation

Index

Examples

Constants

This section is empty.

Variables

This section is empty.

Functions

This section is empty.

Types

type AlternatorNodes

type AlternatorNodes struct {
	// contains filtered or unexported fields
}

AlternatorNodes holds the configuration for the load balanced alternator nodes as well as some locks for the load balancing thread.

Example
// To run this example, just run:
//       go test example_test.go

package main

import (
	"context"
	"crypto/tls"
	"crypto/x509"
	"fmt"
	"github.com/aws/aws-sdk-go-v2/service/dynamodb"
	alternatorlb "github.com/scylladb/alternator-load-balancing/go/v2"
	"net/http"
	"time"
)

var customPEMCertificate = []byte(nil)

func main() {
	ctx := context.Background()

	// Uncomment to use Amazon DynamoDB configured in ~/.aws/
	//    cfg, _ := config.LoadDefaultConfig(ctx)
	//    db := dynamodb.NewFromConfig(cfg)

	// Use the local Alternator with our silly testing alternator/secret_pass
	// authentication - and the new load balancing code.
	alternatorNodes := alternatorlb.NewAlternatorNodes("http", 8000, "127.0.0.1")

	// To add custom CA certificate on top of system CA certificates:
	if customPEMCertificate != nil {
		systemPool, err := x509.SystemCertPool()
		if err != nil {
			panic(err)
		}

		if !systemPool.AppendCertsFromPEM(customPEMCertificate) {
			panic("failed to append custom certificate")
		}

		alternatorNodes.SetHTTPClient(&http.Client{
			Transport: &http.Transport{
				TLSClientConfig: &tls.Config{
					RootCAs: systemPool,
				},
			},
		})
	}

	alternatorNodes.Start(ctx, 1*time.Second)
	defer alternatorNodes.Stop()

	cfg := alternatorNodes.Config("dog.scylladb.com", "alternator", "secret_pass")
	db := dynamodb.NewFromConfig(cfg)

	for i := 1; i < 20; i++ {
		time.Sleep(300 * time.Millisecond)
		// Do the simplest possible request - DescribeEndpoints
		result, err := db.DescribeEndpoints(ctx, &dynamodb.DescribeEndpointsInput{})
		if err != nil {
			fmt.Println(err.Error())
		} else {
			fmt.Println("response:", *result.Endpoints[0].Address)
		}
	}
}
Output:

func NewAlternatorNodes

func NewAlternatorNodes(scheme string, port int, initialNodes ...string) *AlternatorNodes

NewAlternatorNodes creates a new, unstarted instance of the alternator nodes loadbalancing.

func (*AlternatorNodes) Config

func (n *AlternatorNodes) Config(fake_domain string, key string, secret_key string) aws.Config

Config produces a conf for the AWS SDK that will integrate the alternator loadbalancing with the AWS SDK.

func (*AlternatorNodes) SetHTTPClient

func (n *AlternatorNodes) SetHTTPClient(client httpClient)

SetHTTPClient changes underlying http client used for fetching the list of alternator nodes.

func (*AlternatorNodes) Start

func (n *AlternatorNodes) Start(ctx context.Context, fetchInterval time.Duration)

Start will start the loadbalancing thread that keep available alternator instances in sync and selectes the next instance for load balancing.

func (*AlternatorNodes) Stop

func (n *AlternatorNodes) Stop()

Stop will stop the loadbalancing thread and should be called once you are done with the AWS SDK session.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL