feat(alerting): Implement alert-level provider overrides (#929)
* feat(alerting): Implement alert-level provider overrides Fixes #96 * Fix tests * Add missing test cases for alerting providers * feat(alerting): Implement alert-level overrides on all providers * chore: Add config.yaml to .gitignore * fix typo in discord provider * test: Start fixing tests for alerting providers * test: Fix GitLab tests * Fix all tests * test: Improve coverage * test: Improve coverage * Rename override to provider-override * docs: Mention new provider-override config * test: Improve coverage * test: Improve coverage * chore: Rename Alert.OverrideAsBytes to Alert.ProviderOverrideAsBytes
This commit is contained in:
		| @ -11,37 +11,41 @@ import ( | ||||
| 	"github.com/TwiN/gatus/v5/test" | ||||
| ) | ||||
|  | ||||
| func TestAlertDefaultProvider_IsValid(t *testing.T) { | ||||
| func TestAlertProvider_Validate(t *testing.T) { | ||||
| 	scenarios := []struct { | ||||
| 		Name     string | ||||
| 		Provider AlertProvider | ||||
| 		Expected bool | ||||
| 		Name          string | ||||
| 		Provider      AlertProvider | ||||
| 		ExpectedError bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			Name:     "invalid", | ||||
| 			Provider: AlertProvider{WebhookURL: "", AuthorizationKey: ""}, | ||||
| 			Expected: false, | ||||
| 			Name:          "invalid", | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: "", AuthorizationKey: ""}}, | ||||
| 			ExpectedError: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "missing-webhook-url", | ||||
| 			Provider: AlertProvider{WebhookURL: "", AuthorizationKey: "12345"}, | ||||
| 			Expected: false, | ||||
| 			Name:          "missing-webhook-url", | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: "", AuthorizationKey: "12345"}}, | ||||
| 			ExpectedError: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "missing-authorization-key", | ||||
| 			Provider: AlertProvider{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: ""}, | ||||
| 			Expected: false, | ||||
| 			Name:          "missing-authorization-key", | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: "https://gitlab.com/whatever/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: ""}}, | ||||
| 			ExpectedError: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "invalid-url", | ||||
| 			Provider: AlertProvider{WebhookURL: " http://foo.com", AuthorizationKey: "12345"}, | ||||
| 			Expected: false, | ||||
| 			Name:          "invalid-url", | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: " http://foo.com", AuthorizationKey: "12345"}}, | ||||
| 			ExpectedError: true, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, scenario := range scenarios { | ||||
| 		t.Run(scenario.Name, func(t *testing.T) { | ||||
| 			if scenario.Provider.IsValid() != scenario.Expected { | ||||
| 				t.Errorf("expected %t, got %t", scenario.Expected, scenario.Provider.IsValid()) | ||||
| 			err := scenario.Provider.Validate() | ||||
| 			if scenario.ExpectedError && err == nil { | ||||
| 				t.Error("expected error, got none") | ||||
| 			} | ||||
| 			if !scenario.ExpectedError && err != nil && !strings.Contains(err.Error(), "user does not exist") && !strings.Contains(err.Error(), "no such host") { | ||||
| 				t.Error("expected no error, got", err.Error()) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| @ -61,7 +65,7 @@ func TestAlertProvider_Send(t *testing.T) { | ||||
| 	}{ | ||||
| 		{ | ||||
| 			Name:          "triggered-error", | ||||
| 			Provider:      AlertProvider{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}, | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}}, | ||||
| 			Alert:         alert.Alert{Description: &firstDescription, SuccessThreshold: 5, FailureThreshold: 3}, | ||||
| 			Resolved:      false, | ||||
| 			ExpectedError: false, | ||||
| @ -71,7 +75,7 @@ func TestAlertProvider_Send(t *testing.T) { | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:          "resolved-error", | ||||
| 			Provider:      AlertProvider{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}, | ||||
| 			Provider:      AlertProvider{DefaultConfig: Config{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}}, | ||||
| 			Alert:         alert.Alert{Description: &secondDescription, SuccessThreshold: 5, FailureThreshold: 3}, | ||||
| 			Resolved:      true, | ||||
| 			ExpectedError: false, | ||||
| @ -116,21 +120,26 @@ func TestAlertProvider_buildAlertBody(t *testing.T) { | ||||
| 		{ | ||||
| 			Name:         "triggered", | ||||
| 			Endpoint:     endpoint.Endpoint{Name: "endpoint-name", URL: "https://example.org"}, | ||||
| 			Provider:     AlertProvider{}, | ||||
| 			Provider:     AlertProvider{DefaultConfig: Config{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}}, | ||||
| 			Alert:        alert.Alert{Description: &firstDescription, FailureThreshold: 3}, | ||||
| 			ExpectedBody: "{\"title\":\"alert(gatus): endpoint-name\",\"description\":\"An alert for *endpoint-name* has been triggered due to having failed 3 time(s) in a row:\\n\\u003e description-1\\n\\n## Condition results\\n- :white_check_mark: - `[CONNECTED] == true`\\n- :x: - `[STATUS] == 200`\\n\",\"start_time\":\"0001-01-01T00:00:00Z\",\"service\":\"endpoint-name\",\"monitoring_tool\":\"gatus\",\"hosts\":\"https://example.org\"}", | ||||
| 			ExpectedBody: "{\"title\":\"alert(gatus): endpoint-name\",\"description\":\"An alert for *endpoint-name* has been triggered due to having failed 3 time(s) in a row:\\n\\u003e description-1\\n\\n## Condition results\\n- :white_check_mark: - `[CONNECTED] == true`\\n- :x: - `[STATUS] == 200`\\n\",\"start_time\":\"0001-01-01T00:00:00Z\",\"service\":\"endpoint-name\",\"monitoring_tool\":\"gatus\",\"hosts\":\"https://example.org\",\"severity\":\"critical\"}", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:         "no-description", | ||||
| 			Endpoint:     endpoint.Endpoint{Name: "endpoint-name", URL: "https://example.org"}, | ||||
| 			Provider:     AlertProvider{}, | ||||
| 			Provider:     AlertProvider{DefaultConfig: Config{WebhookURL: "https://gitlab.com/hlidotbe/text/alerts/notify/gatus/xxxxxxxxxxxxxxxx.json", AuthorizationKey: "12345"}}, | ||||
| 			Alert:        alert.Alert{FailureThreshold: 10}, | ||||
| 			ExpectedBody: "{\"title\":\"alert(gatus): endpoint-name\",\"description\":\"An alert for *endpoint-name* has been triggered due to having failed 10 time(s) in a row\\n\\n## Condition results\\n- :white_check_mark: - `[CONNECTED] == true`\\n- :x: - `[STATUS] == 200`\\n\",\"start_time\":\"0001-01-01T00:00:00Z\",\"service\":\"endpoint-name\",\"monitoring_tool\":\"gatus\",\"hosts\":\"https://example.org\"}", | ||||
| 			ExpectedBody: "{\"title\":\"alert(gatus): endpoint-name\",\"description\":\"An alert for *endpoint-name* has been triggered due to having failed 10 time(s) in a row\\n\\n## Condition results\\n- :white_check_mark: - `[CONNECTED] == true`\\n- :x: - `[STATUS] == 200`\\n\",\"start_time\":\"0001-01-01T00:00:00Z\",\"service\":\"endpoint-name\",\"monitoring_tool\":\"gatus\",\"hosts\":\"https://example.org\",\"severity\":\"critical\"}", | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, scenario := range scenarios { | ||||
| 		t.Run(scenario.Name, func(t *testing.T) { | ||||
| 			cfg, err := scenario.Provider.GetConfig("", &scenario.Alert) | ||||
| 			if err != nil { | ||||
| 				t.Error("expected no error, got", err.Error()) | ||||
| 			} | ||||
| 			body := scenario.Provider.buildAlertBody( | ||||
| 				cfg, | ||||
| 				&scenario.Endpoint, | ||||
| 				&scenario.Alert, | ||||
| 				&endpoint.Result{ | ||||
| @ -156,3 +165,59 @@ func TestAlertProvider_GetDefaultAlert(t *testing.T) { | ||||
| 		t.Error("expected default alert to be nil") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestAlertProvider_GetConfig(t *testing.T) { | ||||
| 	scenarios := []struct { | ||||
| 		Name           string | ||||
| 		Provider       AlertProvider | ||||
| 		InputAlert     alert.Alert | ||||
| 		ExpectedOutput Config | ||||
| 	}{ | ||||
| 		{ | ||||
| 			Name: "provider-no-override-should-default", | ||||
| 			Provider: AlertProvider{ | ||||
| 				DefaultConfig: Config{WebhookURL: "https://github.com/TwiN/test", AuthorizationKey: "12345"}, | ||||
| 			}, | ||||
| 			InputAlert:     alert.Alert{}, | ||||
| 			ExpectedOutput: Config{WebhookURL: "https://github.com/TwiN/test", AuthorizationKey: "12345", Severity: DefaultSeverity, MonitoringTool: DefaultMonitoringTool}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name: "provider-with-alert-override", | ||||
| 			Provider: AlertProvider{ | ||||
| 				DefaultConfig: Config{WebhookURL: "https://github.com/TwiN/test", AuthorizationKey: "12345"}, | ||||
| 			}, | ||||
| 			InputAlert:     alert.Alert{ProviderOverride: map[string]any{"repository-url": "https://github.com/TwiN/alert-test", "authorization-key": "54321", "severity": "info", "monitoring-tool": "not-gatus", "environment-name": "prod", "service": "example"}}, | ||||
| 			ExpectedOutput: Config{WebhookURL: "https://github.com/TwiN/test", AuthorizationKey: "54321", Severity: "info", MonitoringTool: "not-gatus", EnvironmentName: "prod", Service: "example"}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, scenario := range scenarios { | ||||
| 		t.Run(scenario.Name, func(t *testing.T) { | ||||
| 			got, err := scenario.Provider.GetConfig("", &scenario.InputAlert) | ||||
| 			if err != nil && !strings.Contains(err.Error(), "user does not exist") && !strings.Contains(err.Error(), "no such host") { | ||||
| 				t.Fatalf("unexpected error: %s", err) | ||||
| 			} | ||||
| 			if got.WebhookURL != scenario.ExpectedOutput.WebhookURL { | ||||
| 				t.Errorf("expected repository URL %s, got %s", scenario.ExpectedOutput.WebhookURL, got.WebhookURL) | ||||
| 			} | ||||
| 			if got.AuthorizationKey != scenario.ExpectedOutput.AuthorizationKey { | ||||
| 				t.Errorf("expected AuthorizationKey %s, got %s", scenario.ExpectedOutput.AuthorizationKey, got.AuthorizationKey) | ||||
| 			} | ||||
| 			if got.Severity != scenario.ExpectedOutput.Severity { | ||||
| 				t.Errorf("expected Severity %s, got %s", scenario.ExpectedOutput.Severity, got.Severity) | ||||
| 			} | ||||
| 			if got.MonitoringTool != scenario.ExpectedOutput.MonitoringTool { | ||||
| 				t.Errorf("expected MonitoringTool %s, got %s", scenario.ExpectedOutput.MonitoringTool, got.MonitoringTool) | ||||
| 			} | ||||
| 			if got.EnvironmentName != scenario.ExpectedOutput.EnvironmentName { | ||||
| 				t.Errorf("expected EnvironmentName %s, got %s", scenario.ExpectedOutput.EnvironmentName, got.EnvironmentName) | ||||
| 			} | ||||
| 			if got.Service != scenario.ExpectedOutput.Service { | ||||
| 				t.Errorf("expected Service %s, got %s", scenario.ExpectedOutput.Service, got.Service) | ||||
| 			} | ||||
| 			// Test ValidateOverrides as well, since it really just calls GetConfig | ||||
| 			if err = scenario.Provider.ValidateOverrides("", &scenario.InputAlert); err != nil { | ||||
| 				t.Errorf("unexpected error: %s", err) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
		Reference in New Issue
	
	Block a user