notifications.go 5.4 KB

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