2020-06-28 11:26:22 +00:00
package notifier
import (
2022-02-02 12:11:41 +00:00
"flag"
2020-06-28 11:26:22 +00:00
"fmt"
2022-02-02 14:18:40 +00:00
"strings"
2022-02-02 12:11:41 +00:00
"time"
2020-06-28 11:26:22 +00:00
2020-06-29 19:21:03 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/flagutil"
2022-02-02 12:11:41 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promauth"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal"
2020-06-28 11:26:22 +00:00
)
var (
2022-02-02 12:11:41 +00:00
configPath = flag . String ( "notifier.config" , "" , "Path to configuration file for notifiers" )
suppressDuplicateTargetErrors = flag . Bool ( "notifier.suppressDuplicateTargetErrors" , false , "Whether to suppress 'duplicate target' errors during discovery" )
2021-11-29 23:18:48 +00:00
addrs = flagutil . NewArray ( "notifier.url" , "Prometheus alertmanager URL, e.g. http://127.0.0.1:9093" )
2020-12-24 10:48:59 +00:00
basicAuthUsername = flagutil . NewArray ( "notifier.basicAuth.username" , "Optional basic auth username for -notifier.url" )
basicAuthPassword = flagutil . NewArray ( "notifier.basicAuth.password" , "Optional basic auth password for -notifier.url" )
2020-06-29 19:21:03 +00:00
2020-12-22 20:23:04 +00:00
tlsInsecureSkipVerify = flagutil . NewArrayBool ( "notifier.tlsInsecureSkipVerify" , "Whether to skip tls verification when connecting to -notifier.url" )
2020-06-29 19:21:03 +00:00
tlsCertFile = flagutil . NewArray ( "notifier.tlsCertFile" , "Optional path to client-side TLS certificate file to use when connecting to -notifier.url" )
tlsKeyFile = flagutil . NewArray ( "notifier.tlsKeyFile" , "Optional path to client-side TLS certificate key to use when connecting to -notifier.url" )
tlsCAFile = flagutil . NewArray ( "notifier.tlsCAFile" , "Optional path to TLS CA file to use for verifying connections to -notifier.url. " +
2020-06-28 11:26:22 +00:00
"By default system CA is used" )
2020-06-29 19:21:03 +00:00
tlsServerName = flagutil . NewArray ( "notifier.tlsServerName" , "Optional TLS server name to use for connections to -notifier.url. " +
2020-06-28 11:26:22 +00:00
"By default the server name from -notifier.url is used" )
)
2022-02-02 12:11:41 +00:00
// cw holds a configWatcher for configPath configuration file
// configWatcher provides a list of Notifier objects discovered
// from static config or via service discovery.
// cw is not nil only if configPath is provided.
var cw * configWatcher
// Reload checks the changes in configPath configuration file
// and applies changes if any.
func Reload ( ) error {
if cw == nil {
return nil
}
return cw . reload ( * configPath )
}
var staticNotifiersFn func ( ) [ ] Notifier
// Init returns a function for retrieving actual list of Notifier objects.
// Init works in two mods:
// * configuration via flags (for backward compatibility). Is always static
// and don't support live reloads.
// * configuration via file. Supports live reloads and service discovery.
// Init returns an error if both mods are used.
func Init ( gen AlertURLGenerator ) ( func ( ) [ ] Notifier , error ) {
if * configPath == "" && len ( * addrs ) == 0 {
return nil , nil
}
if * configPath != "" && len ( * addrs ) > 0 {
return nil , fmt . Errorf ( "only one of -notifier.config or -notifier.url flags must be specified" )
}
if len ( * addrs ) > 0 {
notifiers , err := notifiersFromFlags ( gen )
if err != nil {
return nil , fmt . Errorf ( "failed to create notifier from flag values: %s" , err )
}
staticNotifiersFn = func ( ) [ ] Notifier {
return notifiers
}
return staticNotifiersFn , nil
}
var err error
cw , err = newWatcher ( * configPath , gen )
if err != nil {
return nil , fmt . Errorf ( "failed to init config watcher: %s" , err )
}
return cw . notifiers , nil
}
func notifiersFromFlags ( gen AlertURLGenerator ) ( [ ] Notifier , error ) {
2020-06-29 19:21:03 +00:00
var notifiers [ ] Notifier
for i , addr := range * addrs {
2022-02-02 12:11:41 +00:00
authCfg := promauth . HTTPClientConfig {
TLSConfig : & promauth . TLSConfig {
CAFile : tlsCAFile . GetOptionalArg ( i ) ,
CertFile : tlsCertFile . GetOptionalArg ( i ) ,
KeyFile : tlsKeyFile . GetOptionalArg ( i ) ,
ServerName : tlsServerName . GetOptionalArg ( i ) ,
InsecureSkipVerify : tlsInsecureSkipVerify . GetOptionalArg ( i ) ,
} ,
BasicAuth : & promauth . BasicAuthConfig {
Username : basicAuthUsername . GetOptionalArg ( i ) ,
Password : promauth . NewSecret ( basicAuthPassword . GetOptionalArg ( i ) ) ,
} ,
}
2022-02-02 14:18:40 +00:00
addr = strings . TrimSuffix ( addr , "/" )
2022-02-02 12:11:41 +00:00
am , err := NewAlertManager ( addr + alertManagerPath , gen , authCfg , time . Minute )
2020-06-29 19:21:03 +00:00
if err != nil {
2022-02-02 12:11:41 +00:00
return nil , err
2020-06-29 19:21:03 +00:00
}
notifiers = append ( notifiers , am )
2020-06-28 11:26:22 +00:00
}
2020-06-29 19:21:03 +00:00
return notifiers , nil
2020-06-28 11:26:22 +00:00
}
2022-02-02 12:11:41 +00:00
// Target represents a Notifier and optional
// list of labels added during discovery.
type Target struct {
Notifier
Labels [ ] prompbmarshal . Label
}
// TargetType defines how the Target was discovered
type TargetType string
const (
// TargetStatic is for targets configured statically
TargetStatic TargetType = "static"
// TargetConsul is for targets discovered via Consul
TargetConsul TargetType = "consulSD"
)
// GetTargets returns list of static or discovered targets
// via notifier configuration.
func GetTargets ( ) map [ TargetType ] [ ] Target {
var targets = make ( map [ TargetType ] [ ] Target )
if staticNotifiersFn != nil {
for _ , ns := range staticNotifiersFn ( ) {
targets [ TargetStatic ] = append ( targets [ TargetStatic ] , Target {
Notifier : ns ,
} )
}
}
if cw != nil {
cw . targetsMu . RLock ( )
for key , ns := range cw . targets {
targets [ key ] = append ( targets [ key ] , ns ... )
}
cw . targetsMu . RUnlock ( )
}
return targets
}