email.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. package notifiers
  2. import (
  3. "os"
  4. "strings"
  5. "github.com/grafana/grafana/pkg/bus"
  6. "github.com/grafana/grafana/pkg/log"
  7. m "github.com/grafana/grafana/pkg/models"
  8. "github.com/grafana/grafana/pkg/services/alerting"
  9. "github.com/grafana/grafana/pkg/setting"
  10. )
  11. func init() {
  12. alerting.RegisterNotifier(&alerting.NotifierPlugin{
  13. Type: "email",
  14. Name: "Email",
  15. Description: "Sends notifications using Grafana server configured SMTP settings",
  16. Factory: NewEmailNotifier,
  17. OptionsTemplate: `
  18. <h3 class="page-heading">Email addresses</h3>
  19. <div class="gf-form">
  20. <textarea rows="7" class="gf-form-input width-27" required ng-model="ctrl.model.settings.addresses"></textarea>
  21. </div>
  22. <div class="gf-form">
  23. <span>You can enter multiple email addresses using a ";" separator</span>
  24. </div>
  25. `,
  26. })
  27. }
  28. type EmailNotifier struct {
  29. NotifierBase
  30. Addresses []string
  31. log log.Logger
  32. }
  33. func NewEmailNotifier(model *m.AlertNotification) (alerting.Notifier, error) {
  34. addressesString := model.Settings.Get("addresses").MustString()
  35. if addressesString == "" {
  36. return nil, alerting.ValidationError{Reason: "Could not find addresses in settings"}
  37. }
  38. // split addresses with a few different ways
  39. addresses := strings.FieldsFunc(addressesString, func(r rune) bool {
  40. switch r {
  41. case ',', ';', '\n':
  42. return true
  43. }
  44. return false
  45. })
  46. return &EmailNotifier{
  47. NotifierBase: NewNotifierBase(model.Id, model.IsDefault, model.Name, model.Type, model.Settings),
  48. Addresses: addresses,
  49. log: log.New("alerting.notifier.email"),
  50. }, nil
  51. }
  52. func (this *EmailNotifier) Notify(evalContext *alerting.EvalContext) error {
  53. this.log.Info("Sending alert notification to", "addresses", this.Addresses)
  54. ruleUrl, err := evalContext.GetRuleUrl()
  55. if err != nil {
  56. this.log.Error("Failed get rule link", "error", err)
  57. return err
  58. }
  59. error := ""
  60. if evalContext.Error != nil {
  61. error = evalContext.Error.Error()
  62. }
  63. cmd := &m.SendEmailCommandSync{
  64. SendEmailCommand: m.SendEmailCommand{
  65. Subject: evalContext.GetNotificationTitle(),
  66. Data: map[string]interface{}{
  67. "Title": evalContext.GetNotificationTitle(),
  68. "State": evalContext.Rule.State,
  69. "Name": evalContext.Rule.Name,
  70. "StateModel": evalContext.GetStateModel(),
  71. "Message": evalContext.Rule.Message,
  72. "Error": error,
  73. "RuleUrl": ruleUrl,
  74. "ImageLink": "",
  75. "EmbededImage": "",
  76. "AlertPageUrl": setting.AppUrl + "alerting",
  77. "EvalMatches": evalContext.EvalMatches,
  78. },
  79. To: this.Addresses,
  80. Template: "alert_notification.html",
  81. EmbededFiles: []string{},
  82. },
  83. }
  84. if evalContext.ImagePublicUrl != "" {
  85. cmd.Data["ImageLink"] = evalContext.ImagePublicUrl
  86. } else {
  87. file, err := os.Stat(evalContext.ImageOnDiskPath)
  88. if err == nil {
  89. cmd.EmbededFiles = []string{evalContext.ImageOnDiskPath}
  90. cmd.Data["EmbededImage"] = file.Name()
  91. }
  92. }
  93. err = bus.DispatchCtx(evalContext.Ctx, cmd)
  94. if err != nil {
  95. this.log.Error("Failed to send alert notification email", "error", err)
  96. return err
  97. }
  98. return nil
  99. }