2021-09-29 15:38:50 +00:00
|
|
|
package serve
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/tarampampam/error-pages/internal/breaker"
|
|
|
|
"github.com/tarampampam/error-pages/internal/config"
|
|
|
|
appHttp "github.com/tarampampam/error-pages/internal/http"
|
2021-10-06 17:38:00 +00:00
|
|
|
"github.com/tarampampam/error-pages/internal/pick"
|
|
|
|
"go.uber.org/zap"
|
2021-09-29 15:38:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewCommand creates `serve` command.
|
|
|
|
func NewCommand(ctx context.Context, log *zap.Logger, configFile *string) *cobra.Command {
|
|
|
|
var (
|
|
|
|
f flags
|
|
|
|
cfg *config.Config
|
|
|
|
)
|
|
|
|
|
|
|
|
cmd := &cobra.Command{
|
|
|
|
Use: "serve",
|
|
|
|
Aliases: []string{"s", "server"},
|
|
|
|
Short: "Start HTTP server",
|
2022-01-27 12:29:49 +00:00
|
|
|
PreRunE: func(cmd *cobra.Command, _ []string) (err error) {
|
2021-09-29 15:38:50 +00:00
|
|
|
if configFile == nil {
|
|
|
|
return errors.New("path to the config file is required for this command")
|
|
|
|
}
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
if err = f.OverrideUsingEnv(cmd.Flags()); err != nil {
|
2021-09-29 15:38:50 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if cfg, err = config.FromYamlFile(*configFile); err != nil {
|
2021-09-29 15:38:50 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
return f.Validate()
|
2021-09-29 15:38:50 +00:00
|
|
|
},
|
2022-04-12 10:34:35 +00:00
|
|
|
RunE: func(*cobra.Command, []string) error { return run(ctx, log, cfg, f) },
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
f.Init(cmd.Flags())
|
2021-09-29 15:38:50 +00:00
|
|
|
|
|
|
|
return cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
// run current command.
|
2022-04-12 10:34:35 +00:00
|
|
|
func run(parentCtx context.Context, log *zap.Logger, cfg *config.Config, f flags) error { //nolint:funlen
|
2021-09-29 15:38:50 +00:00
|
|
|
var (
|
|
|
|
ctx, cancel = context.WithCancel(parentCtx) // serve context creation
|
|
|
|
oss = breaker.NewOSSignals(ctx) // OS signals listener
|
|
|
|
)
|
|
|
|
|
|
|
|
// subscribe for system signals
|
|
|
|
oss.Subscribe(func(sig os.Signal) {
|
|
|
|
log.Warn("Stopping by OS signal..", zap.String("signal", sig.String()))
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
})
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
cancel() // call the cancellation function after all
|
|
|
|
oss.Stop() // stop system signals listening
|
|
|
|
}()
|
|
|
|
|
2021-10-06 17:38:00 +00:00
|
|
|
var (
|
2022-01-27 12:29:49 +00:00
|
|
|
templateNames = cfg.TemplateNames()
|
2022-02-01 14:39:50 +00:00
|
|
|
picker interface{ Pick() string }
|
2022-04-12 10:34:35 +00:00
|
|
|
|
|
|
|
opt = f.ToOptions()
|
2021-10-06 17:38:00 +00:00
|
|
|
)
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
switch opt.Template.Name {
|
2021-10-06 17:38:00 +00:00
|
|
|
case useRandomTemplate:
|
|
|
|
log.Info("A random template will be used")
|
|
|
|
|
|
|
|
picker = pick.NewStringsSlice(templateNames, pick.RandomOnce)
|
|
|
|
|
|
|
|
case useRandomTemplateOnEachRequest:
|
|
|
|
log.Info("A random template on EACH request will be used")
|
|
|
|
|
|
|
|
picker = pick.NewStringsSlice(templateNames, pick.RandomEveryTime)
|
|
|
|
|
2022-02-01 14:39:50 +00:00
|
|
|
case useRandomTemplateDaily:
|
|
|
|
log.Info("A random template will be used and changed once a day")
|
|
|
|
|
|
|
|
picker = pick.NewStringsSliceWithInterval(templateNames, pick.RandomEveryTime, time.Hour*24) //nolint:gomnd
|
|
|
|
|
|
|
|
case useRandomTemplateHourly:
|
|
|
|
log.Info("A random template will be used and changed hourly")
|
|
|
|
|
|
|
|
picker = pick.NewStringsSliceWithInterval(templateNames, pick.RandomEveryTime, time.Hour)
|
|
|
|
|
2021-10-06 17:38:00 +00:00
|
|
|
case "":
|
|
|
|
log.Info("The first template (ordered by name) will be used")
|
|
|
|
|
|
|
|
picker = pick.NewStringsSlice(templateNames, pick.First)
|
|
|
|
|
|
|
|
default:
|
2022-04-12 10:34:35 +00:00
|
|
|
if t, found := cfg.Template(opt.Template.Name); found {
|
2022-01-27 12:29:49 +00:00
|
|
|
log.Info("We will use the requested template", zap.String("name", t.Name()))
|
|
|
|
picker = pick.NewStringsSlice([]string{t.Name()}, pick.First)
|
|
|
|
} else {
|
2022-04-12 10:34:35 +00:00
|
|
|
return errors.New("requested nonexistent template: " + opt.Template.Name)
|
2021-10-06 17:38:00 +00:00
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// create HTTP server
|
|
|
|
server := appHttp.NewServer(log)
|
|
|
|
|
|
|
|
// register server routes, middlewares, etc.
|
2022-04-12 10:34:35 +00:00
|
|
|
if err := server.Register(cfg, picker, opt); err != nil {
|
2022-01-28 15:42:08 +00:00
|
|
|
return err
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2021-10-06 17:38:00 +00:00
|
|
|
startedAt, startingErrCh := time.Now(), make(chan error, 1) // channel for server starting error
|
2021-09-29 15:38:50 +00:00
|
|
|
|
|
|
|
// start HTTP server in separate goroutine
|
|
|
|
go func(errCh chan<- error) {
|
|
|
|
defer close(errCh)
|
|
|
|
|
|
|
|
log.Info("Server starting",
|
2022-04-12 10:34:35 +00:00
|
|
|
zap.String("addr", f.Listen.IP),
|
|
|
|
zap.Uint16("port", f.Listen.Port),
|
|
|
|
zap.String("default error page", opt.Default.PageCode),
|
|
|
|
zap.Uint16("default HTTP response code", opt.Default.HTTPCode),
|
|
|
|
zap.Strings("proxy headers", opt.ProxyHTTPHeaders),
|
|
|
|
zap.Bool("show request details", opt.ShowDetails),
|
|
|
|
zap.Bool("localization disabled", opt.L10n.Disabled),
|
2021-09-29 15:38:50 +00:00
|
|
|
)
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
if err := server.Start(f.Listen.IP, f.Listen.Port); err != nil {
|
2021-09-29 15:38:50 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}(startingErrCh)
|
|
|
|
|
|
|
|
// and wait for...
|
|
|
|
select {
|
|
|
|
case err := <-startingErrCh: // ..server starting error
|
|
|
|
return err
|
|
|
|
|
|
|
|
case <-ctx.Done(): // ..or context cancellation
|
2021-10-06 17:38:00 +00:00
|
|
|
log.Info("Gracefully server stopping", zap.Duration("uptime", time.Since(startedAt)))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-02-01 14:39:50 +00:00
|
|
|
if p, ok := picker.(interface{ Close() error }); ok {
|
|
|
|
if err := p.Close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-29 15:38:50 +00:00
|
|
|
// stop the server using created context above
|
2021-10-06 17:38:00 +00:00
|
|
|
if err := server.Stop(); err != nil {
|
2021-09-29 15:38:50 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|