Files
troubleshoot/pkg/analyze/json_compare.go
Evans Mungai 1444c01725 feat: json compare host analyser (#1582)
* feat: json compore host analyser

Signed-off-by: Evans Mungai <evans@replicated.com>

* Add missing json compare host analyser file

Signed-off-by: Evans Mungai <evans@replicated.com>

* Generate schemas

Signed-off-by: Evans Mungai <evans@replicated.com>

* Fix failing tests

Signed-off-by: Evans Mungai <evans@replicated.com>

* Ensure json compare analyser always has a title

Signed-off-by: Evans Mungai <evans@replicated.com>

---------

Signed-off-by: Evans Mungai <evans@replicated.com>
2024-07-24 14:27:20 +01:00

254 lines
7.1 KiB
Go

package analyzer
import (
"bytes"
"encoding/json"
"fmt"
"path/filepath"
"reflect"
"sort"
"strconv"
"github.com/pkg/errors"
util "github.com/replicatedhq/troubleshoot/internal/util"
troubleshootv1beta2 "github.com/replicatedhq/troubleshoot/pkg/apis/troubleshoot/v1beta2"
iutils "github.com/replicatedhq/troubleshoot/pkg/interfaceutils"
"k8s.io/client-go/util/jsonpath"
)
type AnalyzeJsonCompare struct {
analyzer *troubleshootv1beta2.JsonCompare
}
func (a *AnalyzeJsonCompare) Title() string {
return jsonCompareTitle(a.analyzer)
}
func jsonCompareTitle(analyser *troubleshootv1beta2.JsonCompare) string {
title := analyser.CheckName
if title == "" {
title = analyser.CollectorName
}
if title == "" {
title = "Json Compare"
}
return title
}
func (a *AnalyzeJsonCompare) IsExcluded() (bool, error) {
return isExcluded(a.analyzer.Exclude)
}
func (a *AnalyzeJsonCompare) Analyze(getFile getCollectedFileContents, findFiles getChildCollectedFileContents) ([]*AnalyzeResult, error) {
result, err := analyzeJsonCompare(a.analyzer, getFile, a.Title())
if err != nil {
return nil, err
}
result.Strict = a.analyzer.Strict.BoolOrDefaultFalse()
return []*AnalyzeResult{result}, nil
}
func analyzeJsonCompare(analyzer *troubleshootv1beta2.JsonCompare, getCollectedFileContents func(string) ([]byte, error), title string) (*AnalyzeResult, error) {
fullPath := filepath.Join(analyzer.CollectorName, analyzer.FileName)
collected, err := getCollectedFileContents(fullPath)
if err != nil {
return nil, errors.Wrapf(err, "failed to read collected file name: %s", fullPath)
}
var actual interface{}
err = json.Unmarshal(collected, &actual)
if err != nil {
return nil, errors.Wrap(err, "failed to parse collected data as json")
}
originalActual := actual
if analyzer.Path != "" {
actual, err = iutils.GetAtPath(actual, analyzer.Path)
if err != nil {
return nil, errors.Wrapf(err, "failed to get object at path: %s", analyzer.Path)
}
} else if analyzer.JsonPath != "" {
jsp := jsonpath.New(analyzer.CheckName)
jsp.AllowMissingKeys(true).EnableJSONOutput(true)
err = jsp.Parse(analyzer.JsonPath)
if err != nil {
return nil, errors.Wrapf(err, "failed to parse jsonpath: %s", analyzer.JsonPath)
}
var data bytes.Buffer
err = jsp.Execute(&data, actual)
if err != nil {
return nil, errors.Wrap(err, "failed to execute jsonpath")
}
err = json.NewDecoder(&data).Decode(&actual)
if err != nil {
return nil, errors.Wrap(err, "failed to decode jsonpath result")
}
// If we get back a single result in a slice unwrap it.
// Technically this doesn't strictly follow jsonpath, but it makes
// things easier downstream. Basically we don't want to require
// users to wrap a single result with [].
if a, ok := actual.([]interface{}); ok && len(a) == 1 {
actual = a[0]
}
}
var expected interface{}
err = json.Unmarshal([]byte(analyzer.Value), &expected)
if err != nil {
return nil, errors.Wrap(err, "failed to parse expected value as json")
}
result := &AnalyzeResult{
Title: title,
IconKey: "kubernetes_text_analyze",
IconURI: "https://troubleshoot.sh/images/analyzer-icons/text-analyze.svg",
}
// due to jsp.Execute may return a slice of results unsorted, we need to sort the slice before comparing
equal := deepEqualWithSlicesSorted(actual, expected)
for _, outcome := range analyzer.Outcomes {
if outcome.Fail != nil {
when := false
if outcome.Fail.When != "" {
when, err = strconv.ParseBool(outcome.Fail.When)
if err != nil {
return nil, errors.Wrapf(err, "failed to process when statement: %s", outcome.Fail.When)
}
}
outcome.Fail.Message, err = util.RenderTemplate(outcome.Fail.Message, originalActual)
if err != nil {
return nil, errors.Wrap(err, "failed to render template on outcome message")
}
if when == equal {
result.IsFail = true
result.Message = outcome.Fail.Message
result.URI = outcome.Fail.URI
return result, nil
}
} else if outcome.Warn != nil {
when := false
if outcome.Warn.When != "" {
when, err = strconv.ParseBool(outcome.Warn.When)
if err != nil {
return nil, errors.Wrapf(err, "failed to process when statement: %s", outcome.Warn.When)
}
}
outcome.Warn.Message, err = util.RenderTemplate(outcome.Warn.Message, originalActual)
if err != nil {
return nil, errors.Wrap(err, "failed to render template on outcome message")
}
if when == equal {
result.IsWarn = true
result.Message = outcome.Warn.Message
result.URI = outcome.Warn.URI
return result, nil
}
} else if outcome.Pass != nil {
when := true // default to passing when values are equal
if outcome.Pass.When != "" {
when, err = strconv.ParseBool(outcome.Pass.When)
if err != nil {
return nil, errors.Wrapf(err, "failed to process when statement: %s", outcome.Pass.When)
}
}
outcome.Pass.Message, err = util.RenderTemplate(outcome.Pass.Message, originalActual)
if err != nil {
return nil, errors.Wrap(err, "failed to render template on outcome message")
}
if when == equal {
result.IsPass = true
result.Message = outcome.Pass.Message
result.URI = outcome.Pass.URI
return result, nil
}
}
}
return &AnalyzeResult{
Title: title,
IconKey: "kubernetes_text_analyze",
IconURI: "https://troubleshoot.sh/images/analyzer-icons/text-analyze.svg",
IsFail: true,
Message: "Invalid analyzer",
}, nil
}
// deepEqualWithSlicesSorted compares two interfaces and returns true if they contain the same values
// If the interfaces are slices, they are sorted before comparison to ensure order does not matter
// If the interfaces are not slices, reflect.DeepEqual is used
func deepEqualWithSlicesSorted(actual, expected interface{}) bool {
ra, re := reflect.ValueOf(actual), reflect.ValueOf(expected)
// If types are different, they're not equal
if ra.Kind() != re.Kind() {
return false
}
// If types are slices, compare sorted slices
if ra.Kind() == reflect.Slice {
return compareSortedSlices(ra.Interface().([]interface{}), re.Interface().([]interface{}))
}
// Otherwise, compare values (reflect.DeepEqual)
return reflect.DeepEqual(actual, expected)
}
// compareSortedSlices compares two sorted slices of interfaces and returns true if they contain the same values
func compareSortedSlices(actual, expected []interface{}) bool {
if len(actual) != len(expected) {
return false
}
// Sort slices
sortSliceOfInterfaces(actual)
sortSliceOfInterfaces(expected)
// Compare slices (reflect.DeepEqual)
return reflect.DeepEqual(actual, expected)
}
func sortSliceOfInterfaces(slice []interface{}) {
sort.Slice(slice, func(i, j int) bool {
return order(slice[i], slice[j])
})
}
// order function determines the order of two interface{} values
func order(a, b interface{}) bool {
switch va := a.(type) {
case int:
if vb, ok := b.(int); ok {
return va < vb
}
case float64:
if vb, ok := b.(float64); ok {
return va < vb
}
case string:
if vb, ok := b.(string); ok {
return va < vb
}
case bool:
if vb, ok := b.(bool); ok {
return !va && vb // false < true
}
}
// use string representation for comparison
return fmt.Sprintf("%v", a) < fmt.Sprintf("%v", b)
}