* feat(alerting): Add new providers for Datadog, IFTTT, Line, NewRelic, Plivo, RocketChat, SendGrid, Signal, SIGNL4, Splunk, Squadcast, Vonage, Webex and Zapier Relevant: https://github.com/TwiN/gatus/discussions/1223 Fixes #1073 Fixes #1074 * chore: Clean up code * docs: Fix table formatting * Update alerting/provider/datadog/datadog.go * Update alerting/provider/signal/signal.go * Update alerting/provider/ifttt/ifttt.go * Update alerting/provider/newrelic/newrelic.go * Update alerting/provider/squadcast/squadcast.go * Update alerting/provider/squadcast/squadcast.go
148 lines
4.8 KiB
Go
148 lines
4.8 KiB
Go
package line
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
"testing"
|
|
|
|
"github.com/TwiN/gatus/v5/alerting/alert"
|
|
"github.com/TwiN/gatus/v5/client"
|
|
"github.com/TwiN/gatus/v5/config/endpoint"
|
|
"github.com/TwiN/gatus/v5/test"
|
|
)
|
|
|
|
func TestAlertProvider_Validate(t *testing.T) {
|
|
scenarios := []struct {
|
|
name string
|
|
provider AlertProvider
|
|
expected error
|
|
}{
|
|
{
|
|
name: "valid",
|
|
provider: AlertProvider{DefaultConfig: Config{ChannelAccessToken: "token123", UserIDs: []string{"U123"}}},
|
|
expected: nil,
|
|
},
|
|
{
|
|
name: "invalid-channel-access-token",
|
|
provider: AlertProvider{DefaultConfig: Config{UserIDs: []string{"U123"}}},
|
|
expected: ErrChannelAccessTokenNotSet,
|
|
},
|
|
{
|
|
name: "invalid-user-ids",
|
|
provider: AlertProvider{DefaultConfig: Config{ChannelAccessToken: "token123"}},
|
|
expected: ErrUserIDsNotSet,
|
|
},
|
|
}
|
|
for _, scenario := range scenarios {
|
|
t.Run(scenario.name, func(t *testing.T) {
|
|
err := scenario.provider.Validate()
|
|
if err != scenario.expected {
|
|
t.Errorf("expected %v, got %v", scenario.expected, err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestAlertProvider_Send(t *testing.T) {
|
|
defer client.InjectHTTPClient(nil)
|
|
firstDescription := "description-1"
|
|
secondDescription := "description-2"
|
|
scenarios := []struct {
|
|
name string
|
|
provider AlertProvider
|
|
alert alert.Alert
|
|
resolved bool
|
|
mockRoundTripper test.MockRoundTripper
|
|
expectedError bool
|
|
}{
|
|
{
|
|
name: "triggered",
|
|
provider: AlertProvider{DefaultConfig: Config{ChannelAccessToken: "token123", UserIDs: []string{"U123", "U456"}}},
|
|
alert: alert.Alert{Description: &firstDescription, SuccessThreshold: 5, FailureThreshold: 3},
|
|
resolved: false,
|
|
mockRoundTripper: test.MockRoundTripper(func(r *http.Request) *http.Response {
|
|
if r.URL.Path != "/v2/bot/message/push" {
|
|
t.Errorf("expected path /v2/bot/message/push, got %s", r.URL.Path)
|
|
}
|
|
if r.Header.Get("Authorization") != "Bearer token123" {
|
|
t.Errorf("expected Authorization header to be 'Bearer token123', got %s", r.Header.Get("Authorization"))
|
|
}
|
|
body := make(map[string]interface{})
|
|
json.NewDecoder(r.Body).Decode(&body)
|
|
if body["to"] == nil {
|
|
t.Error("expected 'to' field in request body")
|
|
}
|
|
messages := body["messages"].([]interface{})
|
|
if len(messages) != 1 {
|
|
t.Errorf("expected 1 message, got %d", len(messages))
|
|
}
|
|
return &http.Response{StatusCode: http.StatusOK, Body: http.NoBody}
|
|
}),
|
|
expectedError: false,
|
|
},
|
|
{
|
|
name: "resolved",
|
|
provider: AlertProvider{DefaultConfig: Config{ChannelAccessToken: "token123", UserIDs: []string{"U123"}}},
|
|
alert: alert.Alert{Description: &secondDescription, SuccessThreshold: 5, FailureThreshold: 3},
|
|
resolved: true,
|
|
mockRoundTripper: test.MockRoundTripper(func(r *http.Request) *http.Response {
|
|
body := make(map[string]interface{})
|
|
json.NewDecoder(r.Body).Decode(&body)
|
|
messages := body["messages"].([]interface{})
|
|
message := messages[0].(map[string]interface{})
|
|
text := message["text"].(string)
|
|
if !contains(text, "RESOLVED") {
|
|
t.Errorf("expected message to contain 'RESOLVED', got %s", text)
|
|
}
|
|
return &http.Response{StatusCode: http.StatusOK, Body: http.NoBody}
|
|
}),
|
|
expectedError: false,
|
|
},
|
|
{
|
|
name: "error-response",
|
|
provider: AlertProvider{DefaultConfig: Config{ChannelAccessToken: "token123", UserIDs: []string{"U123"}}},
|
|
alert: alert.Alert{Description: &firstDescription, SuccessThreshold: 5, FailureThreshold: 3},
|
|
resolved: false,
|
|
mockRoundTripper: test.MockRoundTripper(func(r *http.Request) *http.Response {
|
|
return &http.Response{StatusCode: http.StatusBadRequest, Body: http.NoBody}
|
|
}),
|
|
expectedError: true,
|
|
},
|
|
}
|
|
for _, scenario := range scenarios {
|
|
t.Run(scenario.name, func(t *testing.T) {
|
|
client.InjectHTTPClient(&http.Client{Transport: scenario.mockRoundTripper})
|
|
err := scenario.provider.Send(
|
|
&endpoint.Endpoint{Name: "endpoint-name"},
|
|
&scenario.alert,
|
|
&endpoint.Result{
|
|
ConditionResults: []*endpoint.ConditionResult{
|
|
{Condition: "[CONNECTED] == true", Success: scenario.resolved},
|
|
{Condition: "[STATUS] == 200", Success: scenario.resolved},
|
|
},
|
|
},
|
|
scenario.resolved,
|
|
)
|
|
if scenario.expectedError && err == nil {
|
|
t.Error("expected error, got none")
|
|
}
|
|
if !scenario.expectedError && err != nil {
|
|
t.Error("expected no error, got", err.Error())
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestAlertProvider_GetDefaultAlert(t *testing.T) {
|
|
if (&AlertProvider{DefaultAlert: &alert.Alert{}}).GetDefaultAlert() == nil {
|
|
t.Error("expected default alert to be not nil")
|
|
}
|
|
if (&AlertProvider{DefaultAlert: nil}).GetDefaultAlert() != nil {
|
|
t.Error("expected default alert to be nil")
|
|
}
|
|
}
|
|
|
|
func contains(s, substr string) bool {
|
|
return len(s) >= len(substr) && s[0:len(substr)] == substr || len(s) > len(substr) && contains(s[1:], substr)
|
|
}
|