maddy/internal/check/stateless_check.go
fox.cpp 48e21f566e
Extend .debug.* flags and hide them by default
Allow to override DNS resolver address via the -debug.dnsoverride flag
and SMTP port via -debug.smtpport.

All flags are not available unless maddy is built using the 'debugflags'
tag.
2019-12-13 17:31:35 +03:00

165 lines
4.9 KiB
Go

package check
import (
"fmt"
"github.com/emersion/go-message/textproto"
"github.com/foxcpp/maddy/internal/buffer"
"github.com/foxcpp/maddy/internal/config"
"github.com/foxcpp/maddy/internal/dns"
"github.com/foxcpp/maddy/internal/log"
"github.com/foxcpp/maddy/internal/module"
"github.com/foxcpp/maddy/internal/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 Reason 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: dns.DefaultResolver(),
logger: log.Logger{Name: modName},
defaultFailAction: defaultFailAction,
connCheck: connCheck,
senderCheck: senderCheck,
rcptCheck: rcptCheck,
bodyCheck: bodyCheck,
}, nil
})
}