2022-04-25 10:41:36 -04:00
|
|
|
package cluster
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-05-19 14:05:41 -05:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/base64"
|
2022-04-25 10:41:36 -04:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
2022-06-03 13:07:37 -04:00
|
|
|
"testing"
|
|
|
|
"time"
|
2022-04-25 10:41:36 -04:00
|
|
|
|
2022-05-19 14:05:41 -05:00
|
|
|
"github.com/hashicorp/consul/api"
|
2022-04-25 10:41:36 -04:00
|
|
|
"github.com/hashicorp/consul/integration/consul-container/libs/node"
|
2022-06-03 13:07:37 -04:00
|
|
|
"github.com/hashicorp/consul/sdk/testutil/retry"
|
|
|
|
"github.com/stretchr/testify/require"
|
2022-04-25 10:41:36 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// Cluster provides an interface for creating and controlling a Consul cluster
|
|
|
|
// in integration tests, with nodes running in containers.
|
|
|
|
type Cluster struct {
|
2022-05-19 14:05:41 -05:00
|
|
|
Nodes []node.Node
|
|
|
|
EncryptKey string
|
2022-04-25 10:41:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a Consul cluster. A node will be started for each of the given
|
|
|
|
// configs and joined to the cluster.
|
|
|
|
func New(configs []node.Config) (*Cluster, error) {
|
2022-05-19 14:05:41 -05:00
|
|
|
serfKey, err := newSerfEncryptionKey()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
cluster := Cluster{
|
|
|
|
EncryptKey: serfKey,
|
|
|
|
}
|
2022-04-25 10:41:36 -04:00
|
|
|
|
|
|
|
nodes := make([]node.Node, len(configs))
|
|
|
|
for idx, c := range configs {
|
2022-05-19 14:05:41 -05:00
|
|
|
c.HCL += fmt.Sprintf(" encrypt=%q", serfKey)
|
|
|
|
|
2022-04-25 10:41:36 -04:00
|
|
|
n, err := node.NewConsulContainer(context.Background(), c)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
nodes[idx] = n
|
|
|
|
}
|
|
|
|
if err := cluster.AddNodes(nodes); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &cluster, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddNodes joins the given nodes to the cluster.
|
|
|
|
func (c *Cluster) AddNodes(nodes []node.Node) error {
|
|
|
|
var joinAddr string
|
|
|
|
if len(c.Nodes) >= 1 {
|
|
|
|
joinAddr, _ = c.Nodes[0].GetAddr()
|
|
|
|
} else if len(nodes) >= 1 {
|
|
|
|
joinAddr, _ = nodes[0].GetAddr()
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, node := range nodes {
|
|
|
|
err := node.GetClient().Agent().Join(joinAddr, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.Nodes = append(c.Nodes, node)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Terminate will attempt to terminate all nodes in the cluster. If any node
|
|
|
|
// termination fails, Terminate will abort and return an error.
|
|
|
|
func (c *Cluster) Terminate() error {
|
|
|
|
for _, n := range c.Nodes {
|
|
|
|
err := n.Terminate()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Leader returns the cluster leader node, or an error if no leader is
|
|
|
|
// available.
|
|
|
|
func (c *Cluster) Leader() (node.Node, error) {
|
|
|
|
if len(c.Nodes) < 1 {
|
|
|
|
return nil, fmt.Errorf("no node available")
|
|
|
|
}
|
|
|
|
n0 := c.Nodes[0]
|
2022-05-19 14:05:41 -05:00
|
|
|
|
|
|
|
leaderAdd, err := GetLeader(n0.GetClient())
|
2022-04-25 10:41:36 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-05-19 14:05:41 -05:00
|
|
|
|
2022-04-25 10:41:36 -04:00
|
|
|
for _, n := range c.Nodes {
|
|
|
|
addr, _ := n.GetAddr()
|
|
|
|
if strings.Contains(leaderAdd, addr) {
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("leader not found")
|
|
|
|
}
|
2022-05-19 14:05:41 -05:00
|
|
|
|
|
|
|
func newSerfEncryptionKey() (string, error) {
|
|
|
|
key := make([]byte, 32)
|
|
|
|
n, err := rand.Reader.Read(key)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("Error reading random data: %w", err)
|
|
|
|
}
|
|
|
|
if n != 32 {
|
|
|
|
return "", fmt.Errorf("Couldn't read enough entropy. Generate more entropy!")
|
|
|
|
}
|
|
|
|
|
|
|
|
return base64.StdEncoding.EncodeToString(key), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetLeader(client *api.Client) (string, error) {
|
|
|
|
leaderAdd, err := client.Status().Leader()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if leaderAdd == "" {
|
|
|
|
return "", fmt.Errorf("no leader available")
|
|
|
|
}
|
|
|
|
return leaderAdd, nil
|
|
|
|
}
|
2022-06-03 13:07:37 -04:00
|
|
|
|
|
|
|
const retryTimeout = 20 * time.Second
|
|
|
|
const retryFrequency = 500 * time.Millisecond
|
|
|
|
|
|
|
|
func LongFailer() *retry.Timer {
|
|
|
|
return &retry.Timer{Timeout: retryTimeout, Wait: retryFrequency}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WaitForLeader(t *testing.T, cluster *Cluster, client *api.Client) {
|
|
|
|
retry.RunWith(LongFailer(), t, func(r *retry.R) {
|
|
|
|
leader, err := cluster.Leader()
|
|
|
|
require.NoError(r, err)
|
|
|
|
require.NotEmpty(r, leader)
|
|
|
|
})
|
|
|
|
|
|
|
|
if client != nil {
|
|
|
|
waitForLeaderFromClient(t, client)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func waitForLeaderFromClient(t *testing.T, client *api.Client) {
|
|
|
|
retry.RunWith(LongFailer(), t, func(r *retry.R) {
|
|
|
|
leader, err := GetLeader(client)
|
|
|
|
require.NoError(r, err)
|
|
|
|
require.NotEmpty(r, leader)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WaitForMembers(t *testing.T, client *api.Client, expectN int) {
|
|
|
|
retry.RunWith(LongFailer(), t, func(r *retry.R) {
|
|
|
|
members, err := client.Agent().Members(false)
|
|
|
|
require.NoError(r, err)
|
|
|
|
require.Len(r, members, expectN)
|
|
|
|
})
|
|
|
|
}
|