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:
		| @ -2,6 +2,7 @@ package email | ||||
|  | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"strings" | ||||
| @ -10,10 +11,17 @@ import ( | ||||
| 	"github.com/TwiN/gatus/v5/client" | ||||
| 	"github.com/TwiN/gatus/v5/config/endpoint" | ||||
| 	gomail "gopkg.in/mail.v2" | ||||
| 	"gopkg.in/yaml.v3" | ||||
| ) | ||||
|  | ||||
| // AlertProvider is the configuration necessary for sending an alert using SMTP | ||||
| type AlertProvider struct { | ||||
| var ( | ||||
| 	ErrDuplicateGroupOverride = errors.New("duplicate group override") | ||||
| 	ErrMissingFromOrToFields  = errors.New("from and to fields are required") | ||||
| 	ErrInvalidPort            = errors.New("port must be between 1 and 65535 inclusively") | ||||
| 	ErrMissingHost            = errors.New("host is required") | ||||
| ) | ||||
|  | ||||
| type Config struct { | ||||
| 	From     string `yaml:"from"` | ||||
| 	Username string `yaml:"username"` | ||||
| 	Password string `yaml:"password"` | ||||
| @ -23,6 +31,48 @@ type AlertProvider struct { | ||||
|  | ||||
| 	// ClientConfig is the configuration of the client used to communicate with the provider's target | ||||
| 	ClientConfig *client.Config `yaml:"client,omitempty"` | ||||
| } | ||||
|  | ||||
| func (cfg *Config) Validate() error { | ||||
| 	if len(cfg.From) == 0 || len(cfg.To) == 0 { | ||||
| 		return ErrMissingFromOrToFields | ||||
| 	} | ||||
| 	if cfg.Port < 1 || cfg.Port > math.MaxUint16 { | ||||
| 		return ErrInvalidPort | ||||
| 	} | ||||
| 	if len(cfg.Host) == 0 { | ||||
| 		return ErrMissingHost | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (cfg *Config) Merge(override *Config) { | ||||
| 	if override.ClientConfig != nil { | ||||
| 		cfg.ClientConfig = override.ClientConfig | ||||
| 	} | ||||
| 	if len(override.From) > 0 { | ||||
| 		cfg.From = override.From | ||||
| 	} | ||||
| 	if len(override.Username) > 0 { | ||||
| 		cfg.Username = override.Username | ||||
| 	} | ||||
| 	if len(override.Password) > 0 { | ||||
| 		cfg.Password = override.Password | ||||
| 	} | ||||
| 	if len(override.Host) > 0 { | ||||
| 		cfg.Host = override.Host | ||||
| 	} | ||||
| 	if override.Port > 0 { | ||||
| 		cfg.Port = override.Port | ||||
| 	} | ||||
| 	if len(override.To) > 0 { | ||||
| 		cfg.To = override.To | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // AlertProvider is the configuration necessary for sending an alert using SMTP | ||||
| type AlertProvider struct { | ||||
| 	DefaultConfig Config `yaml:",inline"` | ||||
|  | ||||
| 	// DefaultAlert is the default alert configuration to use for endpoints with an alert of the appropriate type | ||||
| 	DefaultAlert *alert.Alert `yaml:"default-alert,omitempty"` | ||||
| @ -33,54 +83,57 @@ type AlertProvider struct { | ||||
|  | ||||
| // Override is a case under which the default integration is overridden | ||||
| type Override struct { | ||||
| 	Group string `yaml:"group"` | ||||
| 	To    string `yaml:"to"` | ||||
| 	Group  string `yaml:"group"` | ||||
| 	Config `yaml:",inline"` | ||||
| } | ||||
|  | ||||
| // IsValid returns whether the provider's configuration is valid | ||||
| func (provider *AlertProvider) IsValid() bool { | ||||
| // Validate the provider's configuration | ||||
| func (provider *AlertProvider) Validate() error { | ||||
| 	registeredGroups := make(map[string]bool) | ||||
| 	if provider.Overrides != nil { | ||||
| 		for _, override := range provider.Overrides { | ||||
| 			if isAlreadyRegistered := registeredGroups[override.Group]; isAlreadyRegistered || override.Group == "" || len(override.To) == 0 { | ||||
| 				return false | ||||
| 				return ErrDuplicateGroupOverride | ||||
| 			} | ||||
| 			registeredGroups[override.Group] = true | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return len(provider.From) > 0 && len(provider.Host) > 0 && len(provider.To) > 0 && provider.Port > 0 && provider.Port < math.MaxUint16 | ||||
| 	return provider.DefaultConfig.Validate() | ||||
| } | ||||
|  | ||||
| // Send an alert using the provider | ||||
| func (provider *AlertProvider) Send(ep *endpoint.Endpoint, alert *alert.Alert, result *endpoint.Result, resolved bool) error { | ||||
| 	cfg, err := provider.GetConfig(ep.Group, alert) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	var username string | ||||
| 	if len(provider.Username) > 0 { | ||||
| 		username = provider.Username | ||||
| 	if len(cfg.Username) > 0 { | ||||
| 		username = cfg.Username | ||||
| 	} else { | ||||
| 		username = provider.From | ||||
| 		username = cfg.From | ||||
| 	} | ||||
| 	subject, body := provider.buildMessageSubjectAndBody(ep, alert, result, resolved) | ||||
| 	m := gomail.NewMessage() | ||||
| 	m.SetHeader("From", provider.From) | ||||
| 	m.SetHeader("To", strings.Split(provider.getToForGroup(ep.Group), ",")...) | ||||
| 	m.SetHeader("From", cfg.From) | ||||
| 	m.SetHeader("To", strings.Split(cfg.To, ",")...) | ||||
| 	m.SetHeader("Subject", subject) | ||||
| 	m.SetBody("text/plain", body) | ||||
| 	var d *gomail.Dialer | ||||
| 	if len(provider.Password) == 0 { | ||||
| 	if len(cfg.Password) == 0 { | ||||
| 		// Get the domain in the From address | ||||
| 		localName := "localhost" | ||||
| 		fromParts := strings.Split(provider.From, `@`) | ||||
| 		fromParts := strings.Split(cfg.From, `@`) | ||||
| 		if len(fromParts) == 2 { | ||||
| 			localName = fromParts[1] | ||||
| 		} | ||||
| 		// Create a dialer with no authentication | ||||
| 		d = &gomail.Dialer{Host: provider.Host, Port: provider.Port, LocalName: localName} | ||||
| 		d = &gomail.Dialer{Host: cfg.Host, Port: cfg.Port, LocalName: localName} | ||||
| 	} else { | ||||
| 		// Create an authenticated dialer | ||||
| 		d = gomail.NewDialer(provider.Host, provider.Port, username, provider.Password) | ||||
| 		d = gomail.NewDialer(cfg.Host, cfg.Port, username, cfg.Password) | ||||
| 	} | ||||
| 	if provider.ClientConfig != nil && provider.ClientConfig.Insecure { | ||||
| 	if cfg.ClientConfig != nil && cfg.ClientConfig.Insecure { | ||||
| 		d.TLSConfig = &tls.Config{InsecureSkipVerify: true} | ||||
| 	} | ||||
| 	return d.DialAndSend(m) | ||||
| @ -116,19 +169,38 @@ func (provider *AlertProvider) buildMessageSubjectAndBody(ep *endpoint.Endpoint, | ||||
| 	return subject, message + description + formattedConditionResults | ||||
| } | ||||
|  | ||||
| // getToForGroup returns the appropriate email integration to for a given group | ||||
| func (provider *AlertProvider) getToForGroup(group string) string { | ||||
| 	if provider.Overrides != nil { | ||||
| 		for _, override := range provider.Overrides { | ||||
| 			if group == override.Group { | ||||
| 				return override.To | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return provider.To | ||||
| } | ||||
|  | ||||
| // GetDefaultAlert returns the provider's default alert configuration | ||||
| func (provider *AlertProvider) GetDefaultAlert() *alert.Alert { | ||||
| 	return provider.DefaultAlert | ||||
| } | ||||
|  | ||||
| // GetConfig returns the configuration for the provider with the overrides applied | ||||
| func (provider *AlertProvider) GetConfig(group string, alert *alert.Alert) (*Config, error) { | ||||
| 	cfg := provider.DefaultConfig | ||||
| 	// Handle group overrides | ||||
| 	if provider.Overrides != nil { | ||||
| 		for _, override := range provider.Overrides { | ||||
| 			if group == override.Group { | ||||
| 				cfg.Merge(&override.Config) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	// Handle alert overrides | ||||
| 	if len(alert.ProviderOverride) != 0 { | ||||
| 		overrideConfig := Config{} | ||||
| 		if err := yaml.Unmarshal(alert.ProviderOverrideAsBytes(), &overrideConfig); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		cfg.Merge(&overrideConfig) | ||||
| 	} | ||||
| 	// Validate the configuration | ||||
| 	err := cfg.Validate() | ||||
| 	return &cfg, err | ||||
| } | ||||
|  | ||||
| // ValidateOverrides validates the alert's provider override and, if present, the group override | ||||
| func (provider *AlertProvider) ValidateOverrides(group string, alert *alert.Alert) error { | ||||
| 	_, err := provider.GetConfig(group, alert) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
		Reference in New Issue
	
	Block a user