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{}}) }