Migrate to latest go-smtp version

Fixes #661 among other minor things.
This commit is contained in:
fox.cpp 2024-01-21 14:41:57 +03:00
parent 301c47d815
commit db0874c2be
No known key found for this signature in database
GPG key ID: 5B991F6215D2FCC0
25 changed files with 139 additions and 113 deletions

View file

@ -284,7 +284,7 @@ func ParseDataSize(s string) (int, error) {
// data unit and allows multiple arguments (they will be added together).
//
// See Map.Custom for description of arguments.
func (m *Map) DataSize(name string, inheritGlobal, required bool, defaultVal int, store *int) {
func (m *Map) DataSize(name string, inheritGlobal, required bool, defaultVal int64, store *int64) {
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
return defaultVal, nil
}, func(_ *Map, node Node) (interface{}, error) {
@ -301,7 +301,7 @@ func (m *Map) DataSize(name string, inheritGlobal, required bool, defaultVal int
return nil, NodeErr(node, "%v", err)
}
return dur, nil
return int64(dur), nil
}, store)
}

View file

@ -22,6 +22,7 @@ import (
"context"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
)
@ -56,7 +57,7 @@ type Delivery interface {
// recipients that can't be used. Note: MsgMetadata object passed to Start
// contains BodyLength field. If it is non-zero, it can be used to check
// storage quota for the user before Body.
AddRcpt(ctx context.Context, rcptTo string) error
AddRcpt(ctx context.Context, rcptTo string, opts smtp.RcptOptions) error
// Body sets the body and header contents for the message.
// If this method fails, message is assumed to be undeliverable

View file

@ -22,6 +22,7 @@ import (
"context"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/config"
)
@ -63,7 +64,7 @@ func (d *Dummy) Start(ctx context.Context, msgMeta *MsgMetadata, mailFrom string
type dummyDelivery struct{}
func (dd dummyDelivery) AddRcpt(ctx context.Context, to string) error {
func (dd dummyDelivery) AddRcpt(ctx context.Context, rcptTo string, opts smtp.RcptOptions) error {
return nil
}

4
go.mod
View file

@ -13,7 +13,7 @@ require (
github.com/emersion/go-milter v0.3.3
github.com/emersion/go-msgauth v0.6.6
github.com/emersion/go-sasl v0.0.0-20220912192320-0145f2c60ead
github.com/emersion/go-smtp v0.16.0
github.com/emersion/go-smtp v0.20.2-0.20240121112028-434ddca4792e
github.com/foxcpp/go-dovecot-sasl v0.0.0-20200522223722-c4699d7a24bf
github.com/foxcpp/go-imap-backend-tests v0.0.0-20220105184719-e80aa29a5e16
github.com/foxcpp/go-imap-i18nlevel v0.0.0-20200208001533-d6ec88553005
@ -52,6 +52,7 @@ require (
golang.org/x/net v0.17.0
golang.org/x/sync v0.2.0
golang.org/x/text v0.14.0
modernc.org/sqlite v1.28.0
)
require (
@ -153,7 +154,6 @@ require (
modernc.org/mathutil v1.6.0 // indirect
modernc.org/memory v1.7.2 // indirect
modernc.org/opt v0.1.3 // indirect
modernc.org/sqlite v1.28.0 // indirect
modernc.org/strutil v1.1.3 // indirect
modernc.org/token v1.0.1 // indirect
)

25
go.sum
View file

@ -234,10 +234,8 @@ github.com/aws/aws-sdk-go-v2/service/sts v1.19.0/go.mod h1:BgQOMsg8av8jset59jely
github.com/aws/smithy-go v1.13.5 h1:hgz0X/DX0dGqTYpGALqXJoRKRj5oQ7150i5FdTePzO8=
github.com/aws/smithy-go v1.13.5/go.mod h1:Tg+OJXh4MB2R/uN61Ko2f6hTZwB/ZYGOtib8J3gBHzA=
github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8=
github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA=
github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs=
github.com/caddyserver/certmagic v0.17.2 h1:o30seC1T/dBqBCNNGNHWwj2i5/I/FMjBbTAhjADP3nE=
github.com/caddyserver/certmagic v0.17.2/go.mod h1:ouWUuC490GOLJzkyN35eXfV8bSbwMwSf4bdhkIxtdQE=
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
@ -258,8 +256,6 @@ github.com/cncf/xds/go v0.0.0-20210805033703-aa0b78936158/go.mod h1:eXthEFrGJvWH
github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
github.com/cpuguy83/go-md2man/v2 v2.0.2 h1:p1EgwI/C7NhT0JmVkwCD2ZBK8j4aeHQX2pMHHBfMQ6w=
github.com/cpuguy83/go-md2man/v2 v2.0.2/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
@ -290,8 +286,8 @@ github.com/emersion/go-sasl v0.0.0-20191210011802-430746ea8b9b/go.mod h1:G/dpzLu
github.com/emersion/go-sasl v0.0.0-20200509203442-7bfe0ed36a21/go.mod h1:iL2twTeMvZnrg54ZoPDNfJaJaqy0xIQFuBdrLsmspwQ=
github.com/emersion/go-sasl v0.0.0-20220912192320-0145f2c60ead h1:fI1Jck0vUrXT8bnphprS1EoVRe2Q5CKCX8iDlpqjQ/Y=
github.com/emersion/go-sasl v0.0.0-20220912192320-0145f2c60ead/go.mod h1:iL2twTeMvZnrg54ZoPDNfJaJaqy0xIQFuBdrLsmspwQ=
github.com/emersion/go-smtp v0.16.0 h1:eB9CY9527WdEZSs5sWisTmilDX7gG+Q/2IdRcmubpa8=
github.com/emersion/go-smtp v0.16.0/go.mod h1:qm27SGYgoIPRot6ubfQ/GpiPy/g3PaZAVRxiO/sDUgQ=
github.com/emersion/go-smtp v0.20.2-0.20240121112028-434ddca4792e h1:WAPhaiA+bDO/mFgCDQJKCQI/RbH/73lCcis4Jb8Y2ec=
github.com/emersion/go-smtp v0.20.2-0.20240121112028-434ddca4792e/go.mod h1:qm27SGYgoIPRot6ubfQ/GpiPy/g3PaZAVRxiO/sDUgQ=
github.com/emersion/go-textwrapper v0.0.0-20160606182133-d0e65e56babe/go.mod h1:aqO8z8wPrjkscevZJFVE1wXJrLpC5LtJG7fqLOsPb2U=
github.com/emersion/go-textwrapper v0.0.0-20200911093747-65d896831594 h1:IbFBtwoTQyw0fIM5xv1HF+Y+3ZijDR839WMulgxCcUY=
github.com/emersion/go-textwrapper v0.0.0-20200911093747-65d896831594/go.mod h1:aqO8z8wPrjkscevZJFVE1wXJrLpC5LtJG7fqLOsPb2U=
@ -321,8 +317,6 @@ github.com/foxcpp/go-imap-mess v0.0.0-20230108134257-b7ec3a649613 h1:fw9OWfPxP1C
github.com/foxcpp/go-imap-mess v0.0.0-20230108134257-b7ec3a649613/go.mod h1:P/O/qz4gaVkefzJ40BUtN/ZzBnaEg0YYe1no/SMp7Aw=
github.com/foxcpp/go-imap-namespace v0.0.0-20200802091432-08496dd8e0ed h1:1Jo7geyvunrPSjL6F6D9EcXoNApS5v3LQaro7aUNPnE=
github.com/foxcpp/go-imap-namespace v0.0.0-20200802091432-08496dd8e0ed/go.mod h1:Shows1vmkBWO40ChOClaUe6DUnZrsP1UPAuoWzIUdgQ=
github.com/foxcpp/go-imap-sql v0.5.1-0.20230313080458-c0176dad679c h1:vqLBcLtG5lcXL2hifcsKjiUaljRukD8xHodVM2rZ+L4=
github.com/foxcpp/go-imap-sql v0.5.1-0.20230313080458-c0176dad679c/go.mod h1:8uUTN2RRWZrETuA9pDvDr4SjV1hCvEYG2WOlXuupj+g=
github.com/foxcpp/go-imap-sql v0.5.1-0.20240120174134-48f9dc0b4abf h1:tqkJhHCPp1LL0tFqe0GwPjw2BMug8ivMtKJbQ7ZUA/g=
github.com/foxcpp/go-imap-sql v0.5.1-0.20240120174134-48f9dc0b4abf/go.mod h1:8uUTN2RRWZrETuA9pDvDr4SjV1hCvEYG2WOlXuupj+g=
github.com/foxcpp/go-mockdns v0.0.0-20191216195825-5eabd8dbfe1f/go.mod h1:tPg4cp4nseejPd+UKxtCVQ2hUxNTZ7qQZJa7CLriIeo=
@ -342,11 +336,9 @@ github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
github.com/go-ldap/ldap/v3 v3.4.4 h1:qPjipEpt+qDa6SI/h1fzuGWoRUY+qqQ9sOZq67/PYUs=
github.com/go-ldap/ldap/v3 v3.4.4/go.mod h1:fe1MsuN5eJJ1FeLT/LEBVdWfNWKh459R7aXgXtJC+aI=
github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs=
github.com/go-sql-driver/mysql v1.4.1/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w=
github.com/go-sql-driver/mysql v1.7.1 h1:lUIinVbN1DY0xBg0eMOzmmtGoHwWBbvnWubQUrtU8EI=
github.com/go-sql-driver/mysql v1.7.1/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI=
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
@ -426,6 +418,7 @@ github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLe
github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ=
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
github.com/google/s2a-go v0.1.4 h1:1kZ/sQM3srePvKs3tXAvQzo66XfcReoqFpIpIccE7Oc=
github.com/google/s2a-go v0.1.4/go.mod h1:Ej+mSEMGRnqRzjc7VtF+jdBwYG5fuJfiZ8ELkjEwM0A=
@ -473,12 +466,10 @@ github.com/johannesboyne/gofakes3 v0.0.0-20210704111953-6a9f95c2941c h1:lx/uPI+m
github.com/johannesboyne/gofakes3 v0.0.0-20210704111953-6a9f95c2941c/go.mod h1:LIAXxPvcUXwOcTIj9LSNSUpE9/eMHalTWxsP/kmWxQI=
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4=
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk=
github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM=
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 h1:Z9n2FFNUXsshfwJMBgNA0RU6/i7WVaAegv3PtuIHPMs=
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8=
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
@ -557,7 +548,6 @@ github.com/minio/minio-go/v7 v7.0.55 h1:ZXqUO/8cgfHzI+08h/zGuTTFpISSA32BZmBE3FCL
github.com/minio/minio-go/v7 v7.0.55/go.mod h1:NUDy4A4oXPq1l2yK6LTSvCEzAMeIcoz9lcj5dbzSrRE=
github.com/minio/sha256-simd v1.0.1 h1:6kaan5IFmwTNynnKKpDHe6FWHohJOHhCPchzK49dzMM=
github.com/minio/sha256-simd v1.0.1/go.mod h1:Pz6AKMiUdngCLpeTL/RJY1M9rUuPMYujV5xJjtbRSN8=
github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
@ -565,12 +555,10 @@ github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
github.com/netauth/netauth v0.6.2-0.20220831214440-1df568cd25d6 h1:TsF5Cl0Mj5JMvPOP2ySVq+CZoiPrTGwvNPbuQotuSAE=
github.com/netauth/netauth v0.6.2-0.20220831214440-1df568cd25d6/go.mod h1:4PEbISVqRCQaXaDAt289w3nK9UhoF8/ZOLy31Hbv7ds=
github.com/netauth/protocol v0.0.0-20210918062754-7fee492ffcbd h1:4yVpQ/+li28lQ/daYCWeDB08obRmjaoAw2qfFFaCQ40=
github.com/netauth/protocol v0.0.0-20210918062754-7fee492ffcbd/go.mod h1:wpK5wqysOJU1w2OxgG65du8M7UqBkxzsNaJdjwiRqAs=
github.com/pelletier/go-toml v1.9.3/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c=
github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ=
github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4=
github.com/pierrec/lz4 v2.5.2+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY=
@ -604,7 +592,6 @@ github.com/ryszard/goskiplist v0.0.0-20150312221310-2dfbae5fcf46 h1:GHRpF1pTW19a
github.com/ryszard/goskiplist v0.0.0-20150312221310-2dfbae5fcf46/go.mod h1:uAQ5PCi+MFsC7HjREoAz1BU+Mq60+05gifQSsHSDG/8=
github.com/shabbyrobe/gocovmerge v0.0.0-20180507124511-f6ea450bfb63 h1:J6qvD6rbmOil46orKqJaRPG+zTpoGlBTUdyv8ki63L0=
github.com/shabbyrobe/gocovmerge v0.0.0-20180507124511-f6ea450bfb63/go.mod h1:n+VKSARF5y/tS9XFSP7vWDfS+GUC5vs/YT7M5XDTUEM=
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
github.com/sirupsen/logrus v1.9.2 h1:oxx1eChJGI6Uks2ZC4W1zpLlVgqB8ner4EuQwV4Ik1Y=
github.com/sirupsen/logrus v1.9.2/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA=
@ -663,7 +650,6 @@ go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqe
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI=
go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60=
@ -1211,7 +1197,6 @@ gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
@ -1230,6 +1215,8 @@ modernc.org/cc/v3 v3.40.0 h1:P3g79IUS/93SYhtoeaHW+kRCIrYaxJ27MFPv+7kaTOw=
modernc.org/cc/v3 v3.40.0/go.mod h1:/bTg4dnWkSXowUO6ssQKnOV0yMVxDYNIsIrzqTFDGH0=
modernc.org/ccgo/v3 v3.16.13 h1:Mkgdzl46i5F/CNR/Kj80Ri59hC8TKAhZrYSaqvkwzUw=
modernc.org/ccgo/v3 v3.16.13/go.mod h1:2Quk+5YgpImhPjv2Qsob1DnZ/4som1lJTodubIcoUkY=
modernc.org/ccorpus v1.11.6 h1:J16RXiiqiCgua6+ZvQot4yUuUy8zxgqbqEEUuGPlISk=
modernc.org/httpfs v1.0.6 h1:AAgIpFZRXuYnkjftxTAZwMIiwEqAfk8aVB2/oA6nAeM=
modernc.org/libc v1.29.0 h1:tTFRFq69YKCF2QyGNuRUQxKBm1uZZLubf6Cjh/pVHXs=
modernc.org/libc v1.29.0/go.mod h1:DaG/4Q3LRRdqpiLyP0C2m1B8ZMGkQ+cCgOIjEtQlYhQ=
modernc.org/mathutil v1.6.0 h1:fRe9+AmYlaej+64JsEEhoWuAYBkOtQiMEU7n/XgfYi4=
@ -1242,8 +1229,10 @@ modernc.org/sqlite v1.28.0 h1:Zx+LyDDmXczNnEQdvPuEfcFVA2ZPyaD7UCZDjef3BHQ=
modernc.org/sqlite v1.28.0/go.mod h1:Qxpazz0zH8Z1xCFyi5GSL3FzbtZ3fvbjmywNogldEW0=
modernc.org/strutil v1.1.3 h1:fNMm+oJklMGYfU9Ylcywl0CO5O6nTfaowNsh2wpPjzY=
modernc.org/strutil v1.1.3/go.mod h1:MEHNA7PdEnEwLvspRMtWTNnp2nnyvMfkimT1NKNAGbw=
modernc.org/tcl v1.15.2 h1:C4ybAYCGJw968e+Me18oW55kD/FexcHbqH2xak1ROSY=
modernc.org/token v1.0.1 h1:A3qvTqOwexpfZZeyI0FeGPDlSWX5pjZu9hF4lU+EKWg=
modernc.org/token v1.0.1/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM=
modernc.org/z v1.7.3 h1:zDJf6iHjrnB+WRD88stbXokugjyc0/pB91ri1gO6LZY=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=

View file

@ -335,7 +335,7 @@ func (s *Session) fetchRDNSName(ctx context.Context) {
s.connState.RDNSName.Set(name, nil)
}
func (s *Session) Rcpt(to string) error {
func (s *Session) Rcpt(to string, opts *smtp.RcptOptions) error {
s.msgLock.Lock()
defer s.msgLock.Unlock()
@ -363,7 +363,7 @@ func (s *Session) Rcpt(to string) error {
rcptCtx, rcptTask := trace.NewTask(s.msgCtx, "RCPT TO")
defer rcptTask.End()
if err := s.rcpt(rcptCtx, to); err != nil {
if err := s.rcpt(rcptCtx, to, opts); err != nil {
if s.loggedRcptErrors < s.endp.maxLoggedRcptErrors {
s.log.Error("RCPT error", err, "rcpt", to, "msg_id", s.msgMeta.ID)
s.loggedRcptErrors++
@ -377,7 +377,7 @@ func (s *Session) Rcpt(to string) error {
return nil
}
func (s *Session) rcpt(ctx context.Context, to string) error {
func (s *Session) rcpt(ctx context.Context, to string, opts *smtp.RcptOptions) error {
// INTERNATIONALIZATION: Do not permit non-ASCII addresses unless SMTPUTF8 is
// used.
if !address.IsASCII(to) && !s.opts.UTF8 {
@ -396,7 +396,7 @@ func (s *Session) rcpt(ctx context.Context, to string) error {
}
}
return s.delivery.AddRcpt(ctx, cleanTo)
return s.delivery.AddRcpt(ctx, cleanTo, *opts)
}
func (s *Session) Logout() error {
@ -413,6 +413,9 @@ func (s *Session) Logout() error {
if s.cancelRDNS != nil {
s.cancelRDNS()
}
s.endp.sessionCnt.Add(-1)
return nil
}

View file

@ -30,6 +30,7 @@ import (
"path/filepath"
"strings"
"sync"
"sync/atomic"
"time"
"github.com/emersion/go-sasl"
@ -67,7 +68,9 @@ type Endpoint struct {
deferServerReject bool
maxLoggedRcptErrors int
maxReceived int
maxHeaderBytes int
maxHeaderBytes int64
sessionCnt atomic.Int32
authNormalize authz.NormalizeFunc
authMap module.Table
@ -401,6 +404,8 @@ func (endp *Endpoint) NewSession(conn *smtp.Conn) (smtp.Session, error) {
return nil, endp.wrapErr("", true, "EHLO", err)
}
endp.sessionCnt.Add(1)
return sess, nil
}
@ -447,6 +452,10 @@ func (endp *Endpoint) newSession(conn *smtp.Conn) *Session {
return s
}
func (endp *Endpoint) ConnectionCount() int {
return int(endp.sessionCnt.Load())
}
func (endp *Endpoint) Close() error {
endp.serv.Close()
endp.listenersWg.Wait()

View file

@ -124,7 +124,7 @@ func submitMsgOpts(t *testing.T, cl *smtp.Client, from string, rcpts []string, o
return err
}
for _, rcpt := range rcpts {
if err := cl.Rcpt(rcpt); err != nil {
if err := cl.Rcpt(rcpt, &smtp.RcptOptions{}); err != nil {
return err
}
}
@ -334,9 +334,9 @@ func TestSMTPDeliver_CheckError_Deferred(t *testing.T) {
}
}
checkErr(cl.Rcpt("test1@example.org"))
checkErr(cl.Rcpt("test1@example.org"))
checkErr(cl.Rcpt("test2@example.org"))
checkErr(cl.Rcpt("test1@example.org", &smtp.RcptOptions{}))
checkErr(cl.Rcpt("test1@example.org", &smtp.RcptOptions{}))
checkErr(cl.Rcpt("test2@example.org", &smtp.RcptOptions{}))
}
func TestSMTPDelivery_Multi(t *testing.T) {
@ -394,7 +394,7 @@ func TestSMTPDelivery_AbortData(t *testing.T) {
if err := cl.Mail("sender@example.org", nil); err != nil {
t.Fatal(err)
}
if err := cl.Rcpt("test@example.com"); err != nil {
if err := cl.Rcpt("test@example.com", &smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
data, err := cl.Data()
@ -432,7 +432,7 @@ func TestSMTPDelivery_EmptyMessage(t *testing.T) {
if err := cl.Mail("sender@example.org", nil); err != nil {
t.Fatal(err)
}
if err := cl.Rcpt("test@example.com"); err != nil {
if err := cl.Rcpt("test@example.com", &smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
data, err := cl.Data()
@ -471,7 +471,7 @@ func TestSMTPDelivery_AbortLogout(t *testing.T) {
if err := cl.Mail("sender@example.org", nil); err != nil {
t.Fatal(err)
}
if err := cl.Rcpt("test@example.com"); err != nil {
if err := cl.Rcpt("test@example.com", &smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -499,7 +499,7 @@ func TestSMTPDelivery_Reset(t *testing.T) {
if err := cl.Mail("from-garbage@example.org", nil); err != nil {
t.Fatal(err)
}
if err := cl.Rcpt("to-garbage@example.org"); err != nil {
if err := cl.Rcpt("to-garbage@example.org", &smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
if err := cl.Reset(); err != nil {

View file

@ -30,6 +30,7 @@ import (
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-msgauth/authres"
"github.com/emersion/go-smtp"
"github.com/foxcpp/go-mockdns"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/exterrors"
@ -59,7 +60,7 @@ func doTestDelivery(t *testing.T, tgt module.DeliveryTarget, from string, to []s
return encodedID, err
}
for _, rcpt := range to {
if err := delivery.AddRcpt(context.Background(), rcpt); err != nil {
if err := delivery.AddRcpt(context.Background(), rcpt, smtp.RcptOptions{}); err != nil {
if err := delivery.Abort(context.Background()); err != nil {
t.Log("delivery.Abort:", err)
}

View file

@ -22,6 +22,7 @@ import (
"context"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/address"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/config"
@ -276,7 +277,7 @@ type msgpipelineDelivery struct {
checkRunner *checkRunner
}
func (dd *msgpipelineDelivery) AddRcpt(ctx context.Context, to string) error {
func (dd *msgpipelineDelivery) AddRcpt(ctx context.Context, to string, opts smtp.RcptOptions) error {
if err := dd.checkRunner.checkRcpt(ctx, dd.d.globalChecks, to); err != nil {
return err
}
@ -363,7 +364,7 @@ func (dd *msgpipelineDelivery) AddRcpt(ctx context.Context, to string) error {
return wrapErr(err)
}
if err := delivery.AddRcpt(ctx, to); err != nil {
if err := delivery.AddRcpt(ctx, to, opts); err != nil {
return wrapErr(err)
}
delivery.recipients = append(delivery.recipients, originalTo)

View file

@ -24,6 +24,7 @@ import (
"testing"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/module"
"github.com/foxcpp/maddy/internal/modify"
@ -422,10 +423,10 @@ func TestMsgPipeline_PerRcptReject(t *testing.T) {
}
}()
if err := delivery.AddRcpt(context.Background(), "rcpt2@example.com"); err == nil {
if err := delivery.AddRcpt(context.Background(), "rcpt2@example.com", smtp.RcptOptions{}); err == nil {
t.Fatalf("expected error for delivery.AddRcpt(rcpt2@example.com), got nil")
}
if err := delivery.AddRcpt(context.Background(), "rcpt1@example.com"); err != nil {
if err := delivery.AddRcpt(context.Background(), "rcpt1@example.com", smtp.RcptOptions{}); err != nil {
t.Fatalf("unexpected AddRcpt err for %s: %v", "rcpt1@example.com", err)
}
if err := delivery.Body(context.Background(), textproto.Header{}, buffer.MemoryBuffer{Slice: []byte("foobar")}); err != nil {

View file

@ -16,7 +16,7 @@ You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
// The package smtpconn contains the code shared between target.smtp and
// Package smtpconn contains the code shared between target.smtp and
// remote modules.
//
// It implements the wrapper over the SMTP connection (go-smtp.Client) object
@ -222,13 +222,9 @@ func (c *C) attemptConnect(ctx context.Context, lmtp bool, endp config.Endpoint,
c.lmtp = lmtp
// This uses initial greeting timeout of 5 minutes (hardcoded).
if lmtp {
cl, err = smtp.NewClientLMTP(conn, endp.Host)
cl = smtp.NewClientLMTP(conn)
} else {
cl, err = smtp.NewClient(conn, endp.Host)
}
if err != nil {
conn.Close()
return false, nil, err
cl = smtp.NewClient(conn)
}
cl.CommandTimeout = c.CommandTimeout
@ -336,9 +332,13 @@ func (c *C) IsLMTP() bool {
//
// If the address is non-ASCII and cannot be converted to ASCII and the remote
// server does not support SMTPUTF8, error will be returned.
func (c *C) Rcpt(ctx context.Context, to string) error {
func (c *C) Rcpt(ctx context.Context, to string, opts smtp.RcptOptions) error {
defer trace.StartRegion(ctx, "smtpconn/RCPT TO").End()
outOpts := &smtp.RcptOptions{
// TODO: DSN support
}
// If necessary, the extension flag is enabled in Start.
if ok, _ := c.cl.Extension("SMTPUTF8"); !address.IsASCII(to) && !ok {
var err error
@ -356,7 +356,7 @@ func (c *C) Rcpt(ctx context.Context, to string) error {
}
}
if err := c.cl.Rcpt(to); err != nil {
if err := c.cl.Rcpt(to, outOpts); err != nil {
return c.wrapClientErr(err, c.serverName)
}

View file

@ -37,7 +37,7 @@ func doTestDelivery(t *testing.T, conn *C, from string, to []string, opts smtp.M
return err
}
for _, rcpt := range to {
if err := conn.Rcpt(context.Background(), rcpt); err != nil {
if err := conn.Rcpt(context.Background(), rcpt, smtp.RcptOptions{}); err != nil {
return err
}
}

View file

@ -25,6 +25,7 @@ import (
"github.com/emersion/go-imap"
"github.com/emersion/go-imap/backend"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
imapsql "github.com/foxcpp/go-imap-sql"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/exterrors"
@ -58,7 +59,7 @@ func userDoesNotExist(actual error) error {
}
}
func (d *delivery) AddRcpt(ctx context.Context, rcptTo string) error {
func (d *delivery) AddRcpt(ctx context.Context, rcptTo string, _ smtp.RcptOptions) error {
defer trace.StartRegion(ctx, "sql/AddRcpt").End()
accountName, err := d.store.deliveryNormalize(ctx, rcptTo)

View file

@ -107,7 +107,7 @@ func (store *Storage) Init(cfg *config.Map) error {
var (
driver string
dsn []string
appendlimitVal = -1
appendlimitVal int64 = -1
compression []string
authNormalize string
deliveryNormalize string
@ -232,7 +232,7 @@ func (store *Storage) Init(cfg *config.Map) error {
} else {
// int is 32-bit on some platforms, so cut off values we can't actually
// use.
if int(uint32(appendlimitVal)) != appendlimitVal {
if int64(uint32(appendlimitVal)) != appendlimitVal {
return errors.New("imapsql: appendlimit value is too big")
}
opts.MaxMsgBytes = new(uint32)

View file

@ -30,12 +30,12 @@ All scheduled deliveries are attempted to the configured DeliveryTarget.
All metadata is preserved on disk.
Failure status is determined on per-recipient basis:
- Delivery.Start fail handled as a failure for all recipients.
- Delivery.AddRcpt fail handled as a failure for the corresponding recipient.
- Delivery.Body fail handled as a failure for all recipients.
- If Delivery implements PartialDelivery, then
PartialDelivery.BodyNonAtomic is used instead. Failures are determined based
on StatusCollector.SetStatus calls done by target in this case.
- Delivery.Start fail handled as a failure for all recipients.
- Delivery.AddRcpt fail handled as a failure for the corresponding recipient.
- Delivery.Body fail handled as a failure for all recipients.
- If Delivery implements PartialDelivery, then
PartialDelivery.BodyNonAtomic is used instead. Failures are determined based
on StatusCollector.SetStatus calls done by target in this case.
For each failure check is done to see if it is a permanent failure
or a temporary one. This is done using exterrors.IsTemporaryOrUnspec.
@ -487,7 +487,7 @@ func (q *Queue) deliver(meta *QueueMetadata, header textproto.Header, body buffe
var acceptedRcpts []string
for _, rcpt := range meta.To {
rcptCtx, rcptTask := trace.NewTask(msgCtx, "RCPT TO")
if err := delivery.AddRcpt(rcptCtx, rcpt); err != nil {
if err := delivery.AddRcpt(rcptCtx, rcpt, smtp.RcptOptions{} /* TODO: DSN support */); err != nil {
dl.Debugf("delivery.AddRcpt %s failed: %v", rcpt, err)
perr.Errs[rcpt] = err
} else {
@ -558,7 +558,7 @@ type queueDelivery struct {
body buffer.Buffer
}
func (qd *queueDelivery) AddRcpt(ctx context.Context, rcptTo string) error {
func (qd *queueDelivery) AddRcpt(ctx context.Context, rcptTo string, _ smtp.RcptOptions) error {
qd.meta.To = append(qd.meta.To, rcptTo)
return nil
}
@ -975,7 +975,7 @@ func (q *Queue) emitDSN(meta *QueueMetadata, header textproto.Header, failedRcpt
}()
rcptCtx, rcptTask := trace.NewTask(msgCtx, "RCPT TO")
if err = dsnDelivery.AddRcpt(rcptCtx, meta.From); err != nil {
if err = dsnDelivery.AddRcpt(rcptCtx, meta.From, smtp.RcptOptions{}); err != nil {
rcptTask.End()
return
}

View file

@ -33,6 +33,7 @@ import (
"time"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/exterrors"
"github.com/foxcpp/maddy/framework/log"
@ -104,7 +105,7 @@ type unreliableTargetDeliveryPartial struct {
*unreliableTargetDelivery
}
func (utd *unreliableTargetDelivery) AddRcpt(ctx context.Context, rcptTo string) error {
func (utd *unreliableTargetDelivery) AddRcpt(ctx context.Context, rcptTo string, _ smtp.RcptOptions) error {
if len(utd.ut.rcptFailures) > utd.ut.passedMessages {
rcptErrs := utd.ut.rcptFailures[utd.ut.passedMessages]
if err := rcptErrs[rcptTo]; err != nil {
@ -610,7 +611,7 @@ func TestQueueDelivery_AbortNoDangling(t *testing.T) {
t.Fatalf("unexpected Start err: %v", err)
}
for _, rcpt := range [...]string{"test@example.org", "test2@example.org"} {
if err := delivery.AddRcpt(context.Background(), rcpt); err != nil {
if err := delivery.AddRcpt(context.Background(), rcpt, smtp.RcptOptions{}); err != nil {
t.Fatalf("unexpected AddRcpt err for %s: %v", rcpt, err)
}
}
@ -790,7 +791,7 @@ func TestQueueDSN_RcptRewrite(t *testing.T) {
t.Fatalf("unexpected Start err: %v", err)
}
for _, rcpt := range [...]string{"test@example.org", "test2@example.org"} {
if err := delivery.AddRcpt(context.Background(), rcpt); err != nil {
if err := delivery.AddRcpt(context.Background(), rcpt, smtp.RcptOptions{}); err != nil {
t.Fatalf("unexpected AddRcpt err for %s: %v", rcpt, err)
}
}

View file

@ -35,6 +35,7 @@ import (
"time"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/address"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/config"
@ -269,7 +270,7 @@ func (rt *Target) Start(ctx context.Context, msgMeta *module.MsgMetadata, mailFr
}, nil
}
func (rd *remoteDelivery) AddRcpt(ctx context.Context, to string) error {
func (rd *remoteDelivery) AddRcpt(ctx context.Context, to string, opts smtp.RcptOptions) error {
defer trace.StartRegion(ctx, "remote/AddRcpt").End()
if rd.msgMeta.Quarantine {
@ -311,7 +312,7 @@ func (rd *remoteDelivery) AddRcpt(ctx context.Context, to string) error {
return err
}
if err := conn.Rcpt(ctx, to); err != nil {
if err := conn.Rcpt(ctx, to, opts); err != nil {
return moduleError(err)
}

View file

@ -154,7 +154,7 @@ func TestRemoteDelivery_NoMXFallback(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err == nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err == nil {
t.Fatal("Expected an error, got none")
}
@ -275,7 +275,7 @@ func TestRemoteDelivery_Abort(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -305,7 +305,7 @@ func TestRemoteDelivery_CommitWithoutBody(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -342,7 +342,7 @@ func TestRemoteDelivery_MAILFROMErr(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
testutils.CheckSMTPErr(t, err, 550, exterrors.EnhancedCode{5, 1, 2}, "mx.example.invalid. said: Hey")
if err := delivery.Abort(context.Background()); err != nil {
@ -368,7 +368,7 @@ func TestRemoteDelivery_NoMX(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err == nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err == nil {
t.Fatal("Expected an error, got none")
}
@ -398,7 +398,7 @@ func TestRemoteDelivery_NullMX(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
testutils.CheckSMTPErr(t, err, 556, exterrors.EnhancedCode{5, 1, 10}, "Domain does not accept email (null MX)")
if err := delivery.Abort(context.Background()); err != nil {
@ -429,7 +429,7 @@ func TestRemoteDelivery_Quarantined(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -475,10 +475,10 @@ func TestRemoteDelivery_MAILFROMErr_Repeated(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
testutils.CheckSMTPErr(t, err, 550, exterrors.EnhancedCode{5, 1, 2}, "mx.example.invalid. said: Hey")
err = delivery.AddRcpt(context.Background(), "test2@example.invalid")
err = delivery.AddRcpt(context.Background(), "test2@example.invalid", smtp.RcptOptions{})
testutils.CheckSMTPErr(t, err, 550, exterrors.EnhancedCode{5, 1, 2}, "mx.example.invalid. said: Hey")
if err := delivery.Abort(context.Background()); err != nil {
@ -515,12 +515,12 @@ func TestRemoteDelivery_RcptErr(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
testutils.CheckSMTPErr(t, err, 550, exterrors.EnhancedCode{5, 1, 2}, "mx.example.invalid. said: Hey")
// It should be possible to, however, add another recipient and continue
// delivery as if nothing happened.
if err := delivery.AddRcpt(context.Background(), "test2@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test2@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -659,14 +659,14 @@ func TestRemoteDelivery_Split_Fail(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
if err == nil {
t.Fatal("Expected an error, got none")
}
// It should be possible to, however, add another recipient and continue
// delivery as if nothing happened.
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -712,7 +712,7 @@ func TestRemoteDelivery_BodyErr(t *testing.T) {
t.Fatal(err)
}
err = delivery.AddRcpt(context.Background(), "test@example.invalid")
err = delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{})
if err != nil {
t.Fatal(err)
}
@ -766,10 +766,10 @@ func TestRemoteDelivery_Split_BodyErr(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
@ -822,13 +822,13 @@ func TestRemoteDelivery_Split_BodyErr_NonAtomic(t *testing.T) {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test2@example.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test2@example.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid"); err != nil {
if err := delivery.AddRcpt(context.Background(), "test@example2.invalid", smtp.RcptOptions{}); err != nil {
t.Fatal(err)
}

View file

@ -251,8 +251,8 @@ func (d *delivery) connect(ctx context.Context) error {
return nil
}
func (d *delivery) AddRcpt(ctx context.Context, rcptTo string) error {
err := d.conn.Rcpt(ctx, rcptTo)
func (d *delivery) AddRcpt(ctx context.Context, rcptTo string, opts smtp.RcptOptions) error {
err := d.conn.Rcpt(ctx, rcptTo, opts)
if err != nil {
return d.u.moduleError(err)
}

View file

@ -29,6 +29,7 @@ import (
"testing"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/module"
)
@ -124,7 +125,7 @@ func BenchDelivery(b *testing.B, target module.DeliveryTarget, sender string, re
for i, rcptTemplate := range recipientTemplates {
rcpt := strings.Replace(rcptTemplate, "X", strconv.Itoa(i), -1)
if err := delivery.AddRcpt(benchCtx, rcpt); err != nil {
if err := delivery.AddRcpt(benchCtx, rcpt, smtp.RcptOptions{}); err != nil {
b.Fatal(err)
}
}

View file

@ -25,6 +25,7 @@ import (
"net"
"reflect"
"sort"
"sync/atomic"
"testing"
"time"
@ -53,10 +54,13 @@ type SMTPBackend struct {
RcptErr map[string]error
DataErr error
LMTPDataErr []error
ActiveSessionsCounter atomic.Int32
}
func (be *SMTPBackend) NewSession(conn *smtp.Conn) (smtp.Session, error) {
be.SessionCounter++
be.ActiveSessionsCounter.Add(1)
if be.SourceEndpoints == nil {
be.SourceEndpoints = make(map[string]struct{})
}
@ -67,6 +71,10 @@ func (be *SMTPBackend) NewSession(conn *smtp.Conn) (smtp.Session, error) {
}, nil
}
func (be *SMTPBackend) ConnectionCount() int {
return int(be.ActiveSessionsCounter.Load())
}
func (be *SMTPBackend) CheckMsg(t *testing.T, indx int, from string, rcptTo []string) {
t.Helper()
@ -104,6 +112,7 @@ func (s *session) Reset() {
}
func (s *session) Logout() error {
s.backend.ActiveSessionsCounter.Add(-1)
return nil
}
@ -129,7 +138,7 @@ func (s *session) Mail(from string, opts *smtp.MailOptions) error {
return nil
}
func (s *session) Rcpt(to string) error {
func (s *session) Rcpt(to string, _ *smtp.RcptOptions) error {
if err := s.backend.RcptErr[to]; err != nil {
return err
}
@ -368,17 +377,23 @@ func SMTPServerTLS(t *testing.T, addr string, fn ...SMTPServerConfigureFunc) (*t
return clientCfg, be, s
}
type smtpBackendConnCounter interface {
ConnectionCount() int
}
func CheckSMTPConnLeak(t *testing.T, srv *smtp.Server) {
t.Helper()
ccb, ok := srv.Backend.(smtpBackendConnCounter)
if !ok {
t.Error("CheckSMTPConnLeak used for smtp.Server with backend without ConnectionCount method")
return
}
// Connection closure is handled asynchronously, so before failing
// wait a bit for handleQuit in go-smtp to do its work.
for i := 0; i < 10; i++ {
found := false
srv.ForEachConn(func(_ *smtp.Conn) {
found = true
})
if !found {
if ccb.ConnectionCount() == 0 {
return
}
time.Sleep(100 * time.Millisecond)

View file

@ -29,6 +29,7 @@ import (
"testing"
"github.com/emersion/go-message/textproto"
"github.com/emersion/go-smtp"
"github.com/foxcpp/maddy/framework/buffer"
"github.com/foxcpp/maddy/framework/config"
"github.com/foxcpp/maddy/framework/exterrors"
@ -100,7 +101,7 @@ func (dt *Target) Start(ctx context.Context, msgMeta *module.MsgMetadata, mailFr
}, dt.StartErr
}
func (dtd *testTargetDelivery) AddRcpt(ctx context.Context, to string) error {
func (dtd *testTargetDelivery) AddRcpt(ctx context.Context, to string, _ smtp.RcptOptions) error {
if dtd.tgt.RcptErr != nil {
if err := dtd.tgt.RcptErr[to]; err != nil {
return err
@ -219,7 +220,7 @@ func DoTestDeliveryNonAtomic(t *testing.T, c module.StatusCollector, tgt module.
}
for _, rcpt := range to {
t.Log("-- delivery.AddRcpt", rcpt)
if err := delivery.AddRcpt(testCtx, rcpt); err != nil {
if err := delivery.AddRcpt(testCtx, rcpt, smtp.RcptOptions{}); err != nil {
t.Log("-- ... delivery.AddRcpt", rcpt, err, exterrors.Fields(err))
t.Log("-- delivery.Abort")
if err := delivery.Abort(testCtx); err != nil {
@ -269,7 +270,7 @@ func DoTestDeliveryErrMeta(t *testing.T, tgt module.DeliveryTarget, from string,
}
for _, rcpt := range to {
t.Log("-- delivery.AddRcpt", rcpt)
if err := delivery.AddRcpt(testCtx, rcpt); err != nil {
if err := delivery.AddRcpt(testCtx, rcpt, smtp.RcptOptions{}); err != nil {
t.Log("-- ... delivery.AddRcpt", rcpt, err, exterrors.Fields(err))
t.Log("-- delivery.Abort")
if err := delivery.Abort(testCtx); err != nil {

View file

@ -51,14 +51,14 @@ func TestConcurrencyLimit(tt *testing.T) {
c1 := t.Conn("smtp")
defer c1.Close()
c1.SMTPNegotation("localhost", nil, nil)
c1.Writeln("MAIL FROM:<testing@maddy.test")
c1.Writeln("MAIL FROM:<testing@maddy.test>")
c1.ExpectPattern("250 *")
// Down on semaphore.
c2 := t.Conn("smtp")
defer c2.Close()
c2.SMTPNegotation("localhost", nil, nil)
c1.Writeln("MAIL FROM:<testing@maddy.test")
c1.Writeln("MAIL FROM:<testing@maddy.test>")
// Temporary error due to lock timeout.
c1.ExpectPattern("451 *")
}
@ -87,21 +87,21 @@ func TestPerIPConcurrency(tt *testing.T) {
c1 := t.Conn("smtp")
defer c1.Close()
c1.SMTPNegotation("localhost", nil, nil)
c1.Writeln("MAIL FROM:<testing@maddy.test")
c1.Writeln("MAIL FROM:<testing@maddy.test>")
c1.ExpectPattern("250 *")
// Down on semaphore.
c3 := t.Conn4("127.0.0.2", "smtp")
defer c3.Close()
c3.SMTPNegotation("localhost", nil, nil)
c3.Writeln("MAIL FROM:<testing@maddy.test")
c3.Writeln("MAIL FROM:<testing@maddy.test>")
c3.ExpectPattern("250 *")
// Down on semaphore (different IP).
c2 := t.Conn("smtp")
defer c2.Close()
c2.SMTPNegotation("localhost", nil, nil)
c1.Writeln("MAIL FROM:<testing@maddy.test")
c1.Writeln("MAIL FROM:<testing@maddy.test>")
// Temporary error due to lock timeout.
c1.ExpectPattern("451 *")
}

View file

@ -60,7 +60,7 @@ func TestSMTPFlood_FullMsg_NoLimits_1Conn(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"DATA",
"From: <from@maddy.test>",
@ -103,7 +103,7 @@ func TestSMTPFlood_FullMsg_NoLimits_10Conns(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"DATA",
"From: <from@maddy.test>",
@ -151,7 +151,7 @@ func TestSMTPFlood_EnvelopeAbort_NoLimits_10Conns(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"RSET",
}, []string{
@ -202,7 +202,7 @@ func TestSMTPFlood_EnvelopeAbort_Ratelimited(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"RSET",
}, []string{
@ -265,7 +265,7 @@ func TestSMTPFlood_FullMsg_Ratelimited_PerSource(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@1.maddy.test",
"MAIL FROM:<from@1.maddy.test>",
"RCPT TO:<to@maddy.test>",
"DATA",
"From: <from@1.maddy.test>",
@ -292,7 +292,7 @@ func TestSMTPFlood_FullMsg_Ratelimited_PerSource(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@2.maddy.test",
"MAIL FROM:<from@2.maddy.test>",
"RCPT TO:<to@maddy.test>",
"DATA",
"From: <from@1.maddy.test>",
@ -364,7 +364,7 @@ func TestSMTPFlood_EnvelopeAbort_Ratelimited_PerIP(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"RSET",
}, []string{
@ -383,7 +383,7 @@ func TestSMTPFlood_EnvelopeAbort_Ratelimited_PerIP(tt *testing.T) {
defer c.Close()
c.SMTPNegotation("helo.maddy.test", nil, nil)
floodSmtp(&c, []string{
"MAIL FROM:<from@maddy.test",
"MAIL FROM:<from@maddy.test>",
"RCPT TO:<to@maddy.test>",
"RSET",
}, []string{