mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 14:07:38 +03:00
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:
parent
d1df9f60be
commit
a4b4706dbb
15 changed files with 91 additions and 48 deletions
|
@ -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)
|
||||||
|
|
26
config.go
26
config.go
|
@ -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])
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
13
imap.go
|
@ -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)
|
||||||
|
|
18
maddy.conf
18
maddy.conf
|
@ -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
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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.
|
||||||
|
|
11
maddy.go
11
maddy.go
|
@ -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}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
2
queue.go
2
queue.go
|
@ -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,
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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,
|
||||||
|
|
9
smtp.go
9
smtp.go
|
@ -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
2
sql.go
|
@ -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"},
|
||||||
|
|
|
@ -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,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue