listmonk/main.go

304 lines
8.1 KiB
Go
Raw Normal View History

2018-10-25 13:51:47 +00:00
package main
import (
"fmt"
"html/template"
"log"
"os"
"path/filepath"
2018-11-02 18:27:07 +00:00
"strings"
2018-10-25 13:51:47 +00:00
"time"
_ "github.com/jinzhu/gorm/dialects/postgres"
"github.com/jmoiron/sqlx"
"github.com/knadh/goyesql"
2019-06-26 11:23:23 +00:00
"github.com/knadh/koanf"
"github.com/knadh/koanf/parsers/toml"
"github.com/knadh/koanf/providers/file"
"github.com/knadh/koanf/providers/posflag"
"github.com/knadh/listmonk/manager"
2018-10-25 13:51:47 +00:00
"github.com/knadh/listmonk/messenger"
"github.com/knadh/listmonk/subimporter"
"github.com/knadh/stuffbin"
2018-10-25 13:51:47 +00:00
"github.com/labstack/echo"
flag "github.com/spf13/pflag"
)
type constants struct {
2019-06-26 11:32:42 +00:00
RootURL string `koanf:"root"`
LogoURL string `koanf:"logo_url"`
FaviconURL string `koanf:"favicon_url"`
UploadPath string `koanf:"upload_path"`
UploadURI string `koanf:"upload_uri"`
FromEmail string `koanf:"from_email"`
NotifyEmails []string `koanf:"notify_emails"`
2018-10-25 13:51:47 +00:00
}
// App contains the "global" components that are
// passed around, especially through HTTP handlers.
type App struct {
Constants *constants
DB *sqlx.DB
Queries *Queries
Importer *subimporter.Importer
Manager *manager.Manager
FS stuffbin.FileSystem
2018-10-25 13:51:47 +00:00
Logger *log.Logger
NotifTpls *template.Template
2018-10-29 09:50:49 +00:00
Messenger messenger.Messenger
2018-10-25 13:51:47 +00:00
}
2019-06-26 11:23:23 +00:00
var (
// Global logger.
logger = log.New(os.Stdout, "", log.Ldate|log.Ltime|log.Lshortfile)
2019-06-26 11:23:23 +00:00
// Global configuration reader.
ko = koanf.New(".")
buildString string
2019-06-26 11:23:23 +00:00
)
2018-10-25 13:51:47 +00:00
func init() {
2018-10-25 13:51:47 +00:00
// Register --help handler.
2019-06-26 11:23:23 +00:00
f := flag.NewFlagSet("config", flag.ContinueOnError)
f.Usage = func() {
fmt.Println(f.FlagUsages())
2018-10-25 13:51:47 +00:00
os.Exit(0)
}
// Setup the default configuration.
2019-06-26 11:23:23 +00:00
f.StringSlice("config", []string{"config.toml"},
2018-10-25 13:51:47 +00:00
"Path to one or more config files (will be merged in order)")
2019-06-26 11:23:23 +00:00
f.Bool("install", false, "Run first time installation")
f.Bool("version", false, "Current version of the build")
f.Bool("new-config", false, "Generate sample config file")
2018-10-25 13:51:47 +00:00
// Process flags.
2019-06-26 11:23:23 +00:00
f.Parse(os.Args[1:])
// Display version.
if v, _ := f.GetBool("version"); v {
fmt.Println(buildString)
os.Exit(0)
}
// Generate new config.
if ok, _ := f.GetBool("new-config"); ok {
if err := newConfigFile(); err != nil {
2019-07-12 11:49:26 +00:00
logger.Println(err)
os.Exit(1)
}
2019-07-12 11:49:26 +00:00
logger.Println("generated config.toml. Edit and run --install")
os.Exit(0)
}
2019-06-26 11:23:23 +00:00
// Load config files.
cFiles, _ := f.GetStringSlice("config")
for _, f := range cFiles {
2019-07-12 11:49:26 +00:00
logger.Printf("reading config: %s", f)
2019-06-26 11:23:23 +00:00
if err := ko.Load(file.Provider(f), toml.Parser()); err != nil {
2019-07-12 11:49:26 +00:00
if os.IsNotExist(err) {
logger.Fatal("config file not found. If there isn't one yet, run --new-config to generate one.")
}
logger.Fatalf("error loadng config: %v.", err)
2018-10-25 13:51:47 +00:00
}
}
2019-06-26 11:23:23 +00:00
ko.Load(posflag.Provider(f, ".", ko), nil)
2018-10-25 13:51:47 +00:00
}
// initFileSystem initializes the stuffbin FileSystem to provide
// access to bunded static assets to the app.
func initFileSystem(binPath string) (stuffbin.FileSystem, error) {
fs, err := stuffbin.UnStuff(os.Args[0])
if err == nil {
return fs, nil
}
// Running in local mode. Load the required static assets into
// the in-memory stuffbin.FileSystem.
logger.Printf("unable to initialize embedded filesystem: %v", err)
logger.Printf("using local filesystem for static assets")
files := []string{
"config.toml.sample",
"queries.sql",
"schema.sql",
"email-templates",
"public",
// The frontend app's static assets are aliased to /frontend
// so that they are accessible at localhost:port/frontend/static/ ...
"frontend/build:/frontend",
}
fs, err = stuffbin.NewLocalFS("/", files...)
if err != nil {
return nil, fmt.Errorf("failed to initialize local file for assets: %v", err)
}
return fs, nil
2018-10-25 13:51:47 +00:00
}
// initMessengers initializes various messaging backends.
func initMessengers(r *manager.Manager) messenger.Messenger {
2018-10-25 13:51:47 +00:00
// Load SMTP configurations for the default e-mail Messenger.
var srv []messenger.Server
2019-06-26 11:23:23 +00:00
for _, name := range ko.MapKeys("smtp") {
if !ko.Bool(fmt.Sprintf("smtp.%s.enabled", name)) {
logger.Printf("skipped SMTP: %s", name)
2018-10-25 13:51:47 +00:00
continue
}
var s messenger.Server
2019-06-26 11:23:23 +00:00
ko.Unmarshal("smtp."+name, &s)
2018-10-25 13:51:47 +00:00
s.Name = name
s.SendTimeout = s.SendTimeout * time.Millisecond
srv = append(srv, s)
logger.Printf("loaded SMTP: %s (%s@%s)", s.Name, s.Username, s.Host)
2018-10-25 13:51:47 +00:00
}
2018-10-29 09:50:49 +00:00
msgr, err := messenger.NewEmailer(srv...)
2018-10-25 13:51:47 +00:00
if err != nil {
logger.Fatalf("error loading e-mail messenger: %v", err)
}
2018-10-29 09:50:49 +00:00
if err := r.AddMessenger(msgr); err != nil {
2018-10-25 13:51:47 +00:00
logger.Printf("error registering messenger %s", err)
}
2018-10-29 09:50:49 +00:00
return msgr
2018-10-25 13:51:47 +00:00
}
func main() {
// Connect to the DB.
2019-06-26 11:23:23 +00:00
db, err := connectDB(ko.String("db.host"),
ko.Int("db.port"),
ko.String("db.user"),
ko.String("db.password"),
ko.String("db.database"),
ko.String("db.ssl_mode"))
2018-10-25 13:51:47 +00:00
if err != nil {
logger.Fatalf("error connecting to DB: %v", err)
}
defer db.Close()
var c constants
2019-06-26 11:23:23 +00:00
ko.Unmarshal("app", &c)
2018-11-02 18:27:07 +00:00
c.RootURL = strings.TrimRight(c.RootURL, "/")
c.UploadURI = filepath.Clean(c.UploadURI)
c.UploadPath = filepath.Clean(c.UploadPath)
// Initialize the static file system into which all
// required static assets (.sql, .js files etc.) are loaded.
fs, err := initFileSystem(os.Args[0])
if err != nil {
logger.Fatal(err)
}
2018-10-25 13:51:47 +00:00
// Initialize the app context that's passed around.
app := &App{
Constants: &c,
DB: db,
Logger: logger,
FS: fs,
2018-10-25 13:51:47 +00:00
}
// Load SQL queries.
qB, err := fs.Read("/queries.sql")
if err != nil {
logger.Fatalf("error reading queries.sql: %v", err)
}
qMap, err := goyesql.ParseBytes(qB)
2018-10-25 13:51:47 +00:00
if err != nil {
logger.Fatalf("error parsing SQL queries: %v", err)
2018-10-25 13:51:47 +00:00
}
// Run the first time installation.
2019-06-26 11:23:23 +00:00
if ko.Bool("install") {
2018-10-25 13:51:47 +00:00
install(app, qMap)
return
}
// Map queries to the query container.
q := &Queries{}
if err := scanQueriesToStruct(q, qMap, db.Unsafe()); err != nil {
logger.Fatalf("no SQL queries loaded: %v", err)
}
app.Queries = q
// Initialize the bulk subscriber importer.
importNotifCB := func(subject string, data map[string]interface{}) error {
go sendNotification(notifTplImport, subject, data, app)
return nil
}
app.Importer = subimporter.New(q.UpsertSubscriber.Stmt,
q.UpsertBlacklistSubscriber.Stmt,
q.UpdateListsDate.Stmt,
db.DB,
importNotifCB)
// Read system e-mail templates.
2019-07-05 08:38:48 +00:00
notifTpls, err := stuffbin.ParseTemplatesGlob(nil, fs, "/email-templates/*.html")
if err != nil {
logger.Fatalf("error loading system e-mail templates: %v", err)
}
app.NotifTpls = notifTpls
2018-10-25 13:51:47 +00:00
// Initialize the campaign manager.
campNotifCB := func(subject string, data map[string]interface{}) error {
return sendNotification(notifTplCampaign, subject, data, app)
}
m := manager.New(manager.Config{
2019-06-26 11:23:23 +00:00
Concurrency: ko.Int("app.concurrency"),
MaxSendErrors: ko.Int("app.max_send_errors"),
FromEmail: app.Constants.FromEmail,
2018-10-25 13:51:47 +00:00
// url.com/unsubscribe/{campaign_uuid}/{subscriber_uuid}
UnsubscribeURL: fmt.Sprintf("%s/unsubscribe/%%s/%%s", app.Constants.RootURL),
// url.com/link/{campaign_uuid}/{subscriber_uuid}/{link_uuid}
LinkTrackURL: fmt.Sprintf("%s/link/%%s/%%s/%%s", app.Constants.RootURL),
// url.com/campaign/{campaign_uuid}/{subscriber_uuid}/px.png
ViewTrackURL: fmt.Sprintf("%s/campaign/%%s/%%s/px.png", app.Constants.RootURL),
}, newManagerDB(q), campNotifCB, logger)
app.Manager = m
2018-10-25 13:51:47 +00:00
// Add messengers.
app.Messenger = initMessengers(app.Manager)
2018-10-25 13:51:47 +00:00
// Initialize the workers that push out messages.
go m.Run(time.Duration(time.Second * 5))
m.SpawnWorkers()
2018-10-25 13:51:47 +00:00
// Initialize the HTTP server.
2018-10-25 13:51:47 +00:00
var srv = echo.New()
srv.HideBanner = true
// Register app (*App) to be injected into all HTTP handlers.
2018-10-25 13:51:47 +00:00
srv.Use(func(next echo.HandlerFunc) echo.HandlerFunc {
return func(c echo.Context) error {
c.Set("app", app)
return next(c)
}
})
// Parse user facing templates.
2019-07-05 08:38:48 +00:00
tpl, err := stuffbin.ParseTemplatesGlob(nil, fs, "/public/templates/*.html")
2018-10-25 13:51:47 +00:00
if err != nil {
logger.Fatalf("error parsing public templates: %v", err)
}
srv.Renderer = &tplRenderer{
templates: tpl,
RootURL: c.RootURL,
LogoURL: c.LogoURL,
FaviconURL: c.FaviconURL}
// Register HTTP handlers and static file servers.
fSrv := app.FS.FileServer()
srv.GET("/public/*", echo.WrapHandler(fSrv))
srv.GET("/frontend/*", echo.WrapHandler(fSrv))
srv.Static(c.UploadURI, c.UploadURI)
2018-10-25 13:51:47 +00:00
registerHandlers(srv)
2019-06-26 11:23:23 +00:00
srv.Logger.Fatal(srv.Start(ko.String("app.address")))
2018-10-25 13:51:47 +00:00
}