mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 14:07:38 +03:00
add max_header_size, check header size in smtp session
This commit is contained in:
parent
2d691f1f7c
commit
426c61880f
3 changed files with 76 additions and 1 deletions
|
@ -40,6 +40,31 @@ import (
|
||||||
"github.com/foxcpp/maddy/framework/module"
|
"github.com/foxcpp/maddy/framework/module"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func limitReader(r io.Reader, n int64, err error) *limitedReader {
|
||||||
|
return &limitedReader{R: r, N: n, E: err, Enabled: true}
|
||||||
|
}
|
||||||
|
|
||||||
|
type limitedReader struct {
|
||||||
|
R io.Reader
|
||||||
|
N int64
|
||||||
|
E error
|
||||||
|
Enabled bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// same as io.LimitedReader.Read except returning the custom error and the option
|
||||||
|
// to be disabled
|
||||||
|
func (l *limitedReader) Read(p []byte) (n int, err error) {
|
||||||
|
if l.Enabled && l.N <= 0 {
|
||||||
|
return 0, l.E
|
||||||
|
}
|
||||||
|
if int64(len(p)) > l.N {
|
||||||
|
p = p[0:l.N]
|
||||||
|
}
|
||||||
|
n, err = l.R.Read(p)
|
||||||
|
l.N -= int64(n)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
type Session struct {
|
type Session struct {
|
||||||
endp *Endpoint
|
endp *Endpoint
|
||||||
|
|
||||||
|
@ -340,7 +365,13 @@ func (s *Session) Logout() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Session) prepareBody(ctx context.Context, r io.Reader) (textproto.Header, buffer.Buffer, error) {
|
func (s *Session) prepareBody(ctx context.Context, r io.Reader) (textproto.Header, buffer.Buffer, error) {
|
||||||
bufr := bufio.NewReader(r)
|
limitr := limitReader(r, int64(s.endp.maxHeaderBytes), &exterrors.SMTPError{
|
||||||
|
Code: 552,
|
||||||
|
EnhancedCode: exterrors.EnhancedCode{5, 3, 4},
|
||||||
|
Message: "Message header size exceeds limit",
|
||||||
|
})
|
||||||
|
|
||||||
|
bufr := bufio.NewReader(limitr)
|
||||||
header, err := textproto.ReadHeader(bufr)
|
header, err := textproto.ReadHeader(bufr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return textproto.Header{}, nil, fmt.Errorf("I/O error while parsing header: %w", err)
|
return textproto.Header{}, nil, fmt.Errorf("I/O error while parsing header: %w", err)
|
||||||
|
@ -353,6 +384,9 @@ func (s *Session) prepareBody(ctx context.Context, r io.Reader) (textproto.Heade
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// the header size check is done. The message size will be checked by go-smtp
|
||||||
|
limitr.Enabled = false
|
||||||
|
|
||||||
buf, err := s.endp.buffer(bufr)
|
buf, err := s.endp.buffer(bufr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return textproto.Header{}, nil, fmt.Errorf("I/O error while writing buffer: %w", err)
|
return textproto.Header{}, nil, fmt.Errorf("I/O error while writing buffer: %w", err)
|
||||||
|
|
|
@ -67,6 +67,7 @@ type Endpoint struct {
|
||||||
deferServerReject bool
|
deferServerReject bool
|
||||||
maxLoggedRcptErrors int
|
maxLoggedRcptErrors int
|
||||||
maxReceived int
|
maxReceived int
|
||||||
|
maxHeaderBytes int
|
||||||
|
|
||||||
listenersWg sync.WaitGroup
|
listenersWg sync.WaitGroup
|
||||||
|
|
||||||
|
@ -245,6 +246,7 @@ func (endp *Endpoint) setConfig(cfg *config.Map) error {
|
||||||
cfg.Duration("write_timeout", false, false, 1*time.Minute, &endp.serv.WriteTimeout)
|
cfg.Duration("write_timeout", false, false, 1*time.Minute, &endp.serv.WriteTimeout)
|
||||||
cfg.Duration("read_timeout", false, false, 10*time.Minute, &endp.serv.ReadTimeout)
|
cfg.Duration("read_timeout", false, false, 10*time.Minute, &endp.serv.ReadTimeout)
|
||||||
cfg.DataSize("max_message_size", false, false, 32*1024*1024, &endp.serv.MaxMessageBytes)
|
cfg.DataSize("max_message_size", false, false, 32*1024*1024, &endp.serv.MaxMessageBytes)
|
||||||
|
cfg.DataSize("max_header_size", false, false, 1*1024*1024, &endp.maxHeaderBytes)
|
||||||
cfg.Int("max_recipients", false, false, 20000, &endp.serv.MaxRecipients)
|
cfg.Int("max_recipients", false, false, 20000, &endp.serv.MaxRecipients)
|
||||||
cfg.Int("max_received", false, false, 50, &endp.maxReceived)
|
cfg.Int("max_received", false, false, 50, &endp.maxReceived)
|
||||||
cfg.Custom("buffer", false, false, func() (interface{}, error) {
|
cfg.Custom("buffer", false, false, func() (interface{}, error) {
|
||||||
|
|
|
@ -24,6 +24,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/foxcpp/go-mockdns"
|
"github.com/foxcpp/go-mockdns"
|
||||||
|
@ -456,3 +457,41 @@ func TestCheckCommand(tt *testing.T) {
|
||||||
conn.Writeln("QUIT")
|
conn.Writeln("QUIT")
|
||||||
conn.ExpectPattern("221 *")
|
conn.ExpectPattern("221 *")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestHeaderSizeConstraint(tt *testing.T) {
|
||||||
|
tt.Parallel()
|
||||||
|
t := tests.NewT(tt)
|
||||||
|
t.DNS(nil)
|
||||||
|
t.Port("smtp")
|
||||||
|
t.Config(`
|
||||||
|
smtp tcp://127.0.0.1:{env:TEST_PORT_smtp} {
|
||||||
|
hostname mx.maddy.test
|
||||||
|
tls off
|
||||||
|
deliver_to dummy
|
||||||
|
max_header_size 1K
|
||||||
|
}
|
||||||
|
`)
|
||||||
|
t.Run(1)
|
||||||
|
defer t.Close()
|
||||||
|
|
||||||
|
conn := t.Conn("smtp")
|
||||||
|
defer conn.Close()
|
||||||
|
conn.SMTPNegotation("localhost", nil, nil)
|
||||||
|
conn.Writeln("MAIL FROM:<testsender@maddy.test>")
|
||||||
|
conn.ExpectPattern("250 *")
|
||||||
|
conn.Writeln("RCPT TO:<testing@maddy.test>")
|
||||||
|
conn.ExpectPattern("250 *")
|
||||||
|
conn.Writeln("DATA")
|
||||||
|
conn.ExpectPattern("354 *")
|
||||||
|
conn.Writeln("From: <testing@sender.test>")
|
||||||
|
conn.Writeln("To: <testing@maddy.test>")
|
||||||
|
conn.Writeln("Subject: " + strings.Repeat("A", 2*1024))
|
||||||
|
conn.Writeln("")
|
||||||
|
conn.Writeln("Hi")
|
||||||
|
conn.Writeln(".")
|
||||||
|
|
||||||
|
conn.ExpectPattern("552 5.3.4 Message header size exceeds limit *")
|
||||||
|
|
||||||
|
conn.Writeln("QUIT")
|
||||||
|
conn.ExpectPattern("221 *")
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue