2013-12-23 13:52:10 -08:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
2014-11-24 00:36:03 -08:00
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
2013-12-23 13:52:10 -08:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2015-06-05 13:44:42 +02:00
|
|
|
"net"
|
2013-12-23 13:52:10 -08:00
|
|
|
"os"
|
2014-09-17 22:31:32 -07:00
|
|
|
"path/filepath"
|
2014-11-24 19:24:32 -08:00
|
|
|
"reflect"
|
2017-05-12 12:13:18 +02:00
|
|
|
"runtime"
|
2017-01-23 21:11:13 -05:00
|
|
|
"strings"
|
2013-12-23 13:52:10 -08:00
|
|
|
"testing"
|
|
|
|
"time"
|
2014-11-19 11:51:25 -08:00
|
|
|
|
pkg refactor
command/agent/* -> agent/*
command/consul/* -> agent/consul/*
command/agent/command{,_test}.go -> command/agent{,_test}.go
command/base/command.go -> command/base.go
command/base/* -> command/*
commands.go -> command/commands.go
The script which did the refactor is:
(
cd $GOPATH/src/github.com/hashicorp/consul
git mv command/agent/command.go command/agent.go
git mv command/agent/command_test.go command/agent_test.go
git mv command/agent/flag_slice_value{,_test}.go command/
git mv command/agent .
git mv command/base/command.go command/base.go
git mv command/base/config_util{,_test}.go command/
git mv commands.go command/
git mv consul agent
rmdir command/base/
gsed -i -e 's|package agent|package command|' command/agent{,_test}.go
gsed -i -e 's|package agent|package command|' command/flag_slice_value{,_test}.go
gsed -i -e 's|package base|package command|' command/base.go command/config_util{,_test}.go
gsed -i -e 's|package main|package command|' command/commands.go
gsed -i -e 's|base.Command|BaseCommand|' command/commands.go
gsed -i -e 's|agent.Command|AgentCommand|' command/commands.go
gsed -i -e 's|\tCommand:|\tBaseCommand:|' command/commands.go
gsed -i -e 's|base\.||' command/commands.go
gsed -i -e 's|command\.||' command/commands.go
gsed -i -e 's|command|c|' main.go
gsed -i -e 's|range Commands|range command.Commands|' main.go
gsed -i -e 's|Commands: Commands|Commands: command.Commands|' main.go
gsed -i -e 's|base\.BoolValue|BoolValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.DurationValue|DurationValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.StringValue|StringValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.UintValue|UintValue|' command/operator_autopilot_set.go
gsed -i -e 's|\bCommand\b|BaseCommand|' command/base.go
gsed -i -e 's|BaseCommand Options|Command Options|' command/base.go
gsed -i -e 's|base.Command|BaseCommand|' command/*.go
gsed -i -e 's|c\.Command|c.BaseCommand|g' command/*.go
gsed -i -e 's|\tCommand:|\tBaseCommand:|' command/*_test.go
gsed -i -e 's|base\.||' command/*_test.go
gsed -i -e 's|\bCommand\b|AgentCommand|' command/agent{,_test}.go
gsed -i -e 's|cmd.AgentCommand|cmd.BaseCommand|' command/agent.go
gsed -i -e 's|cli.AgentCommand = new(Command)|cli.Command = new(AgentCommand)|' command/agent_test.go
gsed -i -e 's|exec.AgentCommand|exec.Command|' command/agent_test.go
gsed -i -e 's|exec.BaseCommand|exec.Command|' command/agent_test.go
gsed -i -e 's|NewTestAgent|agent.NewTestAgent|' command/agent_test.go
gsed -i -e 's|= TestConfig|= agent.TestConfig|' command/agent_test.go
gsed -i -e 's|: RetryJoin|: agent.RetryJoin|' command/agent_test.go
gsed -i -e 's|\.\./\.\./|../|' command/config_util_test.go
gsed -i -e 's|\bverifyUniqueListeners|VerifyUniqueListeners|' agent/config{,_test}.go command/agent.go
gsed -i -e 's|\bserfLANKeyring\b|SerfLANKeyring|g' agent/{agent,keyring,testagent}.go command/agent.go
gsed -i -e 's|\bserfWANKeyring\b|SerfWANKeyring|g' agent/{agent,keyring,testagent}.go command/agent.go
gsed -i -e 's|\bNewAgent\b|agent.New|g' command/agent{,_test}.go
gsed -i -e 's|\bNewAgent|New|' agent/{acl_test,agent,testagent}.go
gsed -i -e 's|\bAgent\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bBool\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bDefaultConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bDevConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bMergeConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bReadConfigPaths\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bParseMetaPair\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bSerfLANKeyring\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bSerfWANKeyring\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|circonus\.agent|circonus|g' command/agent{,_test}.go
gsed -i -e 's|logger\.agent|logger|g' command/agent{,_test}.go
gsed -i -e 's|metrics\.agent|metrics|g' command/agent{,_test}.go
gsed -i -e 's|// agent.Agent|// agent|' command/agent{,_test}.go
gsed -i -e 's|a\.agent\.Config|a.Config|' command/agent{,_test}.go
gsed -i -e 's|agent\.AppendSliceValue|AppendSliceValue|' command/{configtest,validate}.go
gsed -i -e 's|consul/consul|agent/consul|' GNUmakefile
gsed -i -e 's|\.\./test|../../test|' agent/consul/server_test.go
# fix imports
f=$(grep -rl 'github.com/hashicorp/consul/command/agent' * | grep '\.go')
gsed -i -e 's|github.com/hashicorp/consul/command/agent|github.com/hashicorp/consul/agent|' $f
goimports -w $f
f=$(grep -rl 'github.com/hashicorp/consul/consul' * | grep '\.go')
gsed -i -e 's|github.com/hashicorp/consul/consul|github.com/hashicorp/consul/agent/consul|' $f
goimports -w $f
goimports -w command/*.go main.go
)
2017-06-10 00:28:28 +02:00
|
|
|
"github.com/hashicorp/consul/agent/consul"
|
|
|
|
"github.com/hashicorp/consul/agent/consul/structs"
|
2017-04-19 16:00:11 -07:00
|
|
|
"github.com/hashicorp/consul/api"
|
2017-05-12 15:41:13 +02:00
|
|
|
"github.com/hashicorp/consul/testutil"
|
2017-01-17 22:20:11 -08:00
|
|
|
"github.com/hashicorp/consul/types"
|
2017-03-21 16:36:44 -07:00
|
|
|
"github.com/hashicorp/consul/version"
|
2017-01-17 22:20:11 -08:00
|
|
|
"github.com/hashicorp/go-uuid"
|
2016-08-24 17:33:53 -07:00
|
|
|
"github.com/hashicorp/raft"
|
2017-05-15 21:49:13 +02:00
|
|
|
"github.com/pascaldekloe/goe/verify"
|
2013-12-23 13:52:10 -08:00
|
|
|
)
|
|
|
|
|
2017-03-21 16:36:44 -07:00
|
|
|
func init() {
|
|
|
|
version.Version = "0.8.0"
|
|
|
|
}
|
|
|
|
|
2016-11-03 12:58:58 -07:00
|
|
|
func externalIP() (string, error) {
|
|
|
|
addrs, err := net.InterfaceAddrs()
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("Unable to lookup network interfaces: %v", err)
|
|
|
|
}
|
|
|
|
for _, a := range addrs {
|
|
|
|
if ipnet, ok := a.(*net.IPNet); ok && !ipnet.IP.IsLoopback() {
|
|
|
|
if ipnet.IP.To4() != nil {
|
|
|
|
return ipnet.IP.String(), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "", fmt.Errorf("Unable to find a non-loopback interface")
|
|
|
|
}
|
|
|
|
|
2017-05-23 16:04:53 +02:00
|
|
|
func TestAgent_MultiStartStop(t *testing.T) {
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
t.Run("", func(t *testing.T) {
|
2017-05-31 10:56:19 +02:00
|
|
|
t.Parallel()
|
2017-05-23 16:04:53 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
time.Sleep(250 * time.Millisecond)
|
|
|
|
a.Shutdown()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
func TestAgent_StartStop(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
// defer a.Shutdown()
|
2013-12-23 13:52:10 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.Leave(); err != nil {
|
2013-12-23 13:52:10 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.Shutdown(); err != nil {
|
2013-12-23 13:52:10 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
2017-05-21 09:11:09 +02:00
|
|
|
case <-a.ShutdownCh():
|
2013-12-23 13:52:10 -08:00
|
|
|
default:
|
|
|
|
t.Fatalf("should be closed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 17:24:15 -08:00
|
|
|
func TestAgent_RPCPing(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2013-12-23 13:52:10 -08:00
|
|
|
|
|
|
|
var out struct{}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RPC("Status.Ping", struct{}{}, &out); err != nil {
|
2013-12-23 13:52:10 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2014-01-30 13:39:02 -08:00
|
|
|
|
2016-11-03 12:58:58 -07:00
|
|
|
func TestAgent_CheckSerfBindAddrsSettings(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-12 12:13:18 +02:00
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
t.Skip("skip test on macOS to avoid firewall warning dialog")
|
|
|
|
}
|
|
|
|
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
2016-11-03 12:58:58 -07:00
|
|
|
ip, err := externalIP()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to get a non-loopback IP: %v", err)
|
|
|
|
}
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.SerfLanBindAddr = ip
|
|
|
|
cfg.SerfWanBindAddr = ip
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-11-03 12:58:58 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
serfWanBind := a.consulConfig().SerfWANConfig.MemberlistConfig.BindAddr
|
2016-11-03 12:58:58 -07:00
|
|
|
if serfWanBind != ip {
|
|
|
|
t.Fatalf("SerfWanBindAddr is should be a non-loopback IP not %s", serfWanBind)
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
serfLanBind := a.consulConfig().SerfLANConfig.MemberlistConfig.BindAddr
|
2016-11-03 12:58:58 -07:00
|
|
|
if serfLanBind != ip {
|
|
|
|
t.Fatalf("SerfLanBindAddr is should be a non-loopback IP not %s", serfWanBind)
|
|
|
|
}
|
|
|
|
}
|
2015-06-05 13:44:42 +02:00
|
|
|
func TestAgent_CheckAdvertiseAddrsSettings(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.AdvertiseAddrs.SerfLan, _ = net.ResolveTCPAddr("tcp", "127.0.0.42:1233")
|
|
|
|
cfg.AdvertiseAddrs.SerfWan, _ = net.ResolveTCPAddr("tcp", "127.0.0.43:1234")
|
|
|
|
cfg.AdvertiseAddrs.RPC, _ = net.ResolveTCPAddr("tcp", "127.0.0.44:1235")
|
2017-07-27 22:06:31 -05:00
|
|
|
cfg.SetupTaggedAndAdvertiseAddrs()
|
2017-05-22 13:03:59 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-06-05 13:44:42 +02:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
serfLanAddr := a.consulConfig().SerfLANConfig.MemberlistConfig.AdvertiseAddr
|
2015-06-05 13:44:42 +02:00
|
|
|
if serfLanAddr != "127.0.0.42" {
|
|
|
|
t.Fatalf("SerfLan is not properly set to '127.0.0.42': %s", serfLanAddr)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
serfLanPort := a.consulConfig().SerfLANConfig.MemberlistConfig.AdvertisePort
|
2015-06-05 13:44:42 +02:00
|
|
|
if serfLanPort != 1233 {
|
2015-07-14 11:42:51 -07:00
|
|
|
t.Fatalf("SerfLan is not properly set to '1233': %d", serfLanPort)
|
2015-06-05 13:44:42 +02:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
serfWanAddr := a.consulConfig().SerfWANConfig.MemberlistConfig.AdvertiseAddr
|
2015-06-05 13:44:42 +02:00
|
|
|
if serfWanAddr != "127.0.0.43" {
|
|
|
|
t.Fatalf("SerfWan is not properly set to '127.0.0.43': %s", serfWanAddr)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
serfWanPort := a.consulConfig().SerfWANConfig.MemberlistConfig.AdvertisePort
|
2015-06-05 13:44:42 +02:00
|
|
|
if serfWanPort != 1234 {
|
2015-07-14 11:42:51 -07:00
|
|
|
t.Fatalf("SerfWan is not properly set to '1234': %d", serfWanPort)
|
2015-06-05 13:44:42 +02:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
rpc := a.consulConfig().RPCAdvertise
|
2017-05-22 13:03:59 +02:00
|
|
|
if rpc != cfg.AdvertiseAddrs.RPC {
|
|
|
|
t.Fatalf("RPC is not properly set to %v: %s", cfg.AdvertiseAddrs.RPC, rpc)
|
2015-06-05 13:44:42 +02:00
|
|
|
}
|
2016-02-07 10:37:34 -08:00
|
|
|
expected := map[string]string{
|
2017-05-21 09:11:09 +02:00
|
|
|
"lan": a.Config.AdvertiseAddr,
|
|
|
|
"wan": a.Config.AdvertiseAddrWan,
|
2016-02-07 10:37:34 -08:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if !reflect.DeepEqual(a.Config.TaggedAddresses, expected) {
|
|
|
|
t.Fatalf("Tagged addresses not set up properly: %v", a.Config.TaggedAddresses)
|
2015-12-23 16:07:27 -08:00
|
|
|
}
|
2015-06-05 13:44:42 +02:00
|
|
|
}
|
|
|
|
|
2017-07-26 11:03:43 -07:00
|
|
|
func TestAgent_TokenStore(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.ACLToken = "user"
|
|
|
|
cfg.ACLAgentToken = "agent"
|
|
|
|
cfg.ACLAgentMasterToken = "master"
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer a.Shutdown()
|
|
|
|
|
|
|
|
if got, want := a.tokens.UserToken(), "user"; got != want {
|
|
|
|
t.Fatalf("got %q want %q", got, want)
|
|
|
|
}
|
|
|
|
if got, want := a.tokens.AgentToken(), "agent"; got != want {
|
|
|
|
t.Fatalf("got %q want %q", got, want)
|
|
|
|
}
|
|
|
|
if got, want := a.tokens.IsAgentMasterToken("master"), true; got != want {
|
|
|
|
t.Fatalf("got %v want %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-24 17:33:53 -07:00
|
|
|
func TestAgent_CheckPerformanceSettings(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2016-08-24 17:33:53 -07:00
|
|
|
// Try a default config.
|
|
|
|
{
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Bootstrap = false
|
|
|
|
cfg.ConsulConfig = nil
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-08-24 17:33:53 -07:00
|
|
|
|
|
|
|
raftMult := time.Duration(consul.DefaultRaftMultiplier)
|
2017-05-21 09:11:09 +02:00
|
|
|
r := a.consulConfig().RaftConfig
|
2016-08-24 17:33:53 -07:00
|
|
|
def := raft.DefaultConfig()
|
|
|
|
if r.HeartbeatTimeout != raftMult*def.HeartbeatTimeout ||
|
|
|
|
r.ElectionTimeout != raftMult*def.ElectionTimeout ||
|
|
|
|
r.LeaderLeaseTimeout != raftMult*def.LeaderLeaseTimeout {
|
|
|
|
t.Fatalf("bad: %#v", *r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try a multiplier.
|
|
|
|
{
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Bootstrap = false
|
|
|
|
cfg.Performance.RaftMultiplier = 99
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-08-24 17:33:53 -07:00
|
|
|
|
|
|
|
const raftMult time.Duration = 99
|
2017-05-21 09:11:09 +02:00
|
|
|
r := a.consulConfig().RaftConfig
|
2016-08-24 17:33:53 -07:00
|
|
|
def := raft.DefaultConfig()
|
|
|
|
if r.HeartbeatTimeout != raftMult*def.HeartbeatTimeout ||
|
|
|
|
r.ElectionTimeout != raftMult*def.ElectionTimeout ||
|
|
|
|
r.LeaderLeaseTimeout != raftMult*def.LeaderLeaseTimeout {
|
|
|
|
t.Fatalf("bad: %#v", *r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-10 22:46:07 -07:00
|
|
|
func TestAgent_ReconnectConfigSettings(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2016-04-10 22:46:07 -07:00
|
|
|
func() {
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2016-04-10 22:46:07 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
lan := a.consulConfig().SerfLANConfig.ReconnectTimeout
|
2016-04-10 22:46:07 -07:00
|
|
|
if lan != 3*24*time.Hour {
|
|
|
|
t.Fatalf("bad: %s", lan.String())
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
wan := a.consulConfig().SerfWANConfig.ReconnectTimeout
|
2016-04-10 22:46:07 -07:00
|
|
|
if wan != 3*24*time.Hour {
|
|
|
|
t.Fatalf("bad: %s", wan.String())
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
func() {
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.ReconnectTimeoutLan = 24 * time.Hour
|
|
|
|
cfg.ReconnectTimeoutWan = 36 * time.Hour
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-04-10 22:46:07 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
lan := a.consulConfig().SerfLANConfig.ReconnectTimeout
|
2016-04-10 23:31:16 -07:00
|
|
|
if lan != 24*time.Hour {
|
2016-04-10 22:46:07 -07:00
|
|
|
t.Fatalf("bad: %s", lan.String())
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
wan := a.consulConfig().SerfWANConfig.ReconnectTimeout
|
2016-04-10 23:31:16 -07:00
|
|
|
if wan != 36*time.Hour {
|
2016-04-10 22:46:07 -07:00
|
|
|
t.Fatalf("bad: %s", wan.String())
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2017-04-12 22:05:38 -07:00
|
|
|
func TestAgent_setupNodeID(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.NodeID = ""
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2017-01-17 22:20:11 -08:00
|
|
|
|
|
|
|
// The auto-assigned ID should be valid.
|
2017-05-21 09:11:09 +02:00
|
|
|
id := a.consulConfig().NodeID
|
2017-01-17 22:20:11 -08:00
|
|
|
if _, err := uuid.ParseUUID(string(id)); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-01-18 09:52:34 -08:00
|
|
|
// Running again should get the same ID (persisted in the file).
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.NodeID = ""
|
|
|
|
if err := a.setupNodeID(cfg); err != nil {
|
2017-01-18 09:52:34 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if newID := a.consulConfig().NodeID; id != newID {
|
2017-01-18 09:52:34 -08:00
|
|
|
t.Fatalf("bad: %q vs %q", id, newID)
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
// Set an invalid ID via.Config.
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.NodeID = types.NodeID("nope")
|
|
|
|
err := a.setupNodeID(cfg)
|
2017-01-17 22:20:11 -08:00
|
|
|
if err == nil || !strings.Contains(err.Error(), "uuid string is wrong length") {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
// Set a valid ID via.Config.
|
2017-01-17 22:20:11 -08:00
|
|
|
newID, err := uuid.GenerateUUID()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.NodeID = types.NodeID(strings.ToUpper(newID))
|
|
|
|
if err := a.setupNodeID(cfg); err != nil {
|
2017-01-17 22:20:11 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if id := a.consulConfig().NodeID; string(id) != newID {
|
2017-01-17 22:20:11 -08:00
|
|
|
t.Fatalf("bad: %q vs. %q", id, newID)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set an invalid ID via the file.
|
2017-05-22 13:03:59 +02:00
|
|
|
fileID := filepath.Join(cfg.DataDir, "node-id")
|
2017-01-17 22:20:11 -08:00
|
|
|
if err := ioutil.WriteFile(fileID, []byte("adf4238a!882b!9ddc!4a9d!5b6758e4159e"), 0600); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.NodeID = ""
|
|
|
|
err = a.setupNodeID(cfg)
|
2017-01-17 22:20:11 -08:00
|
|
|
if err == nil || !strings.Contains(err.Error(), "uuid is improperly formatted") {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set a valid ID via the file.
|
2017-03-13 19:51:56 -07:00
|
|
|
if err := ioutil.WriteFile(fileID, []byte("ADF4238a-882b-9ddc-4a9d-5b6758e4159e"), 0600); err != nil {
|
2017-01-17 22:20:11 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.NodeID = ""
|
|
|
|
if err := a.setupNodeID(cfg); err != nil {
|
2017-01-17 22:20:11 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if id := a.consulConfig().NodeID; string(id) != "adf4238a-882b-9ddc-4a9d-5b6758e4159e" {
|
2017-01-17 22:20:11 -08:00
|
|
|
t.Fatalf("bad: %q vs. %q", id, newID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-12 22:05:38 -07:00
|
|
|
func TestAgent_makeNodeID(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.NodeID = ""
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2017-04-12 22:05:38 -07:00
|
|
|
|
|
|
|
// We should get a valid host-based ID initially.
|
2017-05-21 09:11:09 +02:00
|
|
|
id, err := a.makeNodeID()
|
2017-04-12 22:05:38 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, err := uuid.ParseUUID(string(id)); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:36:53 -07:00
|
|
|
// Calling again should yield a random ID by default.
|
2017-05-21 09:11:09 +02:00
|
|
|
another, err := a.makeNodeID()
|
2017-04-12 22:05:38 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-06-24 09:36:53 -07:00
|
|
|
if id == another {
|
2017-04-12 22:05:38 -07:00
|
|
|
t.Fatalf("bad: %s vs %s", id, another)
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:36:53 -07:00
|
|
|
// Turn on host-based IDs and try again. We should get the same ID
|
|
|
|
// each time (and a different one from the random one above).
|
|
|
|
a.Config.DisableHostNodeID = Bool(false)
|
|
|
|
id, err = a.makeNodeID()
|
2017-04-12 22:05:38 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if id == another {
|
|
|
|
t.Fatalf("bad: %s vs %s", id, another)
|
|
|
|
}
|
2017-06-24 09:36:53 -07:00
|
|
|
|
|
|
|
// Calling again should yield the host-based ID.
|
|
|
|
another, err = a.makeNodeID()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if id != another {
|
|
|
|
t.Fatalf("bad: %s vs %s", id, another)
|
|
|
|
}
|
2017-04-12 22:05:38 -07:00
|
|
|
}
|
|
|
|
|
2014-01-30 13:39:02 -08:00
|
|
|
func TestAgent_AddService(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
cfg := TestConfig()
|
2017-05-15 21:49:13 +02:00
|
|
|
cfg.NodeName = "node1"
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer a.Shutdown()
|
2014-01-30 13:39:02 -08:00
|
|
|
|
2017-05-15 21:49:13 +02:00
|
|
|
tests := []struct {
|
|
|
|
desc string
|
|
|
|
srv *structs.NodeService
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes []*structs.CheckType
|
2017-05-15 21:49:13 +02:00
|
|
|
healthChks map[string]*structs.HealthCheck
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"one check",
|
|
|
|
&structs.NodeService{
|
|
|
|
ID: "svcid1",
|
|
|
|
Service: "svcname1",
|
|
|
|
Tags: []string{"tag1"},
|
|
|
|
Port: 8100,
|
2015-01-13 17:52:17 -08:00
|
|
|
},
|
2017-06-15 18:46:06 +02:00
|
|
|
[]*structs.CheckType{
|
|
|
|
&structs.CheckType{
|
2017-05-15 21:49:13 +02:00
|
|
|
CheckID: "check1",
|
|
|
|
Name: "name1",
|
|
|
|
TTL: time.Minute,
|
|
|
|
Notes: "note1",
|
|
|
|
},
|
2015-01-13 17:52:17 -08:00
|
|
|
},
|
2017-05-15 21:49:13 +02:00
|
|
|
map[string]*structs.HealthCheck{
|
|
|
|
"check1": &structs.HealthCheck{
|
|
|
|
Node: "node1",
|
|
|
|
CheckID: "check1",
|
|
|
|
Name: "name1",
|
|
|
|
Status: "critical",
|
|
|
|
Notes: "note1",
|
|
|
|
ServiceID: "svcid1",
|
|
|
|
ServiceName: "svcname1",
|
|
|
|
},
|
2015-01-13 17:52:17 -08:00
|
|
|
},
|
2017-05-15 21:49:13 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"multiple checks",
|
|
|
|
&structs.NodeService{
|
|
|
|
ID: "svcid2",
|
|
|
|
Service: "svcname2",
|
|
|
|
Tags: []string{"tag2"},
|
|
|
|
Port: 8200,
|
|
|
|
},
|
2017-06-15 18:46:06 +02:00
|
|
|
[]*structs.CheckType{
|
|
|
|
&structs.CheckType{
|
2017-05-15 21:49:13 +02:00
|
|
|
CheckID: "check1",
|
|
|
|
Name: "name1",
|
|
|
|
TTL: time.Minute,
|
|
|
|
Notes: "note1",
|
|
|
|
},
|
2017-06-15 18:46:06 +02:00
|
|
|
&structs.CheckType{
|
2017-05-15 21:49:13 +02:00
|
|
|
CheckID: "check-noname",
|
|
|
|
TTL: time.Minute,
|
|
|
|
},
|
2017-06-15 18:46:06 +02:00
|
|
|
&structs.CheckType{
|
2017-05-15 21:49:13 +02:00
|
|
|
Name: "check-noid",
|
|
|
|
TTL: time.Minute,
|
|
|
|
},
|
2017-06-15 18:46:06 +02:00
|
|
|
&structs.CheckType{
|
2017-05-15 21:49:13 +02:00
|
|
|
TTL: time.Minute,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
map[string]*structs.HealthCheck{
|
|
|
|
"check1": &structs.HealthCheck{
|
|
|
|
Node: "node1",
|
|
|
|
CheckID: "check1",
|
|
|
|
Name: "name1",
|
|
|
|
Status: "critical",
|
|
|
|
Notes: "note1",
|
|
|
|
ServiceID: "svcid2",
|
|
|
|
ServiceName: "svcname2",
|
|
|
|
},
|
|
|
|
"check-noname": &structs.HealthCheck{
|
|
|
|
Node: "node1",
|
|
|
|
CheckID: "check-noname",
|
|
|
|
Name: "Service 'svcname2' check",
|
|
|
|
Status: "critical",
|
|
|
|
ServiceID: "svcid2",
|
|
|
|
ServiceName: "svcname2",
|
|
|
|
},
|
|
|
|
"service:svcid2:3": &structs.HealthCheck{
|
|
|
|
Node: "node1",
|
|
|
|
CheckID: "service:svcid2:3",
|
|
|
|
Name: "check-noid",
|
|
|
|
Status: "critical",
|
|
|
|
ServiceID: "svcid2",
|
|
|
|
ServiceName: "svcname2",
|
|
|
|
},
|
|
|
|
"service:svcid2:4": &structs.HealthCheck{
|
|
|
|
Node: "node1",
|
|
|
|
CheckID: "service:svcid2:4",
|
|
|
|
Name: "Service 'svcname2' check",
|
|
|
|
Status: "critical",
|
|
|
|
ServiceID: "svcid2",
|
|
|
|
ServiceName: "svcname2",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2015-01-13 17:52:17 -08:00
|
|
|
|
2017-05-15 21:49:13 +02:00
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.desc, func(t *testing.T) {
|
|
|
|
// check the service registration
|
|
|
|
t.Run(tt.srv.ID, func(t *testing.T) {
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddService(tt.srv, tt.chkTypes, false, "")
|
2017-05-15 21:49:13 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
got, want := a.state.Services()[tt.srv.ID], tt.srv
|
2017-05-15 21:49:13 +02:00
|
|
|
verify.Values(t, "", got, want)
|
|
|
|
})
|
|
|
|
|
|
|
|
// check the health checks
|
|
|
|
for k, v := range tt.healthChks {
|
|
|
|
t.Run(k, func(t *testing.T) {
|
2017-05-21 09:11:09 +02:00
|
|
|
got, want := a.state.Checks()[types.CheckID(k)], v
|
2017-05-15 21:49:13 +02:00
|
|
|
verify.Values(t, k, got, want)
|
|
|
|
})
|
|
|
|
}
|
2015-01-13 17:52:17 -08:00
|
|
|
|
2017-05-15 21:49:13 +02:00
|
|
|
// check the ttl checks
|
|
|
|
for k := range tt.healthChks {
|
|
|
|
t.Run(k+" ttl", func(t *testing.T) {
|
2017-05-21 09:11:09 +02:00
|
|
|
chk := a.checkTTLs[types.CheckID(k)]
|
2017-05-15 21:49:13 +02:00
|
|
|
if chk == nil {
|
|
|
|
t.Fatal("got nil want TTL check")
|
|
|
|
}
|
|
|
|
if got, want := string(chk.CheckID), k; got != want {
|
|
|
|
t.Fatalf("got CheckID %v want %v", got, want)
|
|
|
|
}
|
|
|
|
if got, want := chk.TTL, time.Minute; got != want {
|
|
|
|
t.Fatalf("got TTL %v want %v", got, want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
2014-11-06 18:24:04 -08:00
|
|
|
}
|
2014-01-30 13:39:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestAgent_RemoveService(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2014-01-30 13:39:02 -08:00
|
|
|
|
|
|
|
// Remove a service that doesn't exist
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService("redis", false); err != nil {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-01-27 18:10:56 +09:00
|
|
|
// Remove without an ID
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService("", false); err == nil {
|
2015-01-27 18:10:56 +09:00
|
|
|
t.Fatalf("should have errored")
|
|
|
|
}
|
|
|
|
|
2015-01-13 17:52:17 -08:00
|
|
|
// Removing a service with a single check works
|
|
|
|
{
|
|
|
|
srv := &structs.NodeService{
|
|
|
|
ID: "memcache",
|
|
|
|
Service: "memcache",
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes := []*structs.CheckType{&structs.CheckType{TTL: time.Minute}}
|
2015-01-13 17:52:17 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(srv, chkTypes, false, ""); err != nil {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-05-07 15:30:01 -07:00
|
|
|
// Add a check after the fact with a specific check ID
|
2017-06-15 18:46:06 +02:00
|
|
|
check := &structs.CheckDefinition{
|
2015-05-07 15:30:01 -07:00
|
|
|
ID: "check2",
|
|
|
|
Name: "check2",
|
|
|
|
ServiceID: "memcache",
|
2017-05-15 21:49:13 +02:00
|
|
|
TTL: time.Minute,
|
2015-05-07 15:30:01 -07:00
|
|
|
}
|
|
|
|
hc := check.HealthCheck("node1")
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(hc, check.CheckType(), false, ""); err != nil {
|
2015-05-07 15:30:01 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService("memcache", false); err != nil {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["service:memcache"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("have memcache check")
|
2015-05-07 15:30:01 -07:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["check2"]; ok {
|
2015-05-07 15:30:01 -07:00
|
|
|
t.Fatalf("have check2 check")
|
2015-01-13 17:52:17 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Removing a service with multiple checks works
|
|
|
|
{
|
|
|
|
srv := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes := []*structs.CheckType{
|
|
|
|
&structs.CheckType{TTL: time.Minute},
|
|
|
|
&structs.CheckType{TTL: 30 * time.Second},
|
2015-01-13 17:52:17 -08:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(srv, chkTypes, false, ""); err != nil {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService("redis", false); err != nil {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have a state mapping
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("have redis service")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure checks were removed
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["service:redis:1"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("check redis:1 should be removed")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["service:redis:2"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("check redis:2 should be removed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a TTL is setup
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.checkTTLs["service:redis:1"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("check ttl for redis:1 should be removed")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.checkTTLs["service:redis:2"]; ok {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("check ttl for redis:2 should be removed")
|
|
|
|
}
|
2014-01-30 13:39:02 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-15 21:49:13 +02:00
|
|
|
func TestAgent_RemoveServiceRemovesAllChecks(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
cfg := TestConfig()
|
2017-05-15 21:49:13 +02:00
|
|
|
cfg.NodeName = "node1"
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer a.Shutdown()
|
2017-05-15 21:49:13 +02:00
|
|
|
|
|
|
|
svc := &structs.NodeService{ID: "redis", Service: "redis", Port: 8000}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk1 := &structs.CheckType{CheckID: "chk1", Name: "chk1", TTL: time.Minute}
|
|
|
|
chk2 := &structs.CheckType{CheckID: "chk2", Name: "chk2", TTL: 2 * time.Minute}
|
2017-05-15 21:49:13 +02:00
|
|
|
hchk1 := &structs.HealthCheck{Node: "node1", CheckID: "chk1", Name: "chk1", Status: "critical", ServiceID: "redis", ServiceName: "redis"}
|
|
|
|
hchk2 := &structs.HealthCheck{Node: "node1", CheckID: "chk2", Name: "chk2", Status: "critical", ServiceID: "redis", ServiceName: "redis"}
|
|
|
|
|
|
|
|
// register service with chk1
|
2017-06-15 18:46:06 +02:00
|
|
|
if err := a.AddService(svc, []*structs.CheckType{chk1}, false, ""); err != nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Failed to register service", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// verify chk1 exists
|
2017-05-21 09:11:09 +02:00
|
|
|
if a.state.Checks()["chk1"] == nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Could not find health check chk1")
|
|
|
|
}
|
|
|
|
|
|
|
|
// update the service with chk2
|
2017-06-15 18:46:06 +02:00
|
|
|
if err := a.AddService(svc, []*structs.CheckType{chk2}, false, ""); err != nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Failed to update service", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check that both checks are there
|
2017-05-21 09:11:09 +02:00
|
|
|
if got, want := a.state.Checks()["chk1"], hchk1; !verify.Values(t, "", got, want) {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.FailNow()
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if got, want := a.state.Checks()["chk2"], hchk2; !verify.Values(t, "", got, want) {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService("redis", false); err != nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Failed to remove service", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that both checks are gone
|
2017-05-21 09:11:09 +02:00
|
|
|
if a.state.Checks()["chk1"] != nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Found health check chk1 want nil")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if a.state.Checks()["chk2"] != nil {
|
2017-05-15 21:49:13 +02:00
|
|
|
t.Fatal("Found health check chk2 want nil")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-30 13:39:02 -08:00
|
|
|
func TestAgent_AddCheck(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.EnableScriptChecks = true
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-01-30 13:39:02 -08:00
|
|
|
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthCritical,
|
2014-01-30 13:39:02 -08:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2014-01-30 13:39:02 -08:00
|
|
|
Script: "exit 0",
|
|
|
|
Interval: 15 * time.Second,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2014-01-30 13:39:02 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have a check mapping
|
2017-05-21 09:11:09 +02:00
|
|
|
sChk, ok := a.state.Checks()["mem"]
|
2015-04-12 00:53:48 +00:00
|
|
|
if !ok {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("missing mem check")
|
|
|
|
}
|
|
|
|
|
2015-04-12 00:53:48 +00:00
|
|
|
// Ensure our check is in the right state
|
2017-04-19 16:00:11 -07:00
|
|
|
if sChk.Status != api.HealthCritical {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("check not critical")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a TTL is setup
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.checkMonitors["mem"]; !ok {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("missing mem monitor")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAgent_AddCheck_StartPassing(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.EnableScriptChecks = true
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-04-12 00:53:48 +00:00
|
|
|
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2015-04-12 00:53:48 +00:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2015-04-12 00:53:48 +00:00
|
|
|
Script: "exit 0",
|
|
|
|
Interval: 15 * time.Second,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2015-04-12 00:53:48 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have a check mapping
|
2017-05-21 09:11:09 +02:00
|
|
|
sChk, ok := a.state.Checks()["mem"]
|
2015-04-12 00:53:48 +00:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing mem check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure our check is in the right state
|
2017-04-19 16:00:11 -07:00
|
|
|
if sChk.Status != api.HealthPassing {
|
2015-04-12 00:53:48 +00:00
|
|
|
t.Fatalf("check not passing")
|
|
|
|
}
|
|
|
|
|
2014-01-30 13:39:02 -08:00
|
|
|
// Ensure a TTL is setup
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.checkMonitors["mem"]; !ok {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("missing mem monitor")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-21 14:42:42 -07:00
|
|
|
func TestAgent_AddCheck_MinInterval(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.EnableScriptChecks = true
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-04-21 14:42:42 -07:00
|
|
|
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthCritical,
|
2014-04-21 14:42:42 -07:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2014-04-21 14:42:42 -07:00
|
|
|
Script: "exit 0",
|
|
|
|
Interval: time.Microsecond,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2014-04-21 14:42:42 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have a check mapping
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["mem"]; !ok {
|
2014-04-21 14:42:42 -07:00
|
|
|
t.Fatalf("missing mem check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a TTL is setup
|
2017-05-21 09:11:09 +02:00
|
|
|
if mon, ok := a.checkMonitors["mem"]; !ok {
|
2014-04-21 14:42:42 -07:00
|
|
|
t.Fatalf("missing mem monitor")
|
|
|
|
} else if mon.Interval != MinInterval {
|
|
|
|
t.Fatalf("bad mem monitor interval")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-13 17:52:17 -08:00
|
|
|
func TestAgent_AddCheck_MissingService(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.EnableScriptChecks = true
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-01-13 17:52:17 -08:00
|
|
|
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "baz",
|
|
|
|
Name: "baz check 1",
|
|
|
|
ServiceID: "baz",
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2015-01-13 17:52:17 -08:00
|
|
|
Script: "exit 0",
|
|
|
|
Interval: time.Microsecond,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2015-01-13 17:52:17 -08:00
|
|
|
if err == nil || err.Error() != `ServiceID "baz" does not exist` {
|
|
|
|
t.Fatalf("expected service id error, got: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-05 17:33:34 -07:00
|
|
|
func TestAgent_AddCheck_RestoreState(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-06-05 17:33:34 -07:00
|
|
|
|
|
|
|
// Create some state and persist it
|
|
|
|
ttl := &CheckTTL{
|
|
|
|
CheckID: "baz",
|
|
|
|
TTL: time.Minute,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.persistCheckState(ttl, api.HealthPassing, "yup")
|
2015-06-05 17:33:34 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build and register the check definition and initial state
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "baz",
|
|
|
|
Name: "baz check 1",
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2015-06-05 17:33:34 -07:00
|
|
|
TTL: time.Minute,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err = a.AddCheck(health, chk, false, "")
|
2015-06-05 17:33:34 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the check status was restored during registration
|
2017-05-21 09:11:09 +02:00
|
|
|
checks := a.state.Checks()
|
2015-06-05 17:33:34 -07:00
|
|
|
check, ok := checks["baz"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check")
|
|
|
|
}
|
2017-04-19 16:00:11 -07:00
|
|
|
if check.Status != api.HealthPassing {
|
2015-06-05 17:33:34 -07:00
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
|
|
|
if check.Output != "yup" {
|
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-17 11:20:35 -07:00
|
|
|
func TestAgent_AddCheck_ExecDisable(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-07-17 11:20:35 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2014-01-30 13:39:02 -08:00
|
|
|
|
2017-07-17 11:20:35 -07:00
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
|
|
|
Status: api.HealthCritical,
|
|
|
|
}
|
|
|
|
chk := &structs.CheckType{
|
|
|
|
Script: "exit 0",
|
|
|
|
Interval: 15 * time.Second,
|
|
|
|
}
|
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2017-07-19 22:15:04 -07:00
|
|
|
if err == nil || !strings.Contains(err.Error(), "Scripts are disabled on this agent") {
|
2017-07-17 11:20:35 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we don't have a check mapping
|
|
|
|
if memChk := a.state.Checks()["mem"]; memChk != nil {
|
|
|
|
t.Fatalf("should be missing mem check")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAgent_RemoveCheck(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.EnableScriptChecks = true
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer a.Shutdown()
|
|
|
|
|
2014-01-30 13:39:02 -08:00
|
|
|
// Remove check that doesn't exist
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveCheck("mem", false); err != nil {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-01-27 18:10:56 +09:00
|
|
|
// Remove without an ID
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveCheck("", false); err == nil {
|
2015-01-27 18:10:56 +09:00
|
|
|
t.Fatalf("should have errored")
|
|
|
|
}
|
|
|
|
|
2014-01-30 13:39:02 -08:00
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthCritical,
|
2014-01-30 13:39:02 -08:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2014-01-30 13:39:02 -08:00
|
|
|
Script: "exit 0",
|
|
|
|
Interval: 15 * time.Second,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2014-01-30 13:39:02 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove check
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveCheck("mem", false); err != nil {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have a check mapping
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()["mem"]; ok {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("have mem check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a TTL is setup
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.checkMonitors["mem"]; ok {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("have mem monitor")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-16 00:05:55 -07:00
|
|
|
func TestAgent_updateTTLCheck(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2014-01-30 13:39:02 -08:00
|
|
|
|
|
|
|
health := &structs.HealthCheck{
|
|
|
|
Node: "foo",
|
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory util",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthCritical,
|
2014-01-30 13:39:02 -08:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chk := &structs.CheckType{
|
2014-01-30 13:39:02 -08:00
|
|
|
TTL: 15 * time.Second,
|
|
|
|
}
|
2016-08-16 00:05:55 -07:00
|
|
|
|
|
|
|
// Add check and update it.
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.AddCheck(health, chk, false, "")
|
2014-01-30 13:39:02 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.updateTTLCheck("mem", api.HealthPassing, "foo"); err != nil {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2016-08-16 00:05:55 -07:00
|
|
|
// Ensure we have a check mapping.
|
2017-05-21 09:11:09 +02:00
|
|
|
status := a.state.Checks()["mem"]
|
2017-04-19 16:00:11 -07:00
|
|
|
if status.Status != api.HealthPassing {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("bad: %v", status)
|
|
|
|
}
|
2014-04-21 16:20:22 -07:00
|
|
|
if status.Output != "foo" {
|
2014-01-30 13:39:02 -08:00
|
|
|
t.Fatalf("bad: %v", status)
|
|
|
|
}
|
|
|
|
}
|
2014-10-14 15:05:41 -07:00
|
|
|
|
2014-11-24 00:36:03 -08:00
|
|
|
func TestAgent_PersistService(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Server = false
|
|
|
|
cfg.DataDir = testutil.TempDir(t, "agent") // we manage the data dir
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer os.RemoveAll(cfg.DataDir)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, servicesDir, stringHash(svc.ID))
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
// Check is not persisted unless requested
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2014-11-24 19:24:32 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, err := os.Stat(file); err == nil {
|
|
|
|
t.Fatalf("should not persist")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Persists to file if requested
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, true, "mytoken"); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, err := os.Stat(file); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2015-04-28 12:44:46 -07:00
|
|
|
expected, err := json.Marshal(persistedService{
|
2015-05-04 17:36:17 -07:00
|
|
|
Token: "mytoken",
|
2015-04-27 22:46:01 -07:00
|
|
|
Service: svc,
|
|
|
|
})
|
2014-11-24 00:36:03 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
content, err := ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(expected, content) {
|
|
|
|
t.Fatalf("bad: %s", string(content))
|
|
|
|
}
|
2015-05-05 22:08:03 -07:00
|
|
|
|
|
|
|
// Updates service definition on disk
|
|
|
|
svc.Port = 8001
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, true, "mytoken"); err != nil {
|
2015-05-05 22:08:03 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
expected, err = json.Marshal(persistedService{
|
|
|
|
Token: "mytoken",
|
|
|
|
Service: svc,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
content, err = ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(expected, content) {
|
|
|
|
t.Fatalf("bad: %s", string(content))
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
a.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
|
|
|
// Should load it back during later start
|
2017-05-31 10:56:19 +02:00
|
|
|
a2 := NewTestAgent(t.Name()+"-a2", cfg)
|
|
|
|
defer a2.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
2017-05-31 10:56:19 +02:00
|
|
|
restored, ok := a2.state.services[svc.ID]
|
2015-05-05 22:08:03 -07:00
|
|
|
if !ok {
|
2017-05-31 10:56:19 +02:00
|
|
|
t.Fatalf("bad: %#v", a2.state.services)
|
2014-11-24 00:36:03 -08:00
|
|
|
}
|
2017-05-31 10:56:19 +02:00
|
|
|
if a2.state.serviceTokens[svc.ID] != "mytoken" {
|
|
|
|
t.Fatalf("bad: %#v", a2.state.services[svc.ID])
|
2015-04-27 22:46:01 -07:00
|
|
|
}
|
2015-05-05 22:08:03 -07:00
|
|
|
if restored.Port != 8001 {
|
|
|
|
t.Fatalf("bad: %#v", restored)
|
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
}
|
2014-11-24 00:36:03 -08:00
|
|
|
|
2015-04-28 12:44:46 -07:00
|
|
|
func TestAgent_persistedService_compat(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2015-04-28 12:18:41 -07:00
|
|
|
// Tests backwards compatibility of persisted services from pre-0.5.1
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-04-28 12:18:41 -07:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encode the NodeService directly. This is what previous versions
|
|
|
|
// would serialize to the file (without the wrapper)
|
|
|
|
encoded, err := json.Marshal(svc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the content to the file
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, servicesDir, stringHash(svc.ID))
|
2015-04-28 12:18:41 -07:00
|
|
|
if err := os.MkdirAll(filepath.Dir(file), 0700); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if err := ioutil.WriteFile(file, encoded, 0600); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the services
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.loadServices(a.Config); err != nil {
|
2015-04-28 12:18:41 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the service was restored
|
2017-05-21 09:11:09 +02:00
|
|
|
services := a.state.Services()
|
2015-04-28 12:18:41 -07:00
|
|
|
result, ok := services["redis"]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing service")
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(result, svc) {
|
|
|
|
t.Fatalf("bad: %#v", result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-25 23:58:02 -08:00
|
|
|
func TestAgent_PurgeService(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2014-11-25 23:58:02 -08:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, servicesDir, stringHash(svc.ID))
|
|
|
|
if err := a.AddService(svc, nil, true, ""); err != nil {
|
2014-11-25 23:58:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not removed
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService(svc.ID, false); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
if _, err := os.Stat(file); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
|
2016-11-09 16:56:54 -05:00
|
|
|
// Re-add the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, true, ""); err != nil {
|
2016-11-09 16:56:54 -05:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2014-11-25 23:58:02 -08:00
|
|
|
// Removed
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveService(svc.ID, true); err != nil {
|
2014-11-25 23:58:02 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if _, err := os.Stat(file); !os.IsNotExist(err) {
|
|
|
|
t.Fatalf("bad: %#v", err)
|
|
|
|
}
|
2014-11-24 00:36:03 -08:00
|
|
|
}
|
|
|
|
|
2014-11-24 19:24:32 -08:00
|
|
|
func TestAgent_PurgeServiceOnDuplicate(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Server = false
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
svc1 := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
|
|
|
// First persist the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc1, nil, true, ""); err != nil {
|
2014-11-24 19:24:32 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
a.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
// Try bringing the agent back up with the service already
|
|
|
|
// existing in the config
|
2017-06-15 18:46:06 +02:00
|
|
|
svc2 := &structs.ServiceDefinition{
|
2014-11-24 19:24:32 -08:00
|
|
|
ID: "redis",
|
|
|
|
Name: "redis",
|
|
|
|
Tags: []string{"bar"},
|
|
|
|
Port: 9000,
|
|
|
|
}
|
|
|
|
|
2017-06-15 18:46:06 +02:00
|
|
|
cfg.Services = []*structs.ServiceDefinition{svc2}
|
2017-05-31 10:56:19 +02:00
|
|
|
a2 := NewTestAgent(t.Name()+"-a2", cfg)
|
|
|
|
defer a2.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, servicesDir, stringHash(svc1.ID))
|
2014-11-24 19:24:32 -08:00
|
|
|
if _, err := os.Stat(file); err == nil {
|
|
|
|
t.Fatalf("should have removed persisted service")
|
|
|
|
}
|
2017-05-31 10:56:19 +02:00
|
|
|
result, ok := a2.state.services[svc2.ID]
|
2014-11-24 19:24:32 -08:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing service registration")
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(result.Tags, svc2.Tags) || result.Port != svc2.Port {
|
|
|
|
t.Fatalf("bad: %#v", result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-24 00:36:03 -08:00
|
|
|
func TestAgent_PersistCheck(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Server = false
|
|
|
|
cfg.DataDir = testutil.TempDir(t, "agent") // we manage the data dir
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg.EnableScriptChecks = true
|
2017-05-22 13:03:59 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer os.RemoveAll(cfg.DataDir)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
|
|
|
check := &structs.HealthCheck{
|
2017-05-22 13:03:59 +02:00
|
|
|
Node: cfg.NodeName,
|
2015-01-13 17:52:17 -08:00
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory check",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2014-11-24 00:36:03 -08:00
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chkType := &structs.CheckType{
|
2014-11-29 12:25:01 -08:00
|
|
|
Script: "/bin/true",
|
|
|
|
Interval: 10 * time.Second,
|
|
|
|
}
|
2014-11-24 00:36:03 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checksDir, checkIDHash(check.CheckID))
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
// Not persisted if not requested
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check, chkType, false, ""); err != nil {
|
2014-11-24 19:24:32 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, err := os.Stat(file); err == nil {
|
|
|
|
t.Fatalf("should not persist")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should persist if requested
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check, chkType, true, "mytoken"); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if _, err := os.Stat(file); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2015-04-28 12:44:46 -07:00
|
|
|
expected, err := json.Marshal(persistedCheck{
|
|
|
|
Check: check,
|
|
|
|
ChkType: chkType,
|
2015-05-04 17:36:17 -07:00
|
|
|
Token: "mytoken",
|
2015-04-28 12:44:46 -07:00
|
|
|
})
|
2014-11-24 00:36:03 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
content, err := ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(expected, content) {
|
|
|
|
t.Fatalf("bad: %s", string(content))
|
|
|
|
}
|
2015-05-05 22:08:03 -07:00
|
|
|
|
|
|
|
// Updates the check definition on disk
|
|
|
|
check.Name = "mem1"
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check, chkType, true, "mytoken"); err != nil {
|
2015-05-05 22:08:03 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
expected, err = json.Marshal(persistedCheck{
|
|
|
|
Check: check,
|
|
|
|
ChkType: chkType,
|
|
|
|
Token: "mytoken",
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
content, err = ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(expected, content) {
|
|
|
|
t.Fatalf("bad: %s", string(content))
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
a.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
|
|
|
// Should load it back during later start
|
2017-05-31 10:56:19 +02:00
|
|
|
a2 := NewTestAgent(t.Name()+"-a2", cfg)
|
|
|
|
defer a2.Shutdown()
|
2014-11-24 00:36:03 -08:00
|
|
|
|
2017-05-31 10:56:19 +02:00
|
|
|
result, ok := a2.state.checks[check.CheckID]
|
2014-11-24 01:15:18 -08:00
|
|
|
if !ok {
|
2017-05-31 10:56:19 +02:00
|
|
|
t.Fatalf("bad: %#v", a2.state.checks)
|
2014-11-24 00:36:03 -08:00
|
|
|
}
|
2017-04-19 16:00:11 -07:00
|
|
|
if result.Status != api.HealthCritical {
|
2014-11-24 01:15:18 -08:00
|
|
|
t.Fatalf("bad: %#v", result)
|
|
|
|
}
|
2015-05-05 22:08:03 -07:00
|
|
|
if result.Name != "mem1" {
|
|
|
|
t.Fatalf("bad: %#v", result)
|
|
|
|
}
|
2014-11-29 12:25:01 -08:00
|
|
|
|
|
|
|
// Should have restored the monitor
|
2017-05-31 10:56:19 +02:00
|
|
|
if _, ok := a2.checkMonitors[check.CheckID]; !ok {
|
|
|
|
t.Fatalf("bad: %#v", a2.checkMonitors)
|
2014-11-29 12:25:01 -08:00
|
|
|
}
|
2017-05-31 10:56:19 +02:00
|
|
|
if a2.state.checkTokens[check.CheckID] != "mytoken" {
|
|
|
|
t.Fatalf("bad: %s", a2.state.checkTokens[check.CheckID])
|
2015-04-27 22:46:01 -07:00
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestAgent_PurgeCheck(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2014-11-25 23:58:02 -08:00
|
|
|
|
|
|
|
check := &structs.HealthCheck{
|
2017-05-21 09:11:09 +02:00
|
|
|
Node: a.Config.NodeName,
|
2015-01-13 17:52:17 -08:00
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory check",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2014-11-25 23:58:02 -08:00
|
|
|
}
|
2014-11-24 00:36:03 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checksDir, checkIDHash(check.CheckID))
|
|
|
|
if err := a.AddCheck(check, nil, true, ""); err != nil {
|
2014-11-25 23:58:02 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not removed
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveCheck(check.CheckID, false); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
if _, err := os.Stat(file); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Removed
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.RemoveCheck(check.CheckID, true); err != nil {
|
2014-11-24 00:36:03 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2014-11-25 23:58:02 -08:00
|
|
|
if _, err := os.Stat(file); !os.IsNotExist(err) {
|
|
|
|
t.Fatalf("bad: %#v", err)
|
|
|
|
}
|
2014-11-24 00:36:03 -08:00
|
|
|
}
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
func TestAgent_PurgeCheckOnDuplicate(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Server = false
|
|
|
|
cfg.DataDir = testutil.TempDir(t, "agent") // we manage the data dir
|
2017-07-17 11:20:35 -07:00
|
|
|
cfg.EnableScriptChecks = true
|
2017-05-22 13:03:59 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
|
|
|
defer os.RemoveAll(cfg.DataDir)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
check1 := &structs.HealthCheck{
|
2017-05-22 13:03:59 +02:00
|
|
|
Node: cfg.NodeName,
|
2015-01-13 17:52:17 -08:00
|
|
|
CheckID: "mem",
|
|
|
|
Name: "memory check",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2014-11-24 19:24:32 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// First persist the check
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check1, nil, true, ""); err != nil {
|
2014-11-24 19:24:32 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
a.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
|
|
|
// Start again with the check registered in config
|
2017-06-15 18:46:06 +02:00
|
|
|
check2 := &structs.CheckDefinition{
|
2017-05-15 21:49:13 +02:00
|
|
|
ID: "mem",
|
|
|
|
Name: "memory check",
|
|
|
|
Notes: "my cool notes",
|
|
|
|
Script: "/bin/check-redis.py",
|
|
|
|
Interval: 30 * time.Second,
|
2014-11-24 19:24:32 -08:00
|
|
|
}
|
|
|
|
|
2017-06-15 18:46:06 +02:00
|
|
|
cfg.Checks = []*structs.CheckDefinition{check2}
|
2017-05-31 10:56:19 +02:00
|
|
|
a2 := NewTestAgent(t.Name()+"-a2", cfg)
|
|
|
|
defer a2.Shutdown()
|
2014-11-24 19:24:32 -08:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checksDir, checkIDHash(check1.CheckID))
|
2014-11-24 19:24:32 -08:00
|
|
|
if _, err := os.Stat(file); err == nil {
|
|
|
|
t.Fatalf("should have removed persisted check")
|
|
|
|
}
|
2017-05-31 10:56:19 +02:00
|
|
|
result, ok := a2.state.checks[check2.ID]
|
2014-11-24 19:24:32 -08:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check registration")
|
|
|
|
}
|
2017-05-22 13:03:59 +02:00
|
|
|
expected := check2.HealthCheck(cfg.NodeName)
|
2014-11-24 19:24:32 -08:00
|
|
|
if !reflect.DeepEqual(expected, result) {
|
|
|
|
t.Fatalf("bad: %#v", result)
|
|
|
|
}
|
|
|
|
}
|
2015-01-07 22:26:40 -08:00
|
|
|
|
2015-04-28 12:44:46 -07:00
|
|
|
func TestAgent_loadChecks_token(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
2017-06-15 18:46:06 +02:00
|
|
|
cfg.Checks = append(cfg.Checks, &structs.CheckDefinition{
|
2015-04-28 12:44:46 -07:00
|
|
|
ID: "rabbitmq",
|
|
|
|
Name: "rabbitmq",
|
|
|
|
Token: "abc123",
|
2017-05-15 21:49:13 +02:00
|
|
|
TTL: 10 * time.Second,
|
2015-04-28 12:44:46 -07:00
|
|
|
})
|
2017-05-22 13:03:59 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-04-28 12:44:46 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
checks := a.state.Checks()
|
2015-04-28 12:44:46 -07:00
|
|
|
if _, ok := checks["rabbitmq"]; !ok {
|
|
|
|
t.Fatalf("missing check")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if token := a.state.CheckToken("rabbitmq"); token != "abc123" {
|
2015-04-28 12:44:46 -07:00
|
|
|
t.Fatalf("bad: %s", token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-07 22:26:40 -08:00
|
|
|
func TestAgent_unloadChecks(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-01-07 22:26:40 -08:00
|
|
|
|
2015-01-13 17:52:17 -08:00
|
|
|
// First register a service
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2015-01-13 17:52:17 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a check
|
2015-01-07 22:26:40 -08:00
|
|
|
check1 := &structs.HealthCheck{
|
2017-05-21 09:11:09 +02:00
|
|
|
Node: a.Config.NodeName,
|
2015-01-13 17:52:17 -08:00
|
|
|
CheckID: "service:redis",
|
2015-01-07 22:26:40 -08:00
|
|
|
Name: "redischeck",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2015-01-07 22:26:40 -08:00
|
|
|
ServiceID: "redis",
|
|
|
|
ServiceName: "redis",
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check1, nil, false, ""); err != nil {
|
2015-01-07 22:26:40 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
found := false
|
2017-05-21 09:11:09 +02:00
|
|
|
for check := range a.state.Checks() {
|
2015-01-07 22:26:40 -08:00
|
|
|
if check == check1.CheckID {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatalf("check should have been registered")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unload all of the checks
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.unloadChecks(); err != nil {
|
2015-01-07 22:26:40 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure it was unloaded
|
2017-05-21 09:11:09 +02:00
|
|
|
for check := range a.state.Checks() {
|
2015-01-07 22:26:40 -08:00
|
|
|
if check == check1.CheckID {
|
|
|
|
t.Fatalf("should have unloaded checks")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-28 12:44:46 -07:00
|
|
|
func TestAgent_loadServices_token(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
2017-06-15 18:46:06 +02:00
|
|
|
cfg.Services = append(cfg.Services, &structs.ServiceDefinition{
|
2015-04-28 12:44:46 -07:00
|
|
|
ID: "rabbitmq",
|
|
|
|
Name: "rabbitmq",
|
|
|
|
Port: 5672,
|
|
|
|
Token: "abc123",
|
|
|
|
})
|
2017-05-22 13:03:59 +02:00
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-04-28 12:44:46 -07:00
|
|
|
|
2017-05-21 09:11:09 +02:00
|
|
|
services := a.state.Services()
|
2015-04-28 12:44:46 -07:00
|
|
|
if _, ok := services["rabbitmq"]; !ok {
|
|
|
|
t.Fatalf("missing service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if token := a.state.ServiceToken("rabbitmq"); token != "abc123" {
|
2015-04-28 12:44:46 -07:00
|
|
|
t.Fatalf("bad: %s", token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-07 22:26:40 -08:00
|
|
|
func TestAgent_unloadServices(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-01-07 22:26:40 -08:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2015-01-07 22:26:40 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
found := false
|
2017-05-21 09:11:09 +02:00
|
|
|
for id := range a.state.Services() {
|
2015-01-07 22:26:40 -08:00
|
|
|
if id == svc.ID {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatalf("should have registered service")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unload all services
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.unloadServices(); err != nil {
|
2015-01-07 22:26:40 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2017-07-13 22:33:47 -07:00
|
|
|
if len(a.state.Services()) != 0 {
|
|
|
|
t.Fatalf("should have unloaded services")
|
2015-01-07 22:26:40 -08:00
|
|
|
}
|
|
|
|
}
|
2015-01-15 00:25:36 -08:00
|
|
|
|
2016-08-16 00:05:55 -07:00
|
|
|
func TestAgent_Service_MaintenanceMode(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-01-15 00:25:36 -08:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2015-01-15 00:25:36 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enter maintenance mode for the service
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.EnableServiceMaintenance("redis", "broken", "mytoken"); err != nil {
|
2015-01-15 00:25:36 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the critical health check was added
|
2015-01-15 12:20:57 -08:00
|
|
|
checkID := serviceMaintCheckID("redis")
|
2017-05-21 09:11:09 +02:00
|
|
|
check, ok := a.state.Checks()[checkID]
|
2015-01-21 12:21:57 -08:00
|
|
|
if !ok {
|
2015-01-15 11:26:14 -08:00
|
|
|
t.Fatalf("should have registered critical maintenance check")
|
2015-01-15 00:25:36 -08:00
|
|
|
}
|
|
|
|
|
2015-09-10 12:08:08 -07:00
|
|
|
// Check that the token was used to register the check
|
2017-05-21 09:11:09 +02:00
|
|
|
if token := a.state.CheckToken(checkID); token != "mytoken" {
|
2015-09-10 12:08:08 -07:00
|
|
|
t.Fatalf("expected 'mytoken', got: '%s'", token)
|
|
|
|
}
|
|
|
|
|
2015-01-21 12:21:57 -08:00
|
|
|
// Ensure the reason was set in notes
|
|
|
|
if check.Notes != "broken" {
|
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
|
|
|
|
2015-01-15 00:25:36 -08:00
|
|
|
// Leave maintenance mode
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.DisableServiceMaintenance("redis"); err != nil {
|
2015-01-15 00:25:36 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the check was deregistered
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()[checkID]; ok {
|
2015-01-15 11:26:14 -08:00
|
|
|
t.Fatalf("should have deregistered maintenance check")
|
|
|
|
}
|
2015-01-21 12:21:57 -08:00
|
|
|
|
|
|
|
// Enter service maintenance mode without providing a reason
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.EnableServiceMaintenance("redis", "", ""); err != nil {
|
2015-01-21 12:21:57 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the check was registered with the default notes
|
2017-05-21 09:11:09 +02:00
|
|
|
check, ok = a.state.Checks()[checkID]
|
2015-01-21 12:21:57 -08:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("should have registered critical check")
|
|
|
|
}
|
2015-01-21 14:45:09 -08:00
|
|
|
if check.Notes != defaultServiceMaintReason {
|
2015-01-21 12:21:57 -08:00
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
2015-01-15 11:26:14 -08:00
|
|
|
}
|
|
|
|
|
2016-08-16 00:05:55 -07:00
|
|
|
func TestAgent_Service_Reap(t *testing.T) {
|
2017-07-04 12:44:24 +02:00
|
|
|
// t.Parallel() // timing test. no parallel
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
2017-07-05 11:24:03 +02:00
|
|
|
cfg.CheckReapInterval = 50 * time.Millisecond
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.CheckDeregisterIntervalMin = 0
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-08-16 00:05:55 -07:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes := []*structs.CheckType{
|
|
|
|
&structs.CheckType{
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2017-07-04 12:44:24 +02:00
|
|
|
TTL: 25 * time.Millisecond,
|
|
|
|
DeregisterCriticalServiceAfter: 200 * time.Millisecond,
|
2016-08-16 00:05:55 -07:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register the service.
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, chkTypes, false, ""); err != nil {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure it's there and there's no critical check yet.
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) > 0 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should not have critical checks")
|
|
|
|
}
|
|
|
|
|
2017-07-04 12:44:24 +02:00
|
|
|
// Wait for the check TTL to fail but before the check is reaped.
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) != 1 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have a critical check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pass the TTL.
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.updateTTLCheck("service:redis", api.HealthPassing, "foo"); err != nil {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) > 0 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should not have critical checks")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the check TTL to fail again.
|
2017-07-04 12:44:24 +02:00
|
|
|
time.Sleep(100 * time.Millisecond)
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) != 1 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have a critical check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the reap.
|
2017-07-04 12:44:24 +02:00
|
|
|
time.Sleep(400 * time.Millisecond)
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("redis service should have been reaped")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) > 0 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should not have critical checks")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAgent_Service_NoReap(t *testing.T) {
|
2017-07-04 12:44:24 +02:00
|
|
|
// t.Parallel() // timing test. no parallel
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
2017-07-05 11:24:03 +02:00
|
|
|
cfg.CheckReapInterval = 50 * time.Millisecond
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg.CheckDeregisterIntervalMin = 0
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2016-08-16 00:05:55 -07:00
|
|
|
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes := []*structs.CheckType{
|
|
|
|
&structs.CheckType{
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2017-07-04 12:44:24 +02:00
|
|
|
TTL: 25 * time.Millisecond,
|
2016-08-16 00:05:55 -07:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register the service.
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, chkTypes, false, ""); err != nil {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure it's there and there's no critical check yet.
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) > 0 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should not have critical checks")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the check TTL to fail.
|
2017-07-05 11:24:03 +02:00
|
|
|
time.Sleep(200 * time.Millisecond)
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) != 1 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have a critical check")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait a while and make sure it doesn't reap.
|
2017-07-04 12:44:24 +02:00
|
|
|
time.Sleep(200 * time.Millisecond)
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Services()["redis"]; !ok {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have redis service")
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if checks := a.state.CriticalChecks(); len(checks) != 1 {
|
2016-08-16 00:05:55 -07:00
|
|
|
t.Fatalf("should have a critical check")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-06 12:28:42 -07:00
|
|
|
func TestAgent_addCheck_restoresSnapshot(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-05-06 12:28:42 -07:00
|
|
|
|
|
|
|
// First register a service
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2015-05-06 12:28:42 -07:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a check
|
|
|
|
check1 := &structs.HealthCheck{
|
2017-05-21 09:11:09 +02:00
|
|
|
Node: a.Config.NodeName,
|
2015-05-06 12:28:42 -07:00
|
|
|
CheckID: "service:redis",
|
|
|
|
Name: "redischeck",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2015-05-06 12:28:42 -07:00
|
|
|
ServiceID: "redis",
|
|
|
|
ServiceName: "redis",
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check1, nil, false, ""); err != nil {
|
2015-05-06 12:28:42 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Re-registering the service preserves the state of the check
|
2017-06-15 18:46:06 +02:00
|
|
|
chkTypes := []*structs.CheckType{&structs.CheckType{TTL: 30 * time.Second}}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, chkTypes, false, ""); err != nil {
|
2015-05-06 12:28:42 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
check, ok := a.state.Checks()["service:redis"]
|
2015-05-06 12:28:42 -07:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("missing check")
|
|
|
|
}
|
2017-04-19 16:00:11 -07:00
|
|
|
if check.Status != api.HealthPassing {
|
2015-05-06 12:28:42 -07:00
|
|
|
t.Fatalf("bad: %s", check.Status)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 11:26:14 -08:00
|
|
|
func TestAgent_NodeMaintenanceMode(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-01-15 11:26:14 -08:00
|
|
|
|
|
|
|
// Enter maintenance mode for the node
|
2017-05-21 09:11:09 +02:00
|
|
|
a.EnableNodeMaintenance("broken", "mytoken")
|
2015-01-15 11:26:14 -08:00
|
|
|
|
|
|
|
// Make sure the critical health check was added
|
2017-05-21 09:11:09 +02:00
|
|
|
check, ok := a.state.Checks()[structs.NodeMaint]
|
2015-01-21 12:21:57 -08:00
|
|
|
if !ok {
|
2015-01-15 11:26:14 -08:00
|
|
|
t.Fatalf("should have registered critical node check")
|
|
|
|
}
|
|
|
|
|
2015-09-10 12:08:08 -07:00
|
|
|
// Check that the token was used to register the check
|
2017-05-21 09:11:09 +02:00
|
|
|
if token := a.state.CheckToken(structs.NodeMaint); token != "mytoken" {
|
2015-09-10 12:08:08 -07:00
|
|
|
t.Fatalf("expected 'mytoken', got: '%s'", token)
|
|
|
|
}
|
|
|
|
|
2015-01-21 12:21:57 -08:00
|
|
|
// Ensure the reason was set in notes
|
|
|
|
if check.Notes != "broken" {
|
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
|
|
|
|
2015-01-15 11:26:14 -08:00
|
|
|
// Leave maintenance mode
|
2017-05-21 09:11:09 +02:00
|
|
|
a.DisableNodeMaintenance()
|
2015-01-15 11:26:14 -08:00
|
|
|
|
|
|
|
// Ensure the check was deregistered
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, ok := a.state.Checks()[structs.NodeMaint]; ok {
|
2015-01-15 11:26:14 -08:00
|
|
|
t.Fatalf("should have deregistered critical node check")
|
2015-01-15 00:25:36 -08:00
|
|
|
}
|
2015-01-21 12:21:57 -08:00
|
|
|
|
|
|
|
// Enter maintenance mode without passing a reason
|
2017-05-21 09:11:09 +02:00
|
|
|
a.EnableNodeMaintenance("", "")
|
2015-01-21 12:21:57 -08:00
|
|
|
|
|
|
|
// Make sure the check was registered with the default note
|
2017-05-21 09:11:09 +02:00
|
|
|
check, ok = a.state.Checks()[structs.NodeMaint]
|
2015-01-21 12:21:57 -08:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("should have registered critical node check")
|
|
|
|
}
|
2015-01-21 14:45:09 -08:00
|
|
|
if check.Notes != defaultNodeMaintReason {
|
2015-01-21 12:21:57 -08:00
|
|
|
t.Fatalf("bad: %#v", check)
|
|
|
|
}
|
2015-01-15 00:25:36 -08:00
|
|
|
}
|
2015-02-17 12:00:04 -08:00
|
|
|
|
|
|
|
func TestAgent_checkStateSnapshot(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-02-17 12:00:04 -08:00
|
|
|
|
|
|
|
// First register a service
|
|
|
|
svc := &structs.NodeService{
|
|
|
|
ID: "redis",
|
|
|
|
Service: "redis",
|
|
|
|
Tags: []string{"foo"},
|
|
|
|
Port: 8000,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddService(svc, nil, false, ""); err != nil {
|
2015-02-17 12:00:04 -08:00
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a check
|
|
|
|
check1 := &structs.HealthCheck{
|
2017-05-21 09:11:09 +02:00
|
|
|
Node: a.Config.NodeName,
|
2015-02-17 12:00:04 -08:00
|
|
|
CheckID: "service:redis",
|
|
|
|
Name: "redischeck",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2015-02-17 12:00:04 -08:00
|
|
|
ServiceID: "redis",
|
|
|
|
ServiceName: "redis",
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.AddCheck(check1, nil, true, ""); err != nil {
|
2015-02-17 12:00:04 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Snapshot the state
|
2017-05-21 09:11:09 +02:00
|
|
|
snap := a.snapshotCheckState()
|
2015-02-17 12:00:04 -08:00
|
|
|
|
|
|
|
// Unload all of the checks
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.unloadChecks(); err != nil {
|
2015-02-17 12:00:04 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reload the checks
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.loadChecks(a.Config); err != nil {
|
2015-02-17 12:00:04 -08:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Restore the state
|
2017-05-21 09:11:09 +02:00
|
|
|
a.restoreCheckState(snap)
|
2015-02-17 12:00:04 -08:00
|
|
|
|
|
|
|
// Search for the check
|
2017-05-21 09:11:09 +02:00
|
|
|
out, ok := a.state.Checks()[check1.CheckID]
|
2015-02-17 12:00:04 -08:00
|
|
|
if !ok {
|
|
|
|
t.Fatalf("check should have been registered")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure state was restored
|
2017-04-19 16:00:11 -07:00
|
|
|
if out.Status != api.HealthPassing {
|
2015-02-17 12:00:04 -08:00
|
|
|
t.Fatalf("should have restored check state")
|
|
|
|
}
|
|
|
|
}
|
2015-03-11 16:13:19 -07:00
|
|
|
|
|
|
|
func TestAgent_loadChecks_checkFails(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-03-11 16:13:19 -07:00
|
|
|
|
|
|
|
// Persist a health check with an invalid service ID
|
|
|
|
check := &structs.HealthCheck{
|
2017-05-21 09:11:09 +02:00
|
|
|
Node: a.Config.NodeName,
|
2015-03-11 16:13:19 -07:00
|
|
|
CheckID: "service:redis",
|
|
|
|
Name: "redischeck",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthPassing,
|
2015-03-11 16:13:19 -07:00
|
|
|
ServiceID: "nope",
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.persistCheck(check, nil); err != nil {
|
2015-03-11 16:13:19 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check to make sure the check was persisted
|
2016-06-06 01:53:30 -07:00
|
|
|
checkHash := checkIDHash(check.CheckID)
|
2017-05-21 09:11:09 +02:00
|
|
|
checkPath := filepath.Join(a.Config.DataDir, checksDir, checkHash)
|
2015-03-11 16:13:19 -07:00
|
|
|
if _, err := os.Stat(checkPath); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try loading the checks from the persisted files
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.loadChecks(a.Config); err != nil {
|
2015-03-11 16:13:19 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the erroneous check was purged
|
|
|
|
if _, err := os.Stat(checkPath); err == nil {
|
|
|
|
t.Fatalf("should have purged check")
|
|
|
|
}
|
|
|
|
}
|
2015-06-05 16:45:05 -07:00
|
|
|
|
2015-06-05 16:59:41 -07:00
|
|
|
func TestAgent_persistCheckState(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-06-05 16:45:05 -07:00
|
|
|
|
|
|
|
// Create the TTL check to persist
|
|
|
|
check := &CheckTTL{
|
|
|
|
CheckID: "check1",
|
|
|
|
TTL: 10 * time.Minute,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Persist some check state for the check
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.persistCheckState(check, api.HealthCritical, "nope")
|
2015-06-05 16:45:05 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the persisted file exists and has the content
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checkStateDir, stringHash("check1"))
|
2015-06-05 16:45:05 -07:00
|
|
|
buf, err := ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decode the state
|
|
|
|
var p persistedCheckState
|
|
|
|
if err := json.Unmarshal(buf, &p); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the fields
|
|
|
|
if p.CheckID != "check1" {
|
|
|
|
t.Fatalf("bad: %#v", p)
|
|
|
|
}
|
|
|
|
if p.Output != "nope" {
|
|
|
|
t.Fatalf("bad: %#v", p)
|
|
|
|
}
|
2017-04-19 16:00:11 -07:00
|
|
|
if p.Status != api.HealthCritical {
|
2015-06-05 16:45:05 -07:00
|
|
|
t.Fatalf("bad: %#v", p)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the expiration time was set
|
|
|
|
if p.Expires < time.Now().Unix() {
|
|
|
|
t.Fatalf("bad: %#v", p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-08 09:35:10 -07:00
|
|
|
func TestAgent_loadCheckState(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-06-05 16:45:05 -07:00
|
|
|
|
|
|
|
// Create a check whose state will expire immediately
|
|
|
|
check := &CheckTTL{
|
|
|
|
CheckID: "check1",
|
|
|
|
TTL: 0,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Persist the check state
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.persistCheckState(check, api.HealthPassing, "yup")
|
2015-06-05 16:45:05 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-06-08 09:35:10 -07:00
|
|
|
// Try to load the state
|
2015-06-05 16:45:05 -07:00
|
|
|
health := &structs.HealthCheck{
|
|
|
|
CheckID: "check1",
|
2017-04-19 16:00:11 -07:00
|
|
|
Status: api.HealthCritical,
|
2015-06-05 16:45:05 -07:00
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.loadCheckState(health); err != nil {
|
2015-06-05 16:45:05 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should not have restored the status due to expiration
|
2017-04-19 16:00:11 -07:00
|
|
|
if health.Status != api.HealthCritical {
|
2015-06-05 16:45:05 -07:00
|
|
|
t.Fatalf("bad: %#v", health)
|
|
|
|
}
|
|
|
|
if health.Output != "" {
|
|
|
|
t.Fatalf("bad: %#v", health)
|
|
|
|
}
|
|
|
|
|
2015-06-05 16:59:41 -07:00
|
|
|
// Should have purged the state
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checksDir, stringHash("check1"))
|
2015-06-05 16:59:41 -07:00
|
|
|
if _, err := os.Stat(file); !os.IsNotExist(err) {
|
|
|
|
t.Fatalf("should have purged state")
|
|
|
|
}
|
|
|
|
|
2015-06-05 16:45:05 -07:00
|
|
|
// Set a TTL which will not expire before we check it
|
|
|
|
check.TTL = time.Minute
|
2017-05-21 09:11:09 +02:00
|
|
|
err = a.persistCheckState(check, api.HealthPassing, "yup")
|
2015-06-05 16:45:05 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-06-08 09:35:10 -07:00
|
|
|
// Try to load
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.loadCheckState(health); err != nil {
|
2015-06-05 16:45:05 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should have restored
|
2017-04-19 16:00:11 -07:00
|
|
|
if health.Status != api.HealthPassing {
|
2015-06-05 16:45:05 -07:00
|
|
|
t.Fatalf("bad: %#v", health)
|
|
|
|
}
|
|
|
|
if health.Output != "yup" {
|
|
|
|
t.Fatalf("bad: %#v", health)
|
|
|
|
}
|
|
|
|
}
|
2015-06-05 16:57:14 -07:00
|
|
|
|
|
|
|
func TestAgent_purgeCheckState(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2017-05-21 09:11:09 +02:00
|
|
|
a := NewTestAgent(t.Name(), nil)
|
|
|
|
defer a.Shutdown()
|
2015-06-05 16:57:14 -07:00
|
|
|
|
|
|
|
// No error if the state does not exist
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.purgeCheckState("check1"); err != nil {
|
2015-06-05 16:57:14 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Persist some state to the data dir
|
|
|
|
check := &CheckTTL{
|
|
|
|
CheckID: "check1",
|
|
|
|
TTL: time.Minute,
|
|
|
|
}
|
2017-05-21 09:11:09 +02:00
|
|
|
err := a.persistCheckState(check, api.HealthPassing, "yup")
|
2015-06-05 16:57:14 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Purge the check state
|
2017-05-21 09:11:09 +02:00
|
|
|
if err := a.purgeCheckState("check1"); err != nil {
|
2015-06-05 16:57:14 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Removed the file
|
2017-05-21 09:11:09 +02:00
|
|
|
file := filepath.Join(a.Config.DataDir, checkStateDir, stringHash("check1"))
|
2015-06-05 16:57:14 -07:00
|
|
|
if _, err := os.Stat(file); !os.IsNotExist(err) {
|
|
|
|
t.Fatalf("should have removed file")
|
|
|
|
}
|
|
|
|
}
|
2015-10-15 19:28:31 -07:00
|
|
|
|
|
|
|
func TestAgent_GetCoordinate(t *testing.T) {
|
2017-05-21 09:54:40 +02:00
|
|
|
t.Parallel()
|
2015-10-15 19:28:31 -07:00
|
|
|
check := func(server bool) {
|
2017-05-22 13:03:59 +02:00
|
|
|
cfg := TestConfig()
|
|
|
|
cfg.Server = server
|
|
|
|
a := NewTestAgent(t.Name(), cfg)
|
2017-05-21 09:11:09 +02:00
|
|
|
defer a.Shutdown()
|
2015-10-15 19:28:31 -07:00
|
|
|
|
|
|
|
// This doesn't verify the returned coordinate, but it makes
|
|
|
|
// sure that the agent chooses the correct Serf instance,
|
|
|
|
// depending on how it's configured as a client or a server.
|
|
|
|
// If it chooses the wrong one, this will crash.
|
2017-05-21 09:11:09 +02:00
|
|
|
if _, err := a.GetLANCoordinate(); err != nil {
|
2015-10-15 19:28:31 -07:00
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
check(true)
|
|
|
|
check(false)
|
|
|
|
}
|