// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. package email import ( "bytes" "encoding/json" "fmt" "html/template" "io" "net/http" "net/url" "strings" "github.com/pkg/errors" "github.com/mattermost/mattermost/server/public/model" "github.com/mattermost/mattermost/server/public/shared/i18n" "github.com/mattermost/mattermost/server/public/shared/mlog" "github.com/mattermost/mattermost/server/v8/platform/shared/mail" "github.com/mattermost/mattermost/server/v8/platform/shared/templates" "github.com/microcosm-cc/bluemonday" ) // Returns category if enabled is true (default false) // If "" is returned when enabled is false, the category headers aren't attached to the email func getSendGridCategory(category string, enabled bool) string { if enabled { return category } return "" } func (es *Service) SendChangeUsernameEmail(newUsername, email, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.username_change_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "TeamDisplayName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.username_change_body.title") data.Props["Info"] = T("api.templates.username_change_body.info", map[string]any{"TeamDisplayName": es.config().TeamSettings.SiteName, "NewUsername": newUsername}) data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("email_change_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "ChangeUsernameEmail"); err != nil { return err } return nil } func (es *Service) SendEmailChangeVerifyEmail(newUserEmail, locale, siteURL, token string) error { T := i18n.GetUserTranslations(locale) link := fmt.Sprintf("%s/do_verify_email?token=%s&email=%s", siteURL, token, url.QueryEscape(newUserEmail)) subject := T("api.templates.email_change_verify_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "TeamDisplayName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.email_change_verify_body.title") data.Props["Info"] = T("api.templates.email_change_verify_body.info", map[string]any{"TeamDisplayName": es.config().TeamSettings.SiteName}) data.Props["VerifyUrl"] = link data.Props["VerifyButton"] = T("api.templates.email_change_verify_body.button") data.Props["QuestionTitle"] = T("api.templates.questions_footer.title") data.Props["EmailInfo1"] = T("api.templates.email_us_anytime_at") data.Props["SupportEmail"] = "feedback@mattermost.com" data.Props["FooterV2"] = T("api.templates.email_footer_v2") body, err := es.templatesContainer.RenderToString("email_change_verify_body", data) if err != nil { return err } if err := es.sendMail(newUserEmail, subject, body, "EmailChangeVerifyEmail"); err != nil { return err } return nil } func (es *Service) SendEmailChangeEmail(oldEmail, newEmail, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.email_change_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "TeamDisplayName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.email_change_body.title") data.Props["Info"] = T("api.templates.email_change_body.info", map[string]any{"TeamDisplayName": es.config().TeamSettings.SiteName, "NewEmail": newEmail}) data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("email_change_body", data) if err != nil { return err } if err := es.sendMail(oldEmail, subject, body, "EmailChangeEmail"); err != nil { return err } return nil } func (es *Service) SendVerifyEmail(userEmail, locale, siteURL, token, redirect string) error { T := i18n.GetUserTranslations(locale) link := fmt.Sprintf("%s/do_verify_email?token=%s&email=%s", siteURL, token, url.QueryEscape(userEmail)) if redirect != "" { link += fmt.Sprintf("&redirect_to=%s", redirect) } serverURL := condenseSiteURL(siteURL) subject := T("api.templates.verify_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.verify_body.title") data.Props["SubTitle1"] = T("api.templates.verify_body.subTitle1") data.Props["ServerURL"] = T("api.templates.verify_body.serverURL", map[string]any{"ServerURL": serverURL}) data.Props["SubTitle2"] = T("api.templates.verify_body.subTitle2") data.Props["ButtonURL"] = link data.Props["Button"] = T("api.templates.verify_body.button") data.Props["Info"] = T("api.templates.verify_body.info") data.Props["Info1"] = T("api.templates.verify_body.info1") data.Props["QuestionTitle"] = T("api.templates.questions_footer.title") data.Props["QuestionInfo"] = T("api.templates.questions_footer.info") body, err := es.templatesContainer.RenderToString("verify_body", data) if err != nil { return err } if err := es.sendMail(userEmail, subject, body, "VerifyEmail"); err != nil { return err } return nil } func (es *Service) SendSignInChangeEmail(email, method, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.signin_change_email.subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.signin_change_email.body.title") data.Props["Info"] = T("api.templates.signin_change_email.body.info", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "Method": method}) data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("signin_change_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "SignInChangeEmail"); err != nil { return err } return nil } func (es *Service) SendWelcomeEmail(userID string, email string, verified bool, disableWelcomeEmail bool, locale, siteURL, redirect string) error { if disableWelcomeEmail { return nil } if !*es.config().EmailSettings.SendEmailNotifications && !*es.config().EmailSettings.RequireEmailVerification { return errors.New("send email notifications and require email verification is disabled in the system console") } T := i18n.GetUserTranslations(locale) serverURL := condenseSiteURL(siteURL) subject := T("api.templates.welcome_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "ServerURL": serverURL}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.welcome_body.title") data.Props["SubTitle1"] = T("api.templates.welcome_body.subTitle1") data.Props["ServerURL"] = T("api.templates.welcome_body.serverURL", map[string]any{"ServerURL": serverURL}) data.Props["SubTitle2"] = T("api.templates.welcome_body.subTitle2") data.Props["Button"] = T("api.templates.welcome_body.button") data.Props["Info"] = T("api.templates.welcome_body.info") data.Props["Info1"] = T("api.templates.welcome_body.info1") data.Props["SiteURL"] = siteURL if *es.config().NativeAppSettings.AppDownloadLink != "" { data.Props["AppDownloadTitle"] = T("api.templates.welcome_body.app_download_title") data.Props["AppDownloadInfo"] = T("api.templates.welcome_body.app_download_info") data.Props["AppDownloadButton"] = T("api.templates.welcome_body.app_download_button") data.Props["AppDownloadLink"] = *es.config().NativeAppSettings.AppDownloadLink } if !verified && *es.config().EmailSettings.RequireEmailVerification { token, err := es.CreateVerifyEmailToken(userID, email) if err != nil { return err } link := fmt.Sprintf("%s/do_verify_email?token=%s&email=%s", siteURL, token.Token, url.QueryEscape(email)) if redirect != "" { link += fmt.Sprintf("&redirect_to=%s", redirect) } data.Props["ButtonURL"] = link } body, err := es.templatesContainer.RenderToString("welcome_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "WelcomeEmail"); err != nil { return err } return nil } // SendCloudWelcomeEmail sends the cloud version of the welcome email func (es *Service) SendCloudWelcomeEmail(userEmail, locale, teamInviteID, workSpaceName, dns, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.cloud_welcome_email.subject") data := es.NewEmailTemplateData(locale) data.Props["Title"] = T("api.templates.cloud_welcome_email.title") data.Props["SubTitle"] = T("api.templates.cloud_welcome_email.subtitle") data.Props["SubTitleInfo"] = T("api.templates.cloud_welcome_email.subtitle_info") data.Props["Info"] = T("api.templates.cloud_welcome_email.info") data.Props["Info2"] = T("api.templates.cloud_welcome_email.info2") data.Props["WorkSpacePath"] = siteURL data.Props["DNS"] = dns data.Props["InviteInfo"] = T("api.templates.cloud_welcome_email.invite_info") data.Props["InviteSubInfo"] = T("api.templates.cloud_welcome_email.invite_sub_info", map[string]any{"WorkSpace": workSpaceName}) data.Props["InviteSubInfoLink"] = fmt.Sprintf("%s/signup_user_complete/?id=%s", siteURL, teamInviteID) data.Props["AddAppsInfo"] = T("api.templates.cloud_welcome_email.add_apps_info") data.Props["AddAppsSubInfo"] = T("api.templates.cloud_welcome_email.add_apps_sub_info") data.Props["AppMarketPlace"] = T("api.templates.cloud_welcome_email.app_market_place") data.Props["AppMarketPlaceLink"] = "https://integrations.mattermost.com/" data.Props["DownloadMMInfo"] = T("api.templates.cloud_welcome_email.download_mm_info") data.Props["SignInSubInfo"] = T("api.templates.cloud_welcome_email.signin_sub_info") data.Props["MMApps"] = T("api.templates.cloud_welcome_email.mm_apps") data.Props["SignInSubInfo2"] = T("api.templates.cloud_welcome_email.signin_sub_info2") if es.config().NativeAppSettings.AppDownloadLink != nil && *es.config().NativeAppSettings.AppDownloadLink != "" { data.Props["DownloadMMAppsLink"] = es.config().NativeAppSettings.AppDownloadLink } else { data.Props["DownloadMMAppsLink"] = "https://mattermost.com/pl/download-apps" } data.Props["Button"] = T("api.templates.cloud_welcome_email.button") data.Props["GettingStartedQuestions"] = T("api.templates.cloud_welcome_email.start_questions") body, err := es.templatesContainer.RenderToString("cloud_welcome_email", data) if err != nil { return err } if err := es.sendEmailWithCustomReplyTo(userEmail, subject, body, *es.config().SupportSettings.SupportEmail, "CloudWelcomeEmail"); err != nil { return err } return nil } func (es *Service) SendPasswordChangeEmail(email, method, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.password_change_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "TeamDisplayName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.password_change_body.title") data.Props["Info"] = T("api.templates.password_change_body.info", map[string]any{"TeamDisplayName": es.config().TeamSettings.SiteName, "TeamURL": siteURL, "Method": method}) data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("password_change_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "PasswordChangeEmail"); err != nil { return err } return nil } func (es *Service) SendUserAccessTokenAddedEmail(email, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.user_access_token_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.user_access_token_body.title") data.Props["Info"] = T("api.templates.user_access_token_body.info", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "SiteURL": siteURL}) data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("password_change_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "UserAccessTokenAddedEmail"); err != nil { return err } return nil } func (es *Service) SendPasswordResetEmail(email string, token *model.Token, locale, siteURL string) (bool, error) { T := i18n.GetUserTranslations(locale) link := fmt.Sprintf("%s/reset_password_complete?token=%s", siteURL, url.QueryEscape(token.Token)) subject := T("api.templates.reset_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.reset_body.title") data.Props["SubTitle"] = T("api.templates.reset_body.subTitle") data.Props["Info"] = T("api.templates.reset_body.info") data.Props["ButtonURL"] = link data.Props["Button"] = T("api.templates.reset_body.button") data.Props["QuestionTitle"] = T("api.templates.questions_footer.title") data.Props["QuestionInfo"] = T("api.templates.questions_footer.info") body, err := es.templatesContainer.RenderToString("reset_body", data) if err != nil { return false, err } if err := es.sendMail(email, subject, body, "PasswordResetEmail"); err != nil { return false, err } return true, nil } func (es *Service) SendMfaChangeEmail(email string, activated bool, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.mfa_change_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL if activated { data.Props["Info"] = T("api.templates.mfa_activated_body.info", map[string]any{"SiteURL": siteURL}) data.Props["Title"] = T("api.templates.mfa_activated_body.title") } else { data.Props["Info"] = T("api.templates.mfa_deactivated_body.info", map[string]any{"SiteURL": siteURL}) data.Props["Title"] = T("api.templates.mfa_deactivated_body.title") } data.Props["Warning"] = T("api.templates.email_warning") body, err := es.templatesContainer.RenderToString("mfa_change_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "MfaChangeEmail"); err != nil { return err } return nil } func (es *Service) SendInviteEmails( team *model.Team, senderName string, senderUserId string, invites []string, siteURL string, reminderData *model.TeamInviteReminderData, errorWhenNotSent bool, isSystemAdmin bool, isFirstAdmin bool, ) error { if es.perHourEmailRateLimiter == nil { return NoRateLimiterError } rateLimited, result, err := es.perHourEmailRateLimiter.RateLimit(senderUserId, len(invites)) if err != nil { return SetupRateLimiterError } if rateLimited { mlog.Error("rate limit exceeded", mlog.Duration("RetryAfter", result.RetryAfter), mlog.Duration("ResetAfter", result.ResetAfter), mlog.String("user_id", senderUserId), mlog.String("team_id", team.Id), mlog.String("retry_after_secs", fmt.Sprintf("%f", result.RetryAfter.Seconds())), mlog.String("reset_after_secs", fmt.Sprintf("%f", result.ResetAfter.Seconds()))) return RateLimitExceededError } for _, invite := range invites { if invite != "" { subject := i18n.T("api.templates.invite_subject", map[string]any{"SenderName": senderName, "TeamDisplayName": team.DisplayName, "SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData("") data.Props["SiteURL"] = siteURL data.Props["SubTitle"] = i18n.T("api.templates.invite_body.subTitle") data.Props["Button"] = i18n.T("api.templates.invite_body.button") data.Props["SenderName"] = senderName data.Props["InviteFooterTitle"] = i18n.T("api.templates.invite_body_footer.title") data.Props["InviteFooterInfo"] = i18n.T("api.templates.invite_body_footer.info") data.Props["InviteFooterLearnMore"] = i18n.T("api.templates.invite_body_footer.learn_more") token := model.NewToken( TokenTypeTeamInvitation, model.MapToJSON(map[string]string{"teamId": team.Id, "email": invite}), ) tokenProps := make(map[string]string) tokenProps["email"] = invite tokenProps["display_name"] = team.DisplayName tokenProps["name"] = team.Name title := i18n.T("api.templates.invite_body.title", map[string]any{"SenderName": senderName, "TeamDisplayName": team.DisplayName}) if reminderData != nil { reminder := i18n.T("api.templates.invite_body.title.reminder") title = fmt.Sprintf("%s: %s", reminder, title) tokenProps["reminder_interval"] = reminderData.Interval } data.Props["Title"] = title tokenData := model.MapToJSON(tokenProps) if err := es.store.Token().Save(token); err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) continue } queryString := url.Values{} queryString.Add("d", tokenData) queryString.Add("t", token.Token) queryString.Add("md", "email") queryString.Add("sbr", es.GetTrackFlowStartedByRole(isFirstAdmin, isSystemAdmin)) data.Props["ButtonURL"] = fmt.Sprintf("%s/signup_user_complete/?%s", siteURL, queryString.Encode()) body, err := es.templatesContainer.RenderToString("invite_body", data) if err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) } if err := es.sendMail(invite, subject, body, "InviteEmail"); err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) if errorWhenNotSent { return SendMailError } } } } return nil } func (es *Service) SendGuestInviteEmails( team *model.Team, channels []*model.Channel, senderName string, senderUserId string, senderProfileImage []byte, invites []string, siteURL string, message string, errorWhenNotSent bool, isSystemAdmin bool, isFirstAdmin bool, ) error { if es.perHourEmailRateLimiter == nil { return NoRateLimiterError } rateLimited, result, err := es.perHourEmailRateLimiter.RateLimit(senderUserId, len(invites)) if err != nil { return SetupRateLimiterError } if rateLimited { mlog.Error("rate limit exceeded", mlog.Duration("RetryAfter", result.RetryAfter), mlog.Duration("ResetAfter", result.ResetAfter), mlog.String("user_id", senderUserId), mlog.String("team_id", team.Id), mlog.String("retry_after_secs", fmt.Sprintf("%f", result.RetryAfter.Seconds())), mlog.String("reset_after_secs", fmt.Sprintf("%f", result.ResetAfter.Seconds()))) return RateLimitExceededError } for _, invite := range invites { if invite != "" { subject := i18n.T("api.templates.invite_guest_subject", map[string]any{"SenderName": senderName, "TeamDisplayName": team.DisplayName, "SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData("") data.Props["SiteURL"] = siteURL data.Props["Title"] = i18n.T("api.templates.invite_body.title", map[string]any{"SenderName": senderName, "TeamDisplayName": team.DisplayName}) data.Props["SubTitle"] = i18n.T("api.templates.invite_body_guest.subTitle") data.Props["Button"] = i18n.T("api.templates.invite_body.button") data.Props["SenderName"] = senderName if message != "" { message = bluemonday.NewPolicy().Sanitize(message) } data.Props["Message"] = message data.Props["InviteFooterTitle"] = i18n.T("api.templates.invite_body_footer.title") data.Props["InviteFooterInfo"] = i18n.T("api.templates.invite_body_footer.info") data.Props["InviteFooterLearnMore"] = i18n.T("api.templates.invite_body_footer.learn_more") channelIDs := []string{} for _, channel := range channels { channelIDs = append(channelIDs, channel.Id) } token := model.NewToken( TokenTypeGuestInvitation, model.MapToJSON(map[string]string{ "teamId": team.Id, "channels": strings.Join(channelIDs, " "), "email": invite, "guest": "true", "senderId": senderUserId, }), ) tokenProps := make(map[string]string) tokenProps["email"] = invite tokenProps["display_name"] = team.DisplayName tokenProps["name"] = team.Name tokenData := model.MapToJSON(tokenProps) if err := es.store.Token().Save(token); err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) continue } data.Props["ButtonURL"] = fmt.Sprintf("%s/signup_user_complete/?d=%s&t=%s&sbr=%s", siteURL, url.QueryEscape(tokenData), url.QueryEscape(token.Token), es.GetTrackFlowStartedByRole(isFirstAdmin, isSystemAdmin)) if !*es.config().EmailSettings.SendEmailNotifications { mlog.Info("sending invitation ", mlog.String("to", invite), mlog.String("link", data.Props["ButtonURL"].(string))) } senderPhoto := "" embeddedFiles := make(map[string]io.Reader) if message != "" { if senderProfileImage != nil { senderPhoto = "user-avatar.png" embeddedFiles = map[string]io.Reader{ senderPhoto: bytes.NewReader(senderProfileImage), } } } pData := postData{ SenderName: senderName, Message: template.HTML(message), SenderPhoto: senderPhoto, } data.Props["Posts"] = []postData{pData} body, err := es.templatesContainer.RenderToString("invite_body", data) if err != nil { mlog.Error("Failed to send invite email successfully", mlog.Err(err)) } if nErr := es.SendMailWithEmbeddedFiles(invite, subject, body, embeddedFiles, "", "", "", "InviteEmail"); nErr != nil { mlog.Error("Failed to send invite email successfully", mlog.Err(nErr)) if errorWhenNotSent { return SendMailError } } } } return nil } func (es *Service) SendInviteEmailsToTeamAndChannels( team *model.Team, channels []*model.Channel, senderName string, senderUserId string, senderProfileImage []byte, invites []string, siteURL string, reminderData *model.TeamInviteReminderData, message string, errorWhenNotSent bool, isSystemAdmin bool, isFirstAdmin bool, ) ([]*model.EmailInviteWithError, error) { if es.perHourEmailRateLimiter == nil { return nil, NoRateLimiterError } rateLimited, result, err := es.perHourEmailRateLimiter.RateLimit(senderUserId, len(invites)) if err != nil { return nil, SetupRateLimiterError } if rateLimited { mlog.Error("rate limit exceeded", mlog.Duration("RetryAfter", result.RetryAfter), mlog.Duration("ResetAfter", result.ResetAfter), mlog.String("user_id", senderUserId), mlog.String("team_id", team.Id), mlog.String("retry_after_secs", fmt.Sprintf("%f", result.RetryAfter.Seconds())), mlog.String("reset_after_secs", fmt.Sprintf("%f", result.ResetAfter.Seconds()))) return nil, RateLimitExceededError } channelsLen := len(channels) subject := i18n.T("api.templates.invite_team_and_channels_subject", map[string]any{ "SenderName": senderName, "TeamDisplayName": team.DisplayName, "ChannelsLen": channelsLen, "SiteName": es.config().TeamSettings.SiteName}) title := i18n.T("api.templates.invite_team_and_channels_body.title", map[string]any{ "SenderName": senderName, "ChannelsLen": channelsLen, "TeamDisplayName": team.DisplayName}) if channelsLen == 1 { channelName := channels[0].DisplayName subject = i18n.T("api.templates.invite_team_and_channel_subject", map[string]any{"SenderName": senderName, "TeamDisplayName": team.DisplayName, "ChannelName": channelName, "SiteName": es.config().TeamSettings.SiteName}, ) title = i18n.T("api.templates.invite_team_and_channel_body.title", map[string]any{ "SenderName": senderName, "ChannelName": channelName, "TeamDisplayName": team.DisplayName, }) } var invitesWithErrors []*model.EmailInviteWithError for _, invite := range invites { if invite == "" { continue } channelIDs := []string{} for _, channel := range channels { channelIDs = append(channelIDs, channel.Id) } data := es.NewEmailTemplateData("") data.Props["SiteURL"] = siteURL data.Props["SubTitle"] = i18n.T("api.templates.invite_body.subTitle") data.Props["Button"] = i18n.T("api.templates.invite_body.button") data.Props["SenderName"] = senderName data.Props["InviteFooterTitle"] = i18n.T("api.templates.invite_body_footer.title") data.Props["InviteFooterInfo"] = i18n.T("api.templates.invite_body_footer.info") data.Props["InviteFooterLearnMore"] = i18n.T("api.templates.invite_body_footer.learn_more") if message != "" { message = bluemonday.NewPolicy().Sanitize(message) } data.Props["Message"] = message token := model.NewToken( TokenTypeTeamInvitation, model.MapToJSON(map[string]string{ "teamId": team.Id, "email": invite, "channels": strings.Join(channelIDs, " "), "senderId": senderUserId, }), ) tokenProps := make(map[string]string) tokenProps["email"] = invite tokenProps["display_name"] = team.DisplayName tokenProps["name"] = team.Name if reminderData != nil { reminder := i18n.T("api.templates.invite_body.title.reminder") title = fmt.Sprintf("%s: %s", reminder, title) tokenProps["reminder_interval"] = reminderData.Interval } data.Props["Title"] = title tokenData := model.MapToJSON(tokenProps) if err := es.store.Token().Save(token); err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) continue } data.Props["ButtonURL"] = fmt.Sprintf("%s/signup_user_complete/?d=%s&t=%s&sbr=%s", siteURL, url.QueryEscape(tokenData), url.QueryEscape(token.Token), es.GetTrackFlowStartedByRole(isFirstAdmin, isSystemAdmin)) senderPhoto := "" embeddedFiles := make(map[string]io.Reader) if message != "" { if senderProfileImage != nil { senderPhoto = "user-avatar.png" embeddedFiles = map[string]io.Reader{ senderPhoto: bytes.NewReader(senderProfileImage), } } } pData := postData{ SenderName: senderName, Message: template.HTML(message), SenderPhoto: senderPhoto, } data.Props["Posts"] = []postData{pData} body, err := es.templatesContainer.RenderToString("invite_body", data) if err != nil { mlog.Error("Failed to send invite email successfully ", mlog.Err(err)) } if nErr := es.SendMailWithEmbeddedFiles(invite, subject, body, embeddedFiles, "", "", "", "InviteEmailToTeamsAndChannels"); nErr != nil { mlog.Error("Failed to send invite email successfully", mlog.Err(nErr)) if errorWhenNotSent { inviteWithError := &model.EmailInviteWithError{ Email: invite, Error: &model.AppError{Message: nErr.Error()}, } invitesWithErrors = append(invitesWithErrors, inviteWithError) } } } return invitesWithErrors, nil } func (es *Service) NewEmailTemplateData(locale string) templates.Data { var localT i18n.TranslateFunc if locale != "" { localT = i18n.GetUserTranslations(locale) } else { localT = i18n.T } organization := "" if *es.config().EmailSettings.FeedbackOrganization != "" { organization = localT("api.templates.email_organization") + *es.config().EmailSettings.FeedbackOrganization } return templates.Data{ Props: map[string]any{ "EmailInfo1": localT("api.templates.email_info1"), "EmailInfo2": localT("api.templates.email_info2"), "EmailInfo3": localT("api.templates.email_info3", map[string]any{"SiteName": es.config().TeamSettings.SiteName}), "SupportEmail": *es.config().SupportSettings.SupportEmail, "Footer": localT("api.templates.email_footer"), "FooterV2": localT("api.templates.email_footer_v2"), "Organization": organization, }, HTML: map[string]template.HTML{}, } } func (es *Service) SendDeactivateAccountEmail(email string, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) serverURL := condenseSiteURL(siteURL) subject := T("api.templates.deactivate_subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName, "ServerURL": serverURL}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.deactivate_body.title", map[string]any{"ServerURL": serverURL}) data.Props["Info"] = T("api.templates.deactivate_body.info", map[string]any{"SiteURL": siteURL}) data.Props["Warning"] = T("api.templates.deactivate_body.warning") body, err := es.templatesContainer.RenderToString("deactivate_body", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "DeactivateAccountEmail"); err != nil { // this needs to receive the header options return err } return nil } func (es *Service) SendNotificationMail(to, subject, htmlBody string) error { if !*es.config().EmailSettings.SendEmailNotifications { return nil } return es.sendMail(to, subject, htmlBody, "NotificationEmail") } func (es *Service) sendMail(to, subject, htmlBody, category string) error { return es.sendMailWithCC(to, subject, htmlBody, "", category) } func (es *Service) sendEmailWithCustomReplyTo(to, subject, htmlBody, replyToAddress, category string) error { license := es.license() mailConfig := es.mailServiceConfig(replyToAddress) category = getSendGridCategory(category, license.IsCloud()) return mail.SendMailUsingConfig(to, subject, htmlBody, mailConfig, license != nil && *license.Features.Compliance, "", "", "", "", category) } func (es *Service) sendMailWithCC(to, subject, htmlBody, ccMail, category string) error { license := es.license() mailConfig := es.mailServiceConfig("") category = getSendGridCategory(category, license.IsCloud()) return mail.SendMailUsingConfig(to, subject, htmlBody, mailConfig, license != nil && *license.Features.Compliance, "", "", "", ccMail, category) } func (es *Service) SendMailWithEmbeddedFilesAndCustomReplyTo(to, subject, htmlBody, replyToAddress string, embeddedFiles map[string]io.Reader, category string) error { license := es.license() mailConfig := es.mailServiceConfig(replyToAddress) category = getSendGridCategory(category, license.IsCloud()) return mail.SendMailWithEmbeddedFilesUsingConfig(to, subject, htmlBody, embeddedFiles, mailConfig, license != nil && *license.Features.Compliance, "", "", "", "", category) } func (es *Service) SendMailWithEmbeddedFiles(to, subject, htmlBody string, embeddedFiles map[string]io.Reader, messageID string, inReplyTo string, references string, category string) error { license := es.license() mailConfig := es.mailServiceConfig("") category = getSendGridCategory(category, license.IsCloud()) return mail.SendMailWithEmbeddedFilesUsingConfig(to, subject, htmlBody, embeddedFiles, mailConfig, license != nil && *license.Features.Compliance, messageID, inReplyTo, references, "", category) } func (es *Service) InvalidateVerifyEmailTokensForUser(userID string) *model.AppError { tokens, err := es.store.Token().GetAllTokensByType(TokenTypeVerifyEmail) if err != nil { return model.NewAppError("InvalidateVerifyEmailTokensForUser", "api.user.invalidate_verify_email_tokens.error", nil, "", http.StatusInternalServerError).Wrap(err) } var appErr *model.AppError for _, token := range tokens { tokenExtra := struct { UserId string Email string }{} if err := json.Unmarshal([]byte(token.Extra), &tokenExtra); err != nil { appErr = model.NewAppError("InvalidateVerifyEmailTokensForUser", "api.user.invalidate_verify_email_tokens_parse.error", nil, "", http.StatusInternalServerError).Wrap(err) continue } if tokenExtra.UserId != userID { continue } if err := es.store.Token().Delete(token.Token); err != nil { appErr = model.NewAppError("InvalidateVerifyEmailTokensForUser", "api.user.invalidate_verify_email_tokens_delete.error", nil, "", http.StatusInternalServerError).Wrap(err) } } return appErr } func (es *Service) CreateVerifyEmailToken(userID string, newEmail string) (*model.Token, error) { tokenExtra := struct { UserId string Email string }{ userID, newEmail, } jsonData, err := json.Marshal(tokenExtra) if err != nil { return nil, errors.Wrap(CreateEmailTokenError, err.Error()) } token := model.NewToken(TokenTypeVerifyEmail, string(jsonData)) if err := es.InvalidateVerifyEmailTokensForUser(userID); err != nil { return nil, err } if err = es.store.Token().Save(token); err != nil { return nil, err } return token, nil } func (es *Service) SendLicenseUpForRenewalEmail(email, name, locale, siteURL, ctaTitle, ctaLink, ctaText string, daysToExpiration int) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.license_up_for_renewal_subject") data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.license_up_for_renewal_title") data.Props["SubTitle"] = T("api.templates.license_up_for_renewal_subtitle", map[string]any{"UserName": name, "Days": daysToExpiration}) data.Props["SubTitleTwo"] = ctaTitle data.Props["EmailUs"] = T("api.templates.email_us_anytime_at") data.Props["Button"] = ctaText data.Props["ButtonURL"] = ctaLink data.Props["QuestionTitle"] = T("api.templates.questions_footer.title") data.Props["SupportEmail"] = "feedback@mattermost.com" data.Props["QuestionInfo"] = T("api.templates.questions_footer.info") body, err := es.templatesContainer.RenderToString("license_up_for_renewal", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "LicenseUpForRenewal"); err != nil { return err } return nil } // SendRemoveExpiredLicenseEmail formats an email and uses the email service to send the email to user with link pointing to CWS // to renew the user license func (es *Service) SendRemoveExpiredLicenseEmail(ctaText, ctaLink, email, locale, siteURL string) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.remove_expired_license.subject", map[string]any{"SiteName": es.config().TeamSettings.SiteName}) data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.remove_expired_license.body.title") data.Props["Link"] = ctaLink data.Props["LinkButton"] = ctaText body, err := es.templatesContainer.RenderToString("remove_expired_license", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "RemoveExpiredLicense"); err != nil { return err } return nil } func (es *Service) SendIPFiltersChangedEmail(email string, initiatingUser *model.User, siteURL, portalURL, locale string, isWorkspaceOwner bool) error { T := i18n.GetUserTranslations(locale) subject := T("api.templates.ip_filters_changed.subject") data := es.NewEmailTemplateData(locale) data.Props["SiteURL"] = siteURL data.Props["Title"] = T("api.templates.ip_filters_changed.title") data.Props["SubTitle"] = T("api.templates.ip_filters_changed.subTitle", map[string]any{"InitiatingUsername": initiatingUser.Username, "SiteURL": siteURL}) data.Props["ButtonURL"] = siteURL + "/admin_console/site_config/ip_filtering" data.Props["Button"] = T("api.templates.ip_filters_changed.button") data.Props["TroubleAccessingTitle"] = T("api.templates.ip_filters_changed_footer.title") data.Props["SendAnEmailTo"] = T("api.templates.ip_filters_changed_footer.send_an_email_to", map[string]any{"InitiatingUserEmail": initiatingUser.Email}) data.Props["PortalURL"] = portalURL // If the email we're sending to was the one who initiated the change, we don't want to show their email address as a mailto if email != initiatingUser.Email { data.Props["ActorEmail"] = initiatingUser.Email } if isWorkspaceOwner { data.Props["LogInToCustomerPortal"] = T("api.templates.ip_filters_changed_footer.log_in_to_customer_portal") } data.Props["ContactSupport"] = T("api.templates.ip_filters_changed_footer.contact_support") data.Props["SupportEmail"] = *es.config().SupportSettings.SupportEmail body, err := es.templatesContainer.RenderToString("ip_filters_changed", data) if err != nil { return err } if err := es.sendMail(email, subject, body, "PasswordResetEmail"); err != nil { return err } return nil }