2015-09-25 14:14:30 +00:00
|
|
|
// Copyright 2015 Prometheus Team
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2019-05-13 12:51:26 +00:00
|
|
|
"regexp"
|
2015-10-29 13:34:24 +00:00
|
|
|
"strings"
|
2016-02-26 08:35:00 +00:00
|
|
|
"time"
|
2017-07-27 08:48:24 +00:00
|
|
|
|
2019-05-13 12:51:26 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
commoncfg "github.com/prometheus/common/config"
|
2015-09-25 14:14:30 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2015-12-17 12:05:06 +00:00
|
|
|
// DefaultWebhookConfig defines default values for Webhook configurations.
|
|
|
|
DefaultWebhookConfig = WebhookConfig{
|
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// DefaultEmailConfig defines default values for Email configurations.
|
|
|
|
DefaultEmailConfig = EmailConfig{
|
2015-12-17 12:05:06 +00:00
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: false,
|
|
|
|
},
|
2016-11-21 10:09:49 +00:00
|
|
|
HTML: `{{ template "email.default.html" . }}`,
|
2017-09-07 15:24:19 +00:00
|
|
|
Text: ``,
|
2015-11-20 14:10:38 +00:00
|
|
|
}
|
2015-11-26 17:19:46 +00:00
|
|
|
|
|
|
|
// DefaultEmailSubject defines the default Subject header of an Email.
|
|
|
|
DefaultEmailSubject = `{{ template "email.default.subject" . }}`
|
2015-11-20 14:10:38 +00:00
|
|
|
|
2018-07-05 07:54:28 +00:00
|
|
|
// DefaultPagerdutyDetails defines the default values for PagerDuty details.
|
|
|
|
DefaultPagerdutyDetails = map[string]string{
|
|
|
|
"firing": `{{ template "pagerduty.default.instances" .Alerts.Firing }}`,
|
|
|
|
"resolved": `{{ template "pagerduty.default.instances" .Alerts.Resolved }}`,
|
|
|
|
"num_firing": `{{ .Alerts.Firing | len }}`,
|
|
|
|
"num_resolved": `{{ .Alerts.Resolved | len }}`,
|
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// DefaultPagerdutyConfig defines default values for PagerDuty configurations.
|
|
|
|
DefaultPagerdutyConfig = PagerdutyConfig{
|
2015-12-17 12:05:06 +00:00
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
2015-11-26 17:19:46 +00:00
|
|
|
Description: `{{ template "pagerduty.default.description" .}}`,
|
|
|
|
Client: `{{ template "pagerduty.default.client" . }}`,
|
|
|
|
ClientURL: `{{ template "pagerduty.default.clientURL" . }}`,
|
2015-11-20 14:10:38 +00:00
|
|
|
}
|
|
|
|
|
2015-11-12 11:57:20 +00:00
|
|
|
// DefaultSlackConfig defines default values for Slack configurations.
|
2015-09-25 14:14:30 +00:00
|
|
|
DefaultSlackConfig = SlackConfig{
|
2015-12-17 12:05:06 +00:00
|
|
|
NotifierConfig: NotifierConfig{
|
2016-01-05 18:01:21 +00:00
|
|
|
VSendResolved: false,
|
2015-12-17 12:05:06 +00:00
|
|
|
},
|
2018-10-23 09:22:23 +00:00
|
|
|
Color: `{{ if eq .Status "firing" }}danger{{ else }}good{{ end }}`,
|
|
|
|
Username: `{{ template "slack.default.username" . }}`,
|
|
|
|
Title: `{{ template "slack.default.title" . }}`,
|
|
|
|
TitleLink: `{{ template "slack.default.titlelink" . }}`,
|
|
|
|
IconEmoji: `{{ template "slack.default.iconemoji" . }}`,
|
|
|
|
IconURL: `{{ template "slack.default.iconurl" . }}`,
|
|
|
|
Pretext: `{{ template "slack.default.pretext" . }}`,
|
|
|
|
Text: `{{ template "slack.default.text" . }}`,
|
|
|
|
Fallback: `{{ template "slack.default.fallback" . }}`,
|
|
|
|
CallbackID: `{{ template "slack.default.callbackid" . }}`,
|
|
|
|
Footer: `{{ template "slack.default.footer" . }}`,
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
2015-11-24 22:29:25 +00:00
|
|
|
|
|
|
|
// DefaultOpsGenieConfig defines default values for OpsGenie configurations.
|
|
|
|
DefaultOpsGenieConfig = OpsGenieConfig{
|
2015-12-17 12:05:06 +00:00
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
2016-07-12 12:36:20 +00:00
|
|
|
Message: `{{ template "opsgenie.default.message" . }}`,
|
2015-11-26 17:19:46 +00:00
|
|
|
Description: `{{ template "opsgenie.default.description" . }}`,
|
2015-11-30 10:21:08 +00:00
|
|
|
Source: `{{ template "opsgenie.default.source" . }}`,
|
2015-11-24 22:29:25 +00:00
|
|
|
// TODO: Add a details field with all the alerts.
|
|
|
|
}
|
2016-02-26 08:35:00 +00:00
|
|
|
|
2017-12-09 15:20:22 +00:00
|
|
|
// DefaultWechatConfig defines default values for wechat configurations.
|
|
|
|
DefaultWechatConfig = WechatConfig{
|
|
|
|
NotifierConfig: NotifierConfig{
|
2018-03-02 08:49:41 +00:00
|
|
|
VSendResolved: false,
|
2017-12-09 15:20:22 +00:00
|
|
|
},
|
2019-05-15 02:47:20 +00:00
|
|
|
Message: `{{ template "wechat.default.message" . }}`,
|
|
|
|
ToUser: `{{ template "wechat.default.to_user" . }}`,
|
|
|
|
ToParty: `{{ template "wechat.default.to_party" . }}`,
|
|
|
|
ToTag: `{{ template "wechat.default.to_tag" . }}`,
|
|
|
|
AgentID: `{{ template "wechat.default.agent_id" . }}`,
|
2017-12-09 15:20:22 +00:00
|
|
|
}
|
|
|
|
|
2016-07-03 15:33:44 +00:00
|
|
|
// DefaultVictorOpsConfig defines default values for VictorOps configurations.
|
|
|
|
DefaultVictorOpsConfig = VictorOpsConfig{
|
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
2017-07-03 09:44:53 +00:00
|
|
|
MessageType: `CRITICAL`,
|
|
|
|
StateMessage: `{{ template "victorops.default.state_message" . }}`,
|
|
|
|
EntityDisplayName: `{{ template "victorops.default.entity_display_name" . }}`,
|
|
|
|
MonitoringTool: `{{ template "victorops.default.monitoring_tool" . }}`,
|
2016-07-03 15:33:44 +00:00
|
|
|
}
|
|
|
|
|
2016-02-26 08:35:00 +00:00
|
|
|
// DefaultPushoverConfig defines default values for Pushover configurations.
|
|
|
|
DefaultPushoverConfig = PushoverConfig{
|
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
|
|
|
Title: `{{ template "pushover.default.title" . }}`,
|
|
|
|
Message: `{{ template "pushover.default.message" . }}`,
|
|
|
|
URL: `{{ template "pushover.default.url" . }}`,
|
|
|
|
Priority: `{{ if eq .Status "firing" }}2{{ else }}0{{ end }}`, // emergency (firing) or normal
|
|
|
|
Retry: duration(1 * time.Minute),
|
|
|
|
Expire: duration(1 * time.Hour),
|
2018-12-18 14:15:30 +00:00
|
|
|
HTML: false,
|
2016-02-26 08:35:00 +00:00
|
|
|
}
|
2021-06-10 02:13:26 +00:00
|
|
|
|
|
|
|
// DefaultSNSConfig defines default values for SNS configurations.
|
|
|
|
DefaultSNSConfig = SNSConfig{
|
|
|
|
NotifierConfig: NotifierConfig{
|
|
|
|
VSendResolved: true,
|
|
|
|
},
|
2021-06-16 19:41:30 +00:00
|
|
|
Subject: `{{ template "sns.default.subject" . }}`,
|
|
|
|
Message: `{{ template "sns.default.message" . }}`,
|
2021-06-10 02:13:26 +00:00
|
|
|
}
|
2015-11-20 14:10:38 +00:00
|
|
|
)
|
2015-10-09 10:03:15 +00:00
|
|
|
|
2015-12-17 12:05:06 +00:00
|
|
|
// NotifierConfig contains base options common across all notifier configurations.
|
|
|
|
type NotifierConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
VSendResolved bool `yaml:"send_resolved" json:"send_resolved"`
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
2015-12-17 12:05:06 +00:00
|
|
|
func (nc *NotifierConfig) SendResolved() bool {
|
|
|
|
return nc.VSendResolved
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
2015-11-12 11:57:20 +00:00
|
|
|
// EmailConfig configures notifications via mail.
|
2015-09-25 14:14:30 +00:00
|
|
|
type EmailConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2015-12-17 12:05:06 +00:00
|
|
|
|
2015-09-25 14:14:30 +00:00
|
|
|
// Email address to notify.
|
2018-08-24 09:25:51 +00:00
|
|
|
To string `yaml:"to,omitempty" json:"to,omitempty"`
|
|
|
|
From string `yaml:"from,omitempty" json:"from,omitempty"`
|
|
|
|
Hello string `yaml:"hello,omitempty" json:"hello,omitempty"`
|
2019-08-28 13:04:40 +00:00
|
|
|
Smarthost HostPort `yaml:"smarthost,omitempty" json:"smarthost,omitempty"`
|
2018-08-24 09:25:51 +00:00
|
|
|
AuthUsername string `yaml:"auth_username,omitempty" json:"auth_username,omitempty"`
|
|
|
|
AuthPassword Secret `yaml:"auth_password,omitempty" json:"auth_password,omitempty"`
|
|
|
|
AuthSecret Secret `yaml:"auth_secret,omitempty" json:"auth_secret,omitempty"`
|
|
|
|
AuthIdentity string `yaml:"auth_identity,omitempty" json:"auth_identity,omitempty"`
|
|
|
|
Headers map[string]string `yaml:"headers,omitempty" json:"headers,omitempty"`
|
|
|
|
HTML string `yaml:"html,omitempty" json:"html,omitempty"`
|
|
|
|
Text string `yaml:"text,omitempty" json:"text,omitempty"`
|
|
|
|
RequireTLS *bool `yaml:"require_tls,omitempty" json:"require_tls,omitempty"`
|
|
|
|
TLSConfig commoncfg.TLSConfig `yaml:"tls_config,omitempty" json:"tls_config,omitempty"`
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *EmailConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
2015-10-11 10:34:05 +00:00
|
|
|
*c = DefaultEmailConfig
|
2015-09-25 14:14:30 +00:00
|
|
|
type plain EmailConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-10-29 13:34:24 +00:00
|
|
|
if c.To == "" {
|
|
|
|
return fmt.Errorf("missing to address in email config")
|
|
|
|
}
|
|
|
|
// Header names are case-insensitive, check for collisions.
|
2015-11-20 14:10:38 +00:00
|
|
|
normalizedHeaders := map[string]string{}
|
2015-10-29 13:34:24 +00:00
|
|
|
for h, v := range c.Headers {
|
2016-03-03 14:06:11 +00:00
|
|
|
normalized := strings.Title(h)
|
2015-11-20 14:10:38 +00:00
|
|
|
if _, ok := normalizedHeaders[normalized]; ok {
|
|
|
|
return fmt.Errorf("duplicate header %q in email config", normalized)
|
2015-10-29 13:34:24 +00:00
|
|
|
}
|
2015-11-20 14:10:38 +00:00
|
|
|
normalizedHeaders[normalized] = v
|
2015-10-29 13:34:24 +00:00
|
|
|
}
|
2015-11-20 14:10:38 +00:00
|
|
|
c.Headers = normalizedHeaders
|
2015-10-29 13:34:24 +00:00
|
|
|
|
2018-04-17 14:22:46 +00:00
|
|
|
return nil
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// PagerdutyConfig configures notifications via PagerDuty.
|
|
|
|
type PagerdutyConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2015-12-17 12:05:06 +00:00
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2018-08-24 09:25:51 +00:00
|
|
|
ServiceKey Secret `yaml:"service_key,omitempty" json:"service_key,omitempty"`
|
2017-11-07 10:07:27 +00:00
|
|
|
RoutingKey Secret `yaml:"routing_key,omitempty" json:"routing_key,omitempty"`
|
2018-07-26 10:39:33 +00:00
|
|
|
URL *URL `yaml:"url,omitempty" json:"url,omitempty"`
|
2017-06-20 17:09:14 +00:00
|
|
|
Client string `yaml:"client,omitempty" json:"client,omitempty"`
|
|
|
|
ClientURL string `yaml:"client_url,omitempty" json:"client_url,omitempty"`
|
|
|
|
Description string `yaml:"description,omitempty" json:"description,omitempty"`
|
|
|
|
Details map[string]string `yaml:"details,omitempty" json:"details,omitempty"`
|
2018-10-02 08:45:53 +00:00
|
|
|
Images []PagerdutyImage `yaml:"images,omitempty" json:"images,omitempty"`
|
|
|
|
Links []PagerdutyLink `yaml:"links,omitempty" json:"links,omitempty"`
|
2017-11-07 10:07:27 +00:00
|
|
|
Severity string `yaml:"severity,omitempty" json:"severity,omitempty"`
|
2018-02-09 09:50:18 +00:00
|
|
|
Class string `yaml:"class,omitempty" json:"class,omitempty"`
|
2017-11-07 10:07:27 +00:00
|
|
|
Component string `yaml:"component,omitempty" json:"component,omitempty"`
|
|
|
|
Group string `yaml:"group,omitempty" json:"group,omitempty"`
|
2015-11-20 14:10:38 +00:00
|
|
|
}
|
|
|
|
|
2018-10-02 08:45:53 +00:00
|
|
|
// PagerdutyLink is a link
|
|
|
|
type PagerdutyLink struct {
|
2019-10-29 09:46:40 +00:00
|
|
|
Href string `yaml:"href,omitempty" json:"href,omitempty"`
|
2018-10-02 08:45:53 +00:00
|
|
|
Text string `yaml:"text,omitempty" json:"text,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// PagerdutyImage is an image
|
|
|
|
type PagerdutyImage struct {
|
|
|
|
Src string `yaml:"src,omitempty" json:"src,omitempty"`
|
|
|
|
Alt string `yaml:"alt,omitempty" json:"alt,omitempty"`
|
2019-06-24 10:19:06 +00:00
|
|
|
Href string `yaml:"href,omitempty" json:"href,omitempty"`
|
2018-10-02 08:45:53 +00:00
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *PagerdutyConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultPagerdutyConfig
|
|
|
|
type plain PagerdutyConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-11-07 10:07:27 +00:00
|
|
|
if c.RoutingKey == "" && c.ServiceKey == "" {
|
|
|
|
return fmt.Errorf("missing service or routing key in PagerDuty config")
|
2015-11-20 14:10:38 +00:00
|
|
|
}
|
2018-07-05 07:54:28 +00:00
|
|
|
if c.Details == nil {
|
|
|
|
c.Details = make(map[string]string)
|
|
|
|
}
|
|
|
|
for k, v := range DefaultPagerdutyDetails {
|
|
|
|
if _, ok := c.Details[k]; !ok {
|
|
|
|
c.Details[k] = v
|
|
|
|
}
|
|
|
|
}
|
2018-04-17 14:22:46 +00:00
|
|
|
return nil
|
2015-11-20 14:10:38 +00:00
|
|
|
}
|
|
|
|
|
2018-05-14 15:26:11 +00:00
|
|
|
// SlackAction configures a single Slack action that is sent with each notification.
|
2018-10-29 14:55:43 +00:00
|
|
|
// See https://api.slack.com/docs/message-attachments#action_fields and https://api.slack.com/docs/message-buttons
|
|
|
|
// for more information.
|
2018-05-14 15:26:11 +00:00
|
|
|
type SlackAction struct {
|
2018-10-29 14:55:43 +00:00
|
|
|
Type string `yaml:"type,omitempty" json:"type,omitempty"`
|
|
|
|
Text string `yaml:"text,omitempty" json:"text,omitempty"`
|
|
|
|
URL string `yaml:"url,omitempty" json:"url,omitempty"`
|
|
|
|
Style string `yaml:"style,omitempty" json:"style,omitempty"`
|
|
|
|
Name string `yaml:"name,omitempty" json:"name,omitempty"`
|
|
|
|
Value string `yaml:"value,omitempty" json:"value,omitempty"`
|
|
|
|
ConfirmField *SlackConfirmationField `yaml:"confirm,omitempty" json:"confirm,omitempty"`
|
2018-05-14 15:26:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface for SlackAction.
|
|
|
|
func (c *SlackAction) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
type plain SlackAction
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if c.Type == "" {
|
|
|
|
return fmt.Errorf("missing type in Slack action configuration")
|
|
|
|
}
|
|
|
|
if c.Text == "" {
|
2018-10-29 14:55:43 +00:00
|
|
|
return fmt.Errorf("missing text in Slack action configuration")
|
2018-05-14 15:26:11 +00:00
|
|
|
}
|
2018-10-29 14:55:43 +00:00
|
|
|
if c.URL != "" {
|
|
|
|
// Clear all message action fields.
|
|
|
|
c.Name = ""
|
|
|
|
c.Value = ""
|
|
|
|
c.ConfirmField = nil
|
|
|
|
} else if c.Name != "" {
|
|
|
|
c.URL = ""
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("missing name or url in Slack action configuration")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SlackConfirmationField protect users from destructive actions or particularly distinguished decisions
|
|
|
|
// by asking them to confirm their button click one more time.
|
|
|
|
// See https://api.slack.com/docs/interactive-message-field-guide#confirmation_fields for more information.
|
|
|
|
type SlackConfirmationField struct {
|
|
|
|
Text string `yaml:"text,omitempty" json:"text,omitempty"`
|
|
|
|
Title string `yaml:"title,omitempty" json:"title,omitempty"`
|
|
|
|
OkText string `yaml:"ok_text,omitempty" json:"ok_text,omitempty"`
|
|
|
|
DismissText string `yaml:"dismiss_text,omitempty" json:"dismiss_text,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface for SlackConfirmationField.
|
|
|
|
func (c *SlackConfirmationField) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
type plain SlackConfirmationField
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if c.Text == "" {
|
|
|
|
return fmt.Errorf("missing text in Slack confirmation configuration")
|
2018-05-14 15:26:11 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:58:11 +00:00
|
|
|
// SlackField configures a single Slack field that is sent with each notification.
|
|
|
|
// Each field must contain a title, value, and optionally, a boolean value to indicate if the field
|
|
|
|
// is short enough to be displayed next to other fields designated as short.
|
|
|
|
// See https://api.slack.com/docs/message-attachments#fields for more information.
|
|
|
|
type SlackField struct {
|
|
|
|
Title string `yaml:"title,omitempty" json:"title,omitempty"`
|
|
|
|
Value string `yaml:"value,omitempty" json:"value,omitempty"`
|
|
|
|
Short *bool `yaml:"short,omitempty" json:"short,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface for SlackField.
|
|
|
|
func (c *SlackField) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
type plain SlackField
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if c.Title == "" {
|
|
|
|
return fmt.Errorf("missing title in Slack field configuration")
|
|
|
|
}
|
|
|
|
if c.Value == "" {
|
|
|
|
return fmt.Errorf("missing value in Slack field configuration")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-11-12 11:57:20 +00:00
|
|
|
// SlackConfig configures notifications via Slack.
|
2015-09-25 14:14:30 +00:00
|
|
|
type SlackConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2015-12-17 12:05:06 +00:00
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2021-04-02 01:57:55 +00:00
|
|
|
APIURL *SecretURL `yaml:"api_url,omitempty" json:"api_url,omitempty"`
|
|
|
|
APIURLFile string `yaml:"api_url_file,omitempty" json:"api_url_file,omitempty"`
|
2015-09-25 14:14:30 +00:00
|
|
|
|
|
|
|
// Slack channel override, (like #other-channel or @username).
|
2017-06-20 17:09:14 +00:00
|
|
|
Channel string `yaml:"channel,omitempty" json:"channel,omitempty"`
|
|
|
|
Username string `yaml:"username,omitempty" json:"username,omitempty"`
|
|
|
|
Color string `yaml:"color,omitempty" json:"color,omitempty"`
|
|
|
|
|
2018-05-14 15:26:11 +00:00
|
|
|
Title string `yaml:"title,omitempty" json:"title,omitempty"`
|
|
|
|
TitleLink string `yaml:"title_link,omitempty" json:"title_link,omitempty"`
|
|
|
|
Pretext string `yaml:"pretext,omitempty" json:"pretext,omitempty"`
|
|
|
|
Text string `yaml:"text,omitempty" json:"text,omitempty"`
|
|
|
|
Fields []*SlackField `yaml:"fields,omitempty" json:"fields,omitempty"`
|
2020-07-24 12:58:59 +00:00
|
|
|
ShortFields bool `yaml:"short_fields" json:"short_fields,omitempty"`
|
2018-05-14 15:26:11 +00:00
|
|
|
Footer string `yaml:"footer,omitempty" json:"footer,omitempty"`
|
|
|
|
Fallback string `yaml:"fallback,omitempty" json:"fallback,omitempty"`
|
2018-10-23 09:22:23 +00:00
|
|
|
CallbackID string `yaml:"callback_id,omitempty" json:"callback_id,omitempty"`
|
2018-05-14 15:26:11 +00:00
|
|
|
IconEmoji string `yaml:"icon_emoji,omitempty" json:"icon_emoji,omitempty"`
|
|
|
|
IconURL string `yaml:"icon_url,omitempty" json:"icon_url,omitempty"`
|
2018-08-13 13:14:45 +00:00
|
|
|
ImageURL string `yaml:"image_url,omitempty" json:"image_url,omitempty"`
|
|
|
|
ThumbURL string `yaml:"thumb_url,omitempty" json:"thumb_url,omitempty"`
|
2020-07-24 12:58:59 +00:00
|
|
|
LinkNames bool `yaml:"link_names" json:"link_names,omitempty"`
|
2019-07-31 10:04:59 +00:00
|
|
|
MrkdwnIn []string `yaml:"mrkdwn_in,omitempty" json:"mrkdwn_in,omitempty"`
|
2018-05-14 15:26:11 +00:00
|
|
|
Actions []*SlackAction `yaml:"actions,omitempty" json:"actions,omitempty"`
|
2015-10-09 08:48:25 +00:00
|
|
|
}
|
|
|
|
|
2015-09-25 14:14:30 +00:00
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *SlackConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultSlackConfig
|
|
|
|
type plain SlackConfig
|
2021-04-02 01:57:55 +00:00
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.APIURL != nil && len(c.APIURLFile) > 0 {
|
|
|
|
return fmt.Errorf("at most one of api_url & api_url_file must be configured")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
2015-11-12 11:57:20 +00:00
|
|
|
// WebhookConfig configures notifications via a generic webhook.
|
2015-09-25 14:14:30 +00:00
|
|
|
type WebhookConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2015-12-17 12:05:06 +00:00
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2015-09-25 14:14:30 +00:00
|
|
|
// URL to send POST request to.
|
2018-07-26 10:39:33 +00:00
|
|
|
URL *URL `yaml:"url" json:"url"`
|
2020-06-04 08:07:33 +00:00
|
|
|
// MaxAlerts is the maximum number of alerts to be sent per webhook message.
|
|
|
|
// Alerts exceeding this threshold will be truncated. Setting this to 0
|
|
|
|
// allows an unlimited number of alerts.
|
|
|
|
MaxAlerts uint64 `yaml:"max_alerts" json:"max_alerts"`
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *WebhookConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
2015-12-17 12:05:06 +00:00
|
|
|
*c = DefaultWebhookConfig
|
2015-09-25 14:14:30 +00:00
|
|
|
type plain WebhookConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-07-26 10:39:33 +00:00
|
|
|
if c.URL == nil {
|
2015-09-25 14:14:30 +00:00
|
|
|
return fmt.Errorf("missing URL in webhook config")
|
|
|
|
}
|
2018-07-26 10:39:33 +00:00
|
|
|
if c.URL.Scheme != "https" && c.URL.Scheme != "http" {
|
2017-12-07 14:01:16 +00:00
|
|
|
return fmt.Errorf("scheme required for webhook url")
|
|
|
|
}
|
2018-04-17 14:22:46 +00:00
|
|
|
return nil
|
2015-09-25 14:14:30 +00:00
|
|
|
}
|
2015-11-24 22:29:25 +00:00
|
|
|
|
2017-12-09 15:20:22 +00:00
|
|
|
// WechatConfig configures notifications via Wechat.
|
|
|
|
type WechatConfig struct {
|
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
|
|
|
|
2018-02-06 14:43:49 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2020-07-06 13:56:42 +00:00
|
|
|
APISecret Secret `yaml:"api_secret,omitempty" json:"api_secret,omitempty"`
|
|
|
|
CorpID string `yaml:"corp_id,omitempty" json:"corp_id,omitempty"`
|
|
|
|
Message string `yaml:"message,omitempty" json:"message,omitempty"`
|
|
|
|
APIURL *URL `yaml:"api_url,omitempty" json:"api_url,omitempty"`
|
|
|
|
ToUser string `yaml:"to_user,omitempty" json:"to_user,omitempty"`
|
|
|
|
ToParty string `yaml:"to_party,omitempty" json:"to_party,omitempty"`
|
|
|
|
ToTag string `yaml:"to_tag,omitempty" json:"to_tag,omitempty"`
|
|
|
|
AgentID string `yaml:"agent_id,omitempty" json:"agent_id,omitempty"`
|
|
|
|
MessageType string `yaml:"message_type,omitempty" json:"message_type,omitempty"`
|
2017-12-09 15:20:22 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 13:56:42 +00:00
|
|
|
const wechatValidTypesRe = `^(text|markdown)$`
|
|
|
|
|
|
|
|
var wechatTypeMatcher = regexp.MustCompile(wechatValidTypesRe)
|
|
|
|
|
2017-12-09 15:20:22 +00:00
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *WechatConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultWechatConfig
|
|
|
|
type plain WechatConfig
|
2020-07-06 13:56:42 +00:00
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.MessageType == "" {
|
|
|
|
c.MessageType = "text"
|
|
|
|
}
|
|
|
|
|
|
|
|
if !wechatTypeMatcher.MatchString(c.MessageType) {
|
|
|
|
return errors.Errorf("WeChat message type %q does not match valid options %s", c.MessageType, wechatValidTypesRe)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2017-12-09 15:20:22 +00:00
|
|
|
}
|
|
|
|
|
2015-11-24 22:29:25 +00:00
|
|
|
// OpsGenieConfig configures notifications via OpsGenie.
|
|
|
|
type OpsGenieConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2019-05-13 12:51:26 +00:00
|
|
|
APIKey Secret `yaml:"api_key,omitempty" json:"api_key,omitempty"`
|
|
|
|
APIURL *URL `yaml:"api_url,omitempty" json:"api_url,omitempty"`
|
|
|
|
Message string `yaml:"message,omitempty" json:"message,omitempty"`
|
|
|
|
Description string `yaml:"description,omitempty" json:"description,omitempty"`
|
|
|
|
Source string `yaml:"source,omitempty" json:"source,omitempty"`
|
|
|
|
Details map[string]string `yaml:"details,omitempty" json:"details,omitempty"`
|
|
|
|
Responders []OpsGenieConfigResponder `yaml:"responders,omitempty" json:"responders,omitempty"`
|
|
|
|
Tags string `yaml:"tags,omitempty" json:"tags,omitempty"`
|
|
|
|
Note string `yaml:"note,omitempty" json:"note,omitempty"`
|
|
|
|
Priority string `yaml:"priority,omitempty" json:"priority,omitempty"`
|
2015-11-24 22:29:25 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 09:05:38 +00:00
|
|
|
const opsgenieValidTypesRe = `^(team|user|escalation|schedule)$`
|
2019-05-13 12:51:26 +00:00
|
|
|
|
|
|
|
var opsgenieTypeMatcher = regexp.MustCompile(opsgenieValidTypesRe)
|
|
|
|
|
2015-11-24 22:29:25 +00:00
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *OpsGenieConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultOpsGenieConfig
|
|
|
|
type plain OpsGenieConfig
|
2019-05-13 12:51:26 +00:00
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, r := range c.Responders {
|
|
|
|
if r.ID == "" && r.Username == "" && r.Name == "" {
|
|
|
|
return errors.Errorf("OpsGenieConfig responder %v has to have at least one of id, username or name specified", r)
|
|
|
|
}
|
|
|
|
|
|
|
|
r.Type = strings.ToLower(r.Type)
|
|
|
|
if !opsgenieTypeMatcher.MatchString(r.Type) {
|
|
|
|
return errors.Errorf("OpsGenieConfig responder %v type does not match valid options %s", r, opsgenieValidTypesRe)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type OpsGenieConfigResponder struct {
|
|
|
|
// One of those 3 should be filled.
|
|
|
|
ID string `yaml:"id,omitempty" json:"id,omitempty"`
|
|
|
|
Name string `yaml:"name,omitempty" json:"name,omitempty"`
|
|
|
|
Username string `yaml:"username,omitempty" json:"username,omitempty"`
|
|
|
|
|
|
|
|
// team, user, escalation, schedule etc.
|
|
|
|
Type string `yaml:"type,omitempty" json:"type,omitempty"`
|
2015-11-24 22:29:25 +00:00
|
|
|
}
|
2016-02-26 08:35:00 +00:00
|
|
|
|
2016-07-03 15:33:44 +00:00
|
|
|
// VictorOpsConfig configures notifications via VictorOps.
|
|
|
|
type VictorOpsConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2016-07-03 15:33:44 +00:00
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2021-04-02 01:57:55 +00:00
|
|
|
APIKey Secret `yaml:"api_key,omitempty" json:"api_key,omitempty"`
|
|
|
|
APIKeyFile Secret `yaml:"api_key_file,omitempty" json:"api_key_file,omitempty"`
|
2019-01-15 10:59:05 +00:00
|
|
|
APIURL *URL `yaml:"api_url" json:"api_url"`
|
|
|
|
RoutingKey string `yaml:"routing_key" json:"routing_key"`
|
|
|
|
MessageType string `yaml:"message_type" json:"message_type"`
|
|
|
|
StateMessage string `yaml:"state_message" json:"state_message"`
|
|
|
|
EntityDisplayName string `yaml:"entity_display_name" json:"entity_display_name"`
|
|
|
|
MonitoringTool string `yaml:"monitoring_tool" json:"monitoring_tool"`
|
|
|
|
CustomFields map[string]string `yaml:"custom_fields,omitempty" json:"custom_fields,omitempty"`
|
2016-07-03 15:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *VictorOpsConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultVictorOpsConfig
|
|
|
|
type plain VictorOpsConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if c.RoutingKey == "" {
|
|
|
|
return fmt.Errorf("missing Routing key in VictorOps config")
|
|
|
|
}
|
2019-01-15 10:59:05 +00:00
|
|
|
|
|
|
|
reservedFields := []string{"routing_key", "message_type", "state_message", "entity_display_name", "monitoring_tool", "entity_id", "entity_state"}
|
|
|
|
|
|
|
|
for _, v := range reservedFields {
|
|
|
|
if _, ok := c.CustomFields[v]; ok {
|
|
|
|
return fmt.Errorf("VictorOps config contains custom field %s which cannot be used as it conflicts with the fixed/static fields", v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-17 14:22:46 +00:00
|
|
|
return nil
|
2016-07-03 15:33:44 +00:00
|
|
|
}
|
|
|
|
|
2016-02-26 08:35:00 +00:00
|
|
|
type duration time.Duration
|
|
|
|
|
|
|
|
func (d *duration) UnmarshalText(text []byte) error {
|
|
|
|
parsed, err := time.ParseDuration(string(text))
|
|
|
|
if err == nil {
|
|
|
|
*d = duration(parsed)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d duration) MarshalText() ([]byte, error) {
|
|
|
|
return []byte(time.Duration(d).String()), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type PushoverConfig struct {
|
2016-11-22 14:05:14 +00:00
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
2016-02-26 08:35:00 +00:00
|
|
|
|
2017-07-27 08:48:24 +00:00
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
2017-06-20 17:09:14 +00:00
|
|
|
UserKey Secret `yaml:"user_key,omitempty" json:"user_key,omitempty"`
|
|
|
|
Token Secret `yaml:"token,omitempty" json:"token,omitempty"`
|
|
|
|
Title string `yaml:"title,omitempty" json:"title,omitempty"`
|
|
|
|
Message string `yaml:"message,omitempty" json:"message,omitempty"`
|
|
|
|
URL string `yaml:"url,omitempty" json:"url,omitempty"`
|
2019-03-13 10:09:58 +00:00
|
|
|
URLTitle string `yaml:"url_title,omitempty" json:"url_title,omitempty"`
|
2018-12-18 14:15:30 +00:00
|
|
|
Sound string `yaml:"sound,omitempty" json:"sound,omitempty"`
|
2017-06-20 17:09:14 +00:00
|
|
|
Priority string `yaml:"priority,omitempty" json:"priority,omitempty"`
|
|
|
|
Retry duration `yaml:"retry,omitempty" json:"retry,omitempty"`
|
|
|
|
Expire duration `yaml:"expire,omitempty" json:"expire,omitempty"`
|
2020-07-24 12:58:59 +00:00
|
|
|
HTML bool `yaml:"html" json:"html,omitempty"`
|
2016-02-26 08:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *PushoverConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultPushoverConfig
|
|
|
|
type plain PushoverConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if c.UserKey == "" {
|
|
|
|
return fmt.Errorf("missing user key in Pushover config")
|
|
|
|
}
|
|
|
|
if c.Token == "" {
|
|
|
|
return fmt.Errorf("missing token in Pushover config")
|
|
|
|
}
|
2018-04-17 14:22:46 +00:00
|
|
|
return nil
|
2016-02-26 08:35:00 +00:00
|
|
|
}
|
2021-06-10 02:13:26 +00:00
|
|
|
|
|
|
|
// SigV4Config is the configuration for signing remote write requests with
|
|
|
|
// AWS's SigV4 verification process. Empty values will be retrieved using the
|
|
|
|
// AWS default credentials chain.
|
2021-06-14 23:28:57 +00:00
|
|
|
// TODO: Move to common.
|
2021-06-10 02:13:26 +00:00
|
|
|
type SigV4Config struct {
|
|
|
|
Region string `yaml:"region,omitempty"`
|
|
|
|
AccessKey string `yaml:"access_key,omitempty"`
|
|
|
|
SecretKey Secret `yaml:"secret_key,omitempty"`
|
|
|
|
Profile string `yaml:"profile,omitempty"`
|
|
|
|
RoleARN string `yaml:"role_arn,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type SNSConfig struct {
|
|
|
|
NotifierConfig `yaml:",inline" json:",inline"`
|
|
|
|
|
|
|
|
HTTPConfig *commoncfg.HTTPClientConfig `yaml:"http_config,omitempty" json:"http_config,omitempty"`
|
|
|
|
|
|
|
|
APIUrl string `yaml:"api_url" json:"api_url"`
|
|
|
|
Sigv4 SigV4Config `yaml:"sigv4" json:"sigv4"`
|
|
|
|
TopicARN string `yaml:"topic_arn,omitempty" json:"topic_arn,omitempty"`
|
|
|
|
PhoneNumber string `yaml:"phone_number,omitempty" json:"phone_number,omitempty"`
|
|
|
|
TargetARN string `yaml:"target_arn,omitempty" json:"target_arn,omitempty"`
|
2021-06-11 21:21:15 +00:00
|
|
|
Subject string `yaml:"subject,omitempty" json:"subject,omitempty"`
|
2021-06-10 02:13:26 +00:00
|
|
|
Message string `yaml:"message,omitempty" json:"message,omitempty"`
|
|
|
|
Attributes map[string]string `yaml:"attributes,omitempty" json:"attributes,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
|
|
func (c *SNSConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
|
|
*c = DefaultSNSConfig
|
|
|
|
type plain SNSConfig
|
|
|
|
if err := unmarshal((*plain)(c)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-06-16 19:27:19 +00:00
|
|
|
if (c.TargetARN == "") != (c.TopicARN == "") != (c.PhoneNumber == "") {
|
2021-06-10 02:13:26 +00:00
|
|
|
return fmt.Errorf("must provide either a Target ARN, Topic ARN, or Phone Number for SNS config")
|
|
|
|
}
|
2021-06-15 22:24:52 +00:00
|
|
|
if (c.Sigv4.AccessKey == "") != (c.Sigv4.SecretKey == "") {
|
|
|
|
return fmt.Errorf("must provide a AWS SigV4 Access key and Secret Key if credentials are specified in the SNS config")
|
|
|
|
}
|
2021-06-10 02:13:26 +00:00
|
|
|
return nil
|
|
|
|
}
|