notifications.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. package notifications
  2. import (
  3. "bytes"
  4. "errors"
  5. "fmt"
  6. "html/template"
  7. "net/url"
  8. "path/filepath"
  9. "github.com/grafana/grafana/pkg/bus"
  10. "github.com/grafana/grafana/pkg/events"
  11. "github.com/grafana/grafana/pkg/log"
  12. m "github.com/grafana/grafana/pkg/models"
  13. "github.com/grafana/grafana/pkg/setting"
  14. "github.com/grafana/grafana/pkg/util"
  15. )
  16. var mailTemplates *template.Template
  17. var tmplResetPassword = "reset_password.html"
  18. var tmplSignUpStarted = "signup_started.html"
  19. var tmplWelcomeOnSignUp = "welcome_on_signup.html"
  20. func Init() error {
  21. initMailQueue()
  22. initWebhookQueue()
  23. bus.AddHandler("email", sendResetPasswordEmail)
  24. bus.AddHandler("email", validateResetPasswordCode)
  25. bus.AddHandler("email", sendEmailCommandHandler)
  26. bus.AddHandler("webhook", sendWebhook)
  27. bus.AddEventListener(signUpStartedHandler)
  28. bus.AddEventListener(signUpCompletedHandler)
  29. mailTemplates = template.New("name")
  30. mailTemplates.Funcs(template.FuncMap{
  31. "Subject": subjectTemplateFunc,
  32. })
  33. templatePattern := filepath.Join(setting.StaticRootPath, setting.Smtp.TemplatesPattern)
  34. _, err := mailTemplates.ParseGlob(templatePattern)
  35. if err != nil {
  36. return err
  37. }
  38. if !util.IsEmail(setting.Smtp.FromAddress) {
  39. return errors.New("Invalid email address for smpt from_adress config")
  40. }
  41. if setting.EmailCodeValidMinutes == 0 {
  42. setting.EmailCodeValidMinutes = 120
  43. }
  44. return nil
  45. }
  46. func sendWebhook(cmd *m.SendWebhook) error {
  47. addToWebhookQueue(&Webhook{
  48. Url: cmd.Url,
  49. User: cmd.User,
  50. Password: cmd.Password,
  51. Body: cmd.Body,
  52. })
  53. return nil
  54. }
  55. func subjectTemplateFunc(obj map[string]interface{}, value string) string {
  56. obj["value"] = value
  57. return ""
  58. }
  59. func sendEmailCommandHandler(cmd *m.SendEmailCommand) error {
  60. if !setting.Smtp.Enabled {
  61. return errors.New("Grafana mailing/smtp options not configured, contact your Grafana admin")
  62. }
  63. var buffer bytes.Buffer
  64. var err error
  65. var subjectText interface{}
  66. data := cmd.Data
  67. if data == nil {
  68. data = make(map[string]interface{}, 10)
  69. }
  70. setDefaultTemplateData(data, nil)
  71. err = mailTemplates.ExecuteTemplate(&buffer, cmd.Template, data)
  72. if err != nil {
  73. return err
  74. }
  75. subjectData := data["Subject"].(map[string]interface{})
  76. subjectText, hasSubject := subjectData["value"]
  77. if !hasSubject {
  78. return errors.New(fmt.Sprintf("Missing subject in Template %s", cmd.Template))
  79. }
  80. subjectTmpl, err := template.New("subject").Parse(subjectText.(string))
  81. if err != nil {
  82. return err
  83. }
  84. var subjectBuffer bytes.Buffer
  85. err = subjectTmpl.ExecuteTemplate(&subjectBuffer, "subject", data)
  86. if err != nil {
  87. return err
  88. }
  89. addToMailQueue(&Message{
  90. To: cmd.To,
  91. From: setting.Smtp.FromAddress,
  92. Subject: subjectBuffer.String(),
  93. Body: buffer.String(),
  94. })
  95. return nil
  96. }
  97. func sendResetPasswordEmail(cmd *m.SendResetPasswordEmailCommand) error {
  98. return sendEmailCommandHandler(&m.SendEmailCommand{
  99. To: []string{cmd.User.Email},
  100. Template: tmplResetPassword,
  101. Data: map[string]interface{}{
  102. "Code": createUserEmailCode(cmd.User, nil),
  103. "Name": cmd.User.NameOrFallback(),
  104. },
  105. })
  106. }
  107. func validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
  108. login := getLoginForEmailCode(query.Code)
  109. if login == "" {
  110. return m.ErrInvalidEmailCode
  111. }
  112. userQuery := m.GetUserByLoginQuery{LoginOrEmail: login}
  113. if err := bus.Dispatch(&userQuery); err != nil {
  114. return err
  115. }
  116. if !validateUserEmailCode(userQuery.Result, query.Code) {
  117. return m.ErrInvalidEmailCode
  118. }
  119. query.Result = userQuery.Result
  120. return nil
  121. }
  122. func signUpStartedHandler(evt *events.SignUpStarted) error {
  123. if !setting.VerifyEmailEnabled {
  124. return nil
  125. }
  126. log.Info("User signup started: %s", evt.Email)
  127. if evt.Email == "" {
  128. return nil
  129. }
  130. return sendEmailCommandHandler(&m.SendEmailCommand{
  131. To: []string{evt.Email},
  132. Template: tmplSignUpStarted,
  133. Data: map[string]interface{}{
  134. "Email": evt.Email,
  135. "Code": evt.Code,
  136. "SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
  137. },
  138. })
  139. }
  140. func signUpCompletedHandler(evt *events.SignUpCompleted) error {
  141. if evt.Email == "" || !setting.Smtp.SendWelcomeEmailOnSignUp {
  142. return nil
  143. }
  144. return sendEmailCommandHandler(&m.SendEmailCommand{
  145. To: []string{evt.Email},
  146. Template: tmplWelcomeOnSignUp,
  147. Data: map[string]interface{}{
  148. "Name": evt.Name,
  149. },
  150. })
  151. }