notifications.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. package notifications
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "html/template"
  7. "net/url"
  8. "path/filepath"
  9. "strings"
  10. "github.com/grafana/grafana/pkg/bus"
  11. "github.com/grafana/grafana/pkg/events"
  12. "github.com/grafana/grafana/pkg/log"
  13. m "github.com/grafana/grafana/pkg/models"
  14. "github.com/grafana/grafana/pkg/registry"
  15. "github.com/grafana/grafana/pkg/setting"
  16. "github.com/grafana/grafana/pkg/util"
  17. )
  18. var mailTemplates *template.Template
  19. var tmplResetPassword = "reset_password.html"
  20. var tmplSignUpStarted = "signup_started.html"
  21. var tmplWelcomeOnSignUp = "welcome_on_signup.html"
  22. func init() {
  23. registry.RegisterService(&NotificationService{})
  24. }
  25. type NotificationService struct {
  26. Bus bus.Bus `inject:""`
  27. Cfg *setting.Cfg `inject:""`
  28. mailQueue chan *Message
  29. webhookQueue chan *Webhook
  30. log log.Logger
  31. }
  32. func (ns *NotificationService) Init() error {
  33. ns.log = log.New("notifications")
  34. ns.mailQueue = make(chan *Message, 10)
  35. ns.webhookQueue = make(chan *Webhook, 10)
  36. ns.Bus.AddHandler(ns.sendResetPasswordEmail)
  37. ns.Bus.AddHandler(ns.validateResetPasswordCode)
  38. ns.Bus.AddHandler(ns.sendEmailCommandHandler)
  39. ns.Bus.AddCtxHandler(ns.sendEmailCommandHandlerSync)
  40. ns.Bus.AddCtxHandler(ns.SendWebhookSync)
  41. ns.Bus.AddEventListener(ns.signUpStartedHandler)
  42. ns.Bus.AddEventListener(ns.signUpCompletedHandler)
  43. mailTemplates = template.New("name")
  44. mailTemplates.Funcs(template.FuncMap{
  45. "Subject": subjectTemplateFunc,
  46. })
  47. templatePattern := filepath.Join(setting.StaticRootPath, ns.Cfg.Smtp.TemplatesPattern)
  48. _, err := mailTemplates.ParseGlob(templatePattern)
  49. if err != nil {
  50. return err
  51. }
  52. if !util.IsEmail(ns.Cfg.Smtp.FromAddress) {
  53. return errors.New("Invalid email address for SMTP from_address config")
  54. }
  55. if setting.EmailCodeValidMinutes == 0 {
  56. setting.EmailCodeValidMinutes = 120
  57. }
  58. return nil
  59. }
  60. func (ns *NotificationService) Run(ctx context.Context) error {
  61. for {
  62. select {
  63. case webhook := <-ns.webhookQueue:
  64. err := ns.sendWebRequestSync(context.Background(), webhook)
  65. if err != nil {
  66. ns.log.Error("Failed to send webrequest ", "error", err)
  67. }
  68. case msg := <-ns.mailQueue:
  69. num, err := ns.send(msg)
  70. tos := strings.Join(msg.To, "; ")
  71. info := ""
  72. if err != nil {
  73. if len(msg.Info) > 0 {
  74. info = ", info: " + msg.Info
  75. }
  76. ns.log.Error(fmt.Sprintf("Async sent email %d succeed, not send emails: %s%s err: %s", num, tos, info, err))
  77. } else {
  78. ns.log.Debug(fmt.Sprintf("Async sent email %d succeed, sent emails: %s%s", num, tos, info))
  79. }
  80. case <-ctx.Done():
  81. return ctx.Err()
  82. }
  83. }
  84. return nil
  85. }
  86. func (ns *NotificationService) SendWebhookSync(ctx context.Context, cmd *m.SendWebhookSync) error {
  87. return ns.sendWebRequestSync(ctx, &Webhook{
  88. Url: cmd.Url,
  89. User: cmd.User,
  90. Password: cmd.Password,
  91. Body: cmd.Body,
  92. HttpMethod: cmd.HttpMethod,
  93. HttpHeader: cmd.HttpHeader,
  94. })
  95. }
  96. func subjectTemplateFunc(obj map[string]interface{}, value string) string {
  97. obj["value"] = value
  98. return ""
  99. }
  100. func (ns *NotificationService) sendEmailCommandHandlerSync(ctx context.Context, cmd *m.SendEmailCommandSync) error {
  101. message, err := ns.buildEmailMessage(&m.SendEmailCommand{
  102. Data: cmd.Data,
  103. Info: cmd.Info,
  104. Template: cmd.Template,
  105. To: cmd.To,
  106. EmbededFiles: cmd.EmbededFiles,
  107. Subject: cmd.Subject,
  108. })
  109. if err != nil {
  110. return err
  111. }
  112. _, err = ns.send(message)
  113. return err
  114. }
  115. func (ns *NotificationService) sendEmailCommandHandler(cmd *m.SendEmailCommand) error {
  116. message, err := ns.buildEmailMessage(cmd)
  117. if err != nil {
  118. return err
  119. }
  120. ns.mailQueue <- message
  121. return nil
  122. }
  123. func (ns *NotificationService) sendResetPasswordEmail(cmd *m.SendResetPasswordEmailCommand) error {
  124. return ns.sendEmailCommandHandler(&m.SendEmailCommand{
  125. To: []string{cmd.User.Email},
  126. Template: tmplResetPassword,
  127. Data: map[string]interface{}{
  128. "Code": createUserEmailCode(cmd.User, nil),
  129. "Name": cmd.User.NameOrFallback(),
  130. },
  131. })
  132. }
  133. func (ns *NotificationService) validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
  134. login := getLoginForEmailCode(query.Code)
  135. if login == "" {
  136. return m.ErrInvalidEmailCode
  137. }
  138. userQuery := m.GetUserByLoginQuery{LoginOrEmail: login}
  139. if err := bus.Dispatch(&userQuery); err != nil {
  140. return err
  141. }
  142. if !validateUserEmailCode(userQuery.Result, query.Code) {
  143. return m.ErrInvalidEmailCode
  144. }
  145. query.Result = userQuery.Result
  146. return nil
  147. }
  148. func (ns *NotificationService) signUpStartedHandler(evt *events.SignUpStarted) error {
  149. if !setting.VerifyEmailEnabled {
  150. return nil
  151. }
  152. ns.log.Info("User signup started", "email", evt.Email)
  153. if evt.Email == "" {
  154. return nil
  155. }
  156. err := ns.sendEmailCommandHandler(&m.SendEmailCommand{
  157. To: []string{evt.Email},
  158. Template: tmplSignUpStarted,
  159. Data: map[string]interface{}{
  160. "Email": evt.Email,
  161. "Code": evt.Code,
  162. "SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
  163. },
  164. })
  165. if err != nil {
  166. return err
  167. }
  168. emailSentCmd := m.UpdateTempUserWithEmailSentCommand{Code: evt.Code}
  169. return bus.Dispatch(&emailSentCmd)
  170. }
  171. func (ns *NotificationService) signUpCompletedHandler(evt *events.SignUpCompleted) error {
  172. if evt.Email == "" || !ns.Cfg.Smtp.SendWelcomeEmailOnSignUp {
  173. return nil
  174. }
  175. return ns.sendEmailCommandHandler(&m.SendEmailCommand{
  176. To: []string{evt.Email},
  177. Template: tmplWelcomeOnSignUp,
  178. Data: map[string]interface{}{
  179. "Name": evt.Name,
  180. },
  181. })
  182. }