215 lines
4.4 KiB
Go
215 lines
4.4 KiB
Go
package smtp
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
)
|
|
|
|
// cutPrefixFold is a version of strings.CutPrefix which is case-insensitive.
|
|
func cutPrefixFold(s, prefix string) (string, bool) {
|
|
if len(s) < len(prefix) || !strings.EqualFold(s[:len(prefix)], prefix) {
|
|
return "", false
|
|
}
|
|
return s[len(prefix):], true
|
|
}
|
|
|
|
func parseCmd(line string) (cmd string, arg string, err error) {
|
|
line = strings.TrimRight(line, "\r\n")
|
|
|
|
l := len(line)
|
|
switch {
|
|
case strings.HasPrefix(strings.ToUpper(line), "STARTTLS"):
|
|
return "STARTTLS", "", nil
|
|
case l == 0:
|
|
return "", "", nil
|
|
case l < 4:
|
|
return "", "", fmt.Errorf("command too short: %q", line)
|
|
case l == 4:
|
|
return strings.ToUpper(line), "", nil
|
|
case l == 5:
|
|
// Too long to be only command, too short to have args
|
|
return "", "", fmt.Errorf("mangled command: %q", line)
|
|
}
|
|
|
|
// If we made it here, command is long enough to have args
|
|
if line[4] != ' ' {
|
|
// There wasn't a space after the command?
|
|
return "", "", fmt.Errorf("mangled command: %q", line)
|
|
}
|
|
|
|
return strings.ToUpper(line[0:4]), strings.TrimSpace(line[5:]), nil
|
|
}
|
|
|
|
// Takes the arguments proceeding a command and files them
|
|
// into a map[string]string after uppercasing each key. Sample arg
|
|
// string:
|
|
//
|
|
// " BODY=8BITMIME SIZE=1024 SMTPUTF8"
|
|
//
|
|
// The leading space is mandatory.
|
|
func parseArgs(s string) (map[string]string, error) {
|
|
argMap := map[string]string{}
|
|
for _, arg := range strings.Fields(s) {
|
|
m := strings.Split(arg, "=")
|
|
switch len(m) {
|
|
case 2:
|
|
argMap[strings.ToUpper(m[0])] = m[1]
|
|
case 1:
|
|
argMap[strings.ToUpper(m[0])] = ""
|
|
default:
|
|
return nil, fmt.Errorf("failed to parse arg string: %q", arg)
|
|
}
|
|
}
|
|
return argMap, nil
|
|
}
|
|
|
|
func parseHelloArgument(arg string) (string, error) {
|
|
domain := arg
|
|
if idx := strings.IndexRune(arg, ' '); idx >= 0 {
|
|
domain = arg[:idx]
|
|
}
|
|
if domain == "" {
|
|
return "", fmt.Errorf("invalid domain")
|
|
}
|
|
return domain, nil
|
|
}
|
|
|
|
// parser parses command arguments defined in RFC 5321 section 4.1.2.
|
|
type parser struct {
|
|
s string
|
|
}
|
|
|
|
func (p *parser) peekByte() (byte, bool) {
|
|
if len(p.s) == 0 {
|
|
return 0, false
|
|
}
|
|
return p.s[0], true
|
|
}
|
|
|
|
func (p *parser) readByte() (byte, bool) {
|
|
ch, ok := p.peekByte()
|
|
if ok {
|
|
p.s = p.s[1:]
|
|
}
|
|
return ch, ok
|
|
}
|
|
|
|
func (p *parser) acceptByte(ch byte) bool {
|
|
got, ok := p.peekByte()
|
|
if !ok || got != ch {
|
|
return false
|
|
}
|
|
p.readByte()
|
|
return true
|
|
}
|
|
|
|
func (p *parser) expectByte(ch byte) error {
|
|
if !p.acceptByte(ch) {
|
|
if len(p.s) == 0 {
|
|
return fmt.Errorf("expected '%v', got EOF", string(ch))
|
|
} else {
|
|
return fmt.Errorf("expected '%v', got '%v'", string(ch), string(p.s[0]))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *parser) parseReversePath() (string, error) {
|
|
if strings.HasPrefix(p.s, "<>") {
|
|
p.s = strings.TrimPrefix(p.s, "<>")
|
|
return "", nil
|
|
}
|
|
return p.parsePath()
|
|
}
|
|
|
|
func (p *parser) parsePath() (string, error) {
|
|
hasBracket := p.acceptByte('<')
|
|
if p.acceptByte('@') {
|
|
i := strings.IndexByte(p.s, ':')
|
|
if i < 0 {
|
|
return "", fmt.Errorf("malformed a-d-l")
|
|
}
|
|
p.s = p.s[i+1:]
|
|
}
|
|
mbox, err := p.parseMailbox()
|
|
if err != nil {
|
|
return "", fmt.Errorf("in mailbox: %v", err)
|
|
}
|
|
if hasBracket {
|
|
if err := p.expectByte('>'); err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
return mbox, nil
|
|
}
|
|
|
|
func (p *parser) parseMailbox() (string, error) {
|
|
localPart, err := p.parseLocalPart()
|
|
if err != nil {
|
|
return "", fmt.Errorf("in local-part: %v", err)
|
|
} else if localPart == "" {
|
|
return "", fmt.Errorf("local-part is empty")
|
|
}
|
|
|
|
if err := p.expectByte('@'); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var sb strings.Builder
|
|
sb.WriteString(localPart)
|
|
sb.WriteByte('@')
|
|
|
|
for {
|
|
ch, ok := p.peekByte()
|
|
if !ok {
|
|
break
|
|
}
|
|
if ch == ' ' || ch == '\t' || ch == '>' {
|
|
break
|
|
}
|
|
p.readByte()
|
|
sb.WriteByte(ch)
|
|
}
|
|
|
|
if strings.HasSuffix(sb.String(), "@") {
|
|
return "", fmt.Errorf("domain is empty")
|
|
}
|
|
|
|
return sb.String(), nil
|
|
}
|
|
|
|
func (p *parser) parseLocalPart() (string, error) {
|
|
var sb strings.Builder
|
|
|
|
if p.acceptByte('"') { // quoted-string
|
|
for {
|
|
ch, ok := p.readByte()
|
|
switch ch {
|
|
case '\\':
|
|
ch, ok = p.readByte()
|
|
case '"':
|
|
return sb.String(), nil
|
|
}
|
|
if !ok {
|
|
return "", fmt.Errorf("malformed quoted-string")
|
|
}
|
|
sb.WriteByte(ch)
|
|
}
|
|
} else { // dot-string
|
|
for {
|
|
ch, ok := p.peekByte()
|
|
if !ok {
|
|
return sb.String(), nil
|
|
}
|
|
switch ch {
|
|
case '@':
|
|
return sb.String(), nil
|
|
case '(', ')', '<', '>', '[', ']', ':', ';', '\\', ',', '"', ' ', '\t':
|
|
return "", fmt.Errorf("malformed dot-string")
|
|
}
|
|
p.readByte()
|
|
sb.WriteByte(ch)
|
|
}
|
|
}
|
|
}
|