mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-04 05:37:34 +03:00
msgpipeline: Allow to chain pipelines
This allows for some complex but useful configurations, such as making decision on delivery target based on the result of per-destination address rewriting. One example where that can be useful is aliasing local address to a remote address in a way that can't make the server an open relay.
This commit is contained in:
parent
ed2a7c28e2
commit
af4f180503
6 changed files with 72 additions and 5 deletions
2
dist/vim/syntax/maddy-conf.vim
vendored
2
dist/vim/syntax/maddy-conf.vim
vendored
|
@ -61,6 +61,7 @@ syn keyword maddyModule
|
|||
\ test_check
|
||||
\ test_modifier
|
||||
\ verify_dkim
|
||||
\ aliases_file
|
||||
|
||||
syn keyword maddyDispatchDir
|
||||
\ check
|
||||
|
@ -71,6 +72,7 @@ syn keyword maddyDispatchDir
|
|||
\ destination
|
||||
\ reject
|
||||
\ deliver_to
|
||||
\ reroute
|
||||
|
||||
" grep --no-file -E 'cfg..+\(".+", ' **.go | sed -E 's/.+cfg..+\("([^"]+)", .+/\1/' | sort -u
|
||||
syn keyword maddyModDir
|
||||
|
|
|
@ -340,6 +340,36 @@ default_source {
|
|||
}
|
||||
```
|
||||
|
||||
*Syntax*: reroute { ... } ++
|
||||
*Context*: pipeline configuration, source block, destination block
|
||||
|
||||
This directive allows to make message routing decisions based on the
|
||||
result of modifiers. The block can contain all pipeline directives and they
|
||||
will be handled the same with the exception that source and destination rules
|
||||
will use the final recipient and sender values (e.g. after all modifiers are
|
||||
applied).
|
||||
|
||||
Here is the concrete example how it can be useful:
|
||||
```
|
||||
destination example.org {
|
||||
modify {
|
||||
alias_file /etc/maddy/aliases
|
||||
}
|
||||
reroute {
|
||||
destination example.org {
|
||||
deliver_to local_mailboxes
|
||||
}
|
||||
default_destination {
|
||||
deliver_to remote_queue
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
This configuration allows to specify alias local addresses to remote ones
|
||||
without being an open relay, since remote_queue can be used only if remote
|
||||
address was introduced as a result of rewrite of local address.
|
||||
|
||||
*Syntax*: destination _rule..._ { ... } ++
|
||||
*Context*: pipeline configuration, source block
|
||||
|
||||
|
|
|
@ -72,7 +72,7 @@ func (cr *checkRunner) checkStates(checks []module.Check) ([]module.CheckState,
|
|||
continue
|
||||
}
|
||||
|
||||
cr.log.Debugf("initializing state for %v (%p)", check.(module.Module).InstanceName(), check)
|
||||
cr.log.Debugf("initializing state for %v (%p)", objectName(check), check)
|
||||
state, err := check.CheckStateForMsg(cr.msgMeta)
|
||||
if err != nil {
|
||||
closeStates()
|
||||
|
|
|
@ -230,6 +230,17 @@ func parseMsgPipelineRcptCfg(globals map[string]interface{}, nodes []config.Node
|
|||
}
|
||||
|
||||
rcpt.targets = append(rcpt.targets, mod)
|
||||
case "reroute":
|
||||
if len(node.Children) == 0 {
|
||||
return nil, config.NodeErr(&node, "missing or empty reroute pipeline configuration")
|
||||
}
|
||||
|
||||
pipeline, err := New(globals, node.Children)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
rcpt.targets = append(rcpt.targets, pipeline)
|
||||
case "reject":
|
||||
if len(rcpt.targets) != 0 {
|
||||
return nil, config.NodeErr(&node, "can't use 'reject' and 'deliver_to' together")
|
||||
|
|
|
@ -476,14 +476,12 @@ func (dd *msgpipelineDelivery) getDelivery(tgt module.DeliveryTarget) (*delivery
|
|||
|
||||
deliveryObj, err := tgt.Start(dd.msgMeta, dd.sourceAddr)
|
||||
if err != nil {
|
||||
dd.log.Debugf("tgt.Start(%s) failure, target = %s (%s): %v",
|
||||
dd.sourceAddr, tgt.(module.Module).InstanceName(), tgt.(module.Module).Name(), err)
|
||||
dd.log.Debugf("tgt.Start(%s) failure, target = %s: %v", dd.sourceAddr, objectName(tgt), err)
|
||||
return nil, err
|
||||
}
|
||||
delivery_ = &delivery{Delivery: deliveryObj}
|
||||
|
||||
dd.log.Debugf("tgt.Start(%s) ok, target = %s (%s)",
|
||||
dd.sourceAddr, tgt.(module.Module).InstanceName(), tgt.(module.Module).Name())
|
||||
dd.log.Debugf("tgt.Start(%s) ok, target = %s (%s)", dd.sourceAddr, objectName(tgt))
|
||||
|
||||
dd.deliveries[tgt] = delivery_
|
||||
return delivery_, nil
|
||||
|
|
26
msgpipeline/objname.go
Normal file
26
msgpipeline/objname.go
Normal file
|
@ -0,0 +1,26 @@
|
|||
package msgpipeline
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/foxcpp/maddy/module"
|
||||
)
|
||||
|
||||
// objectName returns a new that is usable to identify the used external
|
||||
// component (module or some stub) in debug logs.
|
||||
func objectName(x interface{}) string {
|
||||
mod, ok := x.(module.Module)
|
||||
if ok {
|
||||
if mod.InstanceName() == "" {
|
||||
return mod.Name()
|
||||
}
|
||||
return mod.InstanceName()
|
||||
}
|
||||
|
||||
_, pipeline := x.(*MsgPipeline)
|
||||
if pipeline {
|
||||
return "reroute"
|
||||
}
|
||||
|
||||
return fmt.Sprintf("%T", x)
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue