module: Allow config blocks to have more than one name

This allows more readable configuration files without additional
explanations in cases where a single module is used for multiple
purposes.

Also cleans up certain problems with modules that rely on block
names having certain semantics (e.g. endpoint modules).
This commit is contained in:
fox.cpp 2019-08-27 19:39:49 +03:00
parent d1df9f60be
commit a4b4706dbb
No known key found for this signature in database
GPG key ID: E76D97CCEDE90B6C
15 changed files with 91 additions and 48 deletions

View file

@ -59,7 +59,7 @@ func (tcs *testCheckState) Close() error {
} }
func init() { func init() {
module.Register("test_check", func(modName, instanceName string) (module.Module, error) { module.Register("test_check", func(modName, instanceName string, aliases []string) (module.Module, error) {
return &testCheck{}, nil return &testCheck{}, nil
}) })
module.RegisterInstance(&testCheck{}, nil) module.RegisterInstance(&testCheck{}, nil)

View file

@ -16,11 +16,7 @@ Config matchers for module interfaces.
*/ */
// createInlineModule is a helper function for config matchers that can create inline modules. // createInlineModule is a helper function for config matchers that can create inline modules.
func createInlineModule(modName, instName string) (module.Module, error) { func createInlineModule(modName, instName string, aliases []string) (module.Module, error) {
if instName != "" && module.HasInstance(instName) {
return nil, fmt.Errorf("module instance named %s already exists", instName)
}
newMod := module.Get(modName) newMod := module.Get(modName)
if newMod == nil { if newMod == nil {
return nil, fmt.Errorf("unknown module: %s", modName) return nil, fmt.Errorf("unknown module: %s", modName)
@ -28,7 +24,7 @@ func createInlineModule(modName, instName string) (module.Module, error) {
log.Debugln("module create", modName, instName, "(inline)") log.Debugln("module create", modName, instName, "(inline)")
return newMod(modName, instName) return newMod(modName, instName, aliases)
} }
// initInlineModule constructs "faked" config tree and passes it to module // initInlineModule constructs "faked" config tree and passes it to module
@ -37,13 +33,7 @@ func createInlineModule(modName, instName string) (module.Module, error) {
// args must contain at least one argument, otherwise initInlineModule panics. // args must contain at least one argument, otherwise initInlineModule panics.
func initInlineModule(modObj module.Module, globals map[string]interface{}, args []string, block *config.Node) error { func initInlineModule(modObj module.Module, globals map[string]interface{}, args []string, block *config.Node) error {
log.Debugln("module init", modObj.Name(), modObj.InstanceName(), "(inline)") log.Debugln("module init", modObj.Name(), modObj.InstanceName(), "(inline)")
return modObj.Init(config.NewMap(globals, &config.Node{ return modObj.Init(config.NewMap(globals, block))
Name: args[0],
Args: args[1:],
Children: block.Children,
File: block.File,
Line: block.Line,
}))
} }
// moduleFromNode does all work to create or get existing module object with a certain type. // moduleFromNode does all work to create or get existing module object with a certain type.
@ -53,8 +43,8 @@ func initInlineModule(modObj module.Module, globals map[string]interface{}, args
// inlineCfg should contain configuration directives for inline declarations. // inlineCfg should contain configuration directives for inline declarations.
// args should contain values that are used to create module. // args should contain values that are used to create module.
// It should be either module name + instance name or just module name. Further extensions // 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 code using // may add other string arguments (currently, they can be accessed by module instances
// Values field of config.Map passed to Init). // as aliases argument to constructor).
// //
// It checks using reflection whether it is possible to store a module object into modObj // 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. // pointer (e.g. it implements all necessary interfaces) and stores it if everything is fine.
@ -80,15 +70,17 @@ func moduleFromNode(args []string, inlineCfg *config.Node, globals map[string]in
if inlineCfg.Children != nil { if inlineCfg.Children != nil {
modName := args[0] modName := args[0]
modAliases := args[1:]
instName := "" instName := ""
if len(args) == 2 { if len(args) == 2 {
modAliases = args[2:]
instName = args[1] instName = args[1]
} }
modObj, err = createInlineModule(modName, instName) modObj, err = createInlineModule(modName, instName, modAliases)
} else { } else {
if len(args) != 1 { if len(args) != 1 {
return config.NodeErr(inlineCfg, "exactly one argument is required") return config.NodeErr(inlineCfg, "exactly one argument is required for reference to existing module")
} }
modObj, err = module.GetInstance(args[0]) modObj, err = module.GetInstance(args[0])
} }

View file

@ -25,7 +25,7 @@ type ExternalAuth struct {
Log log.Logger Log log.Logger
} }
func NewExternalAuth(modName, instName string) (module.Module, error) { func NewExternalAuth(modName, instName string, _ []string) (module.Module, error) {
ea := &ExternalAuth{ ea := &ExternalAuth{
modName: modName, modName: modName,
instName: instName, instName: instName,

13
imap.go
View file

@ -27,6 +27,7 @@ import (
type IMAPEndpoint struct { type IMAPEndpoint struct {
name string name string
aliases []string
serv *imapserver.Server serv *imapserver.Server
listeners []net.Listener listeners []net.Listener
Auth module.AuthProvider Auth module.AuthProvider
@ -39,10 +40,11 @@ type IMAPEndpoint struct {
Log log.Logger Log log.Logger
} }
func NewIMAPEndpoint(_, instName string) (module.Module, error) { func NewIMAPEndpoint(_, instName string, aliases []string) (module.Module, error) {
endp := &IMAPEndpoint{ endp := &IMAPEndpoint{
name: instName, name: instName,
Log: log.Logger{Name: "imap"}, aliases: aliases,
Log: log.Logger{Name: "imap"},
} }
endp.name = instName endp.name = instName
@ -71,8 +73,9 @@ func (endp *IMAPEndpoint) Init(cfg *config.Map) error {
return fmt.Errorf("imap: storage module %T does not implement imapbackend.BackendUpdater", endp.Store) return fmt.Errorf("imap: storage module %T does not implement imapbackend.BackendUpdater", endp.Store)
} }
addresses := make([]Address, 0, len(cfg.Block.Args)) args := append([]string{endp.name}, endp.aliases...)
for _, addr := range cfg.Block.Args { addresses := make([]Address, 0, len(args))
for _, addr := range args {
saddr, err := standardizeAddress(addr) saddr, err := standardizeAddress(addr)
if err != nil { if err != nil {
return fmt.Errorf("imap: invalid address: %s", endp.name) return fmt.Errorf("imap: invalid address: %s", endp.name)

View file

@ -11,7 +11,7 @@ auth_domains example.org
# Create and initialize sql module, it provides simple authentication and # Create and initialize sql module, it provides simple authentication and
# storage backend using one database for everything. # storage backend using one database for everything.
sql { sql local_mailboxes local_authdb {
driver sqlite3 driver sqlite3
dsn /var/lib/maddy/all.db dsn /var/lib/maddy/all.db
} }
@ -25,7 +25,7 @@ smtp smtp://0.0.0.0:25 {
# All messages for the recipients at example.org should be # All messages for the recipients at example.org should be
# delivered to local mailboxes. # delivered to local mailboxes.
destination example.org { destination example.org {
deliver_to sql deliver_to local_mailboxes
} }
# Other recipients are rejected because we are not an open relay. # Other recipients are rejected because we are not an open relay.
@ -36,21 +36,21 @@ smtp smtp://0.0.0.0:25 {
submission smtps://0.0.0.0:465 smtp://0.0.0.0:587 { submission smtps://0.0.0.0:465 smtp://0.0.0.0:587 {
# Use sql module for authentication. # Use sql module for authentication.
auth sql auth local_auth
# All messages for the recipients at example.org should be # All messages for the recipients at example.org should be
# delivered to local mailboxes directly. # delivered to local mailboxes directly.
destination example.org { destination example.org {
deliver_to sql deliver_to local_mailboxes
} }
# Remaining recipients are scheduled for remote delivery. # Remaining recipients are scheduled for remote delivery.
default_destination { default_destination {
deliver_to out_queue deliver_to remote_queue
} }
} }
queue out_queue { queue remote_queue {
# Try to deliver message up to 8 tries. Note that this counter is not per # Try to deliver message up to 8 tries. Note that this counter is not per
# recipient, but for entire message. # recipient, but for entire message.
max_tries 8 max_tries 8
@ -67,8 +67,6 @@ queue out_queue {
remote { } remote { }
imap imaps://0.0.0.0:993 imap://0.0.0.0:143 { imap imaps://0.0.0.0:993 imap://0.0.0.0:143 {
# Use sql module for authentication. auth local_authdb
auth sql storage local_mailboxes
# And also for storage.
storage sql
} }

View file

@ -174,13 +174,15 @@ module2 config2 {
Generic syntax for module configuration block is as follows: Generic syntax for module configuration block is as follows:
``` ```
module_name config_block_name { module_name config_block_name optional_aliases... {
configuration configuration
directives directives
for_this for_this
module module
} }
``` ```
If you specify more than one config_block_name, they all will be usable.
Basically, they will be aliased to the first name.
If config_block_name is omitted, it will be the same as module_name. If config_block_name is omitted, it will be the same as module_name.
Configuration block name must be unique across all configuration. Configuration block name must be unique across all configuration.

View file

@ -36,10 +36,12 @@ func Start(cfg []config.Node) error {
for _, block := range unmatched { for _, block := range unmatched {
var instName string var instName string
var modAliases []string
if len(block.Args) == 0 { if len(block.Args) == 0 {
instName = block.Name instName = block.Name
} else { } else {
instName = block.Args[0] instName = block.Args[0]
modAliases = block.Args[1:]
} }
modName := block.Name modName := block.Name
@ -54,13 +56,20 @@ func Start(cfg []config.Node) error {
} }
log.Debugln("module create", modName, instName) log.Debugln("module create", modName, instName)
inst, err := factory(modName, instName) inst, err := factory(modName, instName, modAliases)
if err != nil { if err != nil {
return err return err
} }
block := block block := block
module.RegisterInstance(inst, config.NewMap(globals.Values, &block)) module.RegisterInstance(inst, config.NewMap(globals.Values, &block))
for _, alias := range modAliases {
if module.HasInstance(alias) {
return config.NodeErr(&block, "module instance named %s already exists", alias)
}
module.RegisterAlias(alias, instName)
log.Debugln("module alias", alias, "->", instName)
}
instances[instName] = modInfo{instance: inst, cfg: block} instances[instName] = modInfo{instance: inst, cfg: block}
} }

View file

@ -12,13 +12,14 @@ var (
mod Module mod Module
cfg *config.Map cfg *config.Map
}) })
aliases = make(map[string]string)
Initialized = make(map[string]bool) Initialized = make(map[string]bool)
) )
// Register adds module instance to the global registry. // RegisterInstance adds module instance to the global registry.
// //
// Instnace name must be unique. Second RegisterInstance with same instance // Instance name must be unique. Second RegisterInstance with same instance
// name will replace previous. // name will replace previous.
func RegisterInstance(inst Module, cfg *config.Map) { func RegisterInstance(inst Module, cfg *config.Map) {
instances[inst.InstanceName()] = struct { instances[inst.InstanceName()] = struct {
@ -27,7 +28,20 @@ func RegisterInstance(inst Module, cfg *config.Map) {
}{inst, cfg} }{inst, cfg}
} }
// RegisterAlias creates an association between a certain name and instance name.
//
// After RegisterAlias, module.GetInstance(aliasName) will return the same
// result as module.GetInstance(instName).
func RegisterAlias(aliasName, instName string) {
aliases[aliasName] = instName
}
func HasInstance(name string) bool { func HasInstance(name string) bool {
aliasedName := aliases[name]
if aliasedName != "" {
name = aliasedName
}
_, ok := instances[name] _, ok := instances[name]
return ok return ok
} }
@ -38,6 +52,11 @@ func HasInstance(name string) bool {
// Error is returned if module initialization fails or module instance does not // Error is returned if module initialization fails or module instance does not
// exists. // exists.
func GetInstance(name string) (Module, error) { func GetInstance(name string) (Module, error) {
aliasedName := aliases[name]
if aliasedName != "" {
name = aliasedName
}
mod, ok := instances[name] mod, ok := instances[name]
if !ok { if !ok {
return nil, fmt.Errorf("unknown module instance: %s", name) return nil, fmt.Errorf("unknown module instance: %s", name)

View file

@ -46,5 +46,22 @@ type Module interface {
} }
// FuncNewModule is function that creates new instance of module with specified name. // FuncNewModule is function that creates new instance of module with specified name.
// Note that this function should not do any form of initialization. //
type FuncNewModule func(modName, instanceName string) (Module, error) // Module.InstanceName() of the returned module object should return instName.
// aliases slice contains other names that can be used to reference created
// module instance. Here is the example of top-level declarations and the
// corresponding arguments passed to module constructor:
//
// modname { }
// Arguments: "modname", "modname", nil.
//
// modname instname { }
// Arguments: "modname", "instname", nil
//
// modname instname secondname1 secondname2 { }
// Arguments: "modname", "instname", []string{"secondname1", "secondname2"}
//
// Note modules are allowed to attach additional meaning to used names.
// For example, endpoint modules use instance name and aliases as addresses
// to listen on.
type FuncNewModule func(modName, instName string, aliases []string) (Module, error)

View file

@ -91,7 +91,7 @@ type QueueMetadata struct {
LastAttempt time.Time LastAttempt time.Time
} }
func NewQueue(_, instName string) (module.Module, error) { func NewQueue(_, instName string, _ []string) (module.Module, error) {
return &Queue{ return &Queue{
name: instName, name: instName,
initialRetryTime: 15 * time.Minute, initialRetryTime: 15 * time.Minute,

View file

@ -38,7 +38,7 @@ func cleanQueue(t *testing.T, q *Queue) {
} }
func newTestQueueDir(t *testing.T, target module.DeliveryTarget, dir string) *Queue { func newTestQueueDir(t *testing.T, target module.DeliveryTarget, dir string) *Queue {
mod, _ := NewQueue("", "queue") mod, _ := NewQueue("", "queue", nil)
q := mod.(*Queue) q := mod.(*Queue)
q.Log = testLogger(t, "queue") q.Log = testLogger(t, "queue")
q.initialRetryTime = 0 q.initialRetryTime = 0

View file

@ -41,7 +41,7 @@ type RemoteTarget struct {
var _ module.DeliveryTarget = &RemoteTarget{} var _ module.DeliveryTarget = &RemoteTarget{}
func NewRemoteTarget(_, instName string) (module.Module, error) { func NewRemoteTarget(_, instName string, _ []string) (module.Module, error) {
return &RemoteTarget{ return &RemoteTarget{
name: instName, name: instName,
resolver: net.DefaultResolver, resolver: net.DefaultResolver,

View file

@ -138,6 +138,7 @@ type SMTPEndpoint struct {
Auth module.AuthProvider Auth module.AuthProvider
serv *smtp.Server serv *smtp.Server
name string name string
aliases []string
listeners []net.Listener listeners []net.Listener
dispatcher *Dispatcher dispatcher *Dispatcher
@ -158,9 +159,10 @@ func (endp *SMTPEndpoint) InstanceName() string {
return endp.name return endp.name
} }
func NewSMTPEndpoint(modName, instName string) (module.Module, error) { func NewSMTPEndpoint(modName, instName string, aliases []string) (module.Module, error) {
endp := &SMTPEndpoint{ endp := &SMTPEndpoint{
name: instName, name: instName,
aliases: aliases,
submission: modName == "submission", submission: modName == "submission",
Log: log.Logger{Name: "smtp"}, Log: log.Logger{Name: "smtp"},
} }
@ -177,8 +179,9 @@ func (endp *SMTPEndpoint) Init(cfg *config.Map) error {
endp.Log.Debugf("authentication provider: %s %s", endp.Auth.(module.Module).Name(), endp.Auth.(module.Module).InstanceName()) endp.Log.Debugf("authentication provider: %s %s", endp.Auth.(module.Module).Name(), endp.Auth.(module.Module).InstanceName())
} }
addresses := make([]Address, 0, len(cfg.Block.Args)) args := append([]string{endp.name}, endp.aliases...)
for _, addr := range cfg.Block.Args { addresses := make([]Address, 0, len(args))
for _, addr := range args {
saddr, err := standardizeAddress(addr) saddr, err := standardizeAddress(addr)
if err != nil { if err != nil {
return fmt.Errorf("smtp: invalid address: %s", addr) return fmt.Errorf("smtp: invalid address: %s", addr)

2
sql.go
View file

@ -152,7 +152,7 @@ func (sqlm *SQLStorage) InstanceName() string {
return sqlm.instName return sqlm.instName
} }
func NewSQLStorage(_, instName string) (module.Module, error) { func NewSQLStorage(_, instName string, _ []string) (module.Module, error) {
return &SQLStorage{ return &SQLStorage{
instName: instName, instName: instName,
Log: log.Logger{Name: "sql"}, Log: log.Logger{Name: "sql"},

View file

@ -145,7 +145,7 @@ func (c *statelessCheck) InstanceName() string {
// It creates the module and its instance with the specified name that implement module.Check interface // 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. // and runs passed functions when corresponding module.CheckState methods are called.
func RegisterStatelessCheck(name string, connCheck FuncConnCheck, senderCheck FuncSenderCheck, rcptCheck FuncRcptCheck, bodyCheck FuncBodyCheck) { func RegisterStatelessCheck(name string, connCheck FuncConnCheck, senderCheck FuncSenderCheck, rcptCheck FuncRcptCheck, bodyCheck FuncBodyCheck) {
module.Register(name, func(modName, instName string) (module.Module, error) { module.Register(name, func(modName, instName string, aliases []string) (module.Module, error) {
return &statelessCheck{ return &statelessCheck{
modName: modName, modName: modName,
instName: instName, instName: instName,