2022-07-29 23:23:08 +12:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2022-08-06 00:09:20 +12:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2022-10-31 22:22:02 +13:00
|
|
|
"path"
|
2022-08-30 22:42:43 +12:00
|
|
|
"path/filepath"
|
2022-07-29 23:23:08 +12:00
|
|
|
"regexp"
|
2022-10-31 22:13:41 +13:00
|
|
|
"strings"
|
2022-08-04 17:18:07 +12:00
|
|
|
|
|
|
|
"github.com/tg123/go-htpasswd"
|
2022-07-29 23:23:08 +12:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// SMTPListen to listen on <interface>:<port>
|
|
|
|
SMTPListen = "0.0.0.0:1025"
|
|
|
|
|
|
|
|
// HTTPListen to listen on <interface>:<port>
|
|
|
|
HTTPListen = "0.0.0.0:8025"
|
|
|
|
|
2022-08-30 22:42:43 +12:00
|
|
|
// DataFile for mail (optional)
|
|
|
|
DataFile string
|
2022-07-29 23:23:08 +12:00
|
|
|
|
|
|
|
// MaxMessages is the maximum number of messages a mailbox can have (auto-pruned every minute)
|
|
|
|
MaxMessages = 500
|
|
|
|
|
|
|
|
// VerboseLogging for console output
|
|
|
|
VerboseLogging = false
|
|
|
|
|
2022-09-14 16:57:01 +12:00
|
|
|
// QuietLogging for console output (errors only)
|
|
|
|
QuietLogging = false
|
|
|
|
|
2022-08-04 17:18:07 +12:00
|
|
|
// NoLogging for tests
|
2022-07-29 23:23:08 +12:00
|
|
|
NoLogging = false
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
// UISSLCert file
|
|
|
|
UISSLCert string
|
2022-08-06 00:09:20 +12:00
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
// UISSLKey file
|
|
|
|
UISSLKey string
|
2022-08-04 17:18:07 +12:00
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
// UIAuthFile for basic authentication
|
|
|
|
UIAuthFile string
|
2022-08-04 17:18:07 +12:00
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
// UIAuth used for euthentication
|
|
|
|
UIAuth *htpasswd.File
|
|
|
|
|
2022-10-31 22:13:41 +13:00
|
|
|
// Webroot to define the base path for the UI and API
|
|
|
|
Webroot = "/"
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
// SMTPSSLCert file
|
|
|
|
SMTPSSLCert string
|
|
|
|
|
|
|
|
// SMTPSSLKey file
|
|
|
|
SMTPSSLKey string
|
|
|
|
|
|
|
|
// SMTPAuthFile for SMTP authentication
|
|
|
|
SMTPAuthFile string
|
|
|
|
|
|
|
|
// SMTPAuth used for euthentication
|
|
|
|
SMTPAuth *htpasswd.File
|
2022-10-07 19:46:39 +13:00
|
|
|
|
|
|
|
// 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';"
|
2022-10-08 23:23:30 +13:00
|
|
|
|
|
|
|
// Version is the default application version, updated on release
|
|
|
|
Version = "dev"
|
|
|
|
|
|
|
|
// Repo on Github for updater
|
|
|
|
Repo = "axllent/mailpit"
|
|
|
|
|
|
|
|
// RepoBinaryName on Github for updater
|
|
|
|
RepoBinaryName = "mailpit"
|
2022-07-29 23:23:08 +12:00
|
|
|
)
|
|
|
|
|
|
|
|
// VerifyConfig wil do some basic checking
|
|
|
|
func VerifyConfig() error {
|
2022-08-30 22:42:43 +12:00
|
|
|
if DataFile != "" && isDir(DataFile) {
|
|
|
|
DataFile = filepath.Join(DataFile, "mailpit.db")
|
|
|
|
}
|
|
|
|
|
2022-07-29 23:23:08 +12:00
|
|
|
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 <ip>:<port>")
|
|
|
|
}
|
|
|
|
if !re.MatchString(HTTPListen) {
|
|
|
|
return errors.New("HTTP bind should be in the format of <ip>:<port>")
|
|
|
|
}
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
if UIAuthFile != "" {
|
|
|
|
if !isFile(UIAuthFile) {
|
|
|
|
return fmt.Errorf("HTTP password file not found: %s", UIAuthFile)
|
2022-08-06 00:09:20 +12:00
|
|
|
}
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
a, err := htpasswd.New(UIAuthFile, htpasswd.DefaultSystems, nil)
|
2022-08-04 17:18:07 +12:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-08-06 20:00:05 +12:00
|
|
|
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")
|
2022-08-04 17:18:07 +12:00
|
|
|
}
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
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)
|
|
|
|
}
|
2022-08-06 00:09:20 +12:00
|
|
|
}
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
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")
|
2022-08-06 00:09:20 +12:00
|
|
|
}
|
|
|
|
|
2022-08-06 20:00:05 +12:00
|
|
|
a, err := htpasswd.New(SMTPAuthFile, htpasswd.DefaultSystems, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-08-06 00:09:20 +12:00
|
|
|
}
|
2022-08-06 20:00:05 +12:00
|
|
|
SMTPAuth = a
|
2022-08-06 00:09:20 +12:00
|
|
|
}
|
|
|
|
|
2022-10-31 22:13:41 +13:00
|
|
|
if strings.Contains(Webroot, " ") {
|
|
|
|
return fmt.Errorf("Webroot cannot contain spaces (%s)", Webroot)
|
|
|
|
}
|
|
|
|
|
2022-10-31 22:22:02 +13:00
|
|
|
s := path.Join("/", Webroot, "/")
|
2022-10-31 22:13:41 +13:00
|
|
|
Webroot = s
|
|
|
|
|
2022-07-29 23:23:08 +12:00
|
|
|
return nil
|
|
|
|
}
|
2022-08-06 00:09:20 +12:00
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
2022-08-30 22:42:43 +12:00
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|