mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-03 21:27:35 +03:00
392 lines
11 KiB
Go
392 lines
11 KiB
Go
package maddy
|
|
|
|
import (
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"runtime/debug"
|
|
"strings"
|
|
|
|
"github.com/foxcpp/maddy/internal/config"
|
|
"github.com/foxcpp/maddy/internal/hooks"
|
|
"github.com/foxcpp/maddy/internal/log"
|
|
"github.com/foxcpp/maddy/internal/module"
|
|
parser "github.com/foxcpp/maddy/pkg/cfgparser"
|
|
|
|
// Import packages for side-effect of module registration.
|
|
_ "github.com/foxcpp/maddy/internal/auth/external"
|
|
_ "github.com/foxcpp/maddy/internal/auth/pam"
|
|
_ "github.com/foxcpp/maddy/internal/auth/pass_table"
|
|
_ "github.com/foxcpp/maddy/internal/auth/plain_separate"
|
|
_ "github.com/foxcpp/maddy/internal/auth/shadow"
|
|
_ "github.com/foxcpp/maddy/internal/check/command"
|
|
_ "github.com/foxcpp/maddy/internal/check/dkim"
|
|
_ "github.com/foxcpp/maddy/internal/check/dns"
|
|
_ "github.com/foxcpp/maddy/internal/check/dnsbl"
|
|
_ "github.com/foxcpp/maddy/internal/check/requiretls"
|
|
_ "github.com/foxcpp/maddy/internal/check/spf"
|
|
_ "github.com/foxcpp/maddy/internal/endpoint/imap"
|
|
_ "github.com/foxcpp/maddy/internal/endpoint/smtp"
|
|
_ "github.com/foxcpp/maddy/internal/modify"
|
|
_ "github.com/foxcpp/maddy/internal/modify/dkim"
|
|
_ "github.com/foxcpp/maddy/internal/storage/imapsql"
|
|
_ "github.com/foxcpp/maddy/internal/table"
|
|
_ "github.com/foxcpp/maddy/internal/target/queue"
|
|
_ "github.com/foxcpp/maddy/internal/target/remote"
|
|
_ "github.com/foxcpp/maddy/internal/target/smtp_downstream"
|
|
)
|
|
|
|
var (
|
|
Version = "go-build"
|
|
|
|
// ConfigDirectory specifies platform-specific value
|
|
// that should be used as a location of default configuration
|
|
//
|
|
// It should not be changed and is defined as a variable
|
|
// only for purposes of modification using -X linker flag.
|
|
ConfigDirectory = "/etc/maddy"
|
|
|
|
// DefaultStateDirectory specifies platform-specific
|
|
// default for StateDirectory.
|
|
//
|
|
// Most code should use StateDirectory instead since
|
|
// it will contain the effective location of the state
|
|
// directory.
|
|
//
|
|
// It should not be changed and is defined as a variable
|
|
// only for purposes of modification using -X linker flag.
|
|
DefaultStateDirectory = "/var/lib/maddy"
|
|
|
|
// DefaultRuntimeDirectory specifies platform-specific
|
|
// default for RuntimeDirectory.
|
|
//
|
|
// Most code should use RuntimeDirectory instead since
|
|
// it will contain the effective location of the state
|
|
// directory.
|
|
//
|
|
// It should not be changed and is defined as a variable
|
|
// only for purposes of modification using -X linker flag.
|
|
DefaultRuntimeDirectory = "/run/maddy"
|
|
|
|
// DefaultLibexecDirectory specifies platform-specific
|
|
// default for LibexecDirectory.
|
|
//
|
|
// Most code should use LibexecDirectory since it will
|
|
// contain the effective location of the libexec
|
|
// directory.
|
|
//
|
|
// It should not be changed and is defined as a variable
|
|
// only for purposes of modification using -X linker flag.
|
|
DefaultLibexecDirectory = "/usr/lib/maddy"
|
|
|
|
enableDebugFlags = false
|
|
profileEndpoint *string
|
|
blockProfileRate *int
|
|
mutexProfileFract *int
|
|
)
|
|
|
|
func BuildInfo() string {
|
|
version := Version
|
|
if info, ok := debug.ReadBuildInfo(); ok && info.Main.Version != "(devel)" {
|
|
version = info.Main.Version
|
|
}
|
|
|
|
return fmt.Sprintf(`%s %s/%s %s
|
|
|
|
default config: %s
|
|
default state_dir: %s
|
|
default runtime_dir: %s`,
|
|
version, runtime.GOOS, runtime.GOARCH, runtime.Version(),
|
|
filepath.Join(ConfigDirectory, "maddy.conf"),
|
|
DefaultStateDirectory,
|
|
DefaultRuntimeDirectory)
|
|
}
|
|
|
|
// Run is the entry point for all maddy code. It takes care of command line arguments parsing,
|
|
// logging initialization, directives setup, configuration reading. After all that, it
|
|
// calls moduleMain to initialize and run modules.
|
|
func Run() int {
|
|
flag.StringVar(&config.LibexecDirectory, "libexec", DefaultLibexecDirectory, "path to the libexec directory")
|
|
flag.BoolVar(&log.DefaultLogger.Debug, "debug", false, "enable debug logging early")
|
|
|
|
var (
|
|
configPath = flag.String("config", filepath.Join(ConfigDirectory, "maddy.conf"), "path to configuration file")
|
|
logTargets = flag.String("log", "stderr", "default logging target(s)")
|
|
printVersion = flag.Bool("v", false, "print versio, build metadata and exit")
|
|
)
|
|
|
|
if enableDebugFlags {
|
|
profileEndpoint = flag.String("debug.pprof", "", "enable live profiler HTTP endpoint and listen on the specified address")
|
|
blockProfileRate = flag.Int("debug.blockprofrate", 0, "set blocking profile rate")
|
|
mutexProfileFract = flag.Int("debug.mutexproffract", 0, "set mutex profile fraction")
|
|
}
|
|
|
|
flag.Parse()
|
|
|
|
if len(flag.Args()) != 0 {
|
|
fmt.Println("usage:", os.Args[0], "[options]")
|
|
return 2
|
|
}
|
|
|
|
if *printVersion {
|
|
fmt.Println("maddy", BuildInfo())
|
|
return 0
|
|
}
|
|
|
|
var err error
|
|
log.DefaultLogger.Out, err = LogOutputOption(strings.Split(*logTargets, ","))
|
|
if err != nil {
|
|
systemdStatusErr(err)
|
|
log.Println(err)
|
|
return 2
|
|
}
|
|
|
|
initDebug()
|
|
|
|
os.Setenv("PATH", config.LibexecDirectory+string(filepath.ListSeparator)+os.Getenv("PATH"))
|
|
|
|
f, err := os.Open(*configPath)
|
|
if err != nil {
|
|
systemdStatusErr(err)
|
|
log.Println(err)
|
|
return 2
|
|
}
|
|
defer f.Close()
|
|
|
|
cfg, err := parser.Read(f, *configPath)
|
|
if err != nil {
|
|
systemdStatusErr(err)
|
|
log.Println(err)
|
|
return 2
|
|
}
|
|
|
|
if err := moduleMain(cfg); err != nil {
|
|
systemdStatusErr(err)
|
|
log.Println(err)
|
|
return 2
|
|
}
|
|
|
|
return 0
|
|
}
|
|
|
|
func initDebug() {
|
|
if !enableDebugFlags {
|
|
return
|
|
}
|
|
|
|
if *profileEndpoint != "" {
|
|
go func() {
|
|
log.Println("listening on", "http://"+*profileEndpoint, "for profiler requests")
|
|
log.Println("failed to listen on profiler endpoint:", http.ListenAndServe(*profileEndpoint, nil))
|
|
}()
|
|
}
|
|
|
|
// These values can also be affected by environment so set them
|
|
// only if argument is specified.
|
|
if *mutexProfileFract != 0 {
|
|
runtime.SetMutexProfileFraction(*mutexProfileFract)
|
|
}
|
|
if *blockProfileRate != 0 {
|
|
runtime.SetBlockProfileRate(*blockProfileRate)
|
|
}
|
|
}
|
|
|
|
func InitDirs() error {
|
|
if config.StateDirectory == "" {
|
|
config.StateDirectory = DefaultStateDirectory
|
|
}
|
|
if config.RuntimeDirectory == "" {
|
|
config.RuntimeDirectory = DefaultRuntimeDirectory
|
|
}
|
|
if config.LibexecDirectory == "" {
|
|
config.LibexecDirectory = DefaultLibexecDirectory
|
|
}
|
|
|
|
if err := ensureDirectoryWritable(config.StateDirectory); err != nil {
|
|
return err
|
|
}
|
|
if err := ensureDirectoryWritable(config.RuntimeDirectory); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Make sure all paths we are going to use are absolute
|
|
// before we change the working directory.
|
|
if !filepath.IsAbs(config.StateDirectory) {
|
|
return errors.New("statedir should be absolute")
|
|
}
|
|
if !filepath.IsAbs(config.RuntimeDirectory) {
|
|
return errors.New("runtimedir should be absolute")
|
|
}
|
|
if !filepath.IsAbs(config.LibexecDirectory) {
|
|
return errors.New("-libexec should be absolute")
|
|
}
|
|
|
|
// Change the working directory to make all relative paths
|
|
// in configuration relative to state directory.
|
|
if err := os.Chdir(config.StateDirectory); err != nil {
|
|
log.Println(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func ensureDirectoryWritable(path string) error {
|
|
if err := os.MkdirAll(path, 0700); err != nil {
|
|
return err
|
|
}
|
|
|
|
testFile, err := os.Create(filepath.Join(path, "writeable-test"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
testFile.Close()
|
|
if err := os.Remove(testFile.Name()); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func moduleMain(cfg []config.Node) error {
|
|
globals := config.NewMap(nil, config.Node{Children: cfg})
|
|
globals.String("state_dir", false, false, DefaultStateDirectory, &config.StateDirectory)
|
|
globals.String("runtime_dir", false, false, DefaultRuntimeDirectory, &config.RuntimeDirectory)
|
|
globals.String("hostname", false, false, "", nil)
|
|
globals.String("autogenerated_msg_domain", false, false, "", nil)
|
|
globals.Custom("tls", false, false, nil, config.TLSDirective, nil)
|
|
globals.Bool("storage_perdomain", false, false, nil)
|
|
globals.Bool("auth_perdomain", false, false, nil)
|
|
globals.StringList("auth_domains", false, false, nil, nil)
|
|
globals.Custom("log", false, false, defaultLogOutput, logOutput, &log.DefaultLogger.Out)
|
|
globals.Bool("debug", false, log.DefaultLogger.Debug, &log.DefaultLogger.Debug)
|
|
globals.AllowUnknown()
|
|
unknown, err := globals.Process()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := InitDirs(); err != nil {
|
|
return err
|
|
}
|
|
|
|
defer log.DefaultLogger.Out.Close()
|
|
|
|
hooks.AddHook(hooks.EventLogRotate, reinitLogging)
|
|
|
|
_, err = instancesFromConfig(globals.Values, unknown)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
systemdStatus(SDReady, "Listening for incoming connections...")
|
|
|
|
handleSignals()
|
|
|
|
systemdStatus(SDStopping, "Waiting for running transactions to complete...")
|
|
|
|
hooks.RunHooks(hooks.EventShutdown)
|
|
|
|
return nil
|
|
}
|
|
|
|
type modInfo struct {
|
|
instance module.Module
|
|
cfg config.Node
|
|
}
|
|
|
|
func instancesFromConfig(globals map[string]interface{}, nodes []config.Node) ([]module.Module, error) {
|
|
var (
|
|
endpoints []modInfo
|
|
mods = make([]modInfo, 0, len(nodes))
|
|
)
|
|
|
|
for _, block := range nodes {
|
|
var instName string
|
|
var modAliases []string
|
|
if len(block.Args) == 0 {
|
|
instName = block.Name
|
|
} else {
|
|
instName = block.Args[0]
|
|
modAliases = block.Args[1:]
|
|
}
|
|
|
|
modName := block.Name
|
|
|
|
endpFactory := module.GetEndpoint(modName)
|
|
if endpFactory != nil {
|
|
inst, err := endpFactory(modName, block.Args)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
endpoints = append(endpoints, modInfo{instance: inst, cfg: block})
|
|
continue
|
|
}
|
|
|
|
factory := module.Get(modName)
|
|
if factory == nil {
|
|
return nil, config.NodeErr(block, "unknown module or global directive: %s", modName)
|
|
}
|
|
|
|
if module.HasInstance(instName) {
|
|
return nil, config.NodeErr(block, "config block named %s already exists", instName)
|
|
}
|
|
|
|
inst, err := factory(modName, instName, modAliases, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
block := block
|
|
module.RegisterInstance(inst, config.NewMap(globals, block))
|
|
for _, alias := range modAliases {
|
|
if module.HasInstance(alias) {
|
|
return nil, config.NodeErr(block, "config block named %s already exists", alias)
|
|
}
|
|
module.RegisterAlias(alias, instName)
|
|
}
|
|
mods = append(mods, modInfo{instance: inst, cfg: block})
|
|
}
|
|
|
|
if len(endpoints) == 0 {
|
|
return nil, fmt.Errorf("at least one endpoint should be configured")
|
|
}
|
|
|
|
for _, endp := range endpoints {
|
|
if err := endp.instance.Init(config.NewMap(globals, endp.cfg)); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if closer, ok := endp.instance.(io.Closer); ok {
|
|
endp := endp
|
|
hooks.AddHook(hooks.EventShutdown, func() {
|
|
log.Debugf("close %s (%s)", endp.instance.Name(), endp.instance.InstanceName())
|
|
if err := closer.Close(); err != nil {
|
|
log.Printf("module %s (%s) close failed: %v", endp.instance.Name(), endp.instance.InstanceName(), err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
for _, inst := range mods {
|
|
if module.Initialized[inst.instance.InstanceName()] {
|
|
continue
|
|
}
|
|
|
|
return nil, fmt.Errorf("Unused configuration block at %s:%d - %s (%s)",
|
|
inst.cfg.File, inst.cfg.Line, inst.instance.InstanceName(), inst.instance.Name())
|
|
}
|
|
|
|
res := make([]module.Module, 0, len(mods)+len(endpoints))
|
|
for _, endp := range endpoints {
|
|
res = append(res, endp.instance)
|
|
}
|
|
for _, mod := range mods {
|
|
res = append(res, mod.instance)
|
|
}
|
|
return res, nil
|
|
}
|