notifications.go 4.1 KB

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