diff --git a/examples/full/cmd/server.go b/examples/full/cmd/server.go index 1076af7..989e544 100644 --- a/examples/full/cmd/server.go +++ b/examples/full/cmd/server.go @@ -11,6 +11,7 @@ import ( "github.com/go-chi/chi/v5/middleware" "github.com/go-redis/redis/v8" "github.com/pkg/errors" + "github.com/rebuy-de/rebuy-go-sdk/v4/pkg/cmdutil" "github.com/rebuy-de/rebuy-go-sdk/v4/pkg/logutil" "github.com/rebuy-de/rebuy-go-sdk/v4/pkg/redisutil" "github.com/rebuy-de/rebuy-go-sdk/v4/pkg/webutil" @@ -25,26 +26,18 @@ type Server struct { TemplateFS fs.FS } -func (s *Server) Run(ctxRoot context.Context) error { - // Creating a new context, so we can have two stages for the graceful - // shutdown. First is to make pod unready (within the admin api) and the - // seconds is all the rest. - ctx, cancel := context.WithCancel(context.Background()) - defer cancel() - +func (s *Server) Run(ctx context.Context) error { ctx = InstInit(ctx) // Using a errors group is a good practice to manage multiple parallel // running routines and should used once on program startup. group, ctx := errgroup.WithContext(ctx) - // Set up the admin API and use the root context to make sure it gets - // terminated first. We have to use ctxRoot, because this is what should - // canceled first, if any error occours. Afterwards it uses cancel() to - // cancel ctx context. - webutil.AdminAPIListenAndServe(ctxRoot, group, cancel) + // Set up the admin API. The admin API lifecycle differs from the context, + // so it actually is the last thing that gets shut down. + webutil.AdminAPIListenAndServe(ctx) - // Other background processes use the main context. + // Other background processes. s.setupHTTPServer(ctx, group) return errors.WithStack(group.Wait()) @@ -55,6 +48,10 @@ func (s *Server) setupHTTPServer(ctx context.Context, group *errgroup.Group) { // process, so we can see what triggered a specific log message later. ctx = logutil.Start(ctx, "http-server") + // Delay the context cancel by 5s to give Kubernetes some time to redirect + // traffic to another pod. + ctx = cmdutil.ContextWithDelay(ctx, 5*time.Second) + // Prepare some interfaces to later use. vh := webutil.NewViewHandler(s.TemplateFS, webutil.SimpleTemplateFuncMap("prettyTime", PrettyTimeTemplateFunction), @@ -71,7 +68,7 @@ func (s *Server) setupHTTPServer(ctx context.Context, group *errgroup.Group) { group.Go(func() error { logutil.Get(ctx).Info("http server listening on port 8080") - return errors.WithStack(webutil.ListenAndServerWithContext( + return errors.WithStack(webutil.ListenAndServeWithContext( ctx, "0.0.0.0:8080", router)) }) } diff --git a/pkg/cmdutil/context.go b/pkg/cmdutil/context.go index 4325328..ec1a7f6 100644 --- a/pkg/cmdutil/context.go +++ b/pkg/cmdutil/context.go @@ -5,6 +5,7 @@ import ( "os" "os/signal" "syscall" + "time" "github.com/sirupsen/logrus" "github.com/spf13/cobra" @@ -47,3 +48,54 @@ func wrapRootConext(run RunFuncWithContext) RunFunc { } } + +// ContextWithDelay delays the context cancel by the given delay. In the +// background it creates a new context with ContextWithValuesFrom and cancels +// it after the original one got canceled. +func ContextWithDelay(in context.Context, delay time.Duration) context.Context { + out := ContextWithValuesFrom(in) + out, cancel := context.WithCancel(out) + + go func() { + defer cancel() + <-in.Done() + time.Sleep(delay) + }() + return out +} + +type compositeContext struct { + deadline context.Context + done context.Context + err context.Context + value context.Context +} + +func (c compositeContext) Deadline() (deadline time.Time, ok bool) { + return c.deadline.Deadline() +} +func (c compositeContext) Done() <-chan struct{} { + return c.done.Done() +} + +func (c compositeContext) Err() error { + return c.err.Err() +} + +func (c compositeContext) Value(key any) any { + return c.value.Value(key) +} + +// ContextWithValuesFrom creates a new context, but still references the values +// from the given context. This is helpful if a background context is needed +// that needs to have the values of an exiting context. +func ContextWithValuesFrom(value context.Context) context.Context { + bg := context.Background() + + return &compositeContext{ + deadline: bg, + done: bg, + err: bg, + value: value, + } +} diff --git a/pkg/webutil/admin.go b/pkg/webutil/admin.go index b6eb98f..07191e7 100644 --- a/pkg/webutil/admin.go +++ b/pkg/webutil/admin.go @@ -6,13 +6,11 @@ import ( "net/http" "net/http/pprof" - "github.com/pkg/errors" "github.com/prometheus/client_golang/prometheus/promhttp" "github.com/rebuy-de/rebuy-go-sdk/v4/pkg/logutil" - "golang.org/x/sync/errgroup" ) -func AdminAPIListenAndServe(ctx context.Context, group *errgroup.Group, fnDone func()) { +func AdminAPIListenAndServe(ctx context.Context, healthy ...func() error) { ctx = logutil.Start(ctx, "admin-api") mux := http.NewServeMux() @@ -24,6 +22,15 @@ func AdminAPIListenAndServe(ctx context.Context, group *errgroup.Group, fnDone f return } + for _, h := range healthy { + err := h() + if err != nil { + w.WriteHeader(http.StatusServiceUnavailable) + fmt.Fprintln(w, err.Error()) + return + } + } + w.WriteHeader(http.StatusOK) fmt.Fprintln(w, "OK") }) @@ -36,12 +43,19 @@ func AdminAPIListenAndServe(ctx context.Context, group *errgroup.Group, fnDone f mux.HandleFunc("/debug/pprof/symbol", pprof.Symbol) mux.HandleFunc("/debug/pprof/trace", pprof.Trace) - group.Go(func() error { - defer fnDone() + // The admin api gets a its own context, because we want to delay the + // server shutdown as long as possible. The reason for this is that Istio + // starts to block all outgoing connections as soon as there is no + // listening server anymore. Also a graceful shutdown is not needed for the + // admin API, so it is also not necessary to cancel the context. + bg := context.Background() + go func() { logutil.Get(ctx).Debugf("admin api listening on port 8090") - return errors.WithStack(ListenAndServerWithContext( - ctx, "0.0.0.0:8090", mux)) - }) + err := ListenAndServeWithContext(bg, "0.0.0.0:8090", mux) + if err != nil { + logutil.Get(ctx).Error(err.Error()) + } + }() } diff --git a/pkg/webutil/server.go b/pkg/webutil/server.go index 8d4f466..3967d40 100644 --- a/pkg/webutil/server.go +++ b/pkg/webutil/server.go @@ -11,11 +11,11 @@ import ( "golang.org/x/sync/errgroup" ) -// ListenAndServerWithContext does the same as http.ListenAndServe with the +// ListenAndServeWithContext does the same as http.ListenAndServe with the // difference that is properly utilises the context. This means it does a // graceful shutdown when the context is done and a context cancellation gets // propagated down to the actual request context. -func ListenAndServerWithContext(ctx context.Context, addr string, handler http.Handler) error { +func ListenAndServeWithContext(ctx context.Context, addr string, handler http.Handler) error { server := &http.Server{ Addr: addr, Handler: handler,