mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-06 06:27:38 +03:00
Check.NewMessage is renamed to Check.CheckStateForMsg to allow a single module to implement both Modifier and Check interfaces. Due to various implementation details, body and MAIL FROM rewrites are not applied on per-destination basis. Closes #134.
78 lines
1.6 KiB
Go
78 lines
1.6 KiB
Go
package modify
|
|
|
|
import (
|
|
"github.com/emersion/go-message/textproto"
|
|
"github.com/foxcpp/maddy/buffer"
|
|
"github.com/foxcpp/maddy/module"
|
|
)
|
|
|
|
type (
|
|
// Group wraps multiple modifiers and runs them serially.
|
|
Group struct {
|
|
Modifiers []module.Modifier
|
|
}
|
|
|
|
groupState struct {
|
|
states []module.ModifierState
|
|
}
|
|
)
|
|
|
|
func (g Group) ModStateForMsg(msgMeta *module.MsgMetadata) (module.ModifierState, error) {
|
|
gs := groupState{}
|
|
for _, modifier := range g.Modifiers {
|
|
state, err := modifier.ModStateForMsg(msgMeta)
|
|
if err != nil {
|
|
// Free state objects we initialized already.
|
|
for _, state := range gs.states {
|
|
state.Close()
|
|
}
|
|
return nil, err
|
|
}
|
|
gs.states = append(gs.states, state)
|
|
}
|
|
return gs, nil
|
|
}
|
|
|
|
func (gs groupState) RewriteSender(mailFrom string) (string, error) {
|
|
var err error
|
|
for _, state := range gs.states {
|
|
mailFrom, err = state.RewriteSender(mailFrom)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
return mailFrom, nil
|
|
}
|
|
|
|
func (gs groupState) RewriteRcpt(rcptTo string) (string, error) {
|
|
var err error
|
|
for _, state := range gs.states {
|
|
rcptTo, err = state.RewriteRcpt(rcptTo)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
return rcptTo, nil
|
|
}
|
|
|
|
func (gs groupState) RewriteBody(h textproto.Header, body buffer.Buffer) error {
|
|
for _, state := range gs.states {
|
|
if err := state.RewriteBody(h, body); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (gs groupState) Close() error {
|
|
// We still try close all state objects to minimize
|
|
// resource leaks when Close fails for one object..
|
|
|
|
var lastErr error
|
|
for _, state := range gs.states {
|
|
if err := state.Close(); err != nil {
|
|
lastErr = err
|
|
}
|
|
}
|
|
return lastErr
|
|
}
|