mirror of
https://github.com/prymitive/karma
synced 2026-05-05 03:16:51 +00:00
204 lines
6.0 KiB
Go
204 lines
6.0 KiB
Go
package alertmanager
|
|
|
|
import (
|
|
"sort"
|
|
|
|
"github.com/prymitive/karma/internal/config"
|
|
"github.com/prymitive/karma/internal/models"
|
|
"github.com/prymitive/karma/internal/slices"
|
|
"github.com/prymitive/karma/internal/transform"
|
|
)
|
|
|
|
// 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 := []models.AlertGroup{}
|
|
alertStates := map[string][]string{}
|
|
for _, agList := range uniqueGroups {
|
|
alerts := map[string]models.Alert{}
|
|
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, alert.Receiver) {
|
|
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
|
|
}
|
|
// set endsAt to the oldest value we have
|
|
if alert.EndsAt.After(a.EndsAt) {
|
|
a.EndsAt = alert.EndsAt
|
|
}
|
|
// update map
|
|
alerts[alertLFP] = a
|
|
// and append alert state to the slice
|
|
alertStates[alertLFP] = append(alertStates[alertLFP], alert.State)
|
|
} else {
|
|
alerts[alertLFP] = models.Alert(alert)
|
|
// seed alert state slice
|
|
alertStates[alertLFP] = []string{alert.State}
|
|
}
|
|
}
|
|
}
|
|
// skip empty groups
|
|
if len(alerts) == 0 {
|
|
continue
|
|
}
|
|
ag := models.AlertGroup(agList[0])
|
|
ag.Alerts = models.AlertList{}
|
|
for _, alert := range alerts {
|
|
// strip labels and annotations user doesn't want to see in the UI
|
|
alert.Labels = transform.StripLables(config.Config.Labels.Keep, config.Config.Labels.Strip, 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()
|
|
if slices.StringInSlice(alertStates[alertLFP], models.AlertStateActive) {
|
|
alert.State = models.AlertStateActive
|
|
} else if slices.StringInSlice(alertStates[alertLFP], models.AlertStateSuppressed) {
|
|
alert.State = models.AlertStateSuppressed
|
|
} else {
|
|
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)
|
|
}
|
|
sort.Sort(ag.Alerts)
|
|
ag.Hash = ag.ContentFingerprint()
|
|
dedupedGroups = append(dedupedGroups, ag)
|
|
}
|
|
|
|
// sort alert groups so they are always returned in the same order
|
|
// use group ID which is unique and immutable
|
|
sort.Slice(dedupedGroups, func(i, j int) bool {
|
|
return dedupedGroups[i].ID < dedupedGroups[j].ID
|
|
})
|
|
|
|
return dedupedGroups
|
|
}
|
|
|
|
// 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] = map[string]models.LabelColors{}
|
|
}
|
|
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 {
|
|
dedupedAutocomplete := []models.Autocomplete{}
|
|
uniqueAutocomplete := map[string]*models.Autocomplete{}
|
|
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
ac := am.Autocomplete()
|
|
for _, hint := range ac {
|
|
h, found := uniqueAutocomplete[hint.Value]
|
|
if found {
|
|
for _, token := range hint.Tokens {
|
|
if !slices.StringInSlice(h.Tokens, token) {
|
|
h.Tokens = append(h.Tokens, token)
|
|
}
|
|
}
|
|
} else {
|
|
uniqueAutocomplete[hint.Value] = &models.Autocomplete{
|
|
Value: hint.Value,
|
|
Tokens: hint.Tokens,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, hint := range uniqueAutocomplete {
|
|
dedupedAutocomplete = append(dedupedAutocomplete, *hint)
|
|
}
|
|
|
|
return dedupedAutocomplete
|
|
}
|
|
|
|
// DedupKnownLabels returns a deduplicated slice of all known label names
|
|
func DedupKnownLabels() []string {
|
|
dedupedLabels := map[string]bool{}
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
for _, key := range am.KnownLabels() {
|
|
dedupedLabels[key] = true
|
|
}
|
|
}
|
|
|
|
flatLabels := []string{}
|
|
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]bool{}
|
|
upstreams := GetAlertmanagers()
|
|
|
|
for _, am := range upstreams {
|
|
for _, ag := range am.Alerts() {
|
|
for _, alert := range ag.Alerts {
|
|
if val, found := alert.Labels[name]; found {
|
|
dedupedValues[val] = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
flatValues := []string{}
|
|
for key := range dedupedValues {
|
|
flatValues = append(flatValues, key)
|
|
}
|
|
return flatValues
|
|
}
|