mirror of
https://github.com/replicatedhq/troubleshoot.git
synced 2026-04-15 07:16:34 +00:00
* Allowing more comperators for the http analyzer * test * Update pkg/analyze/host_http.go Co-authored-by: Andrew Lavery <laverya@umich.edu> --------- Co-authored-by: Andrew Lavery <laverya@umich.edu>
190 lines
4.9 KiB
Go
190 lines
4.9 KiB
Go
package analyzer
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/casbin/govaluate"
|
|
"github.com/pkg/errors"
|
|
troubleshootv1beta2 "github.com/replicatedhq/troubleshoot/pkg/apis/troubleshoot/v1beta2"
|
|
"github.com/replicatedhq/troubleshoot/pkg/collect"
|
|
)
|
|
|
|
type httpResult struct {
|
|
Error *collect.HTTPError
|
|
Response *collect.HTTPResponse
|
|
}
|
|
|
|
type AnalyzeHostHTTP struct {
|
|
hostAnalyzer *troubleshootv1beta2.HTTPAnalyze
|
|
}
|
|
|
|
func (a *AnalyzeHostHTTP) Title() string {
|
|
return hostAnalyzerTitleOrDefault(a.hostAnalyzer.AnalyzeMeta, "HTTP Request")
|
|
}
|
|
|
|
func (a *AnalyzeHostHTTP) IsExcluded() (bool, error) {
|
|
return isExcluded(a.hostAnalyzer.Exclude)
|
|
}
|
|
|
|
func (a *AnalyzeHostHTTP) Analyze(
|
|
getCollectedFileContents func(string) ([]byte, error), findFiles getChildCollectedFileContents,
|
|
) ([]*AnalyzeResult, error) {
|
|
|
|
collectorName := a.hostAnalyzer.CollectorName
|
|
if collectorName == "" {
|
|
collectorName = "result"
|
|
}
|
|
|
|
const nodeBaseDir = "host-collectors/http"
|
|
localPath := fmt.Sprintf("%s/%s.json", nodeBaseDir, collectorName)
|
|
fileName := fmt.Sprintf("%s.json", collectorName)
|
|
|
|
collectedContents, err := retrieveCollectedContents(
|
|
getCollectedFileContents,
|
|
localPath,
|
|
nodeBaseDir,
|
|
fileName,
|
|
)
|
|
if err != nil {
|
|
return []*AnalyzeResult{{Title: a.Title()}}, err
|
|
}
|
|
|
|
results, err := analyzeHostCollectorResults(collectedContents, a.hostAnalyzer.Outcomes, a.CheckCondition, a.Title())
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to analyze http request")
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
func compareHostHTTPConditionalToActual(conditional string, result *httpResult) (res bool, err error) {
|
|
if conditional == "error" {
|
|
return result.Error != nil, nil
|
|
}
|
|
|
|
conditional = strings.ReplaceAll(conditional, " = ", " == ")
|
|
conditional = strings.ReplaceAll(conditional, " === ", " == ")
|
|
|
|
expression, err := govaluate.NewEvaluableExpression(conditional)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "failed to create evaluable expression")
|
|
}
|
|
|
|
if result.Response == nil {
|
|
return false, nil
|
|
}
|
|
|
|
parameters := make(map[string]interface{}, 8)
|
|
parameters["statusCode"] = result.Response.Status
|
|
|
|
comparisonResult, err := expression.Evaluate(parameters)
|
|
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "failed to evaluate expression")
|
|
}
|
|
|
|
boolResult, ok := comparisonResult.(bool)
|
|
if !ok {
|
|
return false, fmt.Errorf("expression did not evaluate to a boolean value")
|
|
}
|
|
|
|
return boolResult, nil
|
|
}
|
|
|
|
func analyzeHTTPResult(analyzer *troubleshootv1beta2.HTTPAnalyze, fileName string, getCollectedFileContents getCollectedFileContents, title string) ([]*AnalyzeResult, error) {
|
|
contents, err := getCollectedFileContents(fileName)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get collected file")
|
|
}
|
|
|
|
httpInfo := &httpResult{}
|
|
if err := json.Unmarshal(contents, httpInfo); err != nil {
|
|
return nil, errors.Wrap(err, "failed to unmarshal http result")
|
|
}
|
|
|
|
result := &AnalyzeResult{
|
|
Title: title,
|
|
}
|
|
|
|
for _, outcome := range analyzer.Outcomes {
|
|
if outcome.Fail != nil {
|
|
if outcome.Fail.When == "" {
|
|
result.IsFail = true
|
|
result.Message = outcome.Fail.Message
|
|
result.URI = outcome.Fail.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
|
|
isMatch, err := compareHostHTTPConditionalToActual(outcome.Fail.When, httpInfo)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to compare %s", outcome.Fail.When)
|
|
}
|
|
|
|
if isMatch {
|
|
result.IsFail = true
|
|
result.Message = outcome.Fail.Message
|
|
result.URI = outcome.Fail.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
} else if outcome.Warn != nil {
|
|
if outcome.Warn.When == "" {
|
|
result.IsWarn = true
|
|
result.Message = outcome.Warn.Message
|
|
result.URI = outcome.Warn.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
|
|
isMatch, err := compareHostHTTPConditionalToActual(outcome.Warn.When, httpInfo)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to compare %s", outcome.Warn.When)
|
|
}
|
|
|
|
if isMatch {
|
|
result.IsWarn = true
|
|
result.Message = outcome.Warn.Message
|
|
result.URI = outcome.Warn.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
} else if outcome.Pass != nil {
|
|
if outcome.Pass.When == "" {
|
|
result.IsPass = true
|
|
result.Message = outcome.Pass.Message
|
|
result.URI = outcome.Pass.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
|
|
isMatch, err := compareHostHTTPConditionalToActual(outcome.Pass.When, httpInfo)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to compare %s", outcome.Pass.When)
|
|
}
|
|
|
|
if isMatch {
|
|
result.IsPass = true
|
|
result.Message = outcome.Pass.Message
|
|
result.URI = outcome.Pass.URI
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return []*AnalyzeResult{result}, nil
|
|
}
|
|
|
|
func (a *AnalyzeHostHTTP) CheckCondition(when string, data []byte) (bool, error) {
|
|
|
|
var httpInfo httpResult
|
|
if err := json.Unmarshal(data, &httpInfo); err != nil {
|
|
return false, fmt.Errorf("failed to unmarshal data into httpResult: %v", err)
|
|
}
|
|
return compareHostHTTPConditionalToActual(when, &httpInfo)
|
|
}
|