error-pages/internal/cli/serve/command.go

169 lines
4.3 KiB
Go
Raw Normal View History

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",
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")
}
if err = f.overrideUsingEnv(cmd.Flags()); err != nil {
2021-09-29 15:38:50 +00:00
return err
}
if cfg, err = config.FromYamlFile(*configFile); err != nil {
2021-09-29 15:38:50 +00:00
return err
}
return f.validate()
},
RunE: func(*cobra.Command, []string) error { return run(ctx, log, f, cfg) },
}
f.init(cmd.Flags())
return cmd
}
// run current command.
func run(parentCtx context.Context, log *zap.Logger, f flags, cfg *config.Config) error { //nolint:funlen
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 (
templateNames = cfg.TemplateNames()
picker interface{ Pick() string }
2021-10-06 17:38:00 +00:00
)
2021-09-29 15:38:50 +00:00
2021-10-06 17:38:00 +00:00
switch f.template.name {
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)
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:
if t, found := cfg.Template(f.template.name); found {
log.Info("We will use the requested template", zap.String("name", t.Name()))
picker = pick.NewStringsSlice([]string{t.Name()}, pick.First)
} else {
2021-10-06 17:38:00 +00:00
return errors.New("requested nonexistent template: " + f.template.name)
}
2021-09-29 15:38:50 +00:00
}
2022-02-23 06:09:54 +00:00
var proxyHTTPHeaders = f.HeadersToProxy()
2021-09-29 15:38:50 +00:00
// create HTTP server
server := appHttp.NewServer(log)
// register server routes, middlewares, etc.
2022-02-23 06:09:54 +00:00
if err := server.Register(
cfg,
picker,
f.defaultErrorPage,
f.defaultHTTPCode,
f.showDetails,
proxyHTTPHeaders,
); 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",
zap.String("addr", f.listen.ip),
zap.Uint16("port", f.listen.port),
2021-10-06 17:38:00 +00:00
zap.String("default error page", f.defaultErrorPage),
zap.Uint16("default HTTP response code", f.defaultHTTPCode),
2022-02-23 06:09:54 +00:00
zap.Strings("proxy headers", proxyHTTPHeaders),
zap.Bool("show request details", f.showDetails),
2021-09-29 15:38:50 +00:00
)
if err := server.Start(f.listen.ip, f.listen.port); err != nil {
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
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
}