package config import ( "errors" "fmt" "os" "path/filepath" "regexp" "github.com/tg123/go-htpasswd" ) var ( // SMTPListen to listen on : SMTPListen = "0.0.0.0:1025" // HTTPListen to listen on : HTTPListen = "0.0.0.0:8025" // DataFile for mail (optional) DataFile string // MaxMessages is the maximum number of messages a mailbox can have (auto-pruned every minute) MaxMessages = 500 // VerboseLogging for console output VerboseLogging = false // QuietLogging for console output (errors only) QuietLogging = false // NoLogging for tests NoLogging = false // UISSLCert file UISSLCert string // UISSLKey file UISSLKey string // UIAuthFile for basic authentication UIAuthFile string // UIAuth used for euthentication UIAuth *htpasswd.File // SMTPSSLCert file SMTPSSLCert string // SMTPSSLKey file SMTPSSLKey string // SMTPAuthFile for SMTP authentication SMTPAuthFile string // SMTPAuth used for euthentication SMTPAuth *htpasswd.File // ContentSecurityPolicy for HTTP server ContentSecurityPolicy = "default-src 'self'; script-src 'self'; style-src 'self' 'unsafe-inline'; frame-src 'self'; img-src * data: blob:; font-src 'self' data:; media-src 'self'; connect-src 'self' ws: wss:; object-src 'none'; base-uri 'self';" ) // VerifyConfig wil do some basic checking func VerifyConfig() error { if DataFile != "" && isDir(DataFile) { DataFile = filepath.Join(DataFile, "mailpit.db") } re := regexp.MustCompile(`^[a-zA-Z0-9\.\-]{3,}:\d{2,}$`) if !re.MatchString(SMTPListen) { return errors.New("SMTP bind should be in the format of :") } if !re.MatchString(HTTPListen) { return errors.New("HTTP bind should be in the format of :") } if UIAuthFile != "" { if !isFile(UIAuthFile) { return fmt.Errorf("HTTP password file not found: %s", UIAuthFile) } a, err := htpasswd.New(UIAuthFile, htpasswd.DefaultSystems, nil) if err != nil { return err } UIAuth = a } if UISSLCert != "" && UISSLKey == "" || UISSLCert == "" && UISSLKey != "" { return errors.New("you must provide both a UI SSL certificate and a key") } if UISSLCert != "" { if !isFile(UISSLCert) { return fmt.Errorf("SSL certificate not found: %s", UISSLCert) } if !isFile(UISSLKey) { return fmt.Errorf("SSL key not found: %s", UISSLKey) } } if SMTPSSLCert != "" && SMTPSSLKey == "" || SMTPSSLCert == "" && SMTPSSLKey != "" { return errors.New("you must provide both an SMTP SSL certificate and a key") } if SMTPSSLCert != "" { if !isFile(SMTPSSLCert) { return fmt.Errorf("SMTP SSL certificate not found: %s", SMTPSSLCert) } if !isFile(SMTPSSLKey) { return fmt.Errorf("SMTP SSL key not found: %s", SMTPSSLKey) } } if SMTPAuthFile != "" { if !isFile(SMTPAuthFile) { return fmt.Errorf("SMTP password file not found: %s", SMTPAuthFile) } if SMTPSSLCert == "" { return errors.New("SMTP authentication requires SMTP encryption") } a, err := htpasswd.New(SMTPAuthFile, htpasswd.DefaultSystems, nil) if err != nil { return err } SMTPAuth = a } return nil } // IsFile returns if a path is a file func isFile(path string) bool { info, err := os.Stat(path) if os.IsNotExist(err) || !info.Mode().IsRegular() { return false } return true } // IsDir returns whether a path is a directory func isDir(path string) bool { info, err := os.Stat(path) if os.IsNotExist(err) || !info.IsDir() { return false } return true }