mirror of
https://github.com/replicatedhq/troubleshoot.git
synced 2026-04-15 07:16:34 +00:00
* 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>
254 lines
7.1 KiB
Go
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)
|
|
}
|