2014-01-21 02:58:05 +00:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
2015-10-26 19:59:40 +00:00
|
|
|
"errors"
|
2015-01-13 00:09:42 +00:00
|
|
|
"fmt"
|
2014-01-21 02:58:05 +00:00
|
|
|
"log"
|
2015-07-23 11:45:08 +00:00
|
|
|
"net"
|
2015-01-13 00:09:42 +00:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2014-01-21 02:58:05 +00:00
|
|
|
"os"
|
2015-10-27 01:06:55 +00:00
|
|
|
"os/exec"
|
2016-03-03 01:58:01 +00:00
|
|
|
"strings"
|
2016-01-13 05:10:25 +00:00
|
|
|
"sync"
|
2014-01-21 02:58:05 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
2015-01-24 00:07:20 +00:00
|
|
|
|
2015-10-26 19:59:40 +00:00
|
|
|
docker "github.com/fsouza/go-dockerclient"
|
2015-01-24 00:07:20 +00:00
|
|
|
"github.com/hashicorp/consul/consul/structs"
|
|
|
|
"github.com/hashicorp/consul/testutil"
|
2014-01-21 02:58:05 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type MockNotify struct {
|
|
|
|
state map[string]string
|
|
|
|
updates map[string]int
|
2014-04-29 22:28:56 +00:00
|
|
|
output map[string]string
|
2014-01-21 02:58:05 +00:00
|
|
|
}
|
|
|
|
|
2014-04-29 22:28:56 +00:00
|
|
|
func (m *MockNotify) UpdateCheck(id, status, output string) {
|
2014-01-21 02:58:05 +00:00
|
|
|
m.state[id] = status
|
|
|
|
old := m.updates[id]
|
|
|
|
m.updates[id] = old + 1
|
2014-04-29 22:28:56 +00:00
|
|
|
m.output[id] = output
|
2014-01-21 02:58:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func expectStatus(t *testing.T, script, status string) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
2014-04-29 22:28:56 +00:00
|
|
|
output: make(map[string]string),
|
2014-01-21 02:58:05 +00:00
|
|
|
}
|
|
|
|
check := &CheckMonitor{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: script,
|
2014-05-09 01:41:10 +00:00
|
|
|
Interval: 10 * time.Millisecond,
|
2014-01-21 02:58:05 +00:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
2016-01-13 05:10:25 +00:00
|
|
|
ReapLock: &sync.RWMutex{},
|
2014-01-21 02:58:05 +00:00
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
2015-01-24 00:07:20 +00:00
|
|
|
testutil.WaitForResult(func() (bool, error) {
|
|
|
|
// Should have at least 2 updates
|
|
|
|
if mock.updates["foo"] < 2 {
|
|
|
|
return false, fmt.Errorf("should have 2 updates %v", mock.updates)
|
|
|
|
}
|
2014-01-21 02:58:05 +00:00
|
|
|
|
2015-01-24 00:07:20 +00:00
|
|
|
if mock.state["foo"] != status {
|
|
|
|
return false, fmt.Errorf("should be %v %v", status, mock.state)
|
|
|
|
}
|
2014-01-21 02:58:05 +00:00
|
|
|
|
2015-01-24 00:07:20 +00:00
|
|
|
return true, nil
|
|
|
|
}, func(err error) {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
})
|
2014-01-21 02:58:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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-21 03:12:40 +00:00
|
|
|
|
2014-12-18 03:39:11 +00: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 14:00:51 +00:00
|
|
|
Interval: 25 * time.Millisecond,
|
2014-12-18 03:39:11 +00:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
2016-01-13 05:10:25 +00:00
|
|
|
ReapLock: &sync.RWMutex{},
|
2014-12-18 03:39:11 +00:00
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
2014-12-18 14:00:51 +00:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2014-12-18 03:39:11 +00:00
|
|
|
|
|
|
|
// Should have at least 1 update
|
|
|
|
if mock.updates["foo"] < 1 {
|
2014-12-18 14:00:51 +00:00
|
|
|
t.Fatalf("should have 1 or more updates %v", mock.updates)
|
2014-12-18 03:39:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["foo"] != structs.HealthPassing {
|
|
|
|
t.Fatalf("should be %v %v", structs.HealthPassing, mock.state)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-29 22:28:56 +00: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 20:13:56 +00:00
|
|
|
Script: "od -N 81920 /dev/urandom",
|
2014-04-29 22:28:56 +00:00
|
|
|
Interval: 25 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
2016-01-13 05:10:25 +00:00
|
|
|
ReapLock: &sync.RWMutex{},
|
2014-04-29 22:28:56 +00:00
|
|
|
}
|
|
|
|
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-21 03:12:40 +00:00
|
|
|
func TestCheckTTL(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
2014-04-29 22:28:56 +00:00
|
|
|
output: make(map[string]string),
|
2014-01-21 03:12:40 +00:00
|
|
|
}
|
|
|
|
check := &CheckTTL{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
2014-05-06 18:54:27 +00:00
|
|
|
TTL: 100 * time.Millisecond,
|
2014-01-21 03:12:40 +00:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
2014-05-06 18:54:27 +00:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2016-03-03 01:58:01 +00:00
|
|
|
check.SetStatus(structs.HealthPassing, "test-output")
|
2014-01-21 03:12:40 +00: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 18:54:27 +00:00
|
|
|
time.Sleep(75 * time.Millisecond)
|
2014-01-21 03:12:40 +00:00
|
|
|
if mock.updates["foo"] != 1 {
|
|
|
|
t.Fatalf("should have 1 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the TTL to expire
|
2014-05-06 18:54:27 +00:00
|
|
|
time.Sleep(75 * time.Millisecond)
|
2014-01-21 03:12:40 +00: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)
|
|
|
|
}
|
2016-03-03 01:58:01 +00:00
|
|
|
|
|
|
|
if !strings.Contains(mock.output["foo"], "test-output") {
|
|
|
|
t.Fatalf("should have retained output %v", mock.output)
|
|
|
|
}
|
2014-01-21 03:12:40 +00:00
|
|
|
}
|
2015-01-13 00:09:42 +00:00
|
|
|
|
|
|
|
func mockHTTPServer(responseCode int) *httptest.Server {
|
|
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.WriteHeader(responseCode)
|
|
|
|
return
|
|
|
|
})
|
|
|
|
|
|
|
|
return httptest.NewServer(mux)
|
|
|
|
}
|
|
|
|
|
|
|
|
func expectHTTPStatus(t *testing.T, url string, status string) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckHTTP{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
HTTP: url,
|
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
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 TestCheckHTTPCritical(t *testing.T) {
|
|
|
|
// var server *httptest.Server
|
|
|
|
|
|
|
|
server := mockHTTPServer(150)
|
|
|
|
fmt.Println(server.URL)
|
|
|
|
expectHTTPStatus(t, server.URL, "critical")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
// 2xx - 1
|
|
|
|
server = mockHTTPServer(199)
|
|
|
|
expectHTTPStatus(t, server.URL, "critical")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
// 2xx + 1
|
|
|
|
server = mockHTTPServer(300)
|
|
|
|
expectHTTPStatus(t, server.URL, "critical")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
server = mockHTTPServer(400)
|
|
|
|
expectHTTPStatus(t, server.URL, "critical")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
server = mockHTTPServer(500)
|
|
|
|
expectHTTPStatus(t, server.URL, "critical")
|
|
|
|
server.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckHTTPPassing(t *testing.T) {
|
|
|
|
var server *httptest.Server
|
|
|
|
|
|
|
|
server = mockHTTPServer(200)
|
|
|
|
expectHTTPStatus(t, server.URL, "passing")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
server = mockHTTPServer(201)
|
|
|
|
expectHTTPStatus(t, server.URL, "passing")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
server = mockHTTPServer(250)
|
|
|
|
expectHTTPStatus(t, server.URL, "passing")
|
|
|
|
server.Close()
|
|
|
|
|
|
|
|
server = mockHTTPServer(299)
|
|
|
|
expectHTTPStatus(t, server.URL, "passing")
|
|
|
|
server.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckHTTPWarning(t *testing.T) {
|
|
|
|
server := mockHTTPServer(429)
|
|
|
|
expectHTTPStatus(t, server.URL, "warning")
|
|
|
|
server.Close()
|
|
|
|
}
|
2015-02-02 08:30:44 +00:00
|
|
|
|
|
|
|
func mockSlowHTTPServer(responseCode int, sleep time.Duration) *httptest.Server {
|
|
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
time.Sleep(sleep)
|
|
|
|
w.WriteHeader(responseCode)
|
|
|
|
return
|
|
|
|
})
|
|
|
|
|
|
|
|
return httptest.NewServer(mux)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckHTTPTimeout(t *testing.T) {
|
|
|
|
server := mockSlowHTTPServer(200, 10*time.Millisecond)
|
|
|
|
defer server.Close()
|
|
|
|
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
|
|
|
|
check := &CheckHTTP{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "bar",
|
|
|
|
HTTP: server.URL,
|
|
|
|
Timeout: 5 * time.Millisecond,
|
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
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["bar"] < 2 {
|
|
|
|
t.Fatalf("should have at least 2 updates %v", mock.updates)
|
|
|
|
}
|
|
|
|
|
|
|
|
if mock.state["bar"] != "critical" {
|
|
|
|
t.Fatalf("should be critical %v", mock.state)
|
|
|
|
}
|
|
|
|
}
|
2015-03-15 20:30:50 +00:00
|
|
|
|
|
|
|
func TestCheckHTTP_disablesKeepAlives(t *testing.T) {
|
|
|
|
check := &CheckHTTP{
|
|
|
|
CheckID: "foo",
|
|
|
|
HTTP: "http://foo.bar/baz",
|
|
|
|
Interval: 10 * time.Second,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
|
|
|
if !check.httpClient.Transport.(*http.Transport).DisableKeepAlives {
|
|
|
|
t.Fatalf("should have disabled keepalives")
|
|
|
|
}
|
|
|
|
}
|
2015-07-23 11:45:08 +00:00
|
|
|
|
|
|
|
func mockTCPServer(network string) net.Listener {
|
|
|
|
var (
|
|
|
|
addr string
|
|
|
|
)
|
|
|
|
|
|
|
|
if network == `tcp6` {
|
|
|
|
addr = `[::1]:0`
|
|
|
|
} else {
|
|
|
|
addr = `127.0.0.1:0`
|
|
|
|
}
|
|
|
|
|
|
|
|
listener, err := net.Listen(network, addr)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return listener
|
|
|
|
}
|
|
|
|
|
|
|
|
func expectTCPStatus(t *testing.T, tcp string, status string) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckTCP{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
TCP: tcp,
|
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
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 TestCheckTCPCritical(t *testing.T) {
|
|
|
|
var (
|
|
|
|
tcpServer net.Listener
|
|
|
|
)
|
|
|
|
|
|
|
|
tcpServer = mockTCPServer(`tcp`)
|
|
|
|
expectTCPStatus(t, `127.0.0.1:0`, "critical")
|
|
|
|
tcpServer.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckTCPPassing(t *testing.T) {
|
|
|
|
var (
|
|
|
|
tcpServer net.Listener
|
|
|
|
)
|
|
|
|
|
|
|
|
tcpServer = mockTCPServer(`tcp`)
|
|
|
|
expectTCPStatus(t, tcpServer.Addr().String(), "passing")
|
|
|
|
tcpServer.Close()
|
|
|
|
|
|
|
|
tcpServer = mockTCPServer(`tcp6`)
|
|
|
|
expectTCPStatus(t, tcpServer.Addr().String(), "passing")
|
|
|
|
tcpServer.Close()
|
|
|
|
}
|
2015-10-26 19:59:40 +00:00
|
|
|
|
|
|
|
// A fake docker client to test happy path scenario
|
|
|
|
type fakeDockerClientWithNoErrors struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithNoErrors) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithNoErrors) StartExec(id string, opts docker.StartExecOptions) error {
|
2015-10-27 01:06:55 +00:00
|
|
|
fmt.Fprint(opts.OutputStream, "output")
|
2015-10-26 19:59:40 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithNoErrors) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return &docker.ExecInspect{
|
|
|
|
ID: "123",
|
|
|
|
ExitCode: 0,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2015-10-27 01:06:55 +00:00
|
|
|
// A fake docker client to test truncation of output
|
|
|
|
type fakeDockerClientWithLongOutput struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithLongOutput) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithLongOutput) StartExec(id string, opts docker.StartExecOptions) error {
|
|
|
|
b, _ := exec.Command("od", "-N", "81920", "/dev/urandom").Output()
|
|
|
|
fmt.Fprint(opts.OutputStream, string(b))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithLongOutput) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return &docker.ExecInspect{
|
|
|
|
ID: "123",
|
|
|
|
ExitCode: 0,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2015-10-26 19:59:40 +00:00
|
|
|
// A fake docker client to test non-zero exit codes from exec invocation
|
|
|
|
type fakeDockerClientWithExecNonZeroExitCode struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecNonZeroExitCode) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecNonZeroExitCode) StartExec(id string, opts docker.StartExecOptions) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecNonZeroExitCode) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return &docker.ExecInspect{
|
|
|
|
ID: "123",
|
|
|
|
ExitCode: 127,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2015-10-26 23:58:09 +00:00
|
|
|
// A fake docker client to test exit code which result into Warning
|
|
|
|
type fakeDockerClientWithExecExitCodeOne struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecExitCodeOne) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecExitCodeOne) StartExec(id string, opts docker.StartExecOptions) error {
|
2015-10-27 01:06:55 +00:00
|
|
|
fmt.Fprint(opts.OutputStream, "output")
|
2015-10-26 23:58:09 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecExitCodeOne) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return &docker.ExecInspect{
|
|
|
|
ID: "123",
|
|
|
|
ExitCode: 1,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2015-10-26 19:59:40 +00:00
|
|
|
// A fake docker client to simulate create exec failing
|
|
|
|
type fakeDockerClientWithCreateExecFailure struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithCreateExecFailure) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return nil, errors.New("Exec Creation Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithCreateExecFailure) StartExec(id string, opts docker.StartExecOptions) error {
|
|
|
|
return errors.New("Exec doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithCreateExecFailure) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return nil, errors.New("Exec doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
// A fake docker client to simulate start exec failing
|
|
|
|
type fakeDockerClientWithStartExecFailure struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithStartExecFailure) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithStartExecFailure) StartExec(id string, opts docker.StartExecOptions) error {
|
2015-10-26 23:54:51 +00:00
|
|
|
return errors.New("Couldn't Start Exec")
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithStartExecFailure) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return nil, errors.New("Exec doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
// A fake docker client to test exec info query failures
|
|
|
|
type fakeDockerClientWithExecInfoErrors struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecInfoErrors) CreateExec(opts docker.CreateExecOptions) (*docker.Exec, error) {
|
|
|
|
return &docker.Exec{ID: "123"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecInfoErrors) StartExec(id string, opts docker.StartExecOptions) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *fakeDockerClientWithExecInfoErrors) InspectExec(id string) (*docker.ExecInspect, error) {
|
|
|
|
return nil, errors.New("Unable to query exec info")
|
|
|
|
}
|
|
|
|
|
2015-10-27 01:20:02 +00:00
|
|
|
func expectDockerCheckStatus(t *testing.T, dockerClient DockerClient, status string, output string) {
|
2015-10-26 19:59:40 +00:00
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckDocker{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: "/health.sh",
|
2015-11-18 15:40:02 +00:00
|
|
|
DockerContainerID: "54432bad1fc7",
|
2015-10-26 19:59:40 +00:00
|
|
|
Shell: "/bin/sh",
|
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
dockerClient: dockerClient,
|
|
|
|
}
|
|
|
|
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)
|
|
|
|
}
|
2015-10-27 01:06:55 +00:00
|
|
|
|
2015-10-27 01:20:02 +00:00
|
|
|
if mock.output["foo"] != output {
|
|
|
|
t.Fatalf("should be %v %v", output, mock.output)
|
2015-10-27 01:06:55 +00:00
|
|
|
}
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDockerCheckWhenExecReturnsSuccessExitCode(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithNoErrors{}, "passing", "output")
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDockerCheckWhenExecCreationFails(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithCreateExecFailure{}, "critical", "Unable to create Exec, error: Exec Creation Failed")
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDockerCheckWhenExitCodeIsNonZero(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithExecNonZeroExitCode{}, "critical", "")
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
2015-10-26 23:58:09 +00:00
|
|
|
func TestDockerCheckWhenExitCodeIsone(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithExecExitCodeOne{}, "warning", "output")
|
2015-10-26 23:58:09 +00:00
|
|
|
}
|
|
|
|
|
2015-10-26 19:59:40 +00:00
|
|
|
func TestDockerCheckWhenExecStartFails(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithStartExecFailure{}, "critical", "Unable to start Exec: Couldn't Start Exec")
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDockerCheckWhenExecInfoFails(t *testing.T) {
|
2015-10-27 01:20:02 +00:00
|
|
|
expectDockerCheckStatus(t, &fakeDockerClientWithExecInfoErrors{}, "critical", "Unable to inspect Exec: Unable to query exec info")
|
2015-10-27 01:06:55 +00:00
|
|
|
}
|
|
|
|
|
2015-10-27 03:21:50 +00:00
|
|
|
func TestDockerCheckDefaultToSh(t *testing.T) {
|
2015-10-27 03:35:13 +00:00
|
|
|
os.Setenv("SHELL", "")
|
2015-10-27 03:21:50 +00:00
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckDocker{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: "/health.sh",
|
2015-11-18 15:40:02 +00:00
|
|
|
DockerContainerID: "54432bad1fc7",
|
2015-10-27 03:21:50 +00:00
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
dockerClient: &fakeDockerClientWithNoErrors{},
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
|
|
if check.Shell != "/bin/sh" {
|
|
|
|
t.Fatalf("Shell should be: %v , actual: %v", "/bin/sh", check.Shell)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDockerCheckUseShellFromEnv(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
os.Setenv("SHELL", "/bin/bash")
|
|
|
|
check := &CheckDocker{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: "/health.sh",
|
2015-11-18 15:40:02 +00:00
|
|
|
DockerContainerID: "54432bad1fc7",
|
2015-10-27 03:21:50 +00:00
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
dockerClient: &fakeDockerClientWithNoErrors{},
|
|
|
|
}
|
|
|
|
check.Start()
|
|
|
|
defer check.Stop()
|
|
|
|
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
|
|
if check.Shell != "/bin/bash" {
|
|
|
|
t.Fatalf("Shell should be: %v , actual: %v", "/bin/bash", check.Shell)
|
|
|
|
}
|
|
|
|
os.Setenv("SHELL", "")
|
|
|
|
}
|
|
|
|
|
2015-10-27 01:06:55 +00:00
|
|
|
func TestDockerCheckTruncateOutput(t *testing.T) {
|
|
|
|
mock := &MockNotify{
|
|
|
|
state: make(map[string]string),
|
|
|
|
updates: make(map[string]int),
|
|
|
|
output: make(map[string]string),
|
|
|
|
}
|
|
|
|
check := &CheckDocker{
|
|
|
|
Notify: mock,
|
|
|
|
CheckID: "foo",
|
|
|
|
Script: "/health.sh",
|
2015-11-18 15:40:02 +00:00
|
|
|
DockerContainerID: "54432bad1fc7",
|
2015-10-27 01:06:55 +00:00
|
|
|
Shell: "/bin/sh",
|
|
|
|
Interval: 10 * time.Millisecond,
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
dockerClient: &fakeDockerClientWithLongOutput{},
|
|
|
|
}
|
|
|
|
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")
|
|
|
|
}
|
|
|
|
|
2015-10-26 19:59:40 +00:00
|
|
|
}
|