Files
karma/internal/alertmanager/upstream.go
2019-10-21 11:40:34 +01:00

150 lines
3.8 KiB
Go

package alertmanager
import (
"fmt"
"net/http"
"net/url"
"sort"
"sync"
"time"
"github.com/prymitive/karma/internal/models"
"github.com/prymitive/karma/internal/uri"
log "github.com/sirupsen/logrus"
)
// Option allows to pass functional options to NewAlertmanager()
type Option func(am *Alertmanager) error
var (
upstreams = map[string]*Alertmanager{}
)
// NewAlertmanager creates a new Alertmanager instance
func NewAlertmanager(name, upstreamURI string, opts ...Option) (*Alertmanager, error) {
am := &Alertmanager{
URI: upstreamURI,
ExternalURI: "",
RequestTimeout: time.Second * 10,
Name: name,
lock: sync.RWMutex{},
alertGroups: []models.AlertGroup{},
silences: map[string]models.Silence{},
colors: models.LabelsColorMap{},
autocomplete: []models.Autocomplete{},
knownLabels: []string{},
HTTPHeaders: map[string]string{},
Metrics: alertmanagerMetrics{
Errors: map[string]float64{
labelValueErrorsAlerts: 0,
labelValueErrorsSilences: 0,
},
},
status: models.AlertmanagerStatus{},
}
for _, opt := range opts {
err := opt(am)
if err != nil {
return nil, err
}
}
var err error
am.reader, err = uri.NewReader(am.URI, am.RequestTimeout, am.HTTPTransport, am.HTTPHeaders)
if err != nil {
return am, err
}
return am, nil
}
// RegisterAlertmanager will add an Alertmanager instance to the list of
// instances used when pulling alerts from upstreams
func RegisterAlertmanager(am *Alertmanager) error {
if _, found := upstreams[am.Name]; found {
return fmt.Errorf("alertmanager upstream '%s' already exist", am.Name)
}
for _, existingAM := range upstreams {
if existingAM.URI == am.URI {
return fmt.Errorf("alertmanager upstream '%s' already collects from '%s'", existingAM.Name, existingAM.URI)
}
}
upstreams[am.Name] = am
log.Infof("[%s] Configured Alertmanager source at %s (proxied: %v)", am.Name, am.URI, am.ProxyRequests)
return nil
}
// GetAlertmanagers returns a list of all defined Alertmanager instances
func GetAlertmanagers() []*Alertmanager {
ams := []*Alertmanager{}
for _, am := range upstreams {
ams = append(ams, am)
}
sort.Slice(ams[:], func(i, j int) bool {
return ams[i].Name < ams[j].Name
})
return ams
}
// GetAlertmanagerByName returns an instance of Alertmanager by name or nil
// if not found
func GetAlertmanagerByName(name string) *Alertmanager {
am, found := upstreams[name]
if found {
return am
}
return nil
}
// WithProxy option can be passed to NewAlertmanager in order to enable request
// proxying for karma clients
func WithProxy(proxied bool) Option {
return func(am *Alertmanager) error {
am.ProxyRequests = proxied
return nil
}
}
// WithRequestTimeout option can be passed to NewAlertmanager in order to set
// a custom timeout for Alertmanager upstream requests
func WithRequestTimeout(timeout time.Duration) Option {
return func(am *Alertmanager) error {
am.RequestTimeout = timeout
return nil
}
}
// WithHTTPHeaders option can be passed to NewAlertManager in order to set
// a map of headers that will be passed with every request
func WithHTTPHeaders(headers map[string]string) Option {
return func(am *Alertmanager) error {
am.HTTPHeaders = headers
return nil
}
}
// WithHTTPTransport option can be passed to NewAlertmanager in order to set
// a custom HTTP transport (http.RoundTripper implementation)
func WithHTTPTransport(httpTransport http.RoundTripper) Option {
return func(am *Alertmanager) error {
am.HTTPTransport = httpTransport
return nil
}
}
// WithExternalURI option allows to set custom ExternalURI on our instance
func WithExternalURI(uri string) Option {
return func(am *Alertmanager) error {
// first validate that this is a valid URI
_, err := url.Parse(uri)
if err != nil {
return err
}
am.ExternalURI = uri
return nil
}
}