mirror of
https://github.com/grafana/grafana.git
synced 2024-12-02 05:29:42 -06:00
d354f3a8af
* fix(alerting): fixed evaluation for no_value condition, fixes #7244 * feat(alerting): moving null library into grafana, fixing handling on no value / no series
136 lines
3.2 KiB
Go
136 lines
3.2 KiB
Go
package conditions
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/grafana/grafana/pkg/components/null"
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
|
"github.com/grafana/grafana/pkg/services/alerting"
|
|
)
|
|
|
|
var (
|
|
defaultTypes []string = []string{"gt", "lt"}
|
|
rangedTypes []string = []string{"within_range", "outside_range"}
|
|
)
|
|
|
|
type AlertEvaluator interface {
|
|
Eval(reducedValue null.Float) bool
|
|
}
|
|
|
|
type NoValueEvaluator struct{}
|
|
|
|
func (e *NoValueEvaluator) Eval(reducedValue null.Float) bool {
|
|
return reducedValue.Valid == false
|
|
}
|
|
|
|
type ThresholdEvaluator struct {
|
|
Type string
|
|
Threshold float64
|
|
}
|
|
|
|
func newThresholdEvaluator(typ string, model *simplejson.Json) (*ThresholdEvaluator, error) {
|
|
params := model.Get("params").MustArray()
|
|
if len(params) == 0 {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator missing threshold parameter"}
|
|
}
|
|
|
|
firstParam, ok := params[0].(json.Number)
|
|
if !ok {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator has invalid parameter"}
|
|
}
|
|
|
|
defaultEval := &ThresholdEvaluator{Type: typ}
|
|
defaultEval.Threshold, _ = firstParam.Float64()
|
|
return defaultEval, nil
|
|
}
|
|
|
|
func (e *ThresholdEvaluator) Eval(reducedValue null.Float) bool {
|
|
if reducedValue.Valid == false {
|
|
return false
|
|
}
|
|
|
|
switch e.Type {
|
|
case "gt":
|
|
return reducedValue.Float64 > e.Threshold
|
|
case "lt":
|
|
return reducedValue.Float64 < e.Threshold
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
type RangedEvaluator struct {
|
|
Type string
|
|
Lower float64
|
|
Upper float64
|
|
}
|
|
|
|
func newRangedEvaluator(typ string, model *simplejson.Json) (*RangedEvaluator, error) {
|
|
params := model.Get("params").MustArray()
|
|
if len(params) == 0 {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator missing threshold parameter"}
|
|
}
|
|
|
|
firstParam, ok := params[0].(json.Number)
|
|
if !ok {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator has invalid parameter"}
|
|
}
|
|
|
|
secondParam, ok := params[1].(json.Number)
|
|
if !ok {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator has invalid second parameter"}
|
|
}
|
|
|
|
rangedEval := &RangedEvaluator{Type: typ}
|
|
rangedEval.Lower, _ = firstParam.Float64()
|
|
rangedEval.Upper, _ = secondParam.Float64()
|
|
return rangedEval, nil
|
|
}
|
|
|
|
func (e *RangedEvaluator) Eval(reducedValue null.Float) bool {
|
|
if reducedValue.Valid == false {
|
|
return false
|
|
}
|
|
|
|
floatValue := reducedValue.Float64
|
|
|
|
switch e.Type {
|
|
case "within_range":
|
|
return (e.Lower < floatValue && e.Upper > floatValue) || (e.Upper < floatValue && e.Lower > floatValue)
|
|
case "outside_range":
|
|
return (e.Upper < floatValue && e.Lower < floatValue) || (e.Upper > floatValue && e.Lower > floatValue)
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func NewAlertEvaluator(model *simplejson.Json) (AlertEvaluator, error) {
|
|
typ := model.Get("type").MustString()
|
|
if typ == "" {
|
|
return nil, alerting.ValidationError{Reason: "Evaluator missing type property"}
|
|
}
|
|
|
|
if inSlice(typ, defaultTypes) {
|
|
return newThresholdEvaluator(typ, model)
|
|
}
|
|
|
|
if inSlice(typ, rangedTypes) {
|
|
return newRangedEvaluator(typ, model)
|
|
}
|
|
|
|
if typ == "no_value" {
|
|
return &NoValueEvaluator{}, nil
|
|
}
|
|
|
|
return nil, alerting.ValidationError{Reason: "Evaluator invalid evaluator type: " + typ}
|
|
}
|
|
|
|
func inSlice(a string, list []string) bool {
|
|
for _, b := range list {
|
|
if b == a {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|