2015-03-03 02:18:38 +00:00
|
|
|
package testutil
|
|
|
|
|
2015-03-11 04:38:31 +00:00
|
|
|
// TestServer is a test helper. It uses a fork/exec model to create
|
|
|
|
// a test Consul server instance in the background and initialize it
|
|
|
|
// with some data and/or services. The test server can then be used
|
|
|
|
// to run a unit test, and offers an easy API to tear itself down
|
|
|
|
// when the test has completed. The only prerequisite is to have a consul
|
|
|
|
// binary available on the $PATH.
|
|
|
|
//
|
|
|
|
// This package does not use Consul's official API client. This is
|
|
|
|
// because we use TestServer to test the API client, which would
|
|
|
|
// otherwise cause an import cycle.
|
|
|
|
|
2015-03-03 02:18:38 +00:00
|
|
|
import (
|
2017-02-11 02:11:21 +00:00
|
|
|
"context"
|
2015-03-03 02:18:38 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2015-03-11 01:08:14 +00:00
|
|
|
"io"
|
2015-03-03 02:18:38 +00:00
|
|
|
"io/ioutil"
|
2015-03-20 00:44:04 +00:00
|
|
|
"net"
|
2015-03-03 02:18:38 +00:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
2017-01-27 00:55:49 +00:00
|
|
|
"strconv"
|
2015-03-20 00:44:04 +00:00
|
|
|
"strings"
|
2015-10-22 14:47:50 +00:00
|
|
|
|
2015-10-22 18:14:22 +00:00
|
|
|
"github.com/hashicorp/go-cleanhttp"
|
2017-03-27 08:28:54 +00:00
|
|
|
"github.com/hashicorp/go-uuid"
|
2017-03-23 20:26:05 +00:00
|
|
|
"github.com/pkg/errors"
|
2015-03-03 02:18:38 +00:00
|
|
|
)
|
|
|
|
|
2016-08-25 00:33:53 +00:00
|
|
|
// TestPerformanceConfig configures the performance parameters.
|
|
|
|
type TestPerformanceConfig struct {
|
|
|
|
RaftMultiplier uint `json:"raft_multiplier,omitempty"`
|
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestPortConfig configures the various ports used for services
|
|
|
|
// provided by the Consul server.
|
2015-03-03 02:18:38 +00:00
|
|
|
type TestPortConfig struct {
|
|
|
|
DNS int `json:"dns,omitempty"`
|
|
|
|
HTTP int `json:"http,omitempty"`
|
2017-04-14 20:37:29 +00:00
|
|
|
HTTPS int `json:"https,omitempty"`
|
2015-03-03 02:18:38 +00:00
|
|
|
SerfLan int `json:"serf_lan,omitempty"`
|
|
|
|
SerfWan int `json:"serf_wan,omitempty"`
|
|
|
|
Server int `json:"server,omitempty"`
|
2017-03-23 21:48:45 +00:00
|
|
|
|
|
|
|
// Deprecated
|
|
|
|
RPC int `json:"rpc,omitempty"`
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestAddressConfig contains the bind addresses for various
|
|
|
|
// components of the Consul server.
|
2015-03-03 02:18:38 +00:00
|
|
|
type TestAddressConfig struct {
|
|
|
|
HTTP string `json:"http,omitempty"`
|
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestServerConfig is the main server configuration struct.
|
2015-03-03 02:18:38 +00:00
|
|
|
type TestServerConfig struct {
|
2017-04-28 23:15:55 +00:00
|
|
|
NodeName string `json:"node_name"`
|
|
|
|
NodeID string `json:"node_id"`
|
|
|
|
NodeMeta map[string]string `json:"node_meta,omitempty"`
|
|
|
|
Performance *TestPerformanceConfig `json:"performance,omitempty"`
|
|
|
|
Bootstrap bool `json:"bootstrap,omitempty"`
|
|
|
|
Server bool `json:"server,omitempty"`
|
|
|
|
DataDir string `json:"data_dir,omitempty"`
|
|
|
|
Datacenter string `json:"datacenter,omitempty"`
|
|
|
|
DisableCheckpoint bool `json:"disable_update_check"`
|
|
|
|
LogLevel string `json:"log_level,omitempty"`
|
|
|
|
Bind string `json:"bind_addr,omitempty"`
|
|
|
|
Addresses *TestAddressConfig `json:"addresses,omitempty"`
|
|
|
|
Ports *TestPortConfig `json:"ports,omitempty"`
|
|
|
|
RaftProtocol int `json:"raft_protocol,omitempty"`
|
|
|
|
ACLMasterToken string `json:"acl_master_token,omitempty"`
|
|
|
|
ACLDatacenter string `json:"acl_datacenter,omitempty"`
|
|
|
|
ACLDefaultPolicy string `json:"acl_default_policy,omitempty"`
|
|
|
|
ACLEnforceVersion8 bool `json:"acl_enforce_version_8"`
|
|
|
|
Encrypt string `json:"encrypt,omitempty"`
|
|
|
|
CAFile string `json:"ca_file,omitempty"`
|
|
|
|
CertFile string `json:"cert_file,omitempty"`
|
|
|
|
KeyFile string `json:"key_file,omitempty"`
|
|
|
|
VerifyIncoming bool `json:"verify_incoming,omitempty"`
|
|
|
|
VerifyIncomingRPC bool `json:"verify_incoming_rpc,omitempty"`
|
|
|
|
VerifyIncomingHTTPS bool `json:"verify_incoming_https,omitempty"`
|
|
|
|
VerifyOutgoing bool `json:"verify_outgoing,omitempty"`
|
|
|
|
Stdout, Stderr io.Writer `json:"-"`
|
|
|
|
Args []string `json:"-"`
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// ServerConfigCallback is a function interface which can be
|
|
|
|
// passed to NewTestServerConfig to modify the server config.
|
2015-03-03 02:18:38 +00:00
|
|
|
type ServerConfigCallback func(c *TestServerConfig)
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// defaultServerConfig returns a new TestServerConfig struct
|
|
|
|
// with all of the listen ports incremented by one.
|
2015-03-03 02:18:38 +00:00
|
|
|
func defaultServerConfig() *TestServerConfig {
|
2017-03-27 08:28:54 +00:00
|
|
|
nodeID, err := uuid.GenerateUUID()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2015-03-03 02:18:38 +00:00
|
|
|
return &TestServerConfig{
|
Use a random port instead of idx in testutil
The testutil server uses an atomic incrementer to generate unique port
numbers. This works great until tests are run in parallel, _across
packages_. Because each package starts at the same "offset" idx, they
collide.
One way to overcome this is to run each packages' test in isolation, but
that makes the test suite much longer as it does not maximize
parallelization. Alternatively, instead of having "predictable" ports,
we can let the OS choose a random open port automatically.
This still has a (albeit smaller) race condition in that the OS could
return an open port twice, before the server has a chance to actually
start and occupy said port. In practice, I have not been able to hit
this race condition, so it either doesn't happen or it happens far less
frequently that the existing implementation.
I'm not sure how I feel about the panic, but this is just test code, so
I'm including to say it's okay?
2016-12-01 15:24:26 +00:00
|
|
|
NodeName: fmt.Sprintf("node%d", randomPort()),
|
2017-03-27 08:28:54 +00:00
|
|
|
NodeID: nodeID,
|
2015-03-24 02:27:59 +00:00
|
|
|
DisableCheckpoint: true,
|
2016-08-25 00:33:53 +00:00
|
|
|
Performance: &TestPerformanceConfig{
|
|
|
|
RaftMultiplier: 1,
|
|
|
|
},
|
|
|
|
Bootstrap: true,
|
|
|
|
Server: true,
|
|
|
|
LogLevel: "debug",
|
|
|
|
Bind: "127.0.0.1",
|
|
|
|
Addresses: &TestAddressConfig{},
|
2015-03-03 02:18:38 +00:00
|
|
|
Ports: &TestPortConfig{
|
Use a random port instead of idx in testutil
The testutil server uses an atomic incrementer to generate unique port
numbers. This works great until tests are run in parallel, _across
packages_. Because each package starts at the same "offset" idx, they
collide.
One way to overcome this is to run each packages' test in isolation, but
that makes the test suite much longer as it does not maximize
parallelization. Alternatively, instead of having "predictable" ports,
we can let the OS choose a random open port automatically.
This still has a (albeit smaller) race condition in that the OS could
return an open port twice, before the server has a chance to actually
start and occupy said port. In practice, I have not been able to hit
this race condition, so it either doesn't happen or it happens far less
frequently that the existing implementation.
I'm not sure how I feel about the panic, but this is just test code, so
I'm including to say it's okay?
2016-12-01 15:24:26 +00:00
|
|
|
DNS: randomPort(),
|
|
|
|
HTTP: randomPort(),
|
2017-04-14 20:37:29 +00:00
|
|
|
HTTPS: randomPort(),
|
Use a random port instead of idx in testutil
The testutil server uses an atomic incrementer to generate unique port
numbers. This works great until tests are run in parallel, _across
packages_. Because each package starts at the same "offset" idx, they
collide.
One way to overcome this is to run each packages' test in isolation, but
that makes the test suite much longer as it does not maximize
parallelization. Alternatively, instead of having "predictable" ports,
we can let the OS choose a random open port automatically.
This still has a (albeit smaller) race condition in that the OS could
return an open port twice, before the server has a chance to actually
start and occupy said port. In practice, I have not been able to hit
this race condition, so it either doesn't happen or it happens far less
frequently that the existing implementation.
I'm not sure how I feel about the panic, but this is just test code, so
I'm including to say it's okay?
2016-12-01 15:24:26 +00:00
|
|
|
SerfLan: randomPort(),
|
|
|
|
SerfWan: randomPort(),
|
|
|
|
Server: randomPort(),
|
2017-03-23 21:48:45 +00:00
|
|
|
RPC: randomPort(),
|
2015-03-03 02:18:38 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Use a random port instead of idx in testutil
The testutil server uses an atomic incrementer to generate unique port
numbers. This works great until tests are run in parallel, _across
packages_. Because each package starts at the same "offset" idx, they
collide.
One way to overcome this is to run each packages' test in isolation, but
that makes the test suite much longer as it does not maximize
parallelization. Alternatively, instead of having "predictable" ports,
we can let the OS choose a random open port automatically.
This still has a (albeit smaller) race condition in that the OS could
return an open port twice, before the server has a chance to actually
start and occupy said port. In practice, I have not been able to hit
this race condition, so it either doesn't happen or it happens far less
frequently that the existing implementation.
I'm not sure how I feel about the panic, but this is just test code, so
I'm including to say it's okay?
2016-12-01 15:24:26 +00:00
|
|
|
// randomPort asks the kernel for a random port to use.
|
|
|
|
func randomPort() int {
|
|
|
|
l, err := net.Listen("tcp", "127.0.0.1:0")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
defer l.Close()
|
|
|
|
return l.Addr().(*net.TCPAddr).Port
|
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestService is used to serialize a service definition.
|
2015-03-11 01:47:45 +00:00
|
|
|
type TestService struct {
|
|
|
|
ID string `json:",omitempty"`
|
|
|
|
Name string `json:",omitempty"`
|
|
|
|
Tags []string `json:",omitempty"`
|
|
|
|
Address string `json:",omitempty"`
|
|
|
|
Port int `json:",omitempty"`
|
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestCheck is used to serialize a check definition.
|
2015-03-11 01:47:45 +00:00
|
|
|
type TestCheck struct {
|
|
|
|
ID string `json:",omitempty"`
|
|
|
|
Name string `json:",omitempty"`
|
|
|
|
ServiceID string `json:",omitempty"`
|
|
|
|
TTL string `json:",omitempty"`
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:10:07 +00:00
|
|
|
// TestKVResponse is what we use to decode KV data.
|
|
|
|
type TestKVResponse struct {
|
|
|
|
Value string
|
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// TestServer is the main server wrapper struct.
|
2015-03-03 02:18:38 +00:00
|
|
|
type TestServer struct {
|
2016-04-15 12:35:45 +00:00
|
|
|
cmd *exec.Cmd
|
2015-03-11 18:08:08 +00:00
|
|
|
Config *TestServerConfig
|
|
|
|
|
2017-04-14 20:37:29 +00:00
|
|
|
HTTPAddr string
|
|
|
|
HTTPSAddr string
|
|
|
|
LANAddr string
|
|
|
|
WANAddr string
|
2015-03-20 00:44:04 +00:00
|
|
|
|
2017-04-21 00:02:42 +00:00
|
|
|
HTTPClient *http.Client
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// NewTestServer is an easy helper method to create a new Consul
|
|
|
|
// test server with the most basic configuration.
|
2017-03-23 20:26:05 +00:00
|
|
|
func NewTestServer() (*TestServer, error) {
|
|
|
|
return NewTestServerConfig(nil)
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2017-03-23 20:26:05 +00:00
|
|
|
// NewTestServerConfig creates a new TestServer, and makes a call to an optional
|
|
|
|
// callback function to modify the configuration. If there is an error
|
|
|
|
// configuring or starting the server, the server will NOT be running when the
|
|
|
|
// function returns (thus you do not need to stop it).
|
|
|
|
func NewTestServerConfig(cb ServerConfigCallback) (*TestServer, error) {
|
2015-03-03 02:18:38 +00:00
|
|
|
if path, err := exec.LookPath("consul"); err != nil || path == "" {
|
2017-03-23 20:26:05 +00:00
|
|
|
return nil, fmt.Errorf("consul not found on $PATH - download and install " +
|
2017-01-18 00:20:29 +00:00
|
|
|
"consul or skip this test")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dataDir, err := ioutil.TempDir("", "consul")
|
|
|
|
if err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return nil, errors.Wrap(err, "failed creating tempdir")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-05-09 01:11:17 +00:00
|
|
|
configFile, err := ioutil.TempFile(dataDir, "config")
|
2015-03-03 02:18:38 +00:00
|
|
|
if err != nil {
|
2015-03-24 02:27:59 +00:00
|
|
|
defer os.RemoveAll(dataDir)
|
2017-03-23 20:26:05 +00:00
|
|
|
return nil, errors.Wrap(err, "failed creating temp config")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
consulConfig := defaultServerConfig()
|
|
|
|
consulConfig.DataDir = dataDir
|
|
|
|
|
|
|
|
if cb != nil {
|
|
|
|
cb(consulConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
configContent, err := json.Marshal(consulConfig)
|
|
|
|
if err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return nil, errors.Wrap(err, "failed marshaling json")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := configFile.Write(configContent); err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
defer configFile.Close()
|
|
|
|
defer os.RemoveAll(dataDir)
|
|
|
|
return nil, errors.Wrap(err, "failed writing config content")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
configFile.Close()
|
|
|
|
|
2015-06-08 14:16:11 +00:00
|
|
|
stdout := io.Writer(os.Stdout)
|
|
|
|
if consulConfig.Stdout != nil {
|
|
|
|
stdout = consulConfig.Stdout
|
|
|
|
}
|
|
|
|
|
|
|
|
stderr := io.Writer(os.Stderr)
|
|
|
|
if consulConfig.Stderr != nil {
|
|
|
|
stderr = consulConfig.Stderr
|
|
|
|
}
|
|
|
|
|
2015-03-03 02:18:38 +00:00
|
|
|
// Start the server
|
2016-11-30 18:29:42 +00:00
|
|
|
args := []string{"agent", "-config-file", configFile.Name()}
|
|
|
|
args = append(args, consulConfig.Args...)
|
|
|
|
cmd := exec.Command("consul", args...)
|
2015-06-08 14:16:11 +00:00
|
|
|
cmd.Stdout = stdout
|
|
|
|
cmd.Stderr = stderr
|
2015-03-03 02:18:38 +00:00
|
|
|
if err := cmd.Start(); err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return nil, errors.Wrap(err, "failed starting command")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 00:44:04 +00:00
|
|
|
var httpAddr string
|
|
|
|
var client *http.Client
|
|
|
|
if strings.HasPrefix(consulConfig.Addresses.HTTP, "unix://") {
|
|
|
|
httpAddr = consulConfig.Addresses.HTTP
|
2015-10-22 14:47:50 +00:00
|
|
|
trans := cleanhttp.DefaultTransport()
|
2017-02-11 02:11:21 +00:00
|
|
|
trans.DialContext = func(_ context.Context, _, _ string) (net.Conn, error) {
|
2015-10-22 14:47:50 +00:00
|
|
|
return net.Dial("unix", httpAddr[7:])
|
|
|
|
}
|
2015-03-20 00:44:04 +00:00
|
|
|
client = &http.Client{
|
2015-10-22 14:47:50 +00:00
|
|
|
Transport: trans,
|
2015-03-20 00:44:04 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
httpAddr = fmt.Sprintf("127.0.0.1:%d", consulConfig.Ports.HTTP)
|
2015-10-22 14:47:50 +00:00
|
|
|
client = cleanhttp.DefaultClient()
|
2015-03-20 00:44:04 +00:00
|
|
|
}
|
|
|
|
|
2015-03-03 02:18:38 +00:00
|
|
|
server := &TestServer{
|
2015-03-11 18:08:08 +00:00
|
|
|
Config: consulConfig,
|
2016-04-15 12:35:45 +00:00
|
|
|
cmd: cmd,
|
2015-03-11 18:08:08 +00:00
|
|
|
|
2017-04-14 20:37:29 +00:00
|
|
|
HTTPAddr: httpAddr,
|
|
|
|
HTTPSAddr: fmt.Sprintf("127.0.0.1:%d", consulConfig.Ports.HTTPS),
|
|
|
|
LANAddr: fmt.Sprintf("127.0.0.1:%d", consulConfig.Ports.SerfLan),
|
|
|
|
WANAddr: fmt.Sprintf("127.0.0.1:%d", consulConfig.Ports.SerfWan),
|
2015-03-20 00:44:04 +00:00
|
|
|
|
2017-04-21 00:02:42 +00:00
|
|
|
HTTPClient: client,
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-03 03:20:13 +00:00
|
|
|
// Wait for the server to be ready
|
2017-03-23 20:26:05 +00:00
|
|
|
var startErr error
|
2015-05-09 01:16:35 +00:00
|
|
|
if consulConfig.Bootstrap {
|
2017-03-23 20:26:05 +00:00
|
|
|
startErr = server.waitForLeader()
|
2015-05-09 01:16:35 +00:00
|
|
|
} else {
|
2017-03-23 20:26:05 +00:00
|
|
|
startErr = server.waitForAPI()
|
|
|
|
}
|
|
|
|
if startErr != nil {
|
|
|
|
defer server.Stop()
|
2017-03-24 04:40:14 +00:00
|
|
|
return nil, errors.Wrap(startErr, "failed waiting for server to start")
|
2015-05-09 01:11:17 +00:00
|
|
|
}
|
2015-03-03 03:20:13 +00:00
|
|
|
|
2017-03-23 20:26:05 +00:00
|
|
|
return server, nil
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// Stop stops the test Consul server, and removes the Consul data
|
|
|
|
// directory once we are done.
|
2017-03-23 20:26:05 +00:00
|
|
|
func (s *TestServer) Stop() error {
|
2015-03-03 03:20:13 +00:00
|
|
|
defer os.RemoveAll(s.Config.DataDir)
|
2015-03-03 02:18:38 +00:00
|
|
|
|
2017-03-23 20:26:05 +00:00
|
|
|
if s.cmd != nil {
|
|
|
|
if s.cmd.Process != nil {
|
|
|
|
if err := s.cmd.Process.Kill(); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to kill consul server")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait for the process to exit to be sure that the data dir can be
|
|
|
|
// deleted on all platforms.
|
|
|
|
return s.cmd.Wait()
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
2016-04-15 12:35:45 +00:00
|
|
|
|
2017-03-23 20:26:05 +00:00
|
|
|
// There was no process
|
|
|
|
return nil
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
|
|
|
|
2015-05-09 01:11:17 +00:00
|
|
|
// waitForAPI waits for only the agent HTTP endpoint to start
|
|
|
|
// responding. This is an indication that the agent has started,
|
|
|
|
// but will likely return before a leader is elected.
|
2017-03-23 20:26:05 +00:00
|
|
|
func (s *TestServer) waitForAPI() error {
|
|
|
|
if err := WaitForResult(func() (bool, error) {
|
2017-04-21 00:02:42 +00:00
|
|
|
resp, err := s.HTTPClient.Get(s.url("/v1/agent/self"))
|
2015-05-09 01:11:17 +00:00
|
|
|
if err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, errors.Wrap(err, "failed http get")
|
2015-05-09 01:11:17 +00:00
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if err := s.requireOK(resp); err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, errors.Wrap(err, "failed OK response")
|
2015-05-09 01:11:17 +00:00
|
|
|
}
|
|
|
|
return true, nil
|
2017-03-23 20:26:05 +00:00
|
|
|
}); err != nil {
|
|
|
|
return errors.Wrap(err, "failed waiting for API")
|
|
|
|
}
|
|
|
|
return nil
|
2015-05-09 01:11:17 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 04:53:51 +00:00
|
|
|
// waitForLeader waits for the Consul server's HTTP API to become
|
|
|
|
// available, and then waits for a known leader and an index of
|
|
|
|
// 1 or more to be observed to confirm leader election is done.
|
2017-01-27 00:55:49 +00:00
|
|
|
// It then waits to ensure the anti-entropy sync has completed.
|
2017-03-23 20:26:05 +00:00
|
|
|
func (s *TestServer) waitForLeader() error {
|
2017-01-27 00:55:49 +00:00
|
|
|
var index int64
|
2017-03-23 20:26:05 +00:00
|
|
|
if err := WaitForResult(func() (bool, error) {
|
2017-01-27 00:55:49 +00:00
|
|
|
// Query the API and check the status code.
|
2017-02-06 19:52:00 +00:00
|
|
|
url := s.url(fmt.Sprintf("/v1/catalog/nodes?index=%d&wait=2s", index))
|
2017-04-21 00:02:42 +00:00
|
|
|
resp, err := s.HTTPClient.Get(url)
|
2015-03-24 02:27:59 +00:00
|
|
|
if err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, errors.Wrap(err, "failed http get")
|
2015-03-24 02:27:59 +00:00
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if err := s.requireOK(resp); err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, errors.Wrap(err, "failed OK response")
|
2015-03-24 02:27:59 +00:00
|
|
|
}
|
2015-03-03 02:18:38 +00:00
|
|
|
|
2017-01-27 00:55:49 +00:00
|
|
|
// Ensure we have a leader and a node registration.
|
2015-03-03 02:18:38 +00:00
|
|
|
if leader := resp.Header.Get("X-Consul-KnownLeader"); leader != "true" {
|
|
|
|
return false, fmt.Errorf("Consul leader status: %#v", leader)
|
|
|
|
}
|
2017-01-27 00:55:49 +00:00
|
|
|
index, err = strconv.ParseInt(resp.Header.Get("X-Consul-Index"), 10, 64)
|
|
|
|
if err != nil {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, errors.Wrap(err, "bad consul index")
|
2017-01-27 00:55:49 +00:00
|
|
|
}
|
|
|
|
if index == 0 {
|
2017-03-23 20:26:05 +00:00
|
|
|
return false, fmt.Errorf("consul index is 0")
|
2015-03-03 02:18:38 +00:00
|
|
|
}
|
2017-01-26 05:13:03 +00:00
|
|
|
|
2017-01-27 00:55:49 +00:00
|
|
|
// Watch for the anti-entropy sync to finish.
|
2017-01-26 05:13:03 +00:00
|
|
|
var parsed []map[string]interface{}
|
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
if err := dec.Decode(&parsed); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if len(parsed) < 1 {
|
|
|
|
return false, fmt.Errorf("No nodes")
|
|
|
|
}
|
|
|
|
taggedAddresses, ok := parsed[0]["TaggedAddresses"].(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return false, fmt.Errorf("Missing tagged addresses")
|
|
|
|
}
|
|
|
|
if _, ok := taggedAddresses["lan"]; !ok {
|
|
|
|
return false, fmt.Errorf("No lan tagged addresses")
|
|
|
|
}
|
2015-03-03 02:18:38 +00:00
|
|
|
return true, nil
|
2017-03-23 20:26:05 +00:00
|
|
|
}); err != nil {
|
|
|
|
return errors.Wrap(err, "failed waiting for leader")
|
2015-03-11 01:08:14 +00:00
|
|
|
}
|
2015-03-24 02:27:59 +00:00
|
|
|
return nil
|
2015-03-11 01:08:14 +00:00
|
|
|
}
|