diff --git a/api/v1/kubernetes.go b/api/v1/kubernetes.go index d4d9711f..186eef06 100644 --- a/api/v1/kubernetes.go +++ b/api/v1/kubernetes.go @@ -14,6 +14,21 @@ import ( "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" ) +// ReservedAnnotations +const ( + // AnnotationIgnoreConfig excludes the object from being scraped + AnnotationIgnoreConfig = "config-db.flanksource.com/ignore" + + // AnnotationIgnoreChangeByType contains the list of change types to ignore + AnnotationIgnoreChangeByType = "config-db.flanksource.com/ignore-changes" + + // AnnotationIgnoreChangeBySeverity contains the list of severity for the change types to ignore + AnnotationIgnoreChangeBySeverity = "config-db.flanksource.com/ignore-change-severity" + + // AnnotationCustomTags contains the list of tags to add to the scraped config + AnnotationCustomTags = "config-db.flanksource.com/tags" +) + // SeverityKeywords is used to identify the severity // from the Kubernetes Event reason. type SeverityKeywords struct { diff --git a/db/update.go b/db/update.go index 810bc96d..adb5a2e2 100644 --- a/db/update.go +++ b/db/update.go @@ -11,6 +11,7 @@ import ( "github.com/aws/smithy-go/ptr" "github.com/dominikbraun/graph" jsonpatch "github.com/evanphx/json-patch" + "github.com/flanksource/commons/collections" "github.com/flanksource/commons/logger" cUtils "github.com/flanksource/commons/utils" "github.com/flanksource/config-db/api" @@ -115,7 +116,7 @@ func updateCI(ctx api.ScrapeContext, result v1.ScrapeResult, ci, existing *model updates["delete_reason"] = gorm.Expr("NULL") } - changeResult, err := generateConfigChange(*ci, *existing) + changeResult, err := generateConfigChange(ctx, *ci, *existing) if err != nil { logger.Errorf("[%s] failed to check for changes: %v", ci, err) } else if changeResult != nil { @@ -514,7 +515,14 @@ func generateDiff(newConf, prevConfig string) (string, error) { // generateConfigChange calculates the diff (git style) and patches between the // given 2 config items and returns a ConfigChange object if there are any changes. -func generateConfigChange(newConf, prev models.ConfigItem) (*v1.ChangeResult, error) { +func generateConfigChange(ctx api.ScrapeContext, newConf, prev models.ConfigItem) (*v1.ChangeResult, error) { + if changeTypExclusion, ok := lo.FromPtr(newConf.Labels)[v1.AnnotationIgnoreChangeByType]; ok { + if collections.MatchItems("diff", strings.Split(changeTypExclusion, ",")...) { + ctx.Logger.V(4).Infof("excluding diff change for config(%s) with annotation (%s)", newConf, changeTypExclusion) + return nil, nil + } + } + diff, err := generateDiff(*newConf.Config, *prev.Config) if err != nil { return nil, fmt.Errorf("failed to generate diff: %w", err) diff --git a/scrapers/kubernetes/kubernetes.go b/scrapers/kubernetes/kubernetes.go index d5ea7dfc..a82cbb9b 100644 --- a/scrapers/kubernetes/kubernetes.go +++ b/scrapers/kubernetes/kubernetes.go @@ -39,21 +39,6 @@ const ConfigTypePrefix = "Kubernetes::" var resourceIDMapPerCluster PerClusterResourceIDMap -// ReservedAnnotations -const ( - // AnnotationIgnoreConfig excludes the object from being scraped - AnnotationIgnoreConfig = "config-db.flanksource.com/ignore" - - // AnnotationIgnoreChangeByType contains the list of change types to ignore - AnnotationIgnoreChangeByType = "config-db.flanksource.com/ignore-changes" - - // AnnotationIgnoreChangeBySeverity contains the list of severity for the change types to ignore - AnnotationIgnoreChangeBySeverity = "config-db.flanksource.com/ignore-change-severity" - - // AnnotationCustomTags contains the list of tags to add to the scraped config - AnnotationCustomTags = "config-db.flanksource.com/tags" -) - func getConfigTypePrefix(apiVersion string) string { if strings.Contains(apiVersion, ".upbound.io") || strings.Contains(apiVersion, ".crossplane.io") { return "Crossplane::" @@ -181,8 +166,8 @@ func formObjChangeExclusionMap(objs []*unstructured.Unstructured) (map[string]st exclusionByType := make(map[string]string) exclusionBySeverity := make(map[string]string) for _, obj := range objs { - exclusionByType[string(obj.GetUID())] = obj.GetAnnotations()[AnnotationIgnoreChangeByType] - exclusionBySeverity[string(obj.GetUID())] = obj.GetAnnotations()[AnnotationIgnoreChangeBySeverity] + exclusionByType[string(obj.GetUID())] = obj.GetAnnotations()[v1.AnnotationIgnoreChangeByType] + exclusionBySeverity[string(obj.GetUID())] = obj.GetAnnotations()[v1.AnnotationIgnoreChangeBySeverity] } return exclusionByType, exclusionBySeverity @@ -216,11 +201,11 @@ func getObjectChangeExclusionAnnotations(ctx api.ScrapeContext, id string, exclu return "", "", err } else if item != nil && item.Labels != nil { labels := lo.FromPtr(item.Labels) - if v, ok := labels[AnnotationIgnoreChangeByType]; ok { + if v, ok := labels[v1.AnnotationIgnoreChangeByType]; ok { changeTypeExclusion = v } - if v, ok := labels[AnnotationIgnoreChangeBySeverity]; ok { + if v, ok := labels[v1.AnnotationIgnoreChangeBySeverity]; ok { changeSeverityExclusion = v } } @@ -284,12 +269,12 @@ func ExtractResults(ctx api.ScrapeContext, config v1.Kubernetes, objs []*unstruc continue } - if val, ok := obj.GetAnnotations()[AnnotationIgnoreConfig]; ok && val == "true" { - ctx.Tracef("excluding object due to annotation %s: %s/%s/%s", AnnotationIgnoreConfig, obj.GetKind(), obj.GetNamespace(), obj.GetName()) + if val, ok := obj.GetAnnotations()[v1.AnnotationIgnoreConfig]; ok && val == "true" { + ctx.Tracef("excluding object due to annotation %s: %s/%s/%s", v1.AnnotationIgnoreConfig, obj.GetKind(), obj.GetNamespace(), obj.GetName()) continue } - if val, ok := obj.GetAnnotations()[AnnotationCustomTags]; ok { + if val, ok := obj.GetAnnotations()[v1.AnnotationCustomTags]; ok { for k, v := range collections.SelectorToMap(val) { tags = append(tags, v1.Tag{Name: k, Value: v}) } @@ -336,7 +321,7 @@ func ExtractResults(ctx api.ScrapeContext, config v1.Kubernetes, objs []*unstruc if collections.MatchItems(change.ChangeType, strings.Split(changeTypExclusion, ",")...) { ctx.Logger.V(4).Infof("excluding event object %s/%s/%s due to change type matched in annotation %s=%s", event.InvolvedObject.Namespace, event.InvolvedObject.Name, event.InvolvedObject.Kind, - AnnotationIgnoreChangeByType, changeTypExclusion) + v1.AnnotationIgnoreChangeByType, changeTypExclusion) continue } } @@ -345,7 +330,7 @@ func ExtractResults(ctx api.ScrapeContext, config v1.Kubernetes, objs []*unstruc if collections.MatchItems(change.Severity, strings.Split(changeSeverityExclusion, ",")...) { ctx.Logger.V(4).Infof("excluding event object %s/%s/%s due to severity matches in annotation %s=%s", event.InvolvedObject.Namespace, event.InvolvedObject.Name, event.InvolvedObject.Kind, - AnnotationIgnoreChangeBySeverity, changeSeverityExclusion) + v1.AnnotationIgnoreChangeBySeverity, changeSeverityExclusion) continue } } @@ -369,12 +354,12 @@ func ExtractResults(ctx api.ScrapeContext, config v1.Kubernetes, objs []*unstruc labels = obj.GetLabels() } - if v, ok := obj.GetAnnotations()[AnnotationIgnoreChangeBySeverity]; ok { - labels[AnnotationIgnoreChangeBySeverity] = v + if v, ok := obj.GetAnnotations()[v1.AnnotationIgnoreChangeBySeverity]; ok { + labels[v1.AnnotationIgnoreChangeBySeverity] = v } - if v, ok := obj.GetAnnotations()[AnnotationIgnoreChangeByType]; ok { - labels[AnnotationIgnoreChangeByType] = v + if v, ok := obj.GetAnnotations()[v1.AnnotationIgnoreChangeByType]; ok { + labels[v1.AnnotationIgnoreChangeByType] = v } if obj.GetKind() == "Node" {