maddy/config.go
fox.cpp 7583e418cb
Rework how check results are reported and processed
In general, the checks interface with added scoring and quarantining
support was not convenient to use enough. Also it was problematic
to add support for Authentication-Results header field generation.

Per-sender and per-recipient checks were not applied to body.
This is fixed now.

Checks inspecting the message header was able to see header
modifications done by other checks. This could lead to unwanted
side-effects and so now checks can't modify the header directly
and instead can only prepend fields to it by returning them.

Additionally, it allows checks to return values for
Authentication-Results field. Each server handling the message should
add only one field, so it is not possible to implement it using header
prepending.

MsgMetadata.CheckScore is removed, now it is managed internally by
dispatcher code and not exposed where it is not needed.

MsgMetadata.Quarantine is no longer set directly by checks code. Future
refactoring may be remove it altogether as it is discouraged to have
mutable flags in MsgMetadata.

On top of that, tests are added for all new code.
2019-08-31 01:15:48 +03:00

251 lines
7.4 KiB
Go

package maddy
import (
"errors"
"fmt"
"os"
"reflect"
"strconv"
"github.com/foxcpp/maddy/config"
"github.com/foxcpp/maddy/log"
"github.com/foxcpp/maddy/module"
)
/*
Config matchers for module interfaces.
*/
// createInlineModule is a helper function for config matchers that can create inline modules.
func createInlineModule(modName, instName string, aliases []string) (module.Module, error) {
newMod := module.Get(modName)
if newMod == nil {
return nil, fmt.Errorf("unknown module: %s", modName)
}
log.Debugln("module create", modName, instName, "(inline)")
return newMod(modName, instName, aliases)
}
// initInlineModule constructs "faked" config tree and passes it to module
// Init function to make it look like it is defined at top-level.
//
// args must contain at least one argument, otherwise initInlineModule panics.
func initInlineModule(modObj module.Module, globals map[string]interface{}, block *config.Node) error {
log.Debugln("module init", modObj.Name(), modObj.InstanceName(), "(inline)")
return modObj.Init(config.NewMap(globals, block))
}
// moduleFromNode does all work to create or get existing module object with a certain type.
// It is not used by top-level module definitions, only for references from other
// modules configuration blocks.
//
// inlineCfg should contain configuration directives for inline declarations.
// args should contain values that are used to create module.
// It should be either module name + instance name or just module name. Further extensions
// may add other string arguments (currently, they can be accessed by module instances
// as aliases argument to constructor).
//
// It checks using reflection whether it is possible to store a module object into modObj
// pointer (e.g. it implements all necessary interfaces) and stores it if everything is fine.
// If module object doesn't implement necessary module interfaces - error is returned.
// If modObj is not a pointer, moduleFromNode panics.
func moduleFromNode(args []string, inlineCfg *config.Node, globals map[string]interface{}, moduleIface interface{}) error {
// single argument
// - instance name of an existing module
// single argument + block
// - module name, inline definition
// two+ arguments + block
// - module name and instance name, inline definition
// two+ arguments, no block
// - module name and instance name, inline definition, empty config block
if len(args) == 0 {
return config.NodeErr(inlineCfg, "at least one argument is required")
}
var modObj module.Module
var err error
if inlineCfg.Children != nil || len(args) > 1 {
modName := args[0]
modAliases := args[1:]
instName := ""
if len(args) >= 2 {
modAliases = args[2:]
instName = args[1]
}
modObj, err = createInlineModule(modName, instName, modAliases)
} else {
if len(args) != 1 {
return config.NodeErr(inlineCfg, "exactly one argument is to use existing config block")
}
modObj, err = module.GetInstance(args[0])
}
if err != nil {
return config.NodeErr(inlineCfg, "%v", err)
}
// NOTE: This will panic if moduleIface is not a pointer.
modIfaceType := reflect.TypeOf(moduleIface).Elem()
modObjType := reflect.TypeOf(modObj)
if !modObjType.Implements(modIfaceType) && !modObjType.AssignableTo(modIfaceType) {
return config.NodeErr(inlineCfg, "module %s (%s) doesn't implement %v interface", modObj.Name(), modObj.InstanceName(), modIfaceType)
}
reflect.ValueOf(moduleIface).Elem().Set(reflect.ValueOf(modObj))
if inlineCfg.Children != nil {
if err := initInlineModule(modObj, globals, inlineCfg); err != nil {
return err
}
}
return nil
}
// deliveryDirective is a callback for use in config.Map.Custom.
//
// It does all work necessary to create a module instance from the config
// directive with the following structure:
// directive_name mod_name [inst_name] [{
// inline_mod_config
// }]
//
// Note that if used configuration structure lacks directive_name before mod_name - this function
// should not be used (call deliveryTarget directly).
func deliveryDirective(m *config.Map, node *config.Node) (interface{}, error) {
return deliveryTarget(m.Globals, node.Args, node)
}
func deliveryTarget(globals map[string]interface{}, args []string, block *config.Node) (module.DeliveryTarget, error) {
var target module.DeliveryTarget
if err := moduleFromNode(args, block, globals, &target); err != nil {
return nil, err
}
return target, nil
}
func messageCheck(globals map[string]interface{}, args []string, block *config.Node) (module.Check, error) {
var check module.Check
if err := moduleFromNode(args, block, globals, &check); err != nil {
return nil, err
}
return check, nil
}
func authDirective(m *config.Map, node *config.Node) (interface{}, error) {
var provider module.AuthProvider
if err := moduleFromNode(node.Args, node, m.Globals, &provider); err != nil {
return nil, err
}
return provider, nil
}
func storageDirective(m *config.Map, node *config.Node) (interface{}, error) {
var backend module.Storage
if err := moduleFromNode(node.Args, node, m.Globals, &backend); err != nil {
return nil, err
}
return backend, nil
}
type checkFailAction struct {
quarantine bool
reject bool
scoreAdjust int
}
func checkFailActionDirective(m *config.Map, node *config.Node) (interface{}, error) {
if len(node.Args) == 0 {
return nil, m.MatchErr("expected at least 1 argument")
}
if len(node.Children) != 0 {
return nil, m.MatchErr("can't declare block here")
}
switch node.Args[0] {
case "reject", "quarantine":
if len(node.Args) > 1 {
return nil, m.MatchErr("too many arguments")
}
return checkFailAction{
reject: node.Args[0] == "reject",
quarantine: node.Args[0] == "quarantine",
}, nil
case "score":
if len(node.Args) != 2 {
return nil, m.MatchErr("expected 2 arguments")
}
scoreAdj, err := strconv.Atoi(node.Args[1])
if err != nil {
return nil, m.MatchErr("%v", err)
}
return checkFailAction{
scoreAdjust: scoreAdj,
}, nil
default:
return nil, m.MatchErr("invalid action")
}
}
// apply merges the result of check execution with action configuration specified
// in the check configuration.
func (cfa checkFailAction) apply(originalRes module.CheckResult) module.CheckResult {
if originalRes.RejectErr == nil {
return originalRes
}
originalRes.Quarantine = cfa.quarantine
originalRes.ScoreAdjust = int32(cfa.scoreAdjust)
if !cfa.reject {
originalRes.RejectErr = nil
}
return originalRes
}
func logOutput(m *config.Map, node *config.Node) (interface{}, error) {
if len(node.Args) == 0 {
return nil, m.MatchErr("expected at least 1 argument")
}
if len(node.Children) != 0 {
return nil, m.MatchErr("can't declare block here")
}
outs := make([]log.FuncLog, 0, len(node.Args))
for _, arg := range node.Args {
switch arg {
case "stderr":
outs = append(outs, log.StderrLog())
case "syslog":
syslogOut, err := log.Syslog()
if err != nil {
return nil, fmt.Errorf("failed to connect to syslog daemon: %v", err)
}
outs = append(outs, syslogOut)
case "off":
if len(node.Args) != 1 {
return nil, errors.New("'off' can't be combined with other log targets")
}
return nil, nil
default:
w, err := os.OpenFile(arg, os.O_RDWR, os.ModePerm)
if err != nil {
return nil, fmt.Errorf("failed to create log file: %v", err)
}
outs = append(outs, log.WriterLog(w))
}
}
if len(outs) == 1 {
return outs[0], nil
}
return log.MultiLog(outs...), nil
}
func defaultLogOutput() (interface{}, error) {
return log.StderrLog(), nil
}