notifications.go 3.9 KB

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