2022-07-07 00:19:05 +03:00
|
|
|
package mailer
|
|
|
|
|
|
|
|
import (
|
2022-12-13 11:45:59 +02:00
|
|
|
"errors"
|
2022-07-07 00:19:05 +03:00
|
|
|
"fmt"
|
|
|
|
"net/smtp"
|
2022-11-21 14:53:05 +02:00
|
|
|
"strings"
|
2022-07-07 00:19:05 +03:00
|
|
|
|
|
|
|
"github.com/domodwyer/mailyak/v3"
|
2022-11-21 14:53:05 +02:00
|
|
|
"github.com/pocketbase/pocketbase/tools/security"
|
2022-07-07 00:19:05 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
var _ Mailer = (*SmtpClient)(nil)
|
|
|
|
|
2022-12-13 11:45:59 +02:00
|
|
|
const (
|
|
|
|
SmtpAuthPlain = "PLAIN"
|
|
|
|
SmtpAuthLogin = "LOGIN"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Deprecated: Use directly the SmtpClient struct literal.
|
|
|
|
//
|
|
|
|
// NewSmtpClient creates new SmtpClient with the provided configuration.
|
2022-07-07 00:19:05 +03:00
|
|
|
func NewSmtpClient(
|
|
|
|
host string,
|
|
|
|
port int,
|
|
|
|
username string,
|
|
|
|
password string,
|
|
|
|
tls bool,
|
|
|
|
) *SmtpClient {
|
|
|
|
return &SmtpClient{
|
2022-12-13 11:45:59 +02:00
|
|
|
Host: host,
|
|
|
|
Port: port,
|
|
|
|
Username: username,
|
|
|
|
Password: password,
|
|
|
|
Tls: tls,
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SmtpClient defines a SMTP mail client structure that implements
|
|
|
|
// `mailer.Mailer` interface.
|
|
|
|
type SmtpClient struct {
|
2022-12-13 11:45:59 +02:00
|
|
|
Host string
|
|
|
|
Port int
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
Tls bool
|
|
|
|
AuthMethod string // default to "PLAIN"
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Send implements `mailer.Mailer` interface.
|
2022-11-21 14:53:05 +02:00
|
|
|
func (c *SmtpClient) Send(m *Message) error {
|
2022-10-30 10:28:14 +02:00
|
|
|
var smtpAuth smtp.Auth
|
2022-12-13 11:45:59 +02:00
|
|
|
if c.Username != "" || c.Password != "" {
|
|
|
|
switch c.AuthMethod {
|
|
|
|
case SmtpAuthLogin:
|
|
|
|
smtpAuth = &smtpLoginAuth{c.Username, c.Password}
|
|
|
|
default:
|
|
|
|
smtpAuth = smtp.PlainAuth("", c.Username, c.Password, c.Host)
|
|
|
|
}
|
2022-10-30 10:28:14 +02:00
|
|
|
}
|
2022-07-07 00:19:05 +03:00
|
|
|
|
|
|
|
// create mail instance
|
|
|
|
var yak *mailyak.MailYak
|
2022-12-13 11:45:59 +02:00
|
|
|
if c.Tls {
|
2022-07-07 00:19:05 +03:00
|
|
|
var tlsErr error
|
2022-12-13 11:45:59 +02:00
|
|
|
yak, tlsErr = mailyak.NewWithTLS(fmt.Sprintf("%s:%d", c.Host, c.Port), smtpAuth, nil)
|
2022-07-07 00:19:05 +03:00
|
|
|
if tlsErr != nil {
|
|
|
|
return tlsErr
|
|
|
|
}
|
|
|
|
} else {
|
2022-12-13 11:45:59 +02:00
|
|
|
yak = mailyak.New(fmt.Sprintf("%s:%d", c.Host, c.Port), smtpAuth)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
2022-11-21 14:53:05 +02:00
|
|
|
if m.From.Name != "" {
|
|
|
|
yak.FromName(m.From.Name)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
2022-11-21 14:53:05 +02:00
|
|
|
yak.From(m.From.Address)
|
|
|
|
yak.Subject(m.Subject)
|
|
|
|
yak.HTML().Set(m.HTML)
|
2022-07-07 00:19:05 +03:00
|
|
|
|
2022-11-21 14:53:05 +02:00
|
|
|
if m.Text == "" {
|
|
|
|
// try to generate a plain text version of the HTML
|
|
|
|
if plain, err := html2Text(m.HTML); err == nil {
|
|
|
|
yak.Plain().Set(plain)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
yak.Plain().Set(m.Text)
|
|
|
|
}
|
|
|
|
|
2023-02-01 22:07:46 +02:00
|
|
|
if len(m.To) > 0 {
|
|
|
|
yak.To(addressesToStrings(m.To, true)...)
|
|
|
|
}
|
|
|
|
|
2022-11-21 14:53:05 +02:00
|
|
|
if len(m.Bcc) > 0 {
|
2023-02-01 22:07:46 +02:00
|
|
|
yak.Bcc(addressesToStrings(m.Bcc, true)...)
|
2022-11-21 14:53:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(m.Cc) > 0 {
|
2023-02-01 22:07:46 +02:00
|
|
|
yak.Cc(addressesToStrings(m.Cc, true)...)
|
2022-08-26 06:46:34 +03:00
|
|
|
}
|
|
|
|
|
2022-11-21 14:53:05 +02:00
|
|
|
// add attachements (if any)
|
|
|
|
for name, data := range m.Attachments {
|
2022-07-07 00:19:05 +03:00
|
|
|
yak.Attach(name, data)
|
|
|
|
}
|
|
|
|
|
2022-11-21 14:53:05 +02:00
|
|
|
// add custom headers (if any)
|
|
|
|
var hasMessageId bool
|
|
|
|
for k, v := range m.Headers {
|
|
|
|
if strings.EqualFold(k, "Message-ID") {
|
|
|
|
hasMessageId = true
|
|
|
|
}
|
|
|
|
yak.AddHeader(k, v)
|
|
|
|
}
|
|
|
|
if !hasMessageId {
|
|
|
|
// add a default message id if missing
|
|
|
|
fromParts := strings.Split(m.From.Address, "@")
|
|
|
|
if len(fromParts) == 2 {
|
|
|
|
yak.AddHeader("Message-ID", fmt.Sprintf("<%s@%s>",
|
|
|
|
security.PseudorandomString(15),
|
|
|
|
fromParts[1],
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-07 00:19:05 +03:00
|
|
|
return yak.Send()
|
|
|
|
}
|
2022-12-13 11:45:59 +02:00
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
// AUTH LOGIN
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
|
|
|
|
var _ smtp.Auth = (*smtpLoginAuth)(nil)
|
|
|
|
|
|
|
|
// smtpLoginAuth defines an AUTH that implements the LOGIN authentication mechanism.
|
|
|
|
//
|
|
|
|
// AUTH LOGIN is obsolete[1] but some mail services like outlook requires it [2].
|
|
|
|
//
|
|
|
|
// NB!
|
|
|
|
// It will only send the credentials if the connection is using TLS or is connected to localhost.
|
|
|
|
// Otherwise authentication will fail with an error, without sending the credentials.
|
|
|
|
//
|
|
|
|
// [1]: https://github.com/golang/go/issues/40817
|
|
|
|
// [2]: https://support.microsoft.com/en-us/office/outlook-com-no-longer-supports-auth-plain-authentication-07f7d5e9-1697-465f-84d2-4513d4ff0145?ui=en-us&rs=en-us&ad=us
|
|
|
|
type smtpLoginAuth struct {
|
|
|
|
username, password string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start initializes an authentication with the server.
|
|
|
|
//
|
|
|
|
// It is part of the [smtp.Auth] interface.
|
|
|
|
func (a *smtpLoginAuth) Start(server *smtp.ServerInfo) (string, []byte, error) {
|
|
|
|
// Must have TLS, or else localhost server.
|
|
|
|
// Note: If TLS is not true, then we can't trust ANYTHING in ServerInfo.
|
|
|
|
// In particular, it doesn't matter if the server advertises LOGIN auth.
|
|
|
|
// That might just be the attacker saying
|
|
|
|
// "it's ok, you can trust me with your password."
|
|
|
|
if !server.TLS && !isLocalhost(server.Name) {
|
|
|
|
return "", nil, errors.New("unencrypted connection")
|
|
|
|
}
|
|
|
|
|
|
|
|
return "LOGIN", nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next "continues" the auth process by feeding the server with the requested data.
|
|
|
|
//
|
|
|
|
// It is part of the [smtp.Auth] interface.
|
|
|
|
func (a *smtpLoginAuth) Next(fromServer []byte, more bool) ([]byte, error) {
|
|
|
|
if more {
|
|
|
|
switch strings.ToLower(string(fromServer)) {
|
|
|
|
case "username:":
|
|
|
|
return []byte(a.username), nil
|
|
|
|
case "password:":
|
|
|
|
return []byte(a.password), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func isLocalhost(name string) bool {
|
|
|
|
return name == "localhost" || name == "127.0.0.1" || name == "::1"
|
|
|
|
}
|