| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219 |
- package notifications
- import (
- "context"
- "errors"
- "fmt"
- "html/template"
- "net/url"
- "path/filepath"
- "strings"
- "github.com/grafana/grafana/pkg/bus"
- "github.com/grafana/grafana/pkg/events"
- "github.com/grafana/grafana/pkg/log"
- m "github.com/grafana/grafana/pkg/models"
- "github.com/grafana/grafana/pkg/registry"
- "github.com/grafana/grafana/pkg/setting"
- "github.com/grafana/grafana/pkg/util"
- )
- var mailTemplates *template.Template
- var tmplResetPassword = "reset_password.html"
- var tmplSignUpStarted = "signup_started.html"
- var tmplWelcomeOnSignUp = "welcome_on_signup.html"
- func init() {
- registry.RegisterService(&NotificationService{})
- }
- type NotificationService struct {
- Bus bus.Bus `inject:""`
- mailQueue chan *Message
- webhookQueue chan *Webhook
- log log.Logger
- }
- func (ns *NotificationService) Init() error {
- ns.log = log.New("notifications")
- ns.mailQueue = make(chan *Message, 10)
- ns.webhookQueue = make(chan *Webhook, 10)
- ns.Bus.AddHandler(ns.sendResetPasswordEmail)
- ns.Bus.AddHandler(ns.validateResetPasswordCode)
- ns.Bus.AddHandler(ns.sendEmailCommandHandler)
- ns.Bus.AddCtxHandler(ns.sendEmailCommandHandlerSync)
- ns.Bus.AddCtxHandler(ns.SendWebhookSync)
- ns.Bus.AddEventListener(ns.signUpStartedHandler)
- ns.Bus.AddEventListener(ns.signUpCompletedHandler)
- mailTemplates = template.New("name")
- mailTemplates.Funcs(template.FuncMap{
- "Subject": subjectTemplateFunc,
- })
- templatePattern := filepath.Join(setting.StaticRootPath, setting.Smtp.TemplatesPattern)
- _, err := mailTemplates.ParseGlob(templatePattern)
- if err != nil {
- return err
- }
- if !util.IsEmail(setting.Smtp.FromAddress) {
- return errors.New("Invalid email address for SMTP from_address config")
- }
- if setting.EmailCodeValidMinutes == 0 {
- setting.EmailCodeValidMinutes = 120
- }
- return nil
- }
- 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:
- num, err := send(msg)
- 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()
- }
- }
- return nil
- }
- func (ns *NotificationService) SendWebhookSync(ctx context.Context, cmd *m.SendWebhookSync) error {
- return ns.sendWebRequestSync(ctx, &Webhook{
- Url: cmd.Url,
- User: cmd.User,
- Password: cmd.Password,
- Body: cmd.Body,
- HttpMethod: cmd.HttpMethod,
- HttpHeader: cmd.HttpHeader,
- })
- }
- func subjectTemplateFunc(obj map[string]interface{}, value string) string {
- obj["value"] = value
- return ""
- }
- func (ns *NotificationService) sendEmailCommandHandlerSync(ctx context.Context, cmd *m.SendEmailCommandSync) error {
- message, err := buildEmailMessage(&m.SendEmailCommand{
- Data: cmd.Data,
- Info: cmd.Info,
- Template: cmd.Template,
- To: cmd.To,
- EmbededFiles: cmd.EmbededFiles,
- Subject: cmd.Subject,
- })
- if err != nil {
- return err
- }
- _, err = send(message)
- return err
- }
- func (ns *NotificationService) sendEmailCommandHandler(cmd *m.SendEmailCommand) error {
- message, err := buildEmailMessage(cmd)
- if err != nil {
- return err
- }
- ns.mailQueue <- message
- return nil
- }
- func (ns *NotificationService) sendResetPasswordEmail(cmd *m.SendResetPasswordEmailCommand) error {
- return ns.sendEmailCommandHandler(&m.SendEmailCommand{
- To: []string{cmd.User.Email},
- Template: tmplResetPassword,
- Data: map[string]interface{}{
- "Code": createUserEmailCode(cmd.User, nil),
- "Name": cmd.User.NameOrFallback(),
- },
- })
- }
- func (ns *NotificationService) validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
- login := getLoginForEmailCode(query.Code)
- if login == "" {
- return m.ErrInvalidEmailCode
- }
- userQuery := m.GetUserByLoginQuery{LoginOrEmail: login}
- if err := bus.Dispatch(&userQuery); err != nil {
- return err
- }
- if !validateUserEmailCode(userQuery.Result, query.Code) {
- return m.ErrInvalidEmailCode
- }
- query.Result = userQuery.Result
- return nil
- }
- func (ns *NotificationService) signUpStartedHandler(evt *events.SignUpStarted) error {
- if !setting.VerifyEmailEnabled {
- return nil
- }
- ns.log.Info("User signup started", "email", evt.Email)
- if evt.Email == "" {
- return nil
- }
- err := ns.sendEmailCommandHandler(&m.SendEmailCommand{
- To: []string{evt.Email},
- Template: tmplSignUpStarted,
- Data: map[string]interface{}{
- "Email": evt.Email,
- "Code": evt.Code,
- "SignUpUrl": setting.ToAbsUrl(fmt.Sprintf("signup/?email=%s&code=%s", url.QueryEscape(evt.Email), url.QueryEscape(evt.Code))),
- },
- })
- if err != nil {
- return err
- }
- emailSentCmd := m.UpdateTempUserWithEmailSentCommand{Code: evt.Code}
- return bus.Dispatch(&emailSentCmd)
- }
- func (ns *NotificationService) signUpCompletedHandler(evt *events.SignUpCompleted) error {
- if evt.Email == "" || !setting.Smtp.SendWelcomeEmailOnSignUp {
- return nil
- }
- return ns.sendEmailCommandHandler(&m.SendEmailCommand{
- To: []string{evt.Email},
- Template: tmplWelcomeOnSignUp,
- Data: map[string]interface{}{
- "Name": evt.Name,
- },
- })
- }
|