Implement lazy initialization of module instances

It is correct fix for initialization order issue introduced in
https://github.com/emersion/maddy/pull/24.
This commit is contained in:
fox.cpp 2019-04-05 22:56:23 +03:00 committed by Simon Ser
parent 7029bfca0f
commit 0ddf540d35
6 changed files with 95 additions and 52 deletions

View file

@ -15,9 +15,9 @@ Config matchers for module interfaces.
*/ */
func authProvider(modName string) (module.AuthProvider, error) { func authProvider(modName string) (module.AuthProvider, error) {
modObj := module.GetInstance(modName) modObj, err := module.GetInstance(modName)
if modObj == nil { if err != nil {
return nil, fmt.Errorf("unknown auth. provider instance: %s", modName) return nil, err
} }
provider, ok := modObj.(module.AuthProvider) provider, ok := modObj.(module.AuthProvider)
@ -44,9 +44,9 @@ func authDirective(m *config.Map, node *config.Node) (interface{}, error) {
} }
func storageBackend(modName string) (module.Storage, error) { func storageBackend(modName string) (module.Storage, error) {
modObj := module.GetInstance(modName) modObj, err := module.GetInstance(modName)
if modObj == nil { if err != nil {
return nil, fmt.Errorf("unknown storage backend instance: %s", modName) return nil, err
} }
backend, ok := modObj.(module.Storage) backend, ok := modObj.(module.Storage)
@ -73,9 +73,9 @@ func storageDirective(m *config.Map, node *config.Node) (interface{}, error) {
} }
func deliveryTarget(modName string) (module.DeliveryTarget, error) { func deliveryTarget(modName string) (module.DeliveryTarget, error) {
mod := module.GetInstance(modName) mod, err := module.GetInstance(modName)
if mod == nil { if mod == nil {
return nil, fmt.Errorf("unknown delivery target instance: %s", modName) return nil, err
} }
target, ok := mod.(module.DeliveryTarget) target, ok := mod.(module.DeliveryTarget)

View file

@ -37,5 +37,5 @@ func (d Dummy) Init(_ *config.Map) error {
func init() { func init() {
module.Register("dummy", nil) module.Register("dummy", nil)
module.RegisterInstance(&Dummy{instName: "dummy"}) module.RegisterInstance(&Dummy{instName: "dummy"}, nil)
} }

View file

@ -45,7 +45,7 @@ func Start(cfg []config.Node) error {
return fmt.Errorf("%s:%d: unknown module: %s", block.File, block.Line, modName) return fmt.Errorf("%s:%d: unknown module: %s", block.File, block.Line, modName)
} }
if mod := module.GetInstance(instName); mod != nil { if mod := module.GetUninitedInstance(instName); mod != nil {
return fmt.Errorf("%s:%d: module instance named %s already exists", block.File, block.Line, instName) return fmt.Errorf("%s:%d: module instance named %s already exists", block.File, block.Line, instName)
} }
@ -55,14 +55,21 @@ func Start(cfg []config.Node) error {
return err return err
} }
module.RegisterInstance(inst) block := block
module.RegisterInstance(inst, config.NewMap(globals.Values, &block))
instances[instName] = modInfo{instance: inst, cfg: block} instances[instName] = modInfo{instance: inst, cfg: block}
} }
addDefaultModule(instances, "default", createDefaultStorage, defaultStorageConfig) addDefaultModule(instances, "default", createDefaultStorage, globals.Values, defaultStorageConfig)
addDefaultModule(instances, "default_remote_delivery", createDefaultRemoteDelivery, nil) addDefaultModule(instances, "default_remote_delivery", createDefaultRemoteDelivery, globals.Values, nil)
for _, inst := range instances { for _, inst := range instances {
if module.Initialized[inst.instance.InstanceName()] {
log.Debugln("module init", inst.instance.Name(), inst.instance.InstanceName(), "skipped because it was lazily initialized before")
continue
}
module.Initialized[inst.instance.InstanceName()] = true
log.Debugln("module init", inst.instance.Name(), inst.instance.InstanceName()) log.Debugln("module init", inst.instance.Name(), inst.instance.InstanceName())
if err := inst.instance.Init(config.NewMap(globals.Values, &inst.cfg)); err != nil { if err := inst.instance.Init(config.NewMap(globals.Values, &inst.cfg)); err != nil {
return err return err
@ -90,17 +97,17 @@ func Start(cfg []config.Node) error {
return nil return nil
} }
func addDefaultModule(insts map[string]modInfo, name string, factory func(string) (module.Module, error), cfgFactory func(string) config.Node) { func addDefaultModule(insts map[string]modInfo, name string, factory func(string) (module.Module, error), globalCfg map[string]interface{}, cfgFactory func(string) config.Node) {
if _, ok := insts[name]; !ok { if _, ok := insts[name]; !ok {
if mod, err := factory(name); err != nil { if mod, err := factory(name); err != nil {
log.Printf("failed to register %s: %v", name, err) log.Printf("failed to register %s: %v", name, err)
} else { } else {
log.Debugf("module create %s %s (built-in)", mod.Name(), name) log.Debugf("module create %s %s (built-in)", mod.Name(), name)
module.RegisterInstance(mod)
info := modInfo{instance: mod} info := modInfo{instance: mod}
if cfgFactory != nil { if cfgFactory != nil {
info.cfg = cfgFactory(name) info.cfg = cfgFactory(name)
} }
module.RegisterInstance(mod, config.NewMap(globalCfg, &info.cfg))
insts[name] = info insts[name] = info
} }
} else { } else {

69
module/instances.go Normal file
View file

@ -0,0 +1,69 @@
package module
import (
"fmt"
"github.com/emersion/maddy/config"
"github.com/emersion/maddy/log"
)
var (
Instances = make(map[string]struct {
mod Module
cfg *config.Map
})
Initialized = make(map[string]bool)
)
// Register adds module instance to the global registry.
//
// Instnace name must be unique. Second RegisterInstance with same instance
// name will replace previous.
func RegisterInstance(inst Module, cfg *config.Map) {
Instances[inst.InstanceName()] = struct {
mod Module
cfg *config.Map
}{inst, cfg}
}
// GetUninitedInstance returns module instance from global registry.
//
// Nil is returned if no module instance with specified name is Instances.
//
// Note that this function may return uninitialized module. If you need to ensure
// that it is safe to interact with module instance - use GetInstance instead.
func GetUninitedInstance(name string) Module {
mod, ok := Instances[name]
if !ok {
return nil
}
return mod.mod
}
// GetInstance returns module instance from global registry, initializing it if
// necessary.
//
// Error is returned if module initialization fails or module instance does not
// exists.
func GetInstance(name string) (Module, error) {
mod, ok := Instances[name]
if !ok {
return nil, fmt.Errorf("unknown module instance: %s", name)
}
// Break circular dependencies.
if Initialized[name] {
log.Debugln("module init", mod.mod.Name(), name, "(lazy init skipped)")
return mod.mod, nil
}
log.Debugln("module init", mod.mod.Name(), name, "(lazy)")
Initialized[name] = true
if err := mod.mod.Init(mod.cfg); err != nil {
return mod.mod, err
}
return mod.mod, nil
}

View file

@ -33,36 +33,3 @@ func Get(name string) FuncNewModule {
return modules[name] return modules[name]
} }
var instances = make(map[string]Module)
var instancesLck sync.RWMutex
// Register adds module instance to global registry.
//
// Instnace name must be unique. Second RegisterInstance with same instance
// name will replace previous.
func RegisterInstance(inst Module) {
instancesLck.Lock()
defer instancesLck.Unlock()
instances[inst.InstanceName()] = inst
}
// GetInstance returns module instance from global registry.
//
// Nil is returned if no module instance with specified name is registered.
func GetInstance(name string) Module {
instancesLck.RLock()
defer instancesLck.RUnlock()
return instances[name]
}
// UnregisterInstance undoes effect of RegisterInstance, removing instance from
// global registry.
func UnregisterInstance(name string) {
instancesLck.Lock()
defer instancesLck.Unlock()
delete(instances, name)
}

View file

@ -181,8 +181,8 @@ func filterStepFromCfg(node config.Node) (SMTPPipelineStep, error) {
return nil, errors.New("filter: expected at least one argument") return nil, errors.New("filter: expected at least one argument")
} }
modInst := module.GetInstance(node.Args[0]) modInst, err := module.GetInstance(node.Args[0])
if modInst == nil { if err != nil {
return nil, fmt.Errorf("filter: unknown module instance: %s", node.Args[0]) return nil, fmt.Errorf("filter: unknown module instance: %s", node.Args[0])
} }
@ -202,8 +202,8 @@ func deliveryStepFromCfg(node config.Node) (SMTPPipelineStep, error) {
return nil, errors.New("delivery: expected at least one argument") return nil, errors.New("delivery: expected at least one argument")
} }
modInst := module.GetInstance(node.Args[0]) modInst, err := module.GetInstance(node.Args[0])
if modInst == nil { if err != nil {
return nil, fmt.Errorf("delivery: unknown module instance: %s", node.Args[0]) return nil, fmt.Errorf("delivery: unknown module instance: %s", node.Args[0])
} }