2015-06-04 17:23:46 +02:00
|
|
|
package notifications
|
|
|
|
|
|
2015-06-05 11:08:19 +02:00
|
|
|
import (
|
2016-10-03 09:38:03 +02:00
|
|
|
"context"
|
2015-06-05 11:08:19 +02:00
|
|
|
"errors"
|
2015-08-28 13:45:16 +02:00
|
|
|
"fmt"
|
2015-06-05 11:08:19 +02:00
|
|
|
"html/template"
|
2015-08-28 13:45:16 +02:00
|
|
|
"net/url"
|
2015-06-05 11:08:19 +02:00
|
|
|
"path/filepath"
|
2018-04-27 13:01:32 +02:00
|
|
|
"strings"
|
2015-06-04 17:23:46 +02:00
|
|
|
|
2015-06-05 11:08:19 +02:00
|
|
|
"github.com/grafana/grafana/pkg/bus"
|
2015-06-08 17:56:56 +02:00
|
|
|
"github.com/grafana/grafana/pkg/events"
|
2019-05-13 14:45:54 +08:00
|
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
2020-02-29 13:35:15 +01:00
|
|
|
"github.com/grafana/grafana/pkg/models"
|
2018-04-27 13:01:32 +02:00
|
|
|
"github.com/grafana/grafana/pkg/registry"
|
2015-06-05 11:08:19 +02:00
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
|
"github.com/grafana/grafana/pkg/util"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var mailTemplates *template.Template
|
|
|
|
|
var tmplResetPassword = "reset_password.html"
|
2015-08-28 13:45:16 +02:00
|
|
|
var tmplSignUpStarted = "signup_started.html"
|
2015-08-31 11:42:12 +02:00
|
|
|
var tmplWelcomeOnSignUp = "welcome_on_signup.html"
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
func init() {
|
|
|
|
|
registry.RegisterService(&NotificationService{})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type NotificationService struct {
|
2018-04-30 16:21:04 +02:00
|
|
|
Bus bus.Bus `inject:""`
|
|
|
|
|
Cfg *setting.Cfg `inject:""`
|
|
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
mailQueue chan *Message
|
|
|
|
|
webhookQueue chan *Webhook
|
|
|
|
|
log log.Logger
|
|
|
|
|
}
|
2015-06-08 16:51:25 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
func (ns *NotificationService) Init() error {
|
|
|
|
|
ns.log = log.New("notifications")
|
|
|
|
|
ns.mailQueue = make(chan *Message, 10)
|
|
|
|
|
ns.webhookQueue = make(chan *Webhook, 10)
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
ns.Bus.AddHandler(ns.sendResetPasswordEmail)
|
|
|
|
|
ns.Bus.AddHandler(ns.validateResetPasswordCode)
|
|
|
|
|
ns.Bus.AddHandler(ns.sendEmailCommandHandler)
|
2016-10-03 09:38:03 +02:00
|
|
|
|
2018-06-07 13:03:27 -07:00
|
|
|
ns.Bus.AddHandlerCtx(ns.sendEmailCommandHandlerSync)
|
|
|
|
|
ns.Bus.AddHandlerCtx(ns.SendWebhookSync)
|
2016-06-15 14:45:05 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
ns.Bus.AddEventListener(ns.signUpStartedHandler)
|
|
|
|
|
ns.Bus.AddEventListener(ns.signUpCompletedHandler)
|
2015-06-08 17:56:56 +02:00
|
|
|
|
2015-06-05 11:08:19 +02:00
|
|
|
mailTemplates = template.New("name")
|
|
|
|
|
mailTemplates.Funcs(template.FuncMap{
|
|
|
|
|
"Subject": subjectTemplateFunc,
|
|
|
|
|
})
|
|
|
|
|
|
2021-03-10 12:41:29 +01:00
|
|
|
templatePattern := filepath.Join(ns.Cfg.StaticRootPath, ns.Cfg.Smtp.TemplatesPattern)
|
2015-06-05 11:08:19 +02:00
|
|
|
_, err := mailTemplates.ParseGlob(templatePattern)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-30 16:21:04 +02:00
|
|
|
if !util.IsEmail(ns.Cfg.Smtp.FromAddress) {
|
2020-11-05 11:57:20 +01:00
|
|
|
return errors.New("invalid email address for SMTP from_address config")
|
2015-06-05 11:08:19 +02:00
|
|
|
}
|
|
|
|
|
|
2015-06-08 10:57:01 +02:00
|
|
|
if setting.EmailCodeValidMinutes == 0 {
|
|
|
|
|
setting.EmailCodeValidMinutes = 120
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-05 11:08:19 +02:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
func (ns *NotificationService) Run(ctx context.Context) error {
|
|
|
|
|
for {
|
|
|
|
|
select {
|
|
|
|
|
case webhook := <-ns.webhookQueue:
|
|
|
|
|
err := ns.sendWebRequestSync(context.Background(), webhook)
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
ns.log.Error("Failed to send webrequest ", "error", err)
|
|
|
|
|
}
|
|
|
|
|
case msg := <-ns.mailQueue:
|
2021-03-18 20:25:11 +05:30
|
|
|
num, err := ns.Send(msg)
|
2018-04-27 13:01:32 +02:00
|
|
|
tos := strings.Join(msg.To, "; ")
|
|
|
|
|
info := ""
|
|
|
|
|
if err != nil {
|
|
|
|
|
if len(msg.Info) > 0 {
|
|
|
|
|
info = ", info: " + msg.Info
|
|
|
|
|
}
|
|
|
|
|
ns.log.Error(fmt.Sprintf("Async sent email %d succeed, not send emails: %s%s err: %s", num, tos, info, err))
|
|
|
|
|
} else {
|
|
|
|
|
ns.log.Debug(fmt.Sprintf("Async sent email %d succeed, sent emails: %s%s", num, tos, info))
|
|
|
|
|
}
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
|
return ctx.Err()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
func (ns *NotificationService) SendWebhookSync(ctx context.Context, cmd *models.SendWebhookSync) error {
|
2018-04-27 13:01:32 +02:00
|
|
|
return ns.sendWebRequestSync(ctx, &Webhook{
|
2017-03-23 21:53:54 +01:00
|
|
|
Url: cmd.Url,
|
|
|
|
|
User: cmd.User,
|
|
|
|
|
Password: cmd.Password,
|
|
|
|
|
Body: cmd.Body,
|
|
|
|
|
HttpMethod: cmd.HttpMethod,
|
|
|
|
|
HttpHeader: cmd.HttpHeader,
|
|
|
|
|
ContentType: cmd.ContentType,
|
2016-10-03 09:38:03 +02:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-05 11:08:19 +02:00
|
|
|
func subjectTemplateFunc(obj map[string]interface{}, value string) string {
|
|
|
|
|
obj["value"] = value
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
func (ns *NotificationService) sendEmailCommandHandlerSync(ctx context.Context, cmd *models.SendEmailCommandSync) error {
|
|
|
|
|
message, err := ns.buildEmailMessage(&models.SendEmailCommand{
|
2020-06-01 17:11:25 +02:00
|
|
|
Data: cmd.Data,
|
|
|
|
|
Info: cmd.Info,
|
|
|
|
|
Template: cmd.Template,
|
|
|
|
|
To: cmd.To,
|
|
|
|
|
SingleEmail: cmd.SingleEmail,
|
|
|
|
|
EmbeddedFiles: cmd.EmbeddedFiles,
|
|
|
|
|
Subject: cmd.Subject,
|
2020-06-22 18:56:49 +02:00
|
|
|
ReplyTo: cmd.ReplyTo,
|
2016-10-03 09:38:03 +02:00
|
|
|
})
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2015-09-08 10:49:35 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-18 20:25:11 +05:30
|
|
|
_, err = ns.Send(message)
|
2016-10-03 09:38:03 +02:00
|
|
|
return err
|
|
|
|
|
}
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
func (ns *NotificationService) sendEmailCommandHandler(cmd *models.SendEmailCommand) error {
|
2018-04-30 16:21:04 +02:00
|
|
|
message, err := ns.buildEmailMessage(cmd)
|
2015-06-14 06:07:36 +02:00
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
ns.mailQueue <- message
|
2015-06-05 08:15:38 +02:00
|
|
|
return nil
|
2015-06-04 17:23:46 +02:00
|
|
|
}
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
func (ns *NotificationService) sendResetPasswordEmail(cmd *models.SendResetPasswordEmailCommand) error {
|
2019-10-22 14:08:18 +02:00
|
|
|
code, err := createUserEmailCode(cmd.User, nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2020-02-29 13:35:15 +01:00
|
|
|
return ns.sendEmailCommandHandler(&models.SendEmailCommand{
|
2015-06-08 16:51:25 +02:00
|
|
|
To: []string{cmd.User.Email},
|
|
|
|
|
Template: tmplResetPassword,
|
|
|
|
|
Data: map[string]interface{}{
|
2019-10-22 14:08:18 +02:00
|
|
|
"Code": code,
|
2015-06-08 16:51:25 +02:00
|
|
|
"Name": cmd.User.NameOrFallback(),
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
func (ns *NotificationService) validateResetPasswordCode(query *models.ValidateResetPasswordCodeQuery) error {
|
2015-06-08 13:39:02 +02:00
|
|
|
login := getLoginForEmailCode(query.Code)
|
|
|
|
|
if login == "" {
|
2020-02-29 13:35:15 +01:00
|
|
|
return models.ErrInvalidEmailCode
|
2015-06-08 13:39:02 +02:00
|
|
|
}
|
2015-06-05 11:08:19 +02:00
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
userQuery := models.GetUserByLoginQuery{LoginOrEmail: login}
|
2015-06-08 13:39:02 +02:00
|
|
|
if err := bus.Dispatch(&userQuery); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2015-06-08 10:57:01 +02:00
|
|
|
|
2019-10-22 14:08:18 +02:00
|
|
|
validEmailCode, err := validateUserEmailCode(userQuery.Result, query.Code)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
if !validEmailCode {
|
2020-02-29 13:35:15 +01:00
|
|
|
return models.ErrInvalidEmailCode
|
2015-06-08 13:39:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
query.Result = userQuery.Result
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2015-06-08 17:56:56 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
func (ns *NotificationService) signUpStartedHandler(evt *events.SignUpStarted) error {
|
2015-08-31 11:35:07 +02:00
|
|
|
if !setting.VerifyEmailEnabled {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
ns.log.Info("User signup started", "email", evt.Email)
|
2015-06-08 17:56:56 +02:00
|
|
|
|
2015-08-28 13:45:16 +02:00
|
|
|
if evt.Email == "" {
|
2015-06-08 17:56:56 +02:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
err := ns.sendEmailCommandHandler(&models.SendEmailCommand{
|
2015-06-08 17:56:56 +02:00
|
|
|
To: []string{evt.Email},
|
2015-08-28 13:45:16 +02:00
|
|
|
Template: tmplSignUpStarted,
|
2015-06-08 17:56:56 +02:00
|
|
|
Data: map[string]interface{}{
|
2015-08-28 13:45:16 +02:00
|
|
|
"Email": evt.Email,
|
|
|
|
|
"Code": evt.Code,
|
|
|
|
|
"SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
|
2015-06-08 17:56:56 +02:00
|
|
|
},
|
|
|
|
|
})
|
2018-04-27 13:01:32 +02:00
|
|
|
|
2017-06-30 20:21:05 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
emailSentCmd := models.UpdateTempUserWithEmailSentCommand{Code: evt.Code}
|
2017-06-30 20:21:05 +02:00
|
|
|
return bus.Dispatch(&emailSentCmd)
|
2015-06-08 17:56:56 +02:00
|
|
|
}
|
2015-08-31 11:42:12 +02:00
|
|
|
|
2018-04-27 13:01:32 +02:00
|
|
|
func (ns *NotificationService) signUpCompletedHandler(evt *events.SignUpCompleted) error {
|
2018-04-30 16:21:04 +02:00
|
|
|
if evt.Email == "" || !ns.Cfg.Smtp.SendWelcomeEmailOnSignUp {
|
2015-08-31 11:42:12 +02:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 13:35:15 +01:00
|
|
|
return ns.sendEmailCommandHandler(&models.SendEmailCommand{
|
2015-08-31 11:42:12 +02:00
|
|
|
To: []string{evt.Email},
|
2015-09-01 12:35:06 +02:00
|
|
|
Template: tmplWelcomeOnSignUp,
|
2015-08-31 11:42:12 +02:00
|
|
|
Data: map[string]interface{}{
|
|
|
|
|
"Name": evt.Name,
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
}
|