mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 14:07:38 +03:00
It fits poorly with limited amount of checks that are (and will be) implemented in maddy. Advanced filtering that requires "spam score" logic should be performed by external software such as rspamd. At this point duplicating that logic in maddy makes no sense, since it is highly problematic to integrate it with external software.
186 lines
5.5 KiB
Go
186 lines
5.5 KiB
Go
package check
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
|
|
"github.com/emersion/go-message/textproto"
|
|
"github.com/foxcpp/maddy/buffer"
|
|
"github.com/foxcpp/maddy/config"
|
|
"github.com/foxcpp/maddy/dns"
|
|
"github.com/foxcpp/maddy/log"
|
|
"github.com/foxcpp/maddy/module"
|
|
"github.com/foxcpp/maddy/target"
|
|
)
|
|
|
|
type (
|
|
StatelessCheckContext struct {
|
|
// Resolver that should be used by the check for DNS queries.
|
|
Resolver dns.Resolver
|
|
|
|
MsgMeta *module.MsgMetadata
|
|
|
|
// Logger that should be used by the check for logging, note that it is
|
|
// already wrapped to append Msg ID to all messages so check code
|
|
// should not do the same.
|
|
Logger log.Logger
|
|
}
|
|
FuncConnCheck func(checkContext StatelessCheckContext) module.CheckResult
|
|
FuncSenderCheck func(checkContext StatelessCheckContext, mailFrom string) module.CheckResult
|
|
FuncRcptCheck func(checkContext StatelessCheckContext, rcptTo string) module.CheckResult
|
|
FuncBodyCheck func(checkContext StatelessCheckContext, header textproto.Header, body buffer.Buffer) module.CheckResult
|
|
)
|
|
|
|
type statelessCheck struct {
|
|
modName string
|
|
instName string
|
|
resolver dns.Resolver
|
|
logger log.Logger
|
|
|
|
// One used by Init if config option is not passed by a user.
|
|
defaultFailAction FailAction
|
|
// The actual fail action that should be applied.
|
|
failAction FailAction
|
|
|
|
connCheck FuncConnCheck
|
|
senderCheck FuncSenderCheck
|
|
rcptCheck FuncRcptCheck
|
|
bodyCheck FuncBodyCheck
|
|
}
|
|
|
|
type statelessCheckState struct {
|
|
c *statelessCheck
|
|
msgMeta *module.MsgMetadata
|
|
}
|
|
|
|
func (s statelessCheckState) CheckConnection() module.CheckResult {
|
|
if s.c.connCheck == nil {
|
|
return module.CheckResult{}
|
|
}
|
|
|
|
originalRes := s.c.connCheck(StatelessCheckContext{
|
|
Resolver: s.c.resolver,
|
|
MsgMeta: s.msgMeta,
|
|
Logger: target.DeliveryLogger(s.c.logger, s.msgMeta),
|
|
})
|
|
return s.c.failAction.Apply(originalRes)
|
|
}
|
|
|
|
func (s statelessCheckState) CheckSender(mailFrom string) module.CheckResult {
|
|
if s.c.senderCheck == nil {
|
|
return module.CheckResult{}
|
|
}
|
|
|
|
originalRes := s.c.senderCheck(StatelessCheckContext{
|
|
Resolver: s.c.resolver,
|
|
MsgMeta: s.msgMeta,
|
|
Logger: target.DeliveryLogger(s.c.logger, s.msgMeta),
|
|
}, mailFrom)
|
|
return s.c.failAction.Apply(originalRes)
|
|
}
|
|
|
|
func (s statelessCheckState) CheckRcpt(rcptTo string) module.CheckResult {
|
|
if s.c.rcptCheck == nil {
|
|
return module.CheckResult{}
|
|
}
|
|
|
|
originalRes := s.c.rcptCheck(StatelessCheckContext{
|
|
Resolver: s.c.resolver,
|
|
MsgMeta: s.msgMeta,
|
|
Logger: target.DeliveryLogger(s.c.logger, s.msgMeta),
|
|
}, rcptTo)
|
|
return s.c.failAction.Apply(originalRes)
|
|
}
|
|
|
|
func (s statelessCheckState) CheckBody(header textproto.Header, body buffer.Buffer) module.CheckResult {
|
|
if s.c.bodyCheck == nil {
|
|
return module.CheckResult{}
|
|
}
|
|
|
|
originalRes := s.c.bodyCheck(StatelessCheckContext{
|
|
Resolver: s.c.resolver,
|
|
MsgMeta: s.msgMeta,
|
|
Logger: target.DeliveryLogger(s.c.logger, s.msgMeta),
|
|
}, header, body)
|
|
return s.c.failAction.Apply(originalRes)
|
|
}
|
|
|
|
func (s statelessCheckState) Close() error {
|
|
return nil
|
|
}
|
|
|
|
func (c *statelessCheck) CheckStateForMsg(ctx *module.MsgMetadata) (module.CheckState, error) {
|
|
return statelessCheckState{
|
|
c: c,
|
|
msgMeta: ctx,
|
|
}, nil
|
|
}
|
|
|
|
func (c *statelessCheck) Init(cfg *config.Map) error {
|
|
cfg.Bool("debug", true, false, &c.logger.Debug)
|
|
cfg.Custom("fail_action", false, false,
|
|
func() (interface{}, error) {
|
|
return c.defaultFailAction, nil
|
|
}, FailActionDirective, &c.failAction)
|
|
_, err := cfg.Process()
|
|
return err
|
|
}
|
|
|
|
func (c *statelessCheck) Name() string {
|
|
return c.modName
|
|
}
|
|
|
|
func (c *statelessCheck) InstanceName() string {
|
|
return c.instName
|
|
}
|
|
|
|
// RegisterStatelessCheck is helper function to create stateless message check modules
|
|
// that run one simple check during one stage.
|
|
//
|
|
// It creates the module and its instance with the specified name that implement module.Check interface
|
|
// and runs passed functions when corresponding module.CheckState methods are called.
|
|
//
|
|
// Note about CheckResult that is returned by the functions:
|
|
// StatelessCheck supports different action types based on the user configuration, but the particular check
|
|
// code doesn't need to know about it. It should assume that it is always "Reject" and hence it should
|
|
// populate RejectErr field of the result object with the relevant error description.
|
|
func RegisterStatelessCheck(name string, defaultFailAction FailAction, connCheck FuncConnCheck, senderCheck FuncSenderCheck, rcptCheck FuncRcptCheck, bodyCheck FuncBodyCheck) {
|
|
module.Register(name, func(modName, instName string, aliases, inlineArgs []string) (module.Module, error) {
|
|
if len(inlineArgs) != 0 {
|
|
return nil, fmt.Errorf("%s: inline arguments are not used", modName)
|
|
}
|
|
return &statelessCheck{
|
|
modName: modName,
|
|
instName: instName,
|
|
resolver: net.DefaultResolver,
|
|
logger: log.Logger{Name: modName},
|
|
|
|
defaultFailAction: defaultFailAction,
|
|
|
|
connCheck: connCheck,
|
|
senderCheck: senderCheck,
|
|
rcptCheck: rcptCheck,
|
|
bodyCheck: bodyCheck,
|
|
}, nil
|
|
})
|
|
|
|
// Here is the problem with global configuration.
|
|
// We can't grab it here because this function is likely
|
|
// called from init(). This RegisterInstance call
|
|
// needs to be moved somewhere after global config parsing
|
|
// so we will be able to pass globals to config.Map constructed
|
|
// here and then let Init access it.
|
|
// TODO.
|
|
|
|
module.RegisterInstance(&statelessCheck{
|
|
modName: name,
|
|
instName: name,
|
|
resolver: net.DefaultResolver,
|
|
logger: log.Logger{Name: name},
|
|
|
|
connCheck: connCheck,
|
|
senderCheck: senderCheck,
|
|
rcptCheck: rcptCheck,
|
|
bodyCheck: bodyCheck,
|
|
}, &config.Map{Block: &config.Node{}})
|
|
}
|