mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 07:15:55 +03:00
8e6fe7459b
both when parsing our configs, and for incoming on smtp or in messages. so we properly compare things like é and e+accent as equal, and accept the different encodings of that same address.
94 lines
3 KiB
Go
94 lines
3 KiB
Go
package mox
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/mjl-/mox/config"
|
|
"github.com/mjl-/mox/dns"
|
|
"github.com/mjl-/mox/smtp"
|
|
)
|
|
|
|
var (
|
|
ErrDomainNotFound = errors.New("domain not found")
|
|
ErrAccountNotFound = errors.New("account not found")
|
|
)
|
|
|
|
// FindAccount looks up the account for localpart and domain.
|
|
//
|
|
// Can return ErrDomainNotFound and ErrAccountNotFound.
|
|
func FindAccount(localpart smtp.Localpart, domain dns.Domain, allowPostmaster bool) (accountName string, canonicalAddress string, dest config.Destination, rerr error) {
|
|
if strings.EqualFold(string(localpart), "postmaster") {
|
|
localpart = "postmaster"
|
|
}
|
|
|
|
postmasterDomain := func() bool {
|
|
var zerodomain dns.Domain
|
|
if domain == zerodomain || domain == Conf.Static.HostnameDomain {
|
|
return true
|
|
}
|
|
for _, l := range Conf.Static.Listeners {
|
|
if l.SMTP.Enabled && domain == l.HostnameDomain {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check for special mail host addresses.
|
|
if localpart == "postmaster" && postmasterDomain() {
|
|
if !allowPostmaster {
|
|
return "", "", config.Destination{}, ErrAccountNotFound
|
|
}
|
|
return Conf.Static.Postmaster.Account, "postmaster", config.Destination{Mailbox: Conf.Static.Postmaster.Mailbox}, nil
|
|
}
|
|
if localpart == Conf.Static.HostTLSRPT.ParsedLocalpart && domain == Conf.Static.HostnameDomain {
|
|
// Get destination, should always be present.
|
|
canonical := smtp.NewAddress(localpart, domain).String()
|
|
accAddr, ok := Conf.AccountDestination(canonical)
|
|
if !ok {
|
|
return "", "", config.Destination{}, ErrAccountNotFound
|
|
}
|
|
return accAddr.Account, canonical, accAddr.Destination, nil
|
|
}
|
|
|
|
d, ok := Conf.Domain(domain)
|
|
if !ok || d.ReportsOnly {
|
|
// For ReportsOnly, we also return ErrDomainNotFound, so this domain isn't
|
|
// considered local/authoritative during delivery.
|
|
return "", "", config.Destination{}, ErrDomainNotFound
|
|
}
|
|
|
|
localpart, err := CanonicalLocalpart(localpart, d)
|
|
if err != nil {
|
|
return "", "", config.Destination{}, fmt.Errorf("%w: %s", ErrAccountNotFound, err)
|
|
}
|
|
canonical := smtp.NewAddress(localpart, domain).String()
|
|
|
|
accAddr, ok := Conf.AccountDestination(canonical)
|
|
if !ok {
|
|
if accAddr, ok = Conf.AccountDestination("@" + domain.Name()); !ok {
|
|
if localpart == "postmaster" && allowPostmaster {
|
|
return Conf.Static.Postmaster.Account, "postmaster", config.Destination{Mailbox: Conf.Static.Postmaster.Mailbox}, nil
|
|
}
|
|
return "", "", config.Destination{}, ErrAccountNotFound
|
|
}
|
|
canonical = "@" + domain.Name()
|
|
}
|
|
return accAddr.Account, canonical, accAddr.Destination, nil
|
|
}
|
|
|
|
// CanonicalLocalpart returns the canonical localpart, removing optional catchall
|
|
// separator, and optionally lower-casing the string.
|
|
func CanonicalLocalpart(localpart smtp.Localpart, d config.Domain) (smtp.Localpart, error) {
|
|
if d.LocalpartCatchallSeparator != "" {
|
|
t := strings.SplitN(string(localpart), d.LocalpartCatchallSeparator, 2)
|
|
localpart = smtp.Localpart(t[0])
|
|
}
|
|
|
|
if !d.LocalpartCaseSensitive {
|
|
localpart = smtp.Localpart(strings.ToLower(string(localpart)))
|
|
}
|
|
return localpart, nil
|
|
}
|