1
0
Fork 0
mirror of https://github.com/mjl-/mox.git synced 2025-01-15 18:06:27 +03:00
mox/smtp/path.go
Mechiel Lukkien 5742ed1537
when logging email addresses with IDNA domain and/or special characters or utf8 in localpart, log both native utf8 form and form with escape localpart and ascii-only domain
the idea is to make it clear from the logging if non-ascii characters are used.

this is implemented by making mlog recognize if a field value that will be
logged has a LogString method. if so, that value is logged. dns.Domain,
smtp.Address, smtp.Localpart, smtp.Path now have a LogString method.

some explicit calls to String have been replaced to LogString, and some %q
formatting have been replaced with %s, because the escaped localpart would
already have double quotes, and double doublequotes aren't easy to read.
2023-03-09 20:18:34 +01:00

86 lines
2.1 KiB
Go

package smtp
import (
"strconv"
"strings"
"github.com/mjl-/mox/dns"
)
// Path is an SMTP forward/reverse path, as used in MAIL FROM and RCPT TO
// commands.
type Path struct {
Localpart Localpart
IPDomain dns.IPDomain
}
func (p Path) IsZero() bool {
return p.Localpart == "" && p.IPDomain.IsZero()
}
// String returns a string representation with ASCII-only domain name.
func (p Path) String() string {
return p.XString(false)
}
// LogString returns both the ASCII-only and optional UTF-8 representation.
func (p Path) LogString() string {
if p.Localpart == "" && p.IPDomain.IsZero() {
return ""
}
s := p.XString(true)
lp := p.Localpart.String()
qlp := strconv.QuoteToASCII(lp)
escaped := qlp != `"`+lp+`"`
if p.IPDomain.Domain.Unicode != "" || escaped {
if escaped {
lp = qlp
}
s += "/" + lp + "@" + p.IPDomain.XString(false)
}
return s
}
// XString is like String, but returns unicode UTF-8 domain names if utf8 is
// true.
func (p Path) XString(utf8 bool) string {
if p.Localpart == "" && p.IPDomain.IsZero() {
return ""
}
return p.Localpart.String() + "@" + p.IPDomain.XString(utf8)
}
// ASCIIExtra returns an ascii-only path if utf8 is true and the ipdomain is a
// unicode domain. Otherwise returns an empty string.
//
// For use in comments in message headers added during SMTP.
func (p Path) ASCIIExtra(utf8 bool) string {
if utf8 && p.IPDomain.Domain.Unicode != "" {
return p.XString(false)
}
return ""
}
// DSNString returns a string representation as used with DSN with/without
// UTF-8 support.
//
// If utf8 is false, the domain is represented as US-ASCII (IDNA), and the
// localpart is encoded with in 7bit according to RFC 6533.
func (p Path) DSNString(utf8 bool) string {
if utf8 {
return p.XString(utf8)
}
return p.Localpart.DSNString(utf8) + "@" + p.IPDomain.XString(utf8)
}
func (p Path) Equal(o Path) bool {
if p.Localpart != o.Localpart {
return false
}
d0 := p.IPDomain
d1 := o.IPDomain
if len(d0.IP) > 0 || len(d1.IP) > 0 {
return d0.IP.Equal(d1.IP)
}
return strings.EqualFold(d0.Domain.ASCII, d1.Domain.ASCII)
}