query.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package conditions
  2. import (
  3. "fmt"
  4. "github.com/grafana/grafana/pkg/bus"
  5. "github.com/grafana/grafana/pkg/components/simplejson"
  6. m "github.com/grafana/grafana/pkg/models"
  7. "github.com/grafana/grafana/pkg/services/alerting"
  8. "github.com/grafana/grafana/pkg/tsdb"
  9. )
  10. func init() {
  11. alerting.RegisterCondition("query", func(model *simplejson.Json, index int) (alerting.Condition, error) {
  12. return NewQueryCondition(model, index)
  13. })
  14. }
  15. type QueryCondition struct {
  16. Index int
  17. Query AlertQuery
  18. Reducer QueryReducer
  19. Evaluator AlertEvaluator
  20. HandleRequest tsdb.HandleRequestFunc
  21. }
  22. type AlertQuery struct {
  23. Model *simplejson.Json
  24. DatasourceId int64
  25. From string
  26. To string
  27. }
  28. func (c *QueryCondition) Eval(context *alerting.EvalContext) {
  29. seriesList, err := c.executeQuery(context)
  30. if err != nil {
  31. context.Error = err
  32. return
  33. }
  34. emptySerieCount := 0
  35. for _, series := range seriesList {
  36. reducedValue := c.Reducer.Reduce(series)
  37. evalMatch := c.Evaluator.Eval(reducedValue)
  38. if reducedValue == nil {
  39. emptySerieCount++
  40. continue
  41. }
  42. if context.IsTestRun {
  43. context.Logs = append(context.Logs, &alerting.ResultLogEntry{
  44. Message: fmt.Sprintf("Condition[%d]: Eval: %v, Metric: %s, Value: %1.3f", c.Index, evalMatch, series.Name, *reducedValue),
  45. })
  46. }
  47. if evalMatch {
  48. context.EvalMatches = append(context.EvalMatches, &alerting.EvalMatch{
  49. Metric: series.Name,
  50. Value: *reducedValue,
  51. })
  52. }
  53. }
  54. context.NoDataFound = emptySerieCount == len(seriesList)
  55. context.Firing = len(context.EvalMatches) > 0
  56. }
  57. func (c *QueryCondition) executeQuery(context *alerting.EvalContext) (tsdb.TimeSeriesSlice, error) {
  58. getDsInfo := &m.GetDataSourceByIdQuery{
  59. Id: c.Query.DatasourceId,
  60. OrgId: context.Rule.OrgId,
  61. }
  62. if err := bus.Dispatch(getDsInfo); err != nil {
  63. return nil, fmt.Errorf("Could not find datasource")
  64. }
  65. req := c.getRequestForAlertRule(getDsInfo.Result)
  66. result := make(tsdb.TimeSeriesSlice, 0)
  67. resp, err := c.HandleRequest(req)
  68. if err != nil {
  69. return nil, fmt.Errorf("tsdb.HandleRequest() error %v", err)
  70. }
  71. for _, v := range resp.Results {
  72. if v.Error != nil {
  73. return nil, fmt.Errorf("tsdb.HandleRequest() response error %v", v)
  74. }
  75. result = append(result, v.Series...)
  76. if context.IsTestRun {
  77. context.Logs = append(context.Logs, &alerting.ResultLogEntry{
  78. Message: fmt.Sprintf("Condition[%d]: Query Result", c.Index),
  79. Data: v.Series,
  80. })
  81. }
  82. }
  83. return result, nil
  84. }
  85. func (c *QueryCondition) getRequestForAlertRule(datasource *m.DataSource) *tsdb.Request {
  86. req := &tsdb.Request{
  87. TimeRange: tsdb.TimeRange{
  88. From: c.Query.From,
  89. To: c.Query.To,
  90. },
  91. Queries: []*tsdb.Query{
  92. {
  93. RefId: "A",
  94. Query: c.Query.Model.Get("target").MustString(),
  95. DataSource: &tsdb.DataSourceInfo{
  96. Id: datasource.Id,
  97. Name: datasource.Name,
  98. PluginId: datasource.Type,
  99. Url: datasource.Url,
  100. User: datasource.User,
  101. Password: datasource.Password,
  102. Database: datasource.Database,
  103. BasicAuth: datasource.BasicAuth,
  104. BasicAuthUser: datasource.BasicAuthUser,
  105. BasicAuthPassword: datasource.BasicAuthPassword,
  106. },
  107. },
  108. },
  109. }
  110. return req
  111. }
  112. func NewQueryCondition(model *simplejson.Json, index int) (*QueryCondition, error) {
  113. condition := QueryCondition{}
  114. condition.Index = index
  115. condition.HandleRequest = tsdb.HandleRequest
  116. queryJson := model.Get("query")
  117. condition.Query.Model = queryJson.Get("model")
  118. condition.Query.From = queryJson.Get("params").MustArray()[1].(string)
  119. condition.Query.To = queryJson.Get("params").MustArray()[2].(string)
  120. condition.Query.DatasourceId = queryJson.Get("datasourceId").MustInt64()
  121. reducerJson := model.Get("reducer")
  122. condition.Reducer = NewSimpleReducer(reducerJson.Get("type").MustString())
  123. evaluatorJson := model.Get("evaluator")
  124. evaluator, err := NewAlertEvaluator(evaluatorJson)
  125. if err != nil {
  126. return nil, err
  127. }
  128. condition.Evaluator = evaluator
  129. return &condition, nil
  130. }