mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-03 05:07:38 +03:00
parent
f43b164170
commit
9dd34b85ef
2 changed files with 7 additions and 90 deletions
|
@ -86,7 +86,7 @@ Complete configuration example using all of the mentioned directives:
|
|||
check {
|
||||
# Run a check to make sure source SMTP server identification
|
||||
# is legit.
|
||||
require_matching_ehlo
|
||||
spf
|
||||
}
|
||||
|
||||
# Messages coming from senders at example.org will be handled in
|
||||
|
@ -129,12 +129,12 @@ Example:
|
|||
```
|
||||
check {
|
||||
# Reference implicitly defined default configuration for check.
|
||||
require_matching_ehlo
|
||||
spf
|
||||
|
||||
# Inline definition of custom config.
|
||||
require_source_mx {
|
||||
# Configuration for require_source_mx goes here.
|
||||
fail_action reject
|
||||
spf {
|
||||
# Configuration for spf goes here.
|
||||
permerr_action reject
|
||||
}
|
||||
}
|
||||
```
|
||||
|
@ -143,7 +143,8 @@ It is also possible to define the block of checks at the top level
|
|||
as "checks" module and reference it using & syntax. Example:
|
||||
```
|
||||
checks inbound_checks {
|
||||
require_matching_ehlo
|
||||
spf
|
||||
dkim
|
||||
}
|
||||
|
||||
# ... somewhere else ...
|
||||
|
|
|
@ -19,7 +19,6 @@ along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|||
package dns
|
||||
|
||||
import (
|
||||
"net"
|
||||
"strings"
|
||||
|
||||
"github.com/foxcpp/maddy/framework/address"
|
||||
|
@ -156,92 +155,9 @@ func requireMXRecord(ctx check.StatelessCheckContext, mailFrom string) module.Ch
|
|||
|
||||
return module.CheckResult{}
|
||||
}
|
||||
|
||||
func requireMatchingEHLO(ctx check.StatelessCheckContext) module.CheckResult {
|
||||
ctx.Logger.Printf("require_matching_echo is deprecated and will be removed in the next release")
|
||||
|
||||
if ctx.MsgMeta.Conn == nil {
|
||||
ctx.Logger.Printf("locally-generated message, skipping")
|
||||
return module.CheckResult{}
|
||||
}
|
||||
|
||||
tcpAddr, ok := ctx.MsgMeta.Conn.RemoteAddr.(*net.TCPAddr)
|
||||
if !ok {
|
||||
ctx.Logger.Printf("non-TCP/IP source, skipped")
|
||||
return module.CheckResult{}
|
||||
}
|
||||
|
||||
ehlo := ctx.MsgMeta.Conn.Hostname
|
||||
|
||||
if strings.HasPrefix(ehlo, "[") && strings.HasSuffix(ehlo, "]") {
|
||||
// IP in EHLO, checking against source IP directly.
|
||||
|
||||
ehlo = ehlo[1 : len(ehlo)-1]
|
||||
ehlo = strings.TrimPrefix(ehlo, "IPv6:")
|
||||
ehloIP := net.ParseIP(ehlo)
|
||||
|
||||
if ehloIP == nil {
|
||||
return module.CheckResult{
|
||||
Reason: &exterrors.SMTPError{
|
||||
Code: 550,
|
||||
EnhancedCode: exterrors.EnhancedCode{5, 7, 0},
|
||||
Message: "Malformed IP in EHLO",
|
||||
CheckName: "require_matching_ehlo",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
if !ehloIP.Equal(tcpAddr.IP) {
|
||||
return module.CheckResult{
|
||||
Reason: &exterrors.SMTPError{
|
||||
Code: 550,
|
||||
EnhancedCode: exterrors.EnhancedCode{5, 7, 0},
|
||||
Message: "IP in EHLO is not the same as the actual client IP",
|
||||
CheckName: "require_matching_ehlo",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
return module.CheckResult{}
|
||||
}
|
||||
|
||||
srcIPs, err := ctx.Resolver.LookupIPAddr(ctx, dns.FQDN(ehlo))
|
||||
if err != nil {
|
||||
reason, misc := exterrors.UnwrapDNSErr(err)
|
||||
return module.CheckResult{
|
||||
Reason: &exterrors.SMTPError{
|
||||
Code: exterrors.SMTPCode(err, 450, 550),
|
||||
EnhancedCode: exterrors.SMTPEnchCode(err, exterrors.EnhancedCode{0, 7, 0}),
|
||||
Message: "DNS error during policy check",
|
||||
CheckName: "require_matching_ehlo",
|
||||
Err: err,
|
||||
Reason: reason,
|
||||
Misc: misc,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
for _, ip := range srcIPs {
|
||||
if tcpAddr.IP.Equal(ip.IP) {
|
||||
ctx.Logger.Debugf("A/AAA record found for %s for %s domain", tcpAddr.IP, ehlo)
|
||||
return module.CheckResult{}
|
||||
}
|
||||
}
|
||||
return module.CheckResult{
|
||||
Reason: &exterrors.SMTPError{
|
||||
Code: 550,
|
||||
EnhancedCode: exterrors.EnhancedCode{5, 7, 0},
|
||||
Message: "No matching A/AAA records found for the EHLO hostname",
|
||||
CheckName: "require_matching_ehlo",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func init() {
|
||||
check.RegisterStatelessCheck("require_matching_rdns", modconfig.FailAction{Quarantine: true},
|
||||
requireMatchingRDNS, nil, nil, nil)
|
||||
check.RegisterStatelessCheck("require_mx_record", modconfig.FailAction{Quarantine: true},
|
||||
nil, requireMXRecord, nil, nil)
|
||||
check.RegisterStatelessCheck("require_matching_ehlo", modconfig.FailAction{Quarantine: true},
|
||||
requireMatchingEHLO, nil, nil, nil)
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue