2023-04-20 04:05:41 +02:00
|
|
|
package apis
|
|
|
|
|
|
|
|
import (
|
2023-05-08 20:52:40 +02:00
|
|
|
"context"
|
2023-04-20 04:05:41 +02:00
|
|
|
"crypto/tls"
|
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2023-12-08 21:16:48 +02:00
|
|
|
"sync"
|
2023-04-20 04:05:41 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/fatih/color"
|
2023-08-25 10:16:31 +02:00
|
|
|
"github.com/labstack/echo/v5"
|
2023-04-20 04:05:41 +02:00
|
|
|
"github.com/labstack/echo/v5/middleware"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
|
|
"github.com/pocketbase/pocketbase/core"
|
|
|
|
"github.com/pocketbase/pocketbase/migrations"
|
|
|
|
"github.com/pocketbase/pocketbase/migrations/logs"
|
2023-08-25 10:16:31 +02:00
|
|
|
"github.com/pocketbase/pocketbase/tools/list"
|
2023-04-20 04:05:41 +02:00
|
|
|
"github.com/pocketbase/pocketbase/tools/migrate"
|
|
|
|
"golang.org/x/crypto/acme"
|
|
|
|
"golang.org/x/crypto/acme/autocert"
|
|
|
|
)
|
|
|
|
|
2023-06-08 16:59:08 +02:00
|
|
|
// ServeConfig defines a configuration struct for apis.Serve().
|
|
|
|
type ServeConfig struct {
|
|
|
|
// ShowStartBanner indicates whether to show or hide the server start console message.
|
2023-04-20 04:05:41 +02:00
|
|
|
ShowStartBanner bool
|
2023-06-08 16:59:08 +02:00
|
|
|
|
2023-08-25 10:16:31 +02:00
|
|
|
// HttpAddr is the TCP address to listen for the HTTP server (eg. `127.0.0.1:80`).
|
2023-06-08 16:59:08 +02:00
|
|
|
HttpAddr string
|
|
|
|
|
2023-08-25 10:16:31 +02:00
|
|
|
// HttpsAddr is the TCP address to listen for the HTTPS server (eg. `127.0.0.1:443`).
|
2023-06-08 16:59:08 +02:00
|
|
|
HttpsAddr string
|
|
|
|
|
2023-08-25 10:16:31 +02:00
|
|
|
// Optional domains list to use when issuing the TLS certificate.
|
|
|
|
//
|
|
|
|
// If not set, the host from the bound server address will be used.
|
|
|
|
//
|
|
|
|
// For convenience, for each "non-www" domain a "www" entry and
|
|
|
|
// redirect will be automatically added.
|
|
|
|
CertificateDomains []string
|
|
|
|
|
2023-06-08 16:59:08 +02:00
|
|
|
// AllowedOrigins is an optional list of CORS origins (default to "*").
|
|
|
|
AllowedOrigins []string
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Serve starts a new app web server.
|
2023-08-20 17:31:56 +02:00
|
|
|
//
|
|
|
|
// NB! The app should be bootstrapped before starting the web server.
|
|
|
|
//
|
|
|
|
// Example:
|
|
|
|
//
|
2023-12-02 12:43:22 +02:00
|
|
|
// app.Bootstrap()
|
|
|
|
// apis.Serve(app, apis.ServeConfig{
|
|
|
|
// HttpAddr: "127.0.0.1:8080",
|
|
|
|
// ShowStartBanner: false,
|
|
|
|
// })
|
2023-07-16 22:12:14 +02:00
|
|
|
func Serve(app core.App, config ServeConfig) (*http.Server, error) {
|
2023-06-08 16:59:08 +02:00
|
|
|
if len(config.AllowedOrigins) == 0 {
|
|
|
|
config.AllowedOrigins = []string{"*"}
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ensure that the latest migrations are applied before starting the server
|
|
|
|
if err := runMigrations(app); err != nil {
|
2023-07-16 22:12:14 +02:00
|
|
|
return nil, err
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// reload app settings in case a new default value was set with a migration
|
|
|
|
// (or if this is the first time the init migration was executed)
|
|
|
|
if err := app.RefreshSettings(); err != nil {
|
|
|
|
color.Yellow("=====================================")
|
|
|
|
color.Yellow("WARNING: Settings load error! \n%v", err)
|
|
|
|
color.Yellow("Fallback to the application defaults.")
|
|
|
|
color.Yellow("=====================================")
|
|
|
|
}
|
|
|
|
|
|
|
|
router, err := InitApi(app)
|
|
|
|
if err != nil {
|
2023-07-16 22:12:14 +02:00
|
|
|
return nil, err
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// configure cors
|
|
|
|
router.Use(middleware.CORSWithConfig(middleware.CORSConfig{
|
|
|
|
Skipper: middleware.DefaultSkipper,
|
2023-06-08 16:59:08 +02:00
|
|
|
AllowOrigins: config.AllowedOrigins,
|
2023-04-20 04:05:41 +02:00
|
|
|
AllowMethods: []string{http.MethodGet, http.MethodHead, http.MethodPut, http.MethodPatch, http.MethodPost, http.MethodDelete},
|
|
|
|
}))
|
|
|
|
|
|
|
|
// start http server
|
|
|
|
// ---
|
2023-06-08 16:59:08 +02:00
|
|
|
mainAddr := config.HttpAddr
|
|
|
|
if config.HttpsAddr != "" {
|
|
|
|
mainAddr = config.HttpsAddr
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-08-25 10:16:31 +02:00
|
|
|
var wwwRedirects []string
|
|
|
|
|
|
|
|
// extract the host names for the certificate host policy
|
|
|
|
hostNames := config.CertificateDomains
|
|
|
|
if len(hostNames) == 0 {
|
|
|
|
host, _, _ := net.SplitHostPort(mainAddr)
|
|
|
|
hostNames = append(hostNames, host)
|
|
|
|
}
|
|
|
|
for _, host := range hostNames {
|
|
|
|
if strings.HasPrefix(host, "www.") {
|
|
|
|
continue // explicitly set www host
|
|
|
|
}
|
|
|
|
|
|
|
|
wwwHost := "www." + host
|
|
|
|
if !list.ExistInSlice(wwwHost, hostNames) {
|
|
|
|
hostNames = append(hostNames, wwwHost)
|
|
|
|
wwwRedirects = append(wwwRedirects, wwwHost)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implicit www->non-www redirect(s)
|
|
|
|
if len(wwwRedirects) > 0 {
|
|
|
|
router.Pre(func(next echo.HandlerFunc) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) error {
|
|
|
|
host := c.Request().Host
|
|
|
|
|
|
|
|
if strings.HasPrefix(host, "www.") && list.ExistInSlice(host, wwwRedirects) {
|
|
|
|
return c.Redirect(
|
|
|
|
http.StatusTemporaryRedirect,
|
|
|
|
(c.Scheme() + "://" + host[4:] + c.Request().RequestURI),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2023-04-20 04:05:41 +02:00
|
|
|
|
2023-05-29 15:57:50 +02:00
|
|
|
certManager := &autocert.Manager{
|
2023-04-20 04:05:41 +02:00
|
|
|
Prompt: autocert.AcceptTOS,
|
|
|
|
Cache: autocert.DirCache(filepath.Join(app.DataDir(), ".autocert_cache")),
|
2023-08-25 10:16:31 +02:00
|
|
|
HostPolicy: autocert.HostWhitelist(hostNames...),
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-12-08 23:14:00 +02:00
|
|
|
// base request context used for cancelling long running requests
|
|
|
|
// like the SSE connections
|
|
|
|
baseCtx, cancelBaseCtx := context.WithCancel(context.Background())
|
|
|
|
defer cancelBaseCtx()
|
|
|
|
|
2023-06-08 16:59:08 +02:00
|
|
|
server := &http.Server{
|
2023-04-20 04:05:41 +02:00
|
|
|
TLSConfig: &tls.Config{
|
2023-08-25 10:16:31 +02:00
|
|
|
MinVersion: tls.VersionTLS12,
|
2023-04-20 04:05:41 +02:00
|
|
|
GetCertificate: certManager.GetCertificate,
|
|
|
|
NextProtos: []string{acme.ALPNProto},
|
|
|
|
},
|
2023-05-08 20:52:40 +02:00
|
|
|
ReadTimeout: 10 * time.Minute,
|
2023-04-20 04:05:41 +02:00
|
|
|
ReadHeaderTimeout: 30 * time.Second,
|
|
|
|
// WriteTimeout: 60 * time.Second, // breaks sse!
|
|
|
|
Handler: router,
|
|
|
|
Addr: mainAddr,
|
2023-12-08 23:14:00 +02:00
|
|
|
BaseContext: func(l net.Listener) context.Context {
|
|
|
|
return baseCtx
|
|
|
|
},
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-05-29 15:57:50 +02:00
|
|
|
serveEvent := &core.ServeEvent{
|
|
|
|
App: app,
|
|
|
|
Router: router,
|
2023-06-08 16:59:08 +02:00
|
|
|
Server: server,
|
2023-05-29 15:57:50 +02:00
|
|
|
CertManager: certManager,
|
|
|
|
}
|
|
|
|
if err := app.OnBeforeServe().Trigger(serveEvent); err != nil {
|
2023-07-16 22:12:14 +02:00
|
|
|
return nil, err
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-06-08 16:59:08 +02:00
|
|
|
if config.ShowStartBanner {
|
2023-04-20 04:05:41 +02:00
|
|
|
schema := "http"
|
2023-08-25 10:16:31 +02:00
|
|
|
addr := server.Addr
|
|
|
|
|
2023-06-08 16:59:08 +02:00
|
|
|
if config.HttpsAddr != "" {
|
2023-04-20 04:05:41 +02:00
|
|
|
schema = "https"
|
2023-08-25 10:16:31 +02:00
|
|
|
|
|
|
|
if len(config.CertificateDomains) > 0 {
|
|
|
|
addr = config.CertificateDomains[0]
|
|
|
|
}
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
date := new(strings.Builder)
|
|
|
|
log.New(date, "", log.LstdFlags).Print()
|
|
|
|
|
|
|
|
bold := color.New(color.Bold).Add(color.FgGreen)
|
|
|
|
bold.Printf(
|
|
|
|
"%s Server started at %s\n",
|
|
|
|
strings.TrimSpace(date.String()),
|
2023-08-25 10:16:31 +02:00
|
|
|
color.CyanString("%s://%s", schema, addr),
|
2023-04-20 04:05:41 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
regular := color.New()
|
2023-08-25 10:16:31 +02:00
|
|
|
regular.Printf("├─ REST API: %s\n", color.CyanString("%s://%s/api/", schema, addr))
|
|
|
|
regular.Printf("└─ Admin UI: %s\n", color.CyanString("%s://%s/_/", schema, addr))
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-12-08 21:16:48 +02:00
|
|
|
// WaitGroup to block until server.ShutDown() returns because Serve and similar methods exit immediately.
|
|
|
|
// Note that the WaitGroup would not do anything if the app.OnTerminate() hook isn't triggered.
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
2023-05-08 20:52:40 +02:00
|
|
|
// try to gracefully shutdown the server on app termination
|
|
|
|
app.OnTerminate().Add(func(e *core.TerminateEvent) error {
|
2023-12-08 23:14:00 +02:00
|
|
|
cancelBaseCtx()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
2023-05-08 20:52:40 +02:00
|
|
|
defer cancel()
|
2023-12-08 21:16:48 +02:00
|
|
|
|
|
|
|
wg.Add(1)
|
2023-06-08 16:59:08 +02:00
|
|
|
server.Shutdown(ctx)
|
2023-12-08 21:16:48 +02:00
|
|
|
if e.IsRestart {
|
2023-12-08 23:14:00 +02:00
|
|
|
// wait for execve and other handlers up to 5 seconds before exit
|
|
|
|
time.AfterFunc(5*time.Second, func() {
|
2023-12-08 21:16:48 +02:00
|
|
|
wg.Done()
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
wg.Done()
|
|
|
|
}
|
|
|
|
|
2023-05-08 20:52:40 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2023-12-08 21:16:48 +02:00
|
|
|
// wait for the graceful shutdown to complete before exit
|
|
|
|
defer wg.Wait()
|
|
|
|
|
|
|
|
// ---
|
|
|
|
// @todo consider removing the server return value because it is
|
|
|
|
// not really useful when combined with the blocking serve calls
|
|
|
|
// ---
|
|
|
|
|
2023-04-20 04:05:41 +02:00
|
|
|
// start HTTPS server
|
2023-06-08 16:59:08 +02:00
|
|
|
if config.HttpsAddr != "" {
|
2023-04-20 04:05:41 +02:00
|
|
|
// if httpAddr is set, start an HTTP server to redirect the traffic to the HTTPS version
|
2023-06-08 16:59:08 +02:00
|
|
|
if config.HttpAddr != "" {
|
|
|
|
go http.ListenAndServe(config.HttpAddr, certManager.HTTPHandler(nil))
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
2023-07-16 22:12:14 +02:00
|
|
|
return server, server.ListenAndServeTLS("", "")
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// OR start HTTP server
|
2023-07-16 22:12:14 +02:00
|
|
|
return server, server.ListenAndServe()
|
2023-04-20 04:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type migrationsConnection struct {
|
|
|
|
DB *dbx.DB
|
|
|
|
MigrationsList migrate.MigrationsList
|
|
|
|
}
|
|
|
|
|
|
|
|
func runMigrations(app core.App) error {
|
|
|
|
connections := []migrationsConnection{
|
|
|
|
{
|
|
|
|
DB: app.DB(),
|
|
|
|
MigrationsList: migrations.AppMigrations,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
DB: app.LogsDB(),
|
|
|
|
MigrationsList: logs.LogsMigrations,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range connections {
|
|
|
|
runner, err := migrate.NewRunner(c.DB, c.MigrationsList)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := runner.Up(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|