2023-09-01 17:06:12 +00:00
|
|
|
package bridge
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"log/slog"
|
2023-09-03 14:37:02 +00:00
|
|
|
"net/http"
|
2023-09-01 17:06:12 +00:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2023-09-03 11:44:16 +00:00
|
|
|
type fluxInvolvedObject struct {
|
|
|
|
Kind string `json:"kind"`
|
|
|
|
Namespace string `json:"namespace"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
UID string `json:"uid"`
|
|
|
|
APIVersion string `json:"apiVersion"`
|
|
|
|
ResourceVersion string `json:"resourceVersion"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f fluxInvolvedObject) String() string {
|
|
|
|
return strings.ToLower(f.Kind) + "/" + f.Namespace + "." + f.Name
|
|
|
|
}
|
|
|
|
|
2023-09-01 17:06:12 +00:00
|
|
|
type FluxNotification struct {
|
2023-09-03 11:44:16 +00:00
|
|
|
InvolvedObject fluxInvolvedObject `json:"involvedObject"`
|
|
|
|
Severity string `json:"severity"`
|
|
|
|
Timestamp time.Time `json:"timestamp"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
Reason string `json:"reason"`
|
|
|
|
Metadata struct {
|
2023-09-03 11:07:42 +00:00
|
|
|
CommitStatus string `json:"commit_status"`
|
|
|
|
Revision string `json:"revision"`
|
|
|
|
Summary string `json:"summary"`
|
2023-09-01 17:06:12 +00:00
|
|
|
} `json:"metadata"`
|
2023-09-03 11:07:42 +00:00
|
|
|
ReportingController string `json:"reportingController"`
|
|
|
|
ReportingInstance string `json:"reportingInstance"`
|
2023-09-01 17:06:12 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 11:44:16 +00:00
|
|
|
type FluxHandler struct {
|
|
|
|
// Register all modifications of reconciliations
|
|
|
|
reconciliations map[string]bool
|
|
|
|
}
|
2023-09-01 17:06:12 +00:00
|
|
|
|
|
|
|
func NewFluxHandler() FluxHandler {
|
2023-09-03 11:44:16 +00:00
|
|
|
return FluxHandler{
|
|
|
|
reconciliations: make(map[string]bool),
|
|
|
|
}
|
2023-09-01 17:06:12 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 14:37:02 +00:00
|
|
|
func (f FluxHandler) ProduceNotifications(r *http.Request) ([]Notification, error) {
|
2023-09-01 17:06:12 +00:00
|
|
|
l := slog.With(slog.String("handler", "flux"))
|
2023-09-03 14:37:02 +00:00
|
|
|
dec := json.NewDecoder(r.Body)
|
|
|
|
defer r.Body.Close()
|
2023-09-01 17:06:12 +00:00
|
|
|
|
|
|
|
var not FluxNotification
|
|
|
|
if err := dec.Decode(¬); err != nil {
|
2023-09-03 14:37:02 +00:00
|
|
|
l.Error("invalid message format", "error", err)
|
|
|
|
return nil, err
|
2023-09-01 17:06:12 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 11:44:16 +00:00
|
|
|
obj := not.InvolvedObject.String()
|
2023-09-03 10:19:34 +00:00
|
|
|
if not.Reason == "ReconciliationSucceeded" {
|
2023-09-03 11:44:16 +00:00
|
|
|
if ok := f.reconciliations[obj]; !ok {
|
|
|
|
// Filter out spammy ReconciliationSucceeded notification
|
2023-09-03 14:37:02 +00:00
|
|
|
return nil, errSkipNotification
|
2023-09-03 11:44:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// we will print the object so skip it next time it spam
|
|
|
|
f.reconciliations[obj] = false
|
|
|
|
} else {
|
|
|
|
// object has been modified, we can print it next time
|
|
|
|
f.reconciliations[obj] = true
|
2023-09-03 10:19:34 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 11:44:16 +00:00
|
|
|
title := fmt.Sprintf("[%s] %s %s", not.Severity, not.Reason, obj)
|
2023-09-03 11:07:42 +00:00
|
|
|
body := not.Message + "\n\n**revision**\n" + not.Metadata.Revision
|
2023-09-01 17:06:12 +00:00
|
|
|
|
|
|
|
l.Debug("flux notification", slog.Group("notification",
|
|
|
|
slog.String("title", title),
|
|
|
|
slog.String("body", body)))
|
2023-09-03 14:37:02 +00:00
|
|
|
return []Notification{{
|
2023-09-01 17:06:12 +00:00
|
|
|
Title: title,
|
|
|
|
Body: body,
|
|
|
|
IsMarkdown: true,
|
2023-09-03 14:37:02 +00:00
|
|
|
}}, nil
|
2023-09-01 17:06:12 +00:00
|
|
|
}
|