2019-09-21 15:03:26 -07:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2019-10-01 08:26:07 -07:00
|
|
|
"bytes"
|
2021-05-10 21:00:58 -07:00
|
|
|
"errors"
|
2019-10-03 16:30:49 -07:00
|
|
|
"fmt"
|
2019-09-21 15:03:26 -07:00
|
|
|
"os/exec"
|
|
|
|
"text/template"
|
|
|
|
"time"
|
2021-05-10 20:12:18 -07:00
|
|
|
|
|
|
|
"git.iamthefij.com/iamthefij/slog"
|
2019-09-21 15:03:26 -07:00
|
|
|
)
|
|
|
|
|
2021-05-10 21:00:58 -07:00
|
|
|
var (
|
|
|
|
errNoTemplate = errors.New("no template")
|
|
|
|
|
|
|
|
// ErrAlertFailed indicates that an alert failed to send
|
|
|
|
ErrAlertFailed = errors.New("alert failed")
|
|
|
|
)
|
|
|
|
|
2019-10-02 16:09:11 -07:00
|
|
|
// Alert is a config driven mechanism for sending a notice
|
2019-09-21 15:03:26 -07:00
|
|
|
type Alert struct {
|
2022-01-26 16:34:31 -08:00
|
|
|
Name string `hcl:"name,label"`
|
|
|
|
Command []string `hcl:"command,optional"`
|
|
|
|
ShellCommand string `hcl:"shell_command,optional"`
|
2019-10-02 09:37:29 -07:00
|
|
|
commandTemplate []*template.Template
|
2019-10-01 08:26:07 -07:00
|
|
|
commandShellTemplate *template.Template
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|
|
|
|
|
2019-10-02 16:09:11 -07:00
|
|
|
// AlertNotice captures the context for an alert to be sent
|
|
|
|
type AlertNotice struct {
|
2022-01-26 16:34:31 -08:00
|
|
|
AlertCount int
|
|
|
|
FailureCount int
|
2019-10-02 16:09:11 -07:00
|
|
|
IsUp bool
|
2021-05-10 21:00:58 -07:00
|
|
|
LastSuccess time.Time
|
|
|
|
MonitorName string
|
|
|
|
LastCheckOutput string
|
2019-10-02 16:09:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsValid returns a boolean indicating if the Alert has been correctly
|
|
|
|
// configured
|
2019-09-21 15:03:26 -07:00
|
|
|
func (alert Alert) IsValid() bool {
|
2022-01-26 16:34:31 -08:00
|
|
|
hasAtLeastOneCommand := alert.Command != nil || alert.ShellCommand != ""
|
|
|
|
hasAtMostOneCommand := alert.Command == nil || alert.ShellCommand == ""
|
|
|
|
|
|
|
|
return hasAtLeastOneCommand && hasAtMostOneCommand
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|
|
|
|
|
2019-10-02 16:09:11 -07:00
|
|
|
// BuildTemplates compiles command templates for the Alert
|
2019-10-03 16:30:49 -07:00
|
|
|
func (alert *Alert) BuildTemplates() error {
|
2021-05-10 20:12:18 -07:00
|
|
|
slog.Debugf("Building template for alert %s", alert.Name)
|
|
|
|
|
2023-08-10 16:22:30 -04:00
|
|
|
// Time format func factory
|
|
|
|
tff := func(formatString string) func(time.Time) string {
|
|
|
|
return func(t time.Time) string {
|
|
|
|
return t.Format(formatString)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create some functions for formatting datetimes in popular formats
|
|
|
|
timeFormatFuncs := template.FuncMap{
|
|
|
|
"ANSIC": tff(time.ANSIC),
|
|
|
|
"UnixDate": tff(time.UnixDate),
|
|
|
|
"RubyDate": tff(time.RubyDate),
|
|
|
|
"RFC822Z": tff(time.RFC822Z),
|
|
|
|
"RFC850": tff(time.RFC850),
|
|
|
|
"RFC1123": tff(time.RFC1123),
|
|
|
|
"RFC1123Z": tff(time.RFC1123Z),
|
|
|
|
"RFC3339": tff(time.RFC3339),
|
|
|
|
"RFC3339Nano": tff(time.RFC3339Nano),
|
|
|
|
"FormatTime": func(t time.Time, timeFormat string) string {
|
|
|
|
return t.Format(timeFormat)
|
|
|
|
},
|
|
|
|
"InTZ": func(t time.Time, tzName string) (time.Time, error) {
|
|
|
|
tz, err := time.LoadLocation(tzName)
|
|
|
|
if err != nil {
|
|
|
|
return t, fmt.Errorf("failed to convert time to specified tz: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return t.In(tz), nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-05-10 21:00:58 -07:00
|
|
|
switch {
|
2022-01-26 16:34:31 -08:00
|
|
|
case alert.commandTemplate == nil && alert.Command != nil:
|
2019-10-02 09:37:29 -07:00
|
|
|
alert.commandTemplate = []*template.Template{}
|
2022-01-26 16:34:31 -08:00
|
|
|
for i, cmdPart := range alert.Command {
|
2019-10-02 09:37:29 -07:00
|
|
|
alert.commandTemplate = append(alert.commandTemplate, template.Must(
|
2023-08-10 16:22:30 -04:00
|
|
|
template.New(alert.Name+fmt.Sprint(i)).Funcs(timeFormatFuncs).Parse(cmdPart),
|
2019-10-02 09:37:29 -07:00
|
|
|
))
|
|
|
|
}
|
2022-01-26 16:34:31 -08:00
|
|
|
case alert.commandShellTemplate == nil && alert.ShellCommand != "":
|
|
|
|
shellCmd := alert.ShellCommand
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-09-21 15:03:26 -07:00
|
|
|
alert.commandShellTemplate = template.Must(
|
2023-08-10 16:22:30 -04:00
|
|
|
template.New(alert.Name).Funcs(timeFormatFuncs).Parse(shellCmd),
|
2019-09-21 15:03:26 -07:00
|
|
|
)
|
2021-05-10 21:00:58 -07:00
|
|
|
default:
|
|
|
|
return fmt.Errorf("No template provided for alert %s: %w", alert.Name, errNoTemplate)
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|
2019-10-03 16:30:49 -07:00
|
|
|
|
|
|
|
return nil
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|
|
|
|
|
2019-10-02 16:09:11 -07:00
|
|
|
// Send will send an alert notice by executing the command template
|
2020-02-18 00:46:56 +00:00
|
|
|
func (alert Alert) Send(notice AlertNotice) (outputStr string, err error) {
|
2021-05-10 20:12:18 -07:00
|
|
|
slog.Infof("Sending alert %s for %s", alert.Name, notice.MonitorName)
|
|
|
|
|
2019-09-21 15:03:26 -07:00
|
|
|
var cmd *exec.Cmd
|
2021-05-10 21:00:58 -07:00
|
|
|
|
|
|
|
switch {
|
|
|
|
case alert.commandTemplate != nil:
|
2019-10-02 09:37:29 -07:00
|
|
|
command := []string{}
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-02 09:37:29 -07:00
|
|
|
for _, cmdTmp := range alert.commandTemplate {
|
|
|
|
var commandBuffer bytes.Buffer
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-03 16:30:49 -07:00
|
|
|
err = cmdTmp.Execute(&commandBuffer, notice)
|
2019-10-02 09:37:29 -07:00
|
|
|
if err != nil {
|
2019-10-03 16:30:49 -07:00
|
|
|
return
|
2019-10-02 09:37:29 -07:00
|
|
|
}
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-02 09:37:29 -07:00
|
|
|
command = append(command, commandBuffer.String())
|
|
|
|
}
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-02 09:37:29 -07:00
|
|
|
cmd = exec.Command(command[0], command[1:]...)
|
2021-05-10 21:00:58 -07:00
|
|
|
case alert.commandShellTemplate != nil:
|
2019-09-21 15:03:26 -07:00
|
|
|
var commandBuffer bytes.Buffer
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-03 16:30:49 -07:00
|
|
|
err = alert.commandShellTemplate.Execute(&commandBuffer, notice)
|
2019-10-01 08:26:07 -07:00
|
|
|
if err != nil {
|
2019-10-03 16:30:49 -07:00
|
|
|
return
|
2019-10-01 08:26:07 -07:00
|
|
|
}
|
2021-05-10 21:00:58 -07:00
|
|
|
|
2019-10-02 09:37:29 -07:00
|
|
|
shellCommand := commandBuffer.String()
|
2019-10-01 08:26:07 -07:00
|
|
|
|
2019-10-02 09:37:29 -07:00
|
|
|
cmd = ShellCommand(shellCommand)
|
2021-05-10 21:00:58 -07:00
|
|
|
default:
|
|
|
|
err = fmt.Errorf("No templates compiled for alert %s: %w", alert.Name, errNoTemplate)
|
|
|
|
|
2019-10-03 16:30:49 -07:00
|
|
|
return
|
2019-10-02 09:37:29 -07:00
|
|
|
}
|
|
|
|
|
2019-10-03 16:30:49 -07:00
|
|
|
// Exit if we're not ready to run the command
|
|
|
|
if cmd == nil || err != nil {
|
|
|
|
return
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|
2019-10-03 16:30:49 -07:00
|
|
|
|
|
|
|
var output []byte
|
|
|
|
output, err = cmd.CombinedOutput()
|
2020-02-18 00:46:56 +00:00
|
|
|
outputStr = string(output)
|
2021-05-10 20:12:18 -07:00
|
|
|
slog.Debugf("Alert output for: %s\n---\n%s\n---", alert.Name, outputStr)
|
2019-10-03 16:30:49 -07:00
|
|
|
|
2021-05-10 21:00:58 -07:00
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf(
|
2023-08-10 16:23:02 -04:00
|
|
|
"Alert %s failed to send. Returned %w: %w",
|
2021-05-10 21:00:58 -07:00
|
|
|
alert.Name,
|
|
|
|
err,
|
|
|
|
ErrAlertFailed,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-02-18 00:46:56 +00:00
|
|
|
return outputStr, err
|
2019-09-21 15:03:26 -07:00
|
|
|
}
|