Refactor a bit more for testing, update tests
This commit is contained in:
parent
5c5388d683
commit
6aaeeb32d4
@ -8,7 +8,7 @@ Initial target is meant to be roughly compatible requiring only minor changes to
|
|||||||
|
|
||||||
## Differences from Python version
|
## Differences from Python version
|
||||||
|
|
||||||
There are a few key differences between the Python version and the v0.x Go version.
|
There are a few key differences between the Python version and the v0.x Go version.
|
||||||
|
|
||||||
First, configuration keys cannot have multiple types in Go, so a different key must be used when specifying a Shell command as a string rather than a list of args. Instead of `command`, you must use `command_shell`. Eg:
|
First, configuration keys cannot have multiple types in Go, so a different key must be used when specifying a Shell command as a string rather than a list of args. Instead of `command`, you must use `command_shell`. Eg:
|
||||||
|
|
||||||
|
7
alert.go
7
alert.go
@ -38,18 +38,15 @@ func (alert Alert) IsValid() bool {
|
|||||||
|
|
||||||
// BuildTemplates compiles command templates for the Alert
|
// BuildTemplates compiles command templates for the Alert
|
||||||
func (alert *Alert) BuildTemplates() error {
|
func (alert *Alert) BuildTemplates() error {
|
||||||
|
log.Printf("DEBUG: Building template for alert %s", alert.Name)
|
||||||
if alert.commandTemplate == nil && alert.Command != nil {
|
if alert.commandTemplate == nil && alert.Command != nil {
|
||||||
// build template
|
|
||||||
log.Println("Building template for command...")
|
|
||||||
alert.commandTemplate = []*template.Template{}
|
alert.commandTemplate = []*template.Template{}
|
||||||
for i, cmdPart := range alert.Command {
|
for i, cmdPart := range alert.Command {
|
||||||
alert.commandTemplate = append(alert.commandTemplate, template.Must(
|
alert.commandTemplate = append(alert.commandTemplate, template.Must(
|
||||||
template.New(alert.Name+string(i)).Parse(cmdPart),
|
template.New(alert.Name+string(i)).Parse(cmdPart),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
log.Printf("Template built: %v", alert.commandTemplate)
|
|
||||||
} else if alert.commandShellTemplate == nil && alert.CommandShell != "" {
|
} else if alert.commandShellTemplate == nil && alert.CommandShell != "" {
|
||||||
log.Println("Building template for shell command...")
|
|
||||||
alert.commandShellTemplate = template.Must(
|
alert.commandShellTemplate = template.Must(
|
||||||
template.New(alert.Name).Parse(alert.CommandShell),
|
template.New(alert.Name).Parse(alert.CommandShell),
|
||||||
)
|
)
|
||||||
@ -96,7 +93,7 @@ func (alert Alert) Send(notice AlertNotice) (output_str string, err error) {
|
|||||||
var output []byte
|
var output []byte
|
||||||
output, err = cmd.CombinedOutput()
|
output, err = cmd.CombinedOutput()
|
||||||
output_str = string(output)
|
output_str = string(output)
|
||||||
log.Printf("Check %s\n---\n%s\n---", alert.Name, output_str)
|
// log.Printf("DEBUG: Alert output for: %s\n---\n%s\n---", alert.Name, output_str)
|
||||||
|
|
||||||
return output_str, err
|
return output_str, err
|
||||||
}
|
}
|
||||||
|
74
main.go
74
main.go
@ -5,44 +5,56 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func checkMonitors(config *Config) {
|
||||||
|
for _, monitor := range config.Monitors {
|
||||||
|
if monitor.ShouldCheck() {
|
||||||
|
_, alertNotice := monitor.Check()
|
||||||
|
|
||||||
|
// Should probably consider refactoring everything below here
|
||||||
|
if alertNotice != nil {
|
||||||
|
log.Printf("DEBUG: Recieved an alert notice: %v", alertNotice)
|
||||||
|
alertNames := monitor.GetAlertNames(alertNotice.IsUp)
|
||||||
|
if alertNames == nil {
|
||||||
|
// TODO: Should this be a panic? Should this be validated against? Probably
|
||||||
|
log.Printf("WARNING: Found alert, but no alert mechanisms exist: %v", alertNotice)
|
||||||
|
}
|
||||||
|
for _, alertName := range alertNames {
|
||||||
|
if alert, ok := config.Alerts[alertName]; ok {
|
||||||
|
output, err := alert.Send(*alertNotice)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf(
|
||||||
|
"ERROR: Alert '%s' failed. result=%v: output=%s",
|
||||||
|
alert.Name,
|
||||||
|
err,
|
||||||
|
output,
|
||||||
|
)
|
||||||
|
// TODO: Maybe return this error instead of panicking here
|
||||||
|
log.Fatalf(
|
||||||
|
"ERROR: Unsuccessfully triggered alert '%s'. "+
|
||||||
|
"Crashing to avoid false negatives: %v",
|
||||||
|
alert.Name,
|
||||||
|
err,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// TODO: Maybe panic here. Also, probably validate up front
|
||||||
|
log.Printf("ERROR: Alert with name '%s' not found", alertName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
config, err := LoadConfig("config.yml")
|
config, err := LoadConfig("config.yml")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error loading config: %v", err)
|
log.Fatalf("Error loading config: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start main loop
|
||||||
for {
|
for {
|
||||||
for _, monitor := range config.Monitors {
|
checkMonitors(&config)
|
||||||
if monitor.ShouldCheck() {
|
|
||||||
_, alertNotice := monitor.Check()
|
|
||||||
|
|
||||||
// Should probably consider refactoring everything below here
|
|
||||||
if alertNotice != nil {
|
|
||||||
//log.Printf("Recieved an alert notice: %v", alertNotice)
|
|
||||||
var alerts []string
|
|
||||||
if alertNotice.IsUp {
|
|
||||||
alerts = monitor.AlertUp
|
|
||||||
log.Printf("Alert up: %v", monitor.AlertUp)
|
|
||||||
} else {
|
|
||||||
alerts = monitor.AlertDown
|
|
||||||
log.Printf("Alert down: %v", monitor.AlertDown)
|
|
||||||
}
|
|
||||||
if alerts == nil {
|
|
||||||
log.Printf("WARNING: Found alert, but no alert mechanism: %v", alertNotice)
|
|
||||||
}
|
|
||||||
for _, alertName := range alerts {
|
|
||||||
if alert, ok := config.Alerts[alertName]; ok {
|
|
||||||
_, err := alert.Send(*alertNotice)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
log.Printf("WARNING: Could not find alert for %s", alertName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sleepTime := time.Duration(config.CheckInterval) * time.Second
|
sleepTime := time.Duration(config.CheckInterval) * time.Second
|
||||||
time.Sleep(sleepTime)
|
time.Sleep(sleepTime)
|
||||||
|
36
monitor.go
36
monitor.go
@ -56,29 +56,27 @@ func (monitor *Monitor) Check() (bool, *AlertNotice) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
output, err := cmd.CombinedOutput()
|
output, err := cmd.CombinedOutput()
|
||||||
//log.Printf("Check %s\n---\n%s\n---", monitor.Name, string(output))
|
|
||||||
|
|
||||||
isSuccess := (err == nil)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("ERROR: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
monitor.lastCheck = time.Now()
|
monitor.lastCheck = time.Now()
|
||||||
monitor.lastOutput = string(output)
|
monitor.lastOutput = string(output)
|
||||||
|
|
||||||
var alertNotice *AlertNotice
|
var alertNotice *AlertNotice
|
||||||
|
isSuccess := (err == nil)
|
||||||
if isSuccess {
|
if isSuccess {
|
||||||
alertNotice = monitor.success()
|
alertNotice = monitor.success()
|
||||||
} else {
|
} else {
|
||||||
alertNotice = monitor.failure()
|
alertNotice = monitor.failure()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// log.Printf("DEBUG: Command output: %s", monitor.lastOutput)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("DEBUG: Command result: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
log.Printf(
|
log.Printf(
|
||||||
"Check result for %s: %v, %v at %v",
|
"INFO: %s success=%t, alert=%t",
|
||||||
monitor.Name,
|
monitor.Name,
|
||||||
isSuccess,
|
isSuccess,
|
||||||
alertNotice,
|
alertNotice != nil,
|
||||||
monitor.lastCheck,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return isSuccess, alertNotice
|
return isSuccess, alertNotice
|
||||||
@ -89,7 +87,6 @@ func (monitor Monitor) isUp() bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (monitor *Monitor) success() (notice *AlertNotice) {
|
func (monitor *Monitor) success() (notice *AlertNotice) {
|
||||||
log.Printf("Great success!")
|
|
||||||
if !monitor.isUp() {
|
if !monitor.isUp() {
|
||||||
// Alert that we have recovered
|
// Alert that we have recovered
|
||||||
notice = monitor.createAlertNotice(true)
|
notice = monitor.createAlertNotice(true)
|
||||||
@ -102,20 +99,23 @@ func (monitor *Monitor) success() (notice *AlertNotice) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (monitor *Monitor) failure() (notice *AlertNotice) {
|
func (monitor *Monitor) failure() (notice *AlertNotice) {
|
||||||
log.Printf("Devastating failure. :(")
|
|
||||||
monitor.failureCount++
|
monitor.failureCount++
|
||||||
// If we haven't hit the minimum failures, we can exit
|
// If we haven't hit the minimum failures, we can exit
|
||||||
if monitor.failureCount < monitor.getAlertAfter() {
|
if monitor.failureCount < monitor.getAlertAfter() {
|
||||||
log.Printf(
|
log.Printf(
|
||||||
"Have not hit minimum failures. failures: %v alert after: %v",
|
"DEBUG: %s failed but did not hit minimum failures. "+
|
||||||
|
"Count: %v alert after: %v",
|
||||||
|
monitor.Name,
|
||||||
monitor.failureCount,
|
monitor.failureCount,
|
||||||
monitor.getAlertAfter(),
|
monitor.getAlertAfter(),
|
||||||
)
|
)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Take number of failures after minimum
|
||||||
failureCount := (monitor.failureCount - monitor.getAlertAfter())
|
failureCount := (monitor.failureCount - monitor.getAlertAfter())
|
||||||
|
|
||||||
|
// Use alert cadence to determine if we should alert
|
||||||
if monitor.AlertEvery > 0 {
|
if monitor.AlertEvery > 0 {
|
||||||
// Handle integer number of failures before alerting
|
// Handle integer number of failures before alerting
|
||||||
if failureCount%monitor.AlertEvery == 0 {
|
if failureCount%monitor.AlertEvery == 0 {
|
||||||
@ -133,6 +133,7 @@ func (monitor *Monitor) failure() (notice *AlertNotice) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If we're going to alert, increment count
|
||||||
if notice != nil {
|
if notice != nil {
|
||||||
monitor.alertCount++
|
monitor.alertCount++
|
||||||
}
|
}
|
||||||
@ -150,6 +151,15 @@ func (monitor Monitor) getAlertAfter() int16 {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetAlertNames gives a list of alert names for a given monitor status
|
||||||
|
func (monitor Monitor) GetAlertNames(up bool) []string {
|
||||||
|
if up {
|
||||||
|
return monitor.AlertUp
|
||||||
|
} else {
|
||||||
|
return monitor.AlertDown
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (monitor Monitor) createAlertNotice(isUp bool) *AlertNotice {
|
func (monitor Monitor) createAlertNotice(isUp bool) *AlertNotice {
|
||||||
// TODO: Maybe add something about recovery status here
|
// TODO: Maybe add something about recovery status here
|
||||||
return &AlertNotice{
|
return &AlertNotice{
|
||||||
|
@ -85,6 +85,31 @@ func TestMonitorIsUp(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestMonitorGetAlertNames tests that proper alert names are returned
|
||||||
|
func TestMonitorGetAlertNames(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
monitor Monitor
|
||||||
|
up bool
|
||||||
|
expected []string
|
||||||
|
name string
|
||||||
|
}{
|
||||||
|
{Monitor{}, true, nil, "Empty up"},
|
||||||
|
{Monitor{}, false, nil, "Empty down"},
|
||||||
|
{Monitor{AlertUp: []string{"alert"}}, true, []string{"alert"}, "Return up"},
|
||||||
|
{Monitor{AlertDown: []string{"alert"}}, false, []string{"alert"}, "Return down"},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range cases {
|
||||||
|
log.Printf("Testing case %s", c.name)
|
||||||
|
actual := c.monitor.GetAlertNames(c.up)
|
||||||
|
if !EqualSliceString(actual, c.expected) {
|
||||||
|
t.Errorf("GetAlertNames(%v), expected=%v actual=%v", c.name, c.expected, actual)
|
||||||
|
log.Printf("Case failed: %s", c.name)
|
||||||
|
}
|
||||||
|
log.Println("-----")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TestMonitorSuccess tests the Monitor.success()
|
// TestMonitorSuccess tests the Monitor.success()
|
||||||
func TestMonitorSuccess(t *testing.T) {
|
func TestMonitorSuccess(t *testing.T) {
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
|
13
util.go
13
util.go
@ -21,3 +21,16 @@ func ShellCommand(command string) *exec.Cmd {
|
|||||||
//log.Printf("Shell command: %v", shellCommand)
|
//log.Printf("Shell command: %v", shellCommand)
|
||||||
return exec.Command(shellCommand[0], shellCommand[1:]...)
|
return exec.Command(shellCommand[0], shellCommand[1:]...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EqualSliceString checks if two string slices are equivalent
|
||||||
|
func EqualSliceString(a, b []string) bool {
|
||||||
|
if len(a) != len(b) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for i, val := range a {
|
||||||
|
if val != b[i] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
32
util_test.go
Normal file
32
util_test.go
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestUtilEqualSliceString(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
a, b []string
|
||||||
|
expected bool
|
||||||
|
}{
|
||||||
|
// Equal cases
|
||||||
|
{nil, nil, true},
|
||||||
|
{nil, []string{}, true},
|
||||||
|
{[]string{}, nil, true},
|
||||||
|
{[]string{"a"}, []string{"a"}, true},
|
||||||
|
// Inequal cases
|
||||||
|
{nil, []string{"b"}, false},
|
||||||
|
{[]string{"a"}, nil, false},
|
||||||
|
{[]string{"a"}, []string{"b"}, false},
|
||||||
|
{[]string{"a"}, []string{"a", "b"}, false},
|
||||||
|
{[]string{"a", "b"}, []string{"b"}, false},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range cases {
|
||||||
|
actual := EqualSliceString(c.a, c.b)
|
||||||
|
if actual != c.expected {
|
||||||
|
t.Errorf(
|
||||||
|
"EqualSliceString(%v, %v), expected=%v actual=%v",
|
||||||
|
c.a, c.b, c.expected, actual,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user