Alerting: annotation on state change (#36535)

* WIP

* Add annotation on alert state change

* move annotation creation to manager

* praise the linter!

* add debug msg when creating annotation
This commit is contained in:
David Parrott 2021-07-13 09:50:10 -07:00 committed by GitHub
parent db5597ab9a
commit 19f18bcecc
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 55 additions and 2 deletions

View File

@ -1,8 +1,14 @@
package state package state
import ( import (
"fmt"
"strconv"
"time" "time"
"github.com/grafana/grafana/pkg/models"
"github.com/grafana/grafana/pkg/services/annotations"
"github.com/grafana/grafana/pkg/services/sqlstore"
"github.com/grafana/grafana/pkg/infra/log" "github.com/grafana/grafana/pkg/infra/log"
"github.com/grafana/grafana/pkg/services/ngalert/eval" "github.com/grafana/grafana/pkg/services/ngalert/eval"
@ -151,11 +157,12 @@ func (st *Manager) setNextState(alertRule *ngModels.AlertRule, result eval.Resul
EvaluationString: result.EvaluationString, EvaluationString: result.EvaluationString,
}) })
currentState.TrimResults(alertRule) currentState.TrimResults(alertRule)
oldState := currentState.State
st.log.Debug("setting alert state", "uid", alertRule.UID) st.log.Debug("setting alert state", "uid", alertRule.UID)
switch result.State { switch result.State {
case eval.Normal: case eval.Normal:
currentState.resultNormal(result) currentState.resultNormal(alertRule, result)
case eval.Alerting: case eval.Alerting:
currentState.resultAlerting(alertRule, result) currentState.resultAlerting(alertRule, result)
case eval.Error: case eval.Error:
@ -166,6 +173,9 @@ func (st *Manager) setNextState(alertRule *ngModels.AlertRule, result eval.Resul
} }
st.set(currentState) st.set(currentState)
if oldState != currentState.State {
go st.createAlertAnnotation(currentState.State, alertRule, result)
}
return currentState return currentState
} }
@ -211,3 +221,46 @@ func translateInstanceState(state ngModels.InstanceStateType) eval.State {
return eval.Error return eval.Error
} }
} }
func (st *Manager) createAlertAnnotation(new eval.State, alertRule *ngModels.AlertRule, result eval.Result) {
st.log.Debug("alert state changed creating annotation", "alertRuleUID", alertRule.UID, "newState", new.String())
dashUid, ok := alertRule.Annotations["__dashboardUid__"]
if !ok {
return
}
panelUid := alertRule.Annotations["__panelId__"]
panelId, err := strconv.ParseInt(panelUid, 10, 64)
if err != nil {
st.log.Error("error parsing panelUID for alert annotation", "panelUID", panelUid, "alertRuleUID", alertRule.UID, "error", err.Error())
return
}
query := &models.GetDashboardQuery{
Uid: dashUid,
OrgId: alertRule.OrgID,
}
err = sqlstore.GetDashboard(query)
if err != nil {
st.log.Error("error getting dashboard for alert annotation", "dashboardUID", dashUid, "alertRuleUID", alertRule.UID, "error", err.Error())
return
}
annotationText := fmt.Sprintf("%s %s", result.Instance.String(), new.String())
item := &annotations.Item{
OrgId: alertRule.OrgID,
DashboardId: query.Result.Id,
PanelId: panelId,
Text: annotationText,
Epoch: result.EvaluatedAt.UnixNano() / int64(time.Millisecond),
}
annotationRepo := annotations.GetRepository()
if err = annotationRepo.Save(item); err != nil {
st.log.Error("error saving alert annotation", "alertRuleUID", alertRule.UID, "error", err.Error())
return
}
}

View File

@ -30,7 +30,7 @@ type Evaluation struct {
EvaluationString string EvaluationString string
} }
func (a *State) resultNormal(result eval.Result) { func (a *State) resultNormal(alertRule *ngModels.AlertRule, result eval.Result) {
if a.State != eval.Normal { if a.State != eval.Normal {
a.EndsAt = result.EvaluatedAt a.EndsAt = result.EvaluatedAt
a.StartsAt = result.EvaluatedAt a.StartsAt = result.EvaluatedAt