mirror of
https://github.com/grafana/grafana.git
synced 2025-01-18 20:43:26 -06:00
2b8909a9c6
* sqlstore: move GetDashboardUIDById to dashboard service * winding change through the rest of the codebase
208 lines
7.3 KiB
Go
208 lines
7.3 KiB
Go
package alerting
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/grafana/grafana/pkg/services/validations"
|
|
"github.com/grafana/grafana/pkg/tsdb/legacydata"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type conditionStub struct {
|
|
firing bool
|
|
operator string
|
|
matches []*EvalMatch
|
|
noData bool
|
|
}
|
|
|
|
func (c *conditionStub) Eval(context *EvalContext, reqHandler legacydata.RequestHandler) (*ConditionResult, error) {
|
|
return &ConditionResult{Firing: c.firing, EvalMatches: c.matches, Operator: c.operator, NoDataFound: c.noData}, nil
|
|
}
|
|
|
|
func TestAlertingEvaluationHandler(t *testing.T) {
|
|
handler := NewEvalHandler(nil)
|
|
|
|
t.Run("Show return triggered with single passing condition", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{&conditionStub{
|
|
firing: true,
|
|
}},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, true, context.Firing)
|
|
require.Equal(t, "true = true", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return triggered with single passing condition2", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{&conditionStub{firing: true, operator: "and"}},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, true, context.Firing)
|
|
require.Equal(t, "true = true", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return false with not passing asdf", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and", matches: []*EvalMatch{{}, {}}},
|
|
&conditionStub{firing: false, operator: "and"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.Equal(t, "[true AND false] = false", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return true if any of the condition is passing with OR operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, true, context.Firing)
|
|
require.Equal(t, "[true OR false] = true", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return false if any of the condition is failing with AND operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "and"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.Equal(t, "[true AND false] = false", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return true if one condition is failing with nested OR operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, true, context.Firing)
|
|
require.Equal(t, "[[true AND true] OR false] = true", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return false if one condition is passing with nested OR operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "and"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.Equal(t, "[[true AND false] OR false] = false", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return false if a condition is failing with nested AND operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "and"},
|
|
&conditionStub{firing: true, operator: "and"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.Equal(t, "[[true AND false] AND true] = false", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Show return true if a condition is passing with nested OR operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: true, operator: "and"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
&conditionStub{firing: true, operator: "or"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, true, context.Firing)
|
|
require.Equal(t, "[[true OR false] OR true] = true", context.ConditionEvals)
|
|
})
|
|
|
|
t.Run("Should return false if no condition is firing using OR operator", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{firing: false, operator: "or"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
&conditionStub{firing: false, operator: "or"},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.Equal(t, "[[false OR false] OR false] = false", context.ConditionEvals)
|
|
})
|
|
|
|
// FIXME: What should the actual test case name be here?
|
|
t.Run("Should not return NoDataFound if all conditions have data and using OR", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{operator: "or", noData: false},
|
|
&conditionStub{operator: "or", noData: false},
|
|
&conditionStub{operator: "or", noData: false},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.False(t, context.NoDataFound)
|
|
})
|
|
|
|
t.Run("Should return NoDataFound if one condition has no data", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{operator: "and", noData: true},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.Equal(t, false, context.Firing)
|
|
require.True(t, context.NoDataFound)
|
|
})
|
|
|
|
t.Run("Should return no data if at least one condition has no data and using AND", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{operator: "and", noData: true},
|
|
&conditionStub{operator: "and", noData: false},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.True(t, context.NoDataFound)
|
|
})
|
|
|
|
t.Run("Should return no data if at least one condition has no data and using OR", func(t *testing.T) {
|
|
context := NewEvalContext(context.Background(), &Rule{
|
|
Conditions: []Condition{
|
|
&conditionStub{operator: "or", noData: true},
|
|
&conditionStub{operator: "or", noData: false},
|
|
},
|
|
}, &validations.OSSPluginRequestValidator{}, nil, nil)
|
|
|
|
handler.Eval(context)
|
|
require.True(t, context.NoDataFound)
|
|
})
|
|
}
|