2015-01-06 10:40:00 -08:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2016-11-30 13:29:42 -05:00
|
|
|
"io/ioutil"
|
2017-09-25 18:06:49 -05:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2015-01-21 11:08:57 -08:00
|
|
|
"strings"
|
2015-01-06 10:40:00 -08:00
|
|
|
"testing"
|
2016-11-16 16:45:26 -05:00
|
|
|
"time"
|
2016-11-30 13:29:42 -05:00
|
|
|
|
|
|
|
"github.com/hashicorp/consul/testutil"
|
|
|
|
"github.com/hashicorp/serf/serf"
|
2015-01-06 10:40:00 -08:00
|
|
|
)
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentSelf(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
info, err := agent.Self()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-09-27 20:47:40 +02:00
|
|
|
name := info["Config"]["NodeName"].(string)
|
2015-01-06 10:40:00 -08:00
|
|
|
if name == "" {
|
|
|
|
t.Fatalf("bad: %v", info)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-08 01:31:38 -07:00
|
|
|
func TestAPI_AgentMetrics(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t)
|
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
metrics, err := agent.Metrics()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-09-25 15:27:04 -07:00
|
|
|
var found bool
|
|
|
|
for _, g := range metrics.Gauges {
|
|
|
|
if g.Name == "consul.runtime.alloc_bytes" {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
2017-08-08 01:31:38 -07:00
|
|
|
}
|
2017-09-25 15:27:04 -07:00
|
|
|
if !found {
|
|
|
|
t.Fatalf("missing runtime metrics")
|
2017-08-08 01:31:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentReload(t *testing.T) {
|
2016-11-30 13:29:42 -05:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
// Create our initial empty config file, to be overwritten later
|
2017-09-25 17:26:49 -05:00
|
|
|
cfgDir := testutil.TempDir(t, "consul-config")
|
|
|
|
defer os.RemoveAll(cfgDir)
|
|
|
|
|
|
|
|
cfgFilePath := filepath.Join(cfgDir, "reload.json")
|
|
|
|
configFile, err := os.Create(cfgFilePath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to create file %v, got error:%v", cfgFilePath, err)
|
2016-11-30 13:29:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
c, s := makeClientWithConfig(t, nil, func(conf *testutil.TestServerConfig) {
|
|
|
|
conf.Args = []string{"-config-file", configFile.Name()}
|
|
|
|
})
|
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// Update the config file with a service definition
|
|
|
|
config := `{"service":{"name":"redis", "port":1234}}`
|
2017-09-25 17:26:49 -05:00
|
|
|
err = ioutil.WriteFile(configFile.Name(), []byte(config), 0644)
|
2016-11-30 13:29:42 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = agent.Reload(); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
service, ok := services["redis"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
|
|
|
}
|
|
|
|
if service.Port != 1234 {
|
|
|
|
t.Fatalf("bad: %v", service.Port)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-31 17:39:46 -07:00
|
|
|
func TestAPI_AgentMembersOpts(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
c, s1 := makeClient(t)
|
|
|
|
_, s2 := makeClientWithConfig(t, nil, func(c *testutil.TestServerConfig) {
|
|
|
|
c.Datacenter = "dc2"
|
|
|
|
})
|
|
|
|
defer s1.Stop()
|
|
|
|
defer s2.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
s2.JoinWAN(t, s1.WANAddr)
|
|
|
|
|
|
|
|
members, err := agent.MembersOpts(MembersOpts{WAN: true})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(members) != 2 {
|
|
|
|
t.Fatalf("bad: %v", members)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentMembers(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
members, err := agent.Members(false)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(members) != 1 {
|
|
|
|
t.Fatalf("bad: %v", members)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentServices(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg := &AgentServiceRegistration{
|
2015-01-22 13:42:22 -08:00
|
|
|
Name: "foo",
|
|
|
|
Tags: []string{"bar", "baz"},
|
|
|
|
Port: 8000,
|
2015-01-06 10:40:00 -08:00
|
|
|
Check: &AgentServiceCheck{
|
|
|
|
TTL: "15s",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-04-12 00:53:48 +00:00
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, ok := services["foo"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
chk, ok := checks["service:foo"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checks should default to critical
|
2016-04-23 16:01:59 -07:00
|
|
|
if chk.Status != HealthCritical {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("Bad: %#v", chk)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := agent.ServiceDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentServices_CheckPassing(t *testing.T) {
|
2015-05-11 18:53:09 -07:00
|
|
|
t.Parallel()
|
2015-04-12 00:53:48 +00:00
|
|
|
c, s := makeClient(t)
|
2015-05-11 18:53:09 -07:00
|
|
|
defer s.Stop()
|
2015-04-12 00:53:48 +00:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
reg := &AgentServiceRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
Tags: []string{"bar", "baz"},
|
|
|
|
Port: 8000,
|
|
|
|
Check: &AgentServiceCheck{
|
|
|
|
TTL: "15s",
|
2016-04-23 16:01:59 -07:00
|
|
|
Status: HealthPassing,
|
2015-04-12 00:53:48 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, ok := services["foo"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2015-04-12 00:53:48 +00:00
|
|
|
chk, ok := checks["service:foo"]
|
|
|
|
if !ok {
|
2015-01-06 10:40:00 -08:00
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
|
2016-04-23 16:01:59 -07:00
|
|
|
if chk.Status != HealthPassing {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("Bad: %#v", chk)
|
|
|
|
}
|
2015-01-06 10:40:00 -08:00
|
|
|
if err := agent.ServiceDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentServices_CheckBadStatus(t *testing.T) {
|
2015-05-11 18:53:09 -07:00
|
|
|
t.Parallel()
|
2015-04-12 00:53:48 +00:00
|
|
|
c, s := makeClient(t)
|
2015-05-11 18:53:09 -07:00
|
|
|
defer s.Stop()
|
2015-04-12 00:53:48 +00:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
reg := &AgentServiceRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
Tags: []string{"bar", "baz"},
|
|
|
|
Port: 8000,
|
|
|
|
Check: &AgentServiceCheck{
|
|
|
|
TTL: "15s",
|
|
|
|
Status: "fluffy",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg); err == nil {
|
|
|
|
t.Fatalf("bad status accepted")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentServiceAddress(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-22 11:50:20 +01:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-22 11:50:20 +01:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg1 := &AgentServiceRegistration{
|
|
|
|
Name: "foo1",
|
|
|
|
Port: 8000,
|
|
|
|
Address: "192.168.0.42",
|
|
|
|
}
|
|
|
|
reg2 := &AgentServiceRegistration{
|
2015-01-22 13:42:22 -08:00
|
|
|
Name: "foo2",
|
|
|
|
Port: 8000,
|
2015-01-22 11:50:20 +01:00
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg1); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg2); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2015-01-22 13:42:22 -08:00
|
|
|
|
2015-01-22 11:50:20 +01:00
|
|
|
if _, ok := services["foo1"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
if _, ok := services["foo2"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
|
|
|
|
if services["foo1"].Address != "192.168.0.42" {
|
|
|
|
t.Fatalf("missing Address field in service foo1: %v", services)
|
|
|
|
}
|
|
|
|
if services["foo2"].Address != "" {
|
|
|
|
t.Fatalf("missing Address field in service foo2: %v", services)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := agent.ServiceDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentEnableTagOverride(t *testing.T) {
|
2016-02-16 11:45:29 -08:00
|
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t)
|
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg1 := &AgentServiceRegistration{
|
|
|
|
Name: "foo1",
|
|
|
|
Port: 8000,
|
|
|
|
Address: "192.168.0.42",
|
|
|
|
EnableTagOverride: true,
|
|
|
|
}
|
|
|
|
reg2 := &AgentServiceRegistration{
|
|
|
|
Name: "foo2",
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg1); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg2); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := services["foo1"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
if services["foo1"].EnableTagOverride != true {
|
|
|
|
t.Fatalf("tag override not set on service foo1: %v", services)
|
|
|
|
}
|
|
|
|
if _, ok := services["foo2"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
if services["foo2"].EnableTagOverride != false {
|
|
|
|
t.Fatalf("tag override set on service foo2: %v", services)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentServices_MultipleChecks(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-13 19:18:46 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-13 19:18:46 -08:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg := &AgentServiceRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
Tags: []string{"bar", "baz"},
|
|
|
|
Port: 8000,
|
|
|
|
Checks: AgentServiceChecks{
|
|
|
|
&AgentServiceCheck{
|
|
|
|
TTL: "15s",
|
|
|
|
},
|
|
|
|
&AgentServiceCheck{
|
|
|
|
TTL: "30s",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
services, err := agent.Services()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, ok := services["foo"]; !ok {
|
|
|
|
t.Fatalf("missing service: %v", services)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, ok := checks["service:foo:1"]; !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
if _, ok := checks["service:foo:2"]; !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentSetTTLStatus(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg := &AgentServiceRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
Check: &AgentServiceCheck{
|
|
|
|
TTL: "15s",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2016-03-04 15:18:25 -08:00
|
|
|
verify := func(status, output string) {
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
chk, ok := checks["service:foo"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
if chk.Status != status {
|
|
|
|
t.Fatalf("Bad: %#v", chk)
|
|
|
|
}
|
|
|
|
if chk.Output != output {
|
|
|
|
t.Fatalf("Bad: %#v", chk)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := agent.WarnTTL("service:foo", "foo"); err != nil {
|
2015-01-06 10:40:00 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthWarning, "foo")
|
2015-01-06 10:40:00 -08:00
|
|
|
|
2016-03-04 15:18:25 -08:00
|
|
|
if err := agent.PassTTL("service:foo", "bar"); err != nil {
|
2015-01-06 10:40:00 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthPassing, "bar")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
|
|
|
if err := agent.FailTTL("service:foo", "baz"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthCritical, "baz")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
|
|
|
if err := agent.UpdateTTL("service:foo", "foo", "warn"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthWarning, "foo")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
|
|
|
if err := agent.UpdateTTL("service:foo", "bar", "pass"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthPassing, "bar")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
|
|
|
if err := agent.UpdateTTL("service:foo", "baz", "fail"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthCritical, "baz")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
2016-04-23 16:01:59 -07:00
|
|
|
if err := agent.UpdateTTL("service:foo", "foo", HealthWarning); err != nil {
|
2016-03-04 15:18:25 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthWarning, "foo")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
2016-04-23 16:01:59 -07:00
|
|
|
if err := agent.UpdateTTL("service:foo", "bar", HealthPassing); err != nil {
|
2016-03-04 15:18:25 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthPassing, "bar")
|
2016-03-04 15:18:25 -08:00
|
|
|
|
2016-04-23 16:01:59 -07:00
|
|
|
if err := agent.UpdateTTL("service:foo", "baz", HealthCritical); err != nil {
|
2016-03-04 15:18:25 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
verify(HealthCritical, "baz")
|
2015-01-06 10:40:00 -08:00
|
|
|
|
|
|
|
if err := agent.ServiceDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentChecks(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg := &AgentCheckRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
}
|
|
|
|
reg.TTL = "15s"
|
|
|
|
if err := agent.CheckRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2015-04-12 00:53:48 +00:00
|
|
|
chk, ok := checks["foo"]
|
|
|
|
if !ok {
|
2015-01-06 10:40:00 -08:00
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
if chk.Status != HealthCritical {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("check not critical: %v", chk)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := agent.CheckDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentCheckStartPassing(t *testing.T) {
|
2015-05-11 18:53:09 -07:00
|
|
|
t.Parallel()
|
2015-04-12 00:53:48 +00:00
|
|
|
c, s := makeClient(t)
|
2015-05-11 18:53:09 -07:00
|
|
|
defer s.Stop()
|
2015-04-12 00:53:48 +00:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
reg := &AgentCheckRegistration{
|
|
|
|
Name: "foo",
|
|
|
|
AgentServiceCheck: AgentServiceCheck{
|
2016-04-23 16:01:59 -07:00
|
|
|
Status: HealthPassing,
|
2015-04-12 00:53:48 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
reg.TTL = "15s"
|
|
|
|
if err := agent.CheckRegister(reg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
chk, ok := checks["foo"]
|
|
|
|
if !ok {
|
2015-01-06 10:40:00 -08:00
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
2016-04-23 16:01:59 -07:00
|
|
|
if chk.Status != HealthPassing {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("check not passing: %v", chk)
|
|
|
|
}
|
2015-01-06 10:40:00 -08:00
|
|
|
|
|
|
|
if err := agent.CheckDeregister("foo"); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentChecks_serviceBound(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-13 19:18:46 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-13 19:18:46 -08:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// First register a service
|
|
|
|
serviceReg := &AgentServiceRegistration{
|
|
|
|
Name: "redis",
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(serviceReg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a check bound to the service
|
|
|
|
reg := &AgentCheckRegistration{
|
|
|
|
Name: "redischeck",
|
|
|
|
ServiceID: "redis",
|
|
|
|
}
|
|
|
|
reg.TTL = "15s"
|
2016-08-16 00:05:55 -07:00
|
|
|
reg.DeregisterCriticalServiceAfter = "nope"
|
|
|
|
err := agent.CheckRegister(reg)
|
|
|
|
if err == nil || !strings.Contains(err.Error(), "invalid duration") {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
reg.DeregisterCriticalServiceAfter = "90m"
|
2015-01-13 19:18:46 -08:00
|
|
|
if err := agent.CheckRegister(reg); err != nil {
|
2015-11-18 07:40:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
check, ok := checks["redischeck"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
if check.ServiceID != "redis" {
|
|
|
|
t.Fatalf("missing service association for check: %v", check)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentChecks_Docker(t *testing.T) {
|
2015-11-18 07:40:02 -08:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
c, s := makeClientWithConfig(t, nil, func(c *testutil.TestServerConfig) {
|
|
|
|
c.EnableScriptChecks = true
|
|
|
|
})
|
2015-11-18 07:40:02 -08:00
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// First register a service
|
|
|
|
serviceReg := &AgentServiceRegistration{
|
|
|
|
Name: "redis",
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(serviceReg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a check bound to the service
|
|
|
|
reg := &AgentCheckRegistration{
|
|
|
|
Name: "redischeck",
|
|
|
|
ServiceID: "redis",
|
|
|
|
AgentServiceCheck: AgentServiceCheck{
|
|
|
|
DockerContainerID: "f972c95ebf0e",
|
|
|
|
Script: "/bin/true",
|
|
|
|
Shell: "/bin/bash",
|
|
|
|
Interval: "10s",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := agent.CheckRegister(reg); err != nil {
|
2015-01-13 19:18:46 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
check, ok := checks["redischeck"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check: %v", checks)
|
|
|
|
}
|
|
|
|
if check.ServiceID != "redis" {
|
|
|
|
t.Fatalf("missing service association for check: %v", check)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentJoin(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
info, err := agent.Self()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Join ourself
|
2017-09-27 20:47:40 +02:00
|
|
|
addr := info["Config"]["SerfAdvertiseAddrLAN"].(string)
|
|
|
|
// strip off 'tcp://'
|
|
|
|
addr = addr[len("tcp://"):]
|
2015-01-06 10:40:00 -08:00
|
|
|
err = agent.Join(addr, false)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentLeave(t *testing.T) {
|
2016-11-30 13:29:42 -05:00
|
|
|
t.Parallel()
|
|
|
|
c1, s1 := makeClient(t)
|
|
|
|
defer s1.Stop()
|
|
|
|
|
|
|
|
c2, s2 := makeClientWithConfig(t, nil, func(conf *testutil.TestServerConfig) {
|
|
|
|
conf.Server = false
|
|
|
|
conf.Bootstrap = false
|
|
|
|
})
|
|
|
|
defer s2.Stop()
|
|
|
|
|
|
|
|
if err := c2.Agent().Join(s1.LANAddr, false); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-01-26 22:22:18 -08:00
|
|
|
// We sometimes see an EOF response to this one, depending on timing.
|
|
|
|
err := c2.Agent().Leave()
|
2017-03-22 09:56:53 -07:00
|
|
|
if err != nil && !strings.Contains(err.Error(), "EOF") {
|
2016-11-30 13:29:42 -05:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the second agent's status is 'Left'
|
|
|
|
members, err := c1.Agent().Members(false)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
member := members[0]
|
|
|
|
if member.Name == s1.Config.NodeName {
|
|
|
|
member = members[1]
|
|
|
|
}
|
|
|
|
if member.Status != int(serf.StatusLeft) {
|
|
|
|
t.Fatalf("bad: %v", *member)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentForceLeave(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-06 15:26:50 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-06 15:26:50 -08:00
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// Eject somebody
|
|
|
|
err := agent.ForceLeave("foo")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2015-01-21 10:51:26 -08:00
|
|
|
|
2017-06-30 23:05:02 +02:00
|
|
|
func TestAPI_AgentMonitor(t *testing.T) {
|
2016-11-16 16:45:26 -05:00
|
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t)
|
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
2016-11-28 16:13:49 -05:00
|
|
|
logCh, err := agent.Monitor("info", nil, nil)
|
2016-11-16 16:45:26 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the first log message and validate it
|
|
|
|
select {
|
|
|
|
case log := <-logCh:
|
2017-01-17 22:20:11 -08:00
|
|
|
if !strings.Contains(log, "[INFO]") {
|
2016-11-16 16:45:26 -05:00
|
|
|
t.Fatalf("bad: %q", log)
|
|
|
|
}
|
|
|
|
case <-time.After(10 * time.Second):
|
|
|
|
t.Fatalf("failed to get a log message")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 22:58:55 +02:00
|
|
|
func TestAPI_ServiceMaintenance(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-21 10:51:26 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-21 10:51:26 -08:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// First register a service
|
|
|
|
serviceReg := &AgentServiceRegistration{
|
|
|
|
Name: "redis",
|
|
|
|
}
|
|
|
|
if err := agent.ServiceRegister(serviceReg); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable maintenance mode
|
2015-01-21 13:02:47 -08:00
|
|
|
if err := agent.EnableServiceMaintenance("redis", "broken"); err != nil {
|
2015-01-21 10:51:26 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a critical check was added
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
found := false
|
|
|
|
for _, check := range checks {
|
2015-01-21 11:08:57 -08:00
|
|
|
if strings.Contains(check.CheckID, "maintenance") {
|
2015-01-21 10:51:26 -08:00
|
|
|
found = true
|
2016-04-23 16:01:59 -07:00
|
|
|
if check.Status != HealthCritical || check.Notes != "broken" {
|
2015-01-21 10:51:26 -08:00
|
|
|
t.Fatalf("bad: %#v", checks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatalf("bad: %#v", checks)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disable maintenance mode
|
|
|
|
if err := agent.DisableServiceMaintenance("redis"); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the critical health check was removed
|
|
|
|
checks, err = agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
for _, check := range checks {
|
2015-01-21 11:08:57 -08:00
|
|
|
if strings.Contains(check.CheckID, "maintenance") {
|
|
|
|
t.Fatalf("should have removed health check")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-30 22:58:55 +02:00
|
|
|
func TestAPI_NodeMaintenance(t *testing.T) {
|
2015-05-08 10:27:24 -07:00
|
|
|
t.Parallel()
|
2015-01-21 11:08:57 -08:00
|
|
|
c, s := makeClient(t)
|
2015-03-02 18:18:38 -08:00
|
|
|
defer s.Stop()
|
2015-01-21 11:08:57 -08:00
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
// Enable maintenance mode
|
2015-01-21 13:02:47 -08:00
|
|
|
if err := agent.EnableNodeMaintenance("broken"); err != nil {
|
2015-01-21 11:08:57 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that a critical check was added
|
|
|
|
checks, err := agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
found := false
|
|
|
|
for _, check := range checks {
|
|
|
|
if strings.Contains(check.CheckID, "maintenance") {
|
|
|
|
found = true
|
2016-04-23 16:01:59 -07:00
|
|
|
if check.Status != HealthCritical || check.Notes != "broken" {
|
2015-01-21 11:08:57 -08:00
|
|
|
t.Fatalf("bad: %#v", checks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatalf("bad: %#v", checks)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disable maintenance mode
|
|
|
|
if err := agent.DisableNodeMaintenance(); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the check was removed
|
|
|
|
checks, err = agent.Checks()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
for _, check := range checks {
|
|
|
|
if strings.Contains(check.CheckID, "maintenance") {
|
2015-01-21 10:51:26 -08:00
|
|
|
t.Fatalf("should have removed health check")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-07-26 11:03:43 -07:00
|
|
|
|
|
|
|
func TestAPI_AgentUpdateToken(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
c, s := makeACLClient(t)
|
|
|
|
defer s.Stop()
|
|
|
|
|
|
|
|
agent := c.Agent()
|
|
|
|
|
|
|
|
if _, err := agent.UpdateACLToken("root", nil); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := agent.UpdateACLAgentToken("root", nil); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := agent.UpdateACLAgentMasterToken("root", nil); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-08-03 15:39:31 -07:00
|
|
|
|
|
|
|
if _, err := agent.UpdateACLReplicationToken("root", nil); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-07-26 11:03:43 -07:00
|
|
|
}
|