notifications.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. package notifications
  2. import (
  3. "context"
  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.AddCtxHandler("email", sendEmailCommandHandlerSync)
  27. bus.AddCtxHandler("webhook", SendWebhookSync)
  28. bus.AddEventListener(signUpStartedHandler)
  29. bus.AddEventListener(signUpCompletedHandler)
  30. mailTemplates = template.New("name")
  31. mailTemplates.Funcs(template.FuncMap{
  32. "Subject": subjectTemplateFunc,
  33. })
  34. templatePattern := filepath.Join(setting.StaticRootPath, setting.Smtp.TemplatesPattern)
  35. _, err := mailTemplates.ParseGlob(templatePattern)
  36. if err != nil {
  37. return err
  38. }
  39. if !util.IsEmail(setting.Smtp.FromAddress) {
  40. return errors.New("Invalid email address for SMTP from_address config")
  41. }
  42. if setting.EmailCodeValidMinutes == 0 {
  43. setting.EmailCodeValidMinutes = 120
  44. }
  45. return nil
  46. }
  47. func SendWebhookSync(ctx context.Context, cmd *m.SendWebhookSync) error {
  48. return sendWebRequestSync(ctx, &Webhook{
  49. Url: cmd.Url,
  50. User: cmd.User,
  51. Password: cmd.Password,
  52. Body: cmd.Body,
  53. HttpMethod: cmd.HttpMethod,
  54. HttpHeader: cmd.HttpHeader,
  55. })
  56. }
  57. func subjectTemplateFunc(obj map[string]interface{}, value string) string {
  58. obj["value"] = value
  59. return ""
  60. }
  61. func sendEmailCommandHandlerSync(ctx context.Context, cmd *m.SendEmailCommandSync) error {
  62. message, err := buildEmailMessage(&m.SendEmailCommand{
  63. Data: cmd.Data,
  64. Info: cmd.Info,
  65. Template: cmd.Template,
  66. To: cmd.To,
  67. EmbededFiles: cmd.EmbededFiles,
  68. Subject: cmd.Subject,
  69. })
  70. if err != nil {
  71. return err
  72. }
  73. _, err = send(message)
  74. return err
  75. }
  76. func sendEmailCommandHandler(cmd *m.SendEmailCommand) error {
  77. message, err := buildEmailMessage(cmd)
  78. if err != nil {
  79. return err
  80. }
  81. addToMailQueue(message)
  82. return nil
  83. }
  84. func sendResetPasswordEmail(cmd *m.SendResetPasswordEmailCommand) error {
  85. return sendEmailCommandHandler(&m.SendEmailCommand{
  86. To: []string{cmd.User.Email},
  87. Template: tmplResetPassword,
  88. Data: map[string]interface{}{
  89. "Code": createUserEmailCode(cmd.User, nil),
  90. "Name": cmd.User.NameOrFallback(),
  91. },
  92. })
  93. }
  94. func validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
  95. login := getLoginForEmailCode(query.Code)
  96. if login == "" {
  97. return m.ErrInvalidEmailCode
  98. }
  99. userQuery := m.GetUserByLoginQuery{LoginOrEmail: login}
  100. if err := bus.Dispatch(&userQuery); err != nil {
  101. return err
  102. }
  103. if !validateUserEmailCode(userQuery.Result, query.Code) {
  104. return m.ErrInvalidEmailCode
  105. }
  106. query.Result = userQuery.Result
  107. return nil
  108. }
  109. func signUpStartedHandler(evt *events.SignUpStarted) error {
  110. if !setting.VerifyEmailEnabled {
  111. return nil
  112. }
  113. log.Info("User signup started: %s", evt.Email)
  114. if evt.Email == "" {
  115. return nil
  116. }
  117. err := sendEmailCommandHandler(&m.SendEmailCommand{
  118. To: []string{evt.Email},
  119. Template: tmplSignUpStarted,
  120. Data: map[string]interface{}{
  121. "Email": evt.Email,
  122. "Code": evt.Code,
  123. "SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
  124. },
  125. })
  126. if err != nil {
  127. return err
  128. }
  129. emailSentCmd := m.UpdateTempUserWithEmailSentCommand{Code: evt.Code}
  130. return bus.Dispatch(&emailSentCmd)
  131. }
  132. func signUpCompletedHandler(evt *events.SignUpCompleted) error {
  133. if evt.Email == "" || !setting.Smtp.SendWelcomeEmailOnSignUp {
  134. return nil
  135. }
  136. return sendEmailCommandHandler(&m.SendEmailCommand{
  137. To: []string{evt.Email},
  138. Template: tmplWelcomeOnSignUp,
  139. Data: map[string]interface{}{
  140. "Name": evt.Name,
  141. },
  142. })
  143. }