notifications.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  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. ContentType: cmd.ContentType,
  56. })
  57. }
  58. func subjectTemplateFunc(obj map[string]interface{}, value string) string {
  59. obj["value"] = value
  60. return ""
  61. }
  62. func sendEmailCommandHandlerSync(ctx context.Context, cmd *m.SendEmailCommandSync) error {
  63. message, err := buildEmailMessage(&m.SendEmailCommand{
  64. Data: cmd.Data,
  65. Info: cmd.Info,
  66. Template: cmd.Template,
  67. To: cmd.To,
  68. EmbededFiles: cmd.EmbededFiles,
  69. Subject: cmd.Subject,
  70. })
  71. if err != nil {
  72. return err
  73. }
  74. _, err = send(message)
  75. return err
  76. }
  77. func sendEmailCommandHandler(cmd *m.SendEmailCommand) error {
  78. message, err := buildEmailMessage(cmd)
  79. if err != nil {
  80. return err
  81. }
  82. addToMailQueue(message)
  83. return nil
  84. }
  85. func sendResetPasswordEmail(cmd *m.SendResetPasswordEmailCommand) error {
  86. return sendEmailCommandHandler(&m.SendEmailCommand{
  87. To: []string{cmd.User.Email},
  88. Template: tmplResetPassword,
  89. Data: map[string]interface{}{
  90. "Code": createUserEmailCode(cmd.User, nil),
  91. "Name": cmd.User.NameOrFallback(),
  92. },
  93. })
  94. }
  95. func validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
  96. login := getLoginForEmailCode(query.Code)
  97. if login == "" {
  98. return m.ErrInvalidEmailCode
  99. }
  100. userQuery := m.GetUserByLoginQuery{LoginOrEmail: login}
  101. if err := bus.Dispatch(&userQuery); err != nil {
  102. return err
  103. }
  104. if !validateUserEmailCode(userQuery.Result, query.Code) {
  105. return m.ErrInvalidEmailCode
  106. }
  107. query.Result = userQuery.Result
  108. return nil
  109. }
  110. func signUpStartedHandler(evt *events.SignUpStarted) error {
  111. if !setting.VerifyEmailEnabled {
  112. return nil
  113. }
  114. log.Info("User signup started: %s", evt.Email)
  115. if evt.Email == "" {
  116. return nil
  117. }
  118. err := sendEmailCommandHandler(&m.SendEmailCommand{
  119. To: []string{evt.Email},
  120. Template: tmplSignUpStarted,
  121. Data: map[string]interface{}{
  122. "Email": evt.Email,
  123. "Code": evt.Code,
  124. "SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
  125. },
  126. })
  127. if err != nil {
  128. return err
  129. }
  130. emailSentCmd := m.UpdateTempUserWithEmailSentCommand{Code: evt.Code}
  131. return bus.Dispatch(&emailSentCmd)
  132. }
  133. func signUpCompletedHandler(evt *events.SignUpCompleted) error {
  134. if evt.Email == "" || !setting.Smtp.SendWelcomeEmailOnSignUp {
  135. return nil
  136. }
  137. return sendEmailCommandHandler(&m.SendEmailCommand{
  138. To: []string{evt.Email},
  139. Template: tmplWelcomeOnSignUp,
  140. Data: map[string]interface{}{
  141. "Name": evt.Name,
  142. },
  143. })
  144. }