alerting.go 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  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. User: c.SignedInUser,
  114. }
  115. if err := bus.Dispatch(&backendCmd); err != nil {
  116. if validationErr, ok := err.(alerting.ValidationError); ok {
  117. return Error(422, validationErr.Error(), nil)
  118. }
  119. if err == m.ErrDataSourceAccessDenied {
  120. return Error(403, "Access denied to datasource", err)
  121. }
  122. return Error(500, "Failed to test rule", err)
  123. }
  124. res := backendCmd.Result
  125. dtoRes := &dtos.AlertTestResult{
  126. Firing: res.Firing,
  127. ConditionEvals: res.ConditionEvals,
  128. State: res.Rule.State,
  129. }
  130. if res.Error != nil {
  131. dtoRes.Error = res.Error.Error()
  132. }
  133. for _, log := range res.Logs {
  134. dtoRes.Logs = append(dtoRes.Logs, &dtos.AlertTestResultLog{Message: log.Message, Data: log.Data})
  135. }
  136. for _, match := range res.EvalMatches {
  137. dtoRes.EvalMatches = append(dtoRes.EvalMatches, &dtos.EvalMatch{Metric: match.Metric, Value: match.Value})
  138. }
  139. dtoRes.TimeMs = fmt.Sprintf("%1.3fms", res.GetDurationMs())
  140. return JSON(200, dtoRes)
  141. }
  142. // GET /api/alerts/:id
  143. func GetAlert(c *m.ReqContext) Response {
  144. id := c.ParamsInt64(":alertId")
  145. query := m.GetAlertByIdQuery{Id: id}
  146. if err := bus.Dispatch(&query); err != nil {
  147. return Error(500, "List alerts failed", err)
  148. }
  149. return JSON(200, &query.Result)
  150. }
  151. func GetAlertNotifiers(c *m.ReqContext) Response {
  152. return JSON(200, alerting.GetNotifiers())
  153. }
  154. func GetAlertNotifications(c *m.ReqContext) Response {
  155. query := &m.GetAllAlertNotificationsQuery{OrgId: c.OrgId}
  156. if err := bus.Dispatch(query); err != nil {
  157. return Error(500, "Failed to get alert notifications", err)
  158. }
  159. result := make([]*dtos.AlertNotification, 0)
  160. for _, notification := range query.Result {
  161. result = append(result, dtos.NewAlertNotification(notification))
  162. }
  163. return JSON(200, result)
  164. }
  165. func GetAlertNotificationByID(c *m.ReqContext) Response {
  166. query := &m.GetAlertNotificationsQuery{
  167. OrgId: c.OrgId,
  168. Id: c.ParamsInt64("notificationId"),
  169. }
  170. if err := bus.Dispatch(query); err != nil {
  171. return Error(500, "Failed to get alert notifications", err)
  172. }
  173. if query.Result == nil {
  174. return Error(404, "Alert notification not found", nil)
  175. }
  176. return JSON(200, dtos.NewAlertNotification(query.Result))
  177. }
  178. func CreateAlertNotification(c *m.ReqContext, cmd m.CreateAlertNotificationCommand) Response {
  179. cmd.OrgId = c.OrgId
  180. if err := bus.Dispatch(&cmd); err != nil {
  181. return Error(500, "Failed to create alert notification", err)
  182. }
  183. return JSON(200, dtos.NewAlertNotification(cmd.Result))
  184. }
  185. func UpdateAlertNotification(c *m.ReqContext, cmd m.UpdateAlertNotificationCommand) Response {
  186. cmd.OrgId = c.OrgId
  187. if err := bus.Dispatch(&cmd); err != nil {
  188. return Error(500, "Failed to update alert notification", err)
  189. }
  190. return JSON(200, dtos.NewAlertNotification(cmd.Result))
  191. }
  192. func DeleteAlertNotification(c *m.ReqContext) Response {
  193. cmd := m.DeleteAlertNotificationCommand{
  194. OrgId: c.OrgId,
  195. Id: c.ParamsInt64("notificationId"),
  196. }
  197. if err := bus.Dispatch(&cmd); err != nil {
  198. return Error(500, "Failed to delete alert notification", err)
  199. }
  200. return Success("Notification deleted")
  201. }
  202. //POST /api/alert-notifications/test
  203. func NotificationTest(c *m.ReqContext, dto dtos.NotificationTestCommand) Response {
  204. cmd := &alerting.NotificationTestCommand{
  205. Name: dto.Name,
  206. Type: dto.Type,
  207. Settings: dto.Settings,
  208. }
  209. if err := bus.Dispatch(cmd); err != nil {
  210. if err == m.ErrSmtpNotEnabled {
  211. return Error(412, err.Error(), err)
  212. }
  213. return Error(500, "Failed to send alert notifications", err)
  214. }
  215. return Success("Test notification sent")
  216. }
  217. //POST /api/alerts/:alertId/pause
  218. func PauseAlert(c *m.ReqContext, dto dtos.PauseAlertCommand) Response {
  219. alertID := c.ParamsInt64("alertId")
  220. query := m.GetAlertByIdQuery{Id: alertID}
  221. if err := bus.Dispatch(&query); err != nil {
  222. return Error(500, "Get Alert failed", err)
  223. }
  224. guardian := guardian.New(query.Result.DashboardId, c.OrgId, c.SignedInUser)
  225. if canEdit, err := guardian.CanEdit(); err != nil || !canEdit {
  226. if err != nil {
  227. return Error(500, "Error while checking permissions for Alert", err)
  228. }
  229. return Error(403, "Access denied to this dashboard and alert", nil)
  230. }
  231. cmd := m.PauseAlertCommand{
  232. OrgId: c.OrgId,
  233. AlertIds: []int64{alertID},
  234. Paused: dto.Paused,
  235. }
  236. if err := bus.Dispatch(&cmd); err != nil {
  237. return Error(500, "", err)
  238. }
  239. var response m.AlertStateType = m.AlertStateUnknown
  240. pausedState := "un-paused"
  241. if cmd.Paused {
  242. response = m.AlertStatePaused
  243. pausedState = "paused"
  244. }
  245. result := map[string]interface{}{
  246. "alertId": alertID,
  247. "state": response,
  248. "message": "Alert " + pausedState,
  249. }
  250. return JSON(200, result)
  251. }
  252. //POST /api/admin/pause-all-alerts
  253. func PauseAllAlerts(c *m.ReqContext, dto dtos.PauseAllAlertsCommand) Response {
  254. updateCmd := m.PauseAllAlertCommand{
  255. Paused: dto.Paused,
  256. }
  257. if err := bus.Dispatch(&updateCmd); err != nil {
  258. return Error(500, "Failed to pause alerts", err)
  259. }
  260. var response m.AlertStateType = m.AlertStatePending
  261. pausedState := "un paused"
  262. if updateCmd.Paused {
  263. response = m.AlertStatePaused
  264. pausedState = "paused"
  265. }
  266. result := map[string]interface{}{
  267. "state": response,
  268. "message": "alerts " + pausedState,
  269. "alertsAffected": updateCmd.ResultCount,
  270. }
  271. return JSON(200, result)
  272. }