Files
troubleshoot/pkg/analyze/host_filesystem_performance.go
2021-03-01 22:45:16 +00:00

220 lines
6.7 KiB
Go

package analyzer
import (
"encoding/json"
"fmt"
"path/filepath"
"strconv"
"strings"
"time"
"github.com/pkg/errors"
troubleshootv1beta2 "github.com/replicatedhq/troubleshoot/pkg/apis/troubleshoot/v1beta2"
"github.com/replicatedhq/troubleshoot/pkg/collect"
)
type AnalyzeHostFilesystemPerformance struct {
hostAnalyzer *troubleshootv1beta2.FilesystemPerformanceAnalyze
}
func (a *AnalyzeHostFilesystemPerformance) Title() string {
return hostAnalyzerTitleOrDefault(a.hostAnalyzer.AnalyzeMeta, "Filesystem Performance")
}
func (a *AnalyzeHostFilesystemPerformance) IsExcluded() (bool, error) {
return isExcluded(a.hostAnalyzer.Exclude)
}
func (a *AnalyzeHostFilesystemPerformance) Analyze(getCollectedFileContents func(string) ([]byte, error)) (*AnalyzeResult, error) {
hostAnalyzer := a.hostAnalyzer
collectorName := hostAnalyzer.CollectorName
if collectorName == "" {
collectorName = "filesystemPerformance"
}
name := filepath.Join("filesystemPerformance", collectorName+".json")
contents, err := getCollectedFileContents(name)
if err != nil {
return nil, errors.Wrapf(err, "failed to get collected file %s", name)
}
fsPerf := collect.FSPerfResults{}
if err := json.Unmarshal(contents, &fsPerf); err != nil {
return nil, errors.Wrapf(err, "failed to unmarshal filesystem performance results from %s", name)
}
result := AnalyzeResult{}
result.Title = a.Title()
for _, outcome := range hostAnalyzer.Outcomes {
if outcome.Fail != nil {
if outcome.Fail.When == "" {
result.IsFail = true
result.Message = outcome.Fail.Message
result.URI = outcome.Fail.URI
return &result, nil
}
isMatch, err := compareHostFilesystemPerformanceConditionalToActual(outcome.Fail.When, fsPerf)
if err != nil {
return nil, errors.Wrapf(err, "failed to compare %q", outcome.Fail.When)
}
if isMatch {
result.IsFail = true
result.Message = outcome.Fail.Message
result.URI = outcome.Fail.URI
return &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 &result, nil
}
isMatch, err := compareHostFilesystemPerformanceConditionalToActual(outcome.Warn.When, fsPerf)
if err != nil {
return nil, errors.Wrapf(err, "failed to compare %q", outcome.Warn.When)
}
if isMatch {
result.IsWarn = true
result.Message = outcome.Warn.Message
result.URI = outcome.Warn.URI
return &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 &result, nil
}
isMatch, err := compareHostFilesystemPerformanceConditionalToActual(outcome.Pass.When, fsPerf)
if err != nil {
return nil, errors.Wrapf(err, "failed to compare %q", outcome.Pass.When)
}
if isMatch {
result.IsPass = true
result.Message = outcome.Pass.Message
result.URI = outcome.Pass.URI
return &result, nil
}
}
}
return &result, nil
}
func compareHostFilesystemPerformanceConditionalToActual(conditional string, fsPerf collect.FSPerfResults) (res bool, err error) {
parts := strings.Split(conditional, " ")
if len(parts) != 3 {
return false, fmt.Errorf("conditional must have exactly 3 parts, got %d", len(parts))
}
keyword := strings.ToLower(parts[0])
comparator := parts[1]
if keyword == "iops" {
desiredInt, err := strconv.ParseInt(parts[2], 10, 64)
if err != nil {
return false, errors.Wrapf(err, "failed to parse desired iops %q as integer", parts[2])
}
return doCompareHostFilesystemIOPS(comparator, fsPerf.IOPS, int(desiredInt))
}
desiredDuration, err := time.ParseDuration(parts[2])
if err != nil {
return false, errors.Wrapf(err, "failed to parse duration %q", parts[2])
}
switch keyword {
case "min":
return doCompareHostFilesystemPerformance(comparator, fsPerf.Min, desiredDuration)
case "max":
return doCompareHostFilesystemPerformance(comparator, fsPerf.Max, desiredDuration)
case "average":
return doCompareHostFilesystemPerformance(comparator, fsPerf.Average, desiredDuration)
case "p1":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P1, desiredDuration)
case "p5":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P5, desiredDuration)
case "p10":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P10, desiredDuration)
case "p20":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P20, desiredDuration)
case "p30":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P30, desiredDuration)
case "p40":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P40, desiredDuration)
case "p50":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P50, desiredDuration)
case "p60":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P60, desiredDuration)
case "p70":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P70, desiredDuration)
case "p80":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P80, desiredDuration)
case "p90":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P90, desiredDuration)
case "p95":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P95, desiredDuration)
case "p99":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P99, desiredDuration)
case "p995":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P995, desiredDuration)
case "p999":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P999, desiredDuration)
case "p9995":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P9995, desiredDuration)
case "p9999":
return doCompareHostFilesystemPerformance(comparator, fsPerf.P9999, desiredDuration)
}
return false, fmt.Errorf("Unknown filesystem performance keyword %q", keyword)
}
func doCompareHostFilesystemPerformance(operator string, actual time.Duration, desired time.Duration) (bool, error) {
switch operator {
case "<":
return actual < desired, nil
case "<=":
return actual <= desired, nil
case ">":
return actual > desired, nil
case ">=":
return actual >= desired, nil
case "=", "==", "===":
return actual == desired, nil
}
return false, fmt.Errorf("Unknown filesystem performance operator %q", operator)
}
func doCompareHostFilesystemIOPS(operator string, actual int, desired int) (bool, error) {
switch operator {
case "<":
return actual < desired, nil
case "<=":
return actual <= desired, nil
case ">":
return actual > desired, nil
case ">=":
return actual >= desired, nil
case "=", "==", "===":
return actual == desired, nil
}
return false, fmt.Errorf("Unknown filesystem IOPS operator %q", operator)
}