alerting.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. package api
  2. import (
  3. "fmt"
  4. "strconv"
  5. "github.com/grafana/grafana/pkg/api/dtos"
  6. "github.com/grafana/grafana/pkg/bus"
  7. m "github.com/grafana/grafana/pkg/models"
  8. "github.com/grafana/grafana/pkg/services/alerting"
  9. "github.com/grafana/grafana/pkg/services/guardian"
  10. "github.com/grafana/grafana/pkg/services/search"
  11. )
  12. func ValidateOrgAlert(c *m.ReqContext) {
  13. id := c.ParamsInt64(":alertId")
  14. query := m.GetAlertByIdQuery{Id: id}
  15. if err := bus.Dispatch(&query); err != nil {
  16. c.JsonApiErr(404, "Alert not found", nil)
  17. return
  18. }
  19. if c.OrgId != query.Result.OrgId {
  20. c.JsonApiErr(403, "You are not allowed to edit/view alert", nil)
  21. return
  22. }
  23. }
  24. func GetAlertStatesForDashboard(c *m.ReqContext) Response {
  25. dashboardID := c.QueryInt64("dashboardId")
  26. if dashboardID == 0 {
  27. return Error(400, "Missing query parameter dashboardId", nil)
  28. }
  29. query := m.GetAlertStatesForDashboardQuery{
  30. OrgId: c.OrgId,
  31. DashboardId: c.QueryInt64("dashboardId"),
  32. }
  33. if err := bus.Dispatch(&query); err != nil {
  34. return Error(500, "Failed to fetch alert states", err)
  35. }
  36. return JSON(200, query.Result)
  37. }
  38. // GET /api/alerts
  39. func GetAlerts(c *m.ReqContext) Response {
  40. dashboardQuery := c.Query("dashboardQuery")
  41. dashboardTags := c.QueryStrings("dashboardTag")
  42. stringDashboardIDs := c.QueryStrings("dashboardId")
  43. stringFolderIDs := c.QueryStrings("folderId")
  44. dashboardIDs := make([]int64, 0)
  45. for _, id := range stringDashboardIDs {
  46. dashboardID, err := strconv.ParseInt(id, 10, 64)
  47. if err == nil {
  48. dashboardIDs = append(dashboardIDs, dashboardID)
  49. }
  50. }
  51. if dashboardQuery != "" || len(dashboardTags) > 0 || len(stringFolderIDs) > 0 {
  52. folderIDs := make([]int64, 0)
  53. for _, id := range stringFolderIDs {
  54. folderID, err := strconv.ParseInt(id, 10, 64)
  55. if err == nil {
  56. folderIDs = append(folderIDs, folderID)
  57. }
  58. }
  59. searchQuery := search.Query{
  60. Title: dashboardQuery,
  61. Tags: dashboardTags,
  62. SignedInUser: c.SignedInUser,
  63. Limit: 1000,
  64. OrgId: c.OrgId,
  65. DashboardIds: dashboardIDs,
  66. Type: string(search.DashHitDB),
  67. FolderIds: folderIDs,
  68. Permission: m.PERMISSION_VIEW,
  69. }
  70. err := bus.Dispatch(&searchQuery)
  71. if err != nil {
  72. return Error(500, "List alerts failed", err)
  73. }
  74. for _, d := range searchQuery.Result {
  75. if d.Type == search.DashHitDB && d.Id > 0 {
  76. dashboardIDs = append(dashboardIDs, d.Id)
  77. }
  78. }
  79. // if we didn't find any dashboards, return empty result
  80. if len(dashboardIDs) == 0 {
  81. return JSON(200, []*m.AlertListItemDTO{})
  82. }
  83. }
  84. query := m.GetAlertsQuery{
  85. OrgId: c.OrgId,
  86. DashboardIDs: dashboardIDs,
  87. PanelId: c.QueryInt64("panelId"),
  88. Limit: c.QueryInt64("limit"),
  89. User: c.SignedInUser,
  90. Query: c.Query("query"),
  91. }
  92. states := c.QueryStrings("state")
  93. if len(states) > 0 {
  94. query.State = states
  95. }
  96. if err := bus.Dispatch(&query); err != nil {
  97. return Error(500, "List alerts failed", err)
  98. }
  99. for _, alert := range query.Result {
  100. alert.Url = m.GetDashboardUrl(alert.DashboardUid, alert.DashboardSlug)
  101. }
  102. return JSON(200, query.Result)
  103. }
  104. // POST /api/alerts/test
  105. func AlertTest(c *m.ReqContext, dto dtos.AlertTestCommand) Response {
  106. if _, idErr := dto.Dashboard.Get("id").Int64(); idErr != nil {
  107. return Error(400, "The dashboard needs to be saved at least once before you can test an alert rule", nil)
  108. }
  109. backendCmd := alerting.AlertTestCommand{
  110. OrgId: c.OrgId,
  111. Dashboard: dto.Dashboard,
  112. PanelId: dto.PanelId,
  113. }
  114. if err := bus.Dispatch(&backendCmd); err != nil {
  115. if validationErr, ok := err.(alerting.ValidationError); ok {
  116. return Error(422, validationErr.Error(), nil)
  117. }
  118. return Error(500, "Failed to test rule", err)
  119. }
  120. res := backendCmd.Result
  121. dtoRes := &dtos.AlertTestResult{
  122. Firing: res.Firing,
  123. ConditionEvals: res.ConditionEvals,
  124. State: res.Rule.State,
  125. }
  126. if res.Error != nil {
  127. dtoRes.Error = res.Error.Error()
  128. }
  129. for _, log := range res.Logs {
  130. dtoRes.Logs = append(dtoRes.Logs, &dtos.AlertTestResultLog{Message: log.Message, Data: log.Data})
  131. }
  132. for _, match := range res.EvalMatches {
  133. dtoRes.EvalMatches = append(dtoRes.EvalMatches, &dtos.EvalMatch{Metric: match.Metric, Value: match.Value})
  134. }
  135. dtoRes.TimeMs = fmt.Sprintf("%1.3fms", res.GetDurationMs())
  136. return JSON(200, dtoRes)
  137. }
  138. // GET /api/alerts/:id
  139. func GetAlert(c *m.ReqContext) Response {
  140. id := c.ParamsInt64(":alertId")
  141. query := m.GetAlertByIdQuery{Id: id}
  142. if err := bus.Dispatch(&query); err != nil {
  143. return Error(500, "List alerts failed", err)
  144. }
  145. return JSON(200, &query.Result)
  146. }
  147. func GetAlertNotifiers(c *m.ReqContext) Response {
  148. return JSON(200, alerting.GetNotifiers())
  149. }
  150. func GetAlertNotifications(c *m.ReqContext) Response {
  151. query := &m.GetAllAlertNotificationsQuery{OrgId: c.OrgId}
  152. if err := bus.Dispatch(query); err != nil {
  153. return Error(500, "Failed to get alert notifications", err)
  154. }
  155. result := make([]*dtos.AlertNotification, 0)
  156. for _, notification := range query.Result {
  157. result = append(result, dtos.NewAlertNotification(notification))
  158. }
  159. return JSON(200, result)
  160. }
  161. func GetAlertNotificationByID(c *m.ReqContext) Response {
  162. query := &m.GetAlertNotificationsQuery{
  163. OrgId: c.OrgId,
  164. Id: c.ParamsInt64("notificationId"),
  165. }
  166. if err := bus.Dispatch(query); err != nil {
  167. return Error(500, "Failed to get alert notifications", err)
  168. }
  169. return JSON(200, dtos.NewAlertNotification(query.Result))
  170. }
  171. func CreateAlertNotification(c *m.ReqContext, cmd m.CreateAlertNotificationCommand) Response {
  172. cmd.OrgId = c.OrgId
  173. if err := bus.Dispatch(&cmd); err != nil {
  174. return Error(500, "Failed to create alert notification", err)
  175. }
  176. return JSON(200, dtos.NewAlertNotification(cmd.Result))
  177. }
  178. func UpdateAlertNotification(c *m.ReqContext, cmd m.UpdateAlertNotificationCommand) Response {
  179. cmd.OrgId = c.OrgId
  180. if err := bus.Dispatch(&cmd); err != nil {
  181. return Error(500, "Failed to update alert notification", err)
  182. }
  183. return JSON(200, dtos.NewAlertNotification(cmd.Result))
  184. }
  185. func DeleteAlertNotification(c *m.ReqContext) Response {
  186. cmd := m.DeleteAlertNotificationCommand{
  187. OrgId: c.OrgId,
  188. Id: c.ParamsInt64("notificationId"),
  189. }
  190. if err := bus.Dispatch(&cmd); err != nil {
  191. return Error(500, "Failed to delete alert notification", err)
  192. }
  193. return Success("Notification deleted")
  194. }
  195. //POST /api/alert-notifications/test
  196. func NotificationTest(c *m.ReqContext, dto dtos.NotificationTestCommand) Response {
  197. cmd := &alerting.NotificationTestCommand{
  198. Name: dto.Name,
  199. Type: dto.Type,
  200. Settings: dto.Settings,
  201. }
  202. if err := bus.Dispatch(cmd); err != nil {
  203. if err == m.ErrSmtpNotEnabled {
  204. return Error(412, err.Error(), err)
  205. }
  206. return Error(500, "Failed to send alert notifications", err)
  207. }
  208. return Success("Test notification sent")
  209. }
  210. //POST /api/alerts/:alertId/pause
  211. func PauseAlert(c *m.ReqContext, dto dtos.PauseAlertCommand) Response {
  212. alertID := c.ParamsInt64("alertId")
  213. query := m.GetAlertByIdQuery{Id: alertID}
  214. if err := bus.Dispatch(&query); err != nil {
  215. return Error(500, "Get Alert failed", err)
  216. }
  217. guardian := guardian.New(query.Result.DashboardId, c.OrgId, c.SignedInUser)
  218. if canEdit, err := guardian.CanEdit(); err != nil || !canEdit {
  219. if err != nil {
  220. return Error(500, "Error while checking permissions for Alert", err)
  221. }
  222. return Error(403, "Access denied to this dashboard and alert", nil)
  223. }
  224. cmd := m.PauseAlertCommand{
  225. OrgId: c.OrgId,
  226. AlertIds: []int64{alertID},
  227. Paused: dto.Paused,
  228. }
  229. if err := bus.Dispatch(&cmd); err != nil {
  230. return Error(500, "", err)
  231. }
  232. var response m.AlertStateType = m.AlertStatePending
  233. pausedState := "un-paused"
  234. if cmd.Paused {
  235. response = m.AlertStatePaused
  236. pausedState = "paused"
  237. }
  238. result := map[string]interface{}{
  239. "alertId": alertID,
  240. "state": response,
  241. "message": "Alert " + pausedState,
  242. }
  243. return JSON(200, result)
  244. }
  245. //POST /api/admin/pause-all-alerts
  246. func PauseAllAlerts(c *m.ReqContext, dto dtos.PauseAllAlertsCommand) Response {
  247. updateCmd := m.PauseAllAlertCommand{
  248. Paused: dto.Paused,
  249. }
  250. if err := bus.Dispatch(&updateCmd); err != nil {
  251. return Error(500, "Failed to pause alerts", err)
  252. }
  253. var response m.AlertStateType = m.AlertStatePending
  254. pausedState := "un paused"
  255. if updateCmd.Paused {
  256. response = m.AlertStatePaused
  257. pausedState = "paused"
  258. }
  259. result := map[string]interface{}{
  260. "state": response,
  261. "message": "alerts " + pausedState,
  262. "alertsAffected": updateCmd.ResultCount,
  263. }
  264. return JSON(200, result)
  265. }