mirror of
https://github.com/prymitive/karma
synced 2026-05-07 03:26:52 +00:00
265 lines
7.7 KiB
Go
265 lines
7.7 KiB
Go
package alertmanager
|
|
|
|
import (
|
|
"slices"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/prymitive/karma/internal/config"
|
|
"github.com/prymitive/karma/internal/models"
|
|
"github.com/prymitive/karma/internal/transform"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
// DedupAlerts will collect alert groups from all defined Alertmanager
|
|
// upstreams and deduplicate them, so we only return unique alerts
|
|
func DedupAlerts() []models.AlertGroup {
|
|
uniqueGroups := map[string][]models.AlertGroup{}
|
|
|
|
upstreams := GetAlertmanagers()
|
|
for _, am := range upstreams {
|
|
groups := am.Alerts()
|
|
for _, ag := range groups {
|
|
if _, found := uniqueGroups[ag.ID]; !found {
|
|
uniqueGroups[ag.ID] = []models.AlertGroup{}
|
|
}
|
|
uniqueGroups[ag.ID] = append(uniqueGroups[ag.ID], ag)
|
|
}
|
|
}
|
|
|
|
dedupedGroups := make([]models.AlertGroup, 0, len(uniqueGroups))
|
|
alertStates := map[string][]models.AlertState{}
|
|
for _, agList := range uniqueGroups {
|
|
totalAlerts := 0
|
|
for _, ag := range agList {
|
|
totalAlerts += len(ag.Alerts)
|
|
}
|
|
alerts := make(map[string]models.Alert, totalAlerts)
|
|
clear(alertStates)
|
|
for _, ag := range agList {
|
|
for _, alert := range ag.Alerts {
|
|
// remove all alerts for receiver(s) that the user doesn't
|
|
// want to see in the UI
|
|
if transform.StripReceivers(
|
|
config.Config.Receivers.Keep,
|
|
config.Config.Receivers.Strip,
|
|
config.Config.Receivers.CompiledKeepRegex,
|
|
config.Config.Receivers.CompiledStripRegex,
|
|
alert.Receiver,
|
|
) {
|
|
continue
|
|
}
|
|
|
|
keep := true
|
|
for _, am := range upstreams {
|
|
if !am.healthchecksVisible {
|
|
if _, hc := am.IsHealthCheckAlert(&alert); hc != nil {
|
|
log.Debug().Str("fingerprint", alert.Fingerprint).Msg("Skipping healthcheck alert")
|
|
keep = false
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if !keep {
|
|
continue
|
|
}
|
|
|
|
alertLFP := alert.LabelsFingerprint()
|
|
a, found := alerts[alertLFP]
|
|
if found {
|
|
// if we already have an alert with the same fp then just append
|
|
// alertmanager instances to it, this way we end up with all instances
|
|
// for each unique alert merged into a single alert with all
|
|
// alertmanager instances attached to it
|
|
a.Alertmanager = append(a.Alertmanager, alert.Alertmanager...)
|
|
// set startsAt to the earliest value we have
|
|
if alert.StartsAt.Before(a.StartsAt) {
|
|
a.StartsAt = alert.StartsAt
|
|
}
|
|
// update map
|
|
alerts[alertLFP] = a
|
|
// and append alert state to the slice
|
|
alertStates[alertLFP] = append(alertStates[alertLFP], alert.State)
|
|
} else {
|
|
alerts[alertLFP] = alert
|
|
// seed alert state slice
|
|
alertStates[alertLFP] = []models.AlertState{alert.State}
|
|
}
|
|
}
|
|
}
|
|
// skip empty groups
|
|
if len(alerts) == 0 {
|
|
continue
|
|
}
|
|
ag := agList[0]
|
|
ag.Labels = transform.StripLabels(
|
|
config.Config.Labels.Keep, config.Config.Labels.Strip,
|
|
config.Config.Labels.CompiledKeepRegex, config.Config.Labels.CompiledStripRegex,
|
|
ag.Labels)
|
|
ag.Alerts = make(models.AlertList, 0, len(alerts))
|
|
for _, alert := range alerts {
|
|
// strip labels and annotations user doesn't want to see in the UI
|
|
alert.Labels = transform.StripLabels(
|
|
config.Config.Labels.Keep, config.Config.Labels.Strip,
|
|
config.Config.Labels.CompiledKeepRegex, config.Config.Labels.CompiledStripRegex,
|
|
alert.Labels)
|
|
alert.Annotations = transform.StripAnnotations(config.Config.Annotations.Keep, config.Config.Annotations.Strip, alert.Annotations)
|
|
// calculate final alert state based on the most important value found
|
|
// in the list of states from all instances
|
|
alertLFP := alert.LabelsFingerprint()
|
|
switch {
|
|
case slices.Contains(alertStates[alertLFP], models.AlertStateActive):
|
|
alert.State = models.AlertStateActive
|
|
case slices.Contains(alertStates[alertLFP], models.AlertStateSuppressed):
|
|
alert.State = models.AlertStateSuppressed
|
|
default:
|
|
alert.State = models.AlertStateUnprocessed
|
|
}
|
|
|
|
// sort Alertmanager instances for every alert
|
|
sort.Slice(alert.Alertmanager, func(i, j int) bool {
|
|
return alert.Alertmanager[i].Name < alert.Alertmanager[j].Name
|
|
})
|
|
ag.Alerts = append(ag.Alerts, alert)
|
|
}
|
|
ag.Hash = ag.ContentFingerprint()
|
|
dedupedGroups = append(dedupedGroups, ag)
|
|
}
|
|
|
|
return dedupedGroups
|
|
}
|
|
|
|
// DedupSilences returns a deduplicated slice of all known silences
|
|
func DedupSilences() []models.ManagedSilence {
|
|
silenceByCluster := map[string]map[string]models.Silence{}
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
for id, silence := range am.Silences() {
|
|
|
|
if _, found := silenceByCluster[am.Cluster]; !found {
|
|
silenceByCluster[am.Cluster] = map[string]models.Silence{}
|
|
}
|
|
|
|
if _, ok := silenceByCluster[am.Cluster][id]; !ok {
|
|
silenceByCluster[am.Cluster][id] = silence
|
|
}
|
|
}
|
|
}
|
|
|
|
now := time.Now()
|
|
dedupedSilences := make([]models.ManagedSilence, 0, len(silenceByCluster))
|
|
for cluster, silenceMap := range silenceByCluster {
|
|
s := make([]models.ManagedSilence, 0, len(silenceMap))
|
|
for _, silence := range silenceMap {
|
|
managedSilence := models.ManagedSilence{
|
|
Cluster: cluster,
|
|
IsExpired: silence.EndsAt.Before(now),
|
|
Silence: silence,
|
|
}
|
|
s = append(s, managedSilence)
|
|
}
|
|
dedupedSilences = append(dedupedSilences, s...)
|
|
}
|
|
return dedupedSilences
|
|
}
|
|
|
|
// DedupColors returns a color map merged from all Alertmanager upstream color
|
|
// maps
|
|
func DedupColors() models.LabelsColorMap {
|
|
dedupedColors := models.LabelsColorMap{}
|
|
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
colors := am.Colors()
|
|
// map[string]map[string]LabelColors
|
|
for labelName, valueMap := range colors {
|
|
if _, found := dedupedColors[labelName]; !found {
|
|
dedupedColors[labelName] = make(map[string]models.LabelColors, len(valueMap))
|
|
}
|
|
for labelVal, labelColors := range valueMap {
|
|
if _, found := dedupedColors[labelName][labelVal]; !found {
|
|
dedupedColors[labelName][labelVal] = labelColors
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return dedupedColors
|
|
}
|
|
|
|
// DedupAutocomplete returns a list of autocomplete hints merged from all
|
|
// Alertmanager upstreams
|
|
func DedupAutocomplete() []models.Autocomplete {
|
|
upstreams := GetAlertmanagers()
|
|
|
|
var result []models.Autocomplete
|
|
index := map[string]int{}
|
|
|
|
for _, am := range upstreams {
|
|
am.ForEachAutocomplete(func(hint models.Autocomplete) {
|
|
mergeAutocompleteHint(&result, index, hint)
|
|
})
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func mergeAutocompleteHint(result *[]models.Autocomplete, index map[string]int, hint models.Autocomplete) {
|
|
if idx, found := index[hint.Value]; found {
|
|
for _, token := range hint.Tokens {
|
|
if !slices.Contains((*result)[idx].Tokens, token) {
|
|
(*result)[idx].Tokens = append((*result)[idx].Tokens, token)
|
|
}
|
|
}
|
|
} else {
|
|
index[hint.Value] = len(*result)
|
|
*result = append(*result, models.Autocomplete{
|
|
Value: hint.Value,
|
|
Tokens: hint.Tokens,
|
|
})
|
|
}
|
|
}
|
|
|
|
// DedupKnownLabels returns a deduplicated slice of all known label names
|
|
func DedupKnownLabels() []string {
|
|
dedupedLabels := map[string]struct{}{}
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
for _, key := range am.KnownLabels() {
|
|
dedupedLabels[key] = struct{}{}
|
|
}
|
|
}
|
|
|
|
flatLabels := make([]string, 0, len(dedupedLabels))
|
|
for key := range dedupedLabels {
|
|
flatLabels = append(flatLabels, key)
|
|
}
|
|
return flatLabels
|
|
}
|
|
|
|
// DedupKnownLabelValues returns a list of all known values for label $name
|
|
func DedupKnownLabelValues(name string) []string {
|
|
dedupedValues := map[string]struct{}{}
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
for _, ag := range am.Alerts() {
|
|
for _, alert := range ag.Alerts {
|
|
if v := alert.Labels.Get(name); v != "" {
|
|
dedupedValues[v] = struct{}{}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
flatValues := make([]string, 0, len(dedupedValues))
|
|
for key := range dedupedValues {
|
|
flatValues = append(flatValues, key)
|
|
}
|
|
return flatValues
|
|
}
|