1
0
mirror of https://github.com/hrfee/jfa-go.git synced 2025-01-08 17:30:11 +00:00

decouple email content from sender to ensure thread safety

If two emails fired off at once, they would previously replace each
other's content and possibly send the wrong email to the wrong person.
construct* methods now return the email content, which is sent
separately.
This commit is contained in:
Harvey Tindall 2020-09-13 21:18:47 +01:00
parent 51839b5942
commit b8dfb5d6a3
Signed by: hrfee
GPG Key ID: BBC65952848FB1A2
3 changed files with 60 additions and 47 deletions

20
api.go
View File

@ -92,10 +92,11 @@ func (app *appContext) checkInvites() {
for address, settings := range notify { for address, settings := range notify {
if settings["notify-expiry"] { if settings["notify-expiry"] {
go func() { go func() {
if err := app.email.constructExpiry(code, data, app); err != nil { msg, err := app.email.constructExpiry(code, data, app)
if err != nil {
app.err.Printf("%s: Failed to construct expiry notification", code) app.err.Printf("%s: Failed to construct expiry notification", code)
app.debug.Printf("Error: %s", err) app.debug.Printf("Error: %s", err)
} else if err := app.email.send(address); err != nil { } else if err := app.email.send(address, msg); err != nil {
app.err.Printf("%s: Failed to send expiry notification", code) app.err.Printf("%s: Failed to send expiry notification", code)
app.debug.Printf("Error: %s", err) app.debug.Printf("Error: %s", err)
} else { } else {
@ -128,10 +129,11 @@ func (app *appContext) checkInvite(code string, used bool, username string) bool
for address, settings := range notify { for address, settings := range notify {
if settings["notify-expiry"] { if settings["notify-expiry"] {
go func() { go func() {
if err := app.email.constructExpiry(code, inv, app); err != nil { msg, err := app.email.constructExpiry(code, inv, app)
if err != nil {
app.err.Printf("%s: Failed to construct expiry notification", code) app.err.Printf("%s: Failed to construct expiry notification", code)
app.debug.Printf("Error: %s", err) app.debug.Printf("Error: %s", err)
} else if err := app.email.send(address); err != nil { } else if err := app.email.send(address, msg); err != nil {
app.err.Printf("%s: Failed to send expiry notification", code) app.err.Printf("%s: Failed to send expiry notification", code)
app.debug.Printf("Error: %s", err) app.debug.Printf("Error: %s", err)
} else { } else {
@ -220,10 +222,11 @@ func (app *appContext) NewUser(gc *gin.Context) {
for address, settings := range invite.Notify { for address, settings := range invite.Notify {
if settings["notify-creation"] { if settings["notify-creation"] {
go func() { go func() {
if err := app.email.constructCreated(req.Code, req.Username, req.Email, invite, app); err != nil { msg, err := app.email.constructCreated(req.Code, req.Username, req.Email, invite, app)
if err != nil {
app.err.Printf("%s: Failed to construct user creation notification", req.Code) app.err.Printf("%s: Failed to construct user creation notification", req.Code)
app.debug.Printf("%s: Error: %s", req.Code, err) app.debug.Printf("%s: Error: %s", req.Code, err)
} else if err := app.email.send(address); err != nil { } else if err := app.email.send(address, msg); err != nil {
app.err.Printf("%s: Failed to send user creation notification", req.Code) app.err.Printf("%s: Failed to send user creation notification", req.Code)
app.debug.Printf("%s: Error: %s", req.Code, err) app.debug.Printf("%s: Error: %s", req.Code, err)
} else { } else {
@ -304,11 +307,12 @@ func (app *appContext) GenerateInvite(gc *gin.Context) {
if req.Email != "" && app.config.Section("invite_emails").Key("enabled").MustBool(false) { if req.Email != "" && app.config.Section("invite_emails").Key("enabled").MustBool(false) {
app.debug.Printf("%s: Sending invite email", invite_code) app.debug.Printf("%s: Sending invite email", invite_code)
invite.Email = req.Email invite.Email = req.Email
if err := app.email.constructInvite(invite_code, invite, app); err != nil { msg, err := app.email.constructInvite(invite_code, invite, app)
if err != nil {
invite.Email = fmt.Sprintf("Failed to send to %s", req.Email) invite.Email = fmt.Sprintf("Failed to send to %s", req.Email)
app.err.Printf("%s: Failed to construct invite email", invite_code) app.err.Printf("%s: Failed to construct invite email", invite_code)
app.debug.Printf("%s: Error: %s", invite_code, err) app.debug.Printf("%s: Error: %s", invite_code, err)
} else if err := app.email.send(req.Email); err != nil { } else if err := app.email.send(req.Email, msg); err != nil {
invite.Email = fmt.Sprintf("Failed to send to %s", req.Email) invite.Email = fmt.Sprintf("Failed to send to %s", req.Email)
app.err.Printf("%s: %s", invite_code, invite.Email) app.err.Printf("%s: %s", invite_code, invite.Email)
app.debug.Printf("%s: Error: %s", invite_code, err) app.debug.Printf("%s: Error: %s", invite_code, err)

View File

@ -17,14 +17,14 @@ import (
// implements email sending, right now via smtp or mailgun. // implements email sending, right now via smtp or mailgun.
type emailClient interface { type emailClient interface {
send(address, fromName, fromAddr string, email email) error send(address, fromName, fromAddr string, email *Email) error
} }
type Mailgun struct { type Mailgun struct {
client *mailgun.MailgunImpl client *mailgun.MailgunImpl
} }
func (mg *Mailgun) send(address, fromName, fromAddr string, email email) error { func (mg *Mailgun) send(address, fromName, fromAddr string, email *Email) error {
message := mg.client.NewMessage( message := mg.client.NewMessage(
fmt.Sprintf("%s <%s>", fromName, fromAddr), fmt.Sprintf("%s <%s>", fromName, fromAddr),
email.subject, email.subject,
@ -45,7 +45,7 @@ type Smtp struct {
auth smtp.Auth auth smtp.Auth
} }
func (sm *Smtp) send(address, fromName, fromAddr string, email email) error { func (sm *Smtp) send(address, fromName, fromAddr string, email *Email) error {
e := jEmail.NewEmail() e := jEmail.NewEmail()
e.Subject = email.subject e.Subject = email.subject
e.From = fmt.Sprintf("%s <%s>", fromName, fromAddr) e.From = fmt.Sprintf("%s <%s>", fromName, fromAddr)
@ -68,12 +68,12 @@ func (sm *Smtp) send(address, fromName, fromAddr string, email email) error {
// Emailer contains the email sender, email content, and methods to construct message content. // Emailer contains the email sender, email content, and methods to construct message content.
type Emailer struct { type Emailer struct {
content email
fromAddr, fromName string fromAddr, fromName string
sender emailClient sender emailClient
} }
type email struct { // Email stores content.
type Email struct {
subject string subject string
html, text string html, text string
} }
@ -140,10 +140,12 @@ func (emailer *Emailer) NewSMTP(server string, port int, password, host string,
} }
} }
func (email *Emailer) constructInvite(code string, invite Invite, app *appContext) error { func (emailer *Emailer) constructInvite(code string, invite Invite, app *appContext) (*Email, error) {
email.content.subject = app.config.Section("invite_emails").Key("subject").String() email := &Email{
subject: app.config.Section("invite_emails").Key("subject").String(),
}
expiry := invite.ValidTill expiry := invite.ValidTill
d, t, expires_in := email.formatExpiry(expiry, false, app.datePattern, app.timePattern) d, t, expires_in := emailer.formatExpiry(expiry, false, app.datePattern, app.timePattern)
message := app.config.Section("email").Key("message").String() message := app.config.Section("email").Key("message").String()
invite_link := app.config.Section("invite_emails").Key("url_base").String() invite_link := app.config.Section("invite_emails").Key("url_base").String()
invite_link = fmt.Sprintf("%s/%s", invite_link, code) invite_link = fmt.Sprintf("%s/%s", invite_link, code)
@ -152,7 +154,7 @@ func (email *Emailer) constructInvite(code string, invite Invite, app *appContex
fpath := app.config.Section("invite_emails").Key("email_" + key).String() fpath := app.config.Section("invite_emails").Key("email_" + key).String()
tpl, err := template.ParseFiles(fpath) tpl, err := template.ParseFiles(fpath)
if err != nil { if err != nil {
return err return nil, err
} }
var tplData bytes.Buffer var tplData bytes.Buffer
err = tpl.Execute(&tplData, map[string]string{ err = tpl.Execute(&tplData, map[string]string{
@ -163,25 +165,27 @@ func (email *Emailer) constructInvite(code string, invite Invite, app *appContex
"message": message, "message": message,
}) })
if err != nil { if err != nil {
return err return nil, err
} }
if key == "html" { if key == "html" {
email.content.html = tplData.String() email.html = tplData.String()
} else { } else {
email.content.text = tplData.String() email.text = tplData.String()
} }
} }
return nil return email, nil
} }
func (email *Emailer) constructExpiry(code string, invite Invite, app *appContext) error { func (emailer *Emailer) constructExpiry(code string, invite Invite, app *appContext) (*Email, error) {
email.content.subject = "Notice: Invite expired" email := &Email{
subject: "Notice: Invite expired",
}
expiry := app.formatDatetime(invite.ValidTill) expiry := app.formatDatetime(invite.ValidTill)
for _, key := range []string{"html", "text"} { for _, key := range []string{"html", "text"} {
fpath := app.config.Section("notifications").Key("expiry_" + key).String() fpath := app.config.Section("notifications").Key("expiry_" + key).String()
tpl, err := template.ParseFiles(fpath) tpl, err := template.ParseFiles(fpath)
if err != nil { if err != nil {
return err return nil, err
} }
var tplData bytes.Buffer var tplData bytes.Buffer
err = tpl.Execute(&tplData, map[string]string{ err = tpl.Execute(&tplData, map[string]string{
@ -189,19 +193,21 @@ func (email *Emailer) constructExpiry(code string, invite Invite, app *appContex
"expiry": expiry, "expiry": expiry,
}) })
if err != nil { if err != nil {
return err return nil, err
} }
if key == "html" { if key == "html" {
email.content.html = tplData.String() email.html = tplData.String()
} else { } else {
email.content.text = tplData.String() email.text = tplData.String()
} }
} }
return nil return email, nil
} }
func (email *Emailer) constructCreated(code, username, address string, invite Invite, app *appContext) error { func (emailer *Emailer) constructCreated(code, username, address string, invite Invite, app *appContext) (*Email, error) {
email.content.subject = "Notice: User created" email := &Email{
subject: "Notice: User created",
}
created := app.formatDatetime(invite.Created) created := app.formatDatetime(invite.Created)
var tplAddress string var tplAddress string
if app.config.Section("email").Key("no_username").MustBool(false) { if app.config.Section("email").Key("no_username").MustBool(false) {
@ -213,7 +219,7 @@ func (email *Emailer) constructCreated(code, username, address string, invite In
fpath := app.config.Section("notifications").Key("created_" + key).String() fpath := app.config.Section("notifications").Key("created_" + key).String()
tpl, err := template.ParseFiles(fpath) tpl, err := template.ParseFiles(fpath)
if err != nil { if err != nil {
return err return nil, err
} }
var tplData bytes.Buffer var tplData bytes.Buffer
err = tpl.Execute(&tplData, map[string]string{ err = tpl.Execute(&tplData, map[string]string{
@ -223,26 +229,28 @@ func (email *Emailer) constructCreated(code, username, address string, invite In
"time": created, "time": created,
}) })
if err != nil { if err != nil {
return err return nil, err
} }
if key == "html" { if key == "html" {
email.content.html = tplData.String() email.html = tplData.String()
} else { } else {
email.content.text = tplData.String() email.text = tplData.String()
} }
} }
return nil return email, nil
} }
func (email *Emailer) constructReset(pwr Pwr, app *appContext) error { func (emailer *Emailer) constructReset(pwr Pwr, app *appContext) (*Email, error) {
email.content.subject = app.config.Section("password_resets").Key("subject").MustString("Password reset - Jellyfin") email := &Email{
d, t, expires_in := email.formatExpiry(pwr.Expiry, true, app.datePattern, app.timePattern) subject: app.config.Section("password_resets").Key("subject").MustString("Password reset - Jellyfin"),
}
d, t, expires_in := emailer.formatExpiry(pwr.Expiry, true, app.datePattern, app.timePattern)
message := app.config.Section("email").Key("message").String() message := app.config.Section("email").Key("message").String()
for _, key := range []string{"html", "text"} { for _, key := range []string{"html", "text"} {
fpath := app.config.Section("password_resets").Key("email_" + key).String() fpath := app.config.Section("password_resets").Key("email_" + key).String()
tpl, err := template.ParseFiles(fpath) tpl, err := template.ParseFiles(fpath)
if err != nil { if err != nil {
return err return nil, err
} }
var tplData bytes.Buffer var tplData bytes.Buffer
err = tpl.Execute(&tplData, map[string]string{ err = tpl.Execute(&tplData, map[string]string{
@ -254,17 +262,17 @@ func (email *Emailer) constructReset(pwr Pwr, app *appContext) error {
"message": message, "message": message,
}) })
if err != nil { if err != nil {
return err return nil, err
} }
if key == "html" { if key == "html" {
email.content.html = tplData.String() email.html = tplData.String()
} else { } else {
email.content.text = tplData.String() email.text = tplData.String()
} }
} }
return nil return email, nil
} }
func (emailer *Emailer) send(address string) error { func (emailer *Emailer) send(address string, email *Email) error {
return emailer.sender.send(address, emailer.fromName, emailer.fromAddr, emailer.content) return emailer.sender.send(address, emailer.fromName, emailer.fromAddr, email)
} }

View File

@ -71,10 +71,11 @@ func pwrMonitor(app *appContext, watcher *fsnotify.Watcher) {
app.err.Printf("Couldn't find email for user \"%s\". Make sure it's set", pwr.Username) app.err.Printf("Couldn't find email for user \"%s\". Make sure it's set", pwr.Username)
return return
} }
if err := app.email.constructReset(pwr, app); err != nil { msg, err := app.email.constructReset(pwr, app)
if err != nil {
app.err.Printf("Failed to construct password reset email for %s", pwr.Username) app.err.Printf("Failed to construct password reset email for %s", pwr.Username)
app.debug.Printf("%s: Error: %s", pwr.Username, err) app.debug.Printf("%s: Error: %s", pwr.Username, err)
} else if err := app.email.send(address); err != nil { } else if err := app.email.send(address, msg); err != nil {
app.err.Printf("Failed to send password reset email to \"%s\"", address) app.err.Printf("Failed to send password reset email to \"%s\"", address)
app.debug.Printf("%s: Error: %s", pwr.Username, err) app.debug.Printf("%s: Error: %s", pwr.Username, err)
} else { } else {