maddy/internal/endpoint/smtp/submission.go
fox.cpp bf188e454f
Move most code from the repo root into subdirectories
The intention is to keep to repo root clean while the list of packages
is slowly growing.

Additionally, a bunch of small (~30 LoC) files in the repo root is
merged into a single maddy.go file, for the same reason.

Most of the internal code is moved into the internal/ directory. Go
toolchain will make it impossible to import these packages from external
applications.

Some packages are renamed and moved into the pkg/ directory in the root.
According to https://github.com/golang-standards/project-layout this is
the de-facto standard to place "library code that's ok to use by
external applications" in.

To clearly define the purpose of top-level directories, README.md files
are added to each.
2019-12-06 01:35:12 +03:00

130 lines
3.2 KiB
Go

package smtp
import (
"errors"
"fmt"
"net/mail"
"time"
"github.com/emersion/go-message/textproto"
"github.com/foxcpp/maddy/internal/exterrors"
"github.com/foxcpp/maddy/internal/module"
"github.com/google/uuid"
)
var (
msgIDField = func() (string, error) {
id, err := uuid.NewRandom()
if err != nil {
return "", err
}
return id.String(), nil
}
now = time.Now
)
func (s *Session) submissionPrepare(msgMeta *module.MsgMetadata, header *textproto.Header) error {
msgMeta.DontTraceSender = true
if header.Get("Message-ID") == "" {
msgId, err := msgIDField()
if err != nil {
return errors.New("Message-ID generation failed")
}
s.log.Msg("adding missing Message-ID")
header.Set("Message-ID", "<"+msgId+"@"+s.endp.serv.Domain+">")
}
if header.Get("From") == "" {
return &exterrors.SMTPError{
Code: 554,
EnhancedCode: exterrors.EnhancedCode{5, 6, 0},
Message: "Message does not contains a From header field",
Misc: map[string]interface{}{
"modifier": "submission_prepare",
},
}
}
for _, hdr := range [...]string{"Sender"} {
if value := header.Get(hdr); value != "" {
if _, err := mail.ParseAddress(value); err != nil {
return &exterrors.SMTPError{
Code: 554,
EnhancedCode: exterrors.EnhancedCode{5, 6, 0},
Message: fmt.Sprintf("Invalid address in %s", hdr),
Misc: map[string]interface{}{
"modifier": "submission_prepare",
"addr": value,
},
Err: err,
}
}
}
}
for _, hdr := range [...]string{"To", "Cc", "Bcc", "Reply-To"} {
if value := header.Get(hdr); value != "" {
if _, err := mail.ParseAddressList(value); err != nil {
return &exterrors.SMTPError{
Code: 554,
EnhancedCode: exterrors.EnhancedCode{5, 6, 0},
Message: fmt.Sprintf("Invalid address in %s", hdr),
Misc: map[string]interface{}{
"modifier": "submission_prepare",
"addr": value,
},
Err: err,
}
}
}
}
addrs, err := mail.ParseAddressList(header.Get("From"))
if err != nil {
return &exterrors.SMTPError{
Code: 554,
EnhancedCode: exterrors.EnhancedCode{5, 6, 0},
Message: fmt.Sprintf("Invalid address in From"),
Misc: map[string]interface{}{
"modifier": "submission_prepare",
"addr": header.Get("From"),
},
Err: err,
}
}
// https://tools.ietf.org/html/rfc5322#section-3.6.2
// If From contains multiple addresses, Sender field must be present.
if len(addrs) > 1 && header.Get("Sender") == "" {
return &exterrors.SMTPError{
Code: 554,
EnhancedCode: exterrors.EnhancedCode{5, 6, 0},
Message: "Missing Sender header field",
Misc: map[string]interface{}{
"modifier": "submission_prepare",
"from": header.Get("From"),
},
}
}
if dateHdr := header.Get("Date"); dateHdr != "" {
_, err := parseMessageDateTime(dateHdr)
if err != nil {
return &exterrors.SMTPError{
Code: 554,
Message: "Malformed Date header",
Misc: map[string]interface{}{
"modifier": "submission_prepare",
"date": dateHdr,
},
Err: err,
}
}
} else {
s.log.Msg("adding missing Date header")
header.Set("Date", now().UTC().Format("Mon, 2 Jan 2006 15:04:05 -0700"))
}
return nil
}