2014-01-20 16:58:05 -10:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/hashicorp/consul/consul/structs"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type MockNotify struct {
|
|
|
|
state map[string]string
|
|
|
|
updates map[string]int
|
2014-04-29 15:28:56 -07:00
|
|
|
output map[string]string
|
2014-01-20 16:58:05 -10:00
|
|
|
}
|
|
|
|
|
2014-04-29 15:28:56 -07:00
|
|
|
func (m *MockNotify) UpdateCheck(id, status, output string) {
|
2014-01-20 16:58:05 -10:00
|
|
|
m.state[id] = status
|
|
|
|
old := m.updates[id]
|
|
|
|
m.updates[id] = old + 1
|
2014-04-29 15:28:56 -07:00
|
|
|
m.output[id] = output
|
2014-01-20 16:58:05 -10:00
|
|
|
}
|
|
|
|
|
|
|
|
func expectStatus(t *testing.T, script, status string) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
2014-04-29 15:28:56 -07:00
|
|
|
output: make(map[string]string),
|
2014-01-20 16:58:05 -10:00
|
|
|
}
|
|
|
|
check := &CheckMonitor{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: script,
|
2014-05-09 03:41:10 +02:00
|
|
|
Interval: 10 * time.Millisecond,
|
2014-01-20 16:58:05 -10:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
|
|
|
|
|
|
// Should have at least 2 updates
|
|
|
|
if mock.updates["foo"] < 2 {
|
|
|
|
t.Fatalf("should have 2 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["foo"] != status {
|
|
|
|
t.Fatalf("should be %v %v", status, mock.state)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckMonitor_Passing(t *testing.T) {
|
|
|
|
expectStatus(t, "exit 0", structs.HealthPassing)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckMonitor_Warning(t *testing.T) {
|
|
|
|
expectStatus(t, "exit 1", structs.HealthWarning)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckMonitor_Critical(t *testing.T) {
|
|
|
|
expectStatus(t, "exit 2", structs.HealthCritical)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckMonitor_BadCmd(t *testing.T) {
|
|
|
|
expectStatus(t, "foobarbaz", structs.HealthCritical)
|
|
|
|
}
|
2014-01-20 17:12:40 -10:00
|
|
|
|
2014-12-17 22:39:11 -05:00
|
|
|
func TestCheckMonitor_RandomStagger(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckMonitor{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: "exit 0",
|
2014-12-18 09:00:51 -05:00
|
|
|
Interval: 25 * time.Millisecond,
|
2014-12-17 22:39:11 -05:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
2014-12-18 09:00:51 -05:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2014-12-17 22:39:11 -05:00
|
|
|
|
|
|
|
// Should have at least 1 update
|
|
|
|
if mock.updates["foo"] < 1 {
|
2014-12-18 09:00:51 -05:00
|
|
|
t.Fatalf("should have 1 or more updates %v", mock.updates)
|
2014-12-17 22:39:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["foo"] != structs.HealthPassing {
|
|
|
|
t.Fatalf("should be %v %v", structs.HealthPassing, mock.state)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-29 15:28:56 -07:00
|
|
|
func TestCheckMonitor_LimitOutput(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckMonitor{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
2014-05-26 13:13:56 -07:00
|
|
|
Script: "od -N 81920 /dev/urandom",
|
2014-04-29 15:28:56 -07:00
|
|
|
Interval: 25 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
|
|
|
|
|
|
// Allow for extra bytes for the truncation message
|
|
|
|
if len(mock.output["foo"]) > CheckBufSize+100 {
|
|
|
|
t.Fatalf("output size is too long")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-20 17:12:40 -10:00
|
|
|
func TestCheckTTL(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
2014-04-29 15:28:56 -07:00
|
|
|
output: make(map[string]string),
|
2014-01-20 17:12:40 -10:00
|
|
|
}
|
|
|
|
check := &CheckTTL{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
2014-05-06 20:54:27 +02:00
|
|
|
TTL: 100 * time.Millisecond,
|
2014-01-20 17:12:40 -10:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
2014-05-06 20:54:27 +02:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2014-01-21 11:52:25 -08:00
|
|
|
check.SetStatus(structs.HealthPassing, "")
|
2014-01-20 17:12:40 -10:00
|
|
|
|
|
|
|
if mock.updates["foo"] != 1 {
|
|
|
|
t.Fatalf("should have 1 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["foo"] != structs.HealthPassing {
|
|
|
|
t.Fatalf("should be passing %v", mock.state)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we don't fail early
|
2014-05-06 20:54:27 +02:00
|
|
|
time.Sleep(75 * time.Millisecond)
|
2014-01-20 17:12:40 -10:00
|
|
|
if mock.updates["foo"] != 1 {
|
|
|
|
t.Fatalf("should have 1 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the TTL to expire
|
2014-05-06 20:54:27 +02:00
|
|
|
time.Sleep(75 * time.Millisecond)
|
2014-01-20 17:12:40 -10:00
|
|
|
|
|
|
|
if mock.updates["foo"] != 2 {
|
|
|
|
t.Fatalf("should have 2 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["foo"] != structs.HealthCritical {
|
|
|
|
t.Fatalf("should be critical %v", mock.state)
|
|
|
|
}
|
|
|
|
}
|