2020-10-14 22:29:51 +00:00
|
|
|
package httpserver
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"net"
|
|
|
|
|
"net/http"
|
|
|
|
|
"os"
|
|
|
|
|
"sync"
|
2021-11-24 18:16:00 +00:00
|
|
|
"time"
|
2020-10-14 22:29:51 +00:00
|
|
|
|
2024-01-02 18:07:25 +00:00
|
|
|
"github.com/inconshreveable/log15" //nolint:logging // TODO move all logging to sourcegraph/log
|
2021-03-02 16:03:45 +00:00
|
|
|
|
2020-10-14 22:29:51 +00:00
|
|
|
"github.com/sourcegraph/sourcegraph/internal/goroutine"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type server struct {
|
2021-11-25 00:16:40 +00:00
|
|
|
server *http.Server
|
|
|
|
|
makeListener func() (net.Listener, error)
|
|
|
|
|
once sync.Once
|
|
|
|
|
preShutdownPause time.Duration
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type ServerOptions func(s *server)
|
|
|
|
|
|
|
|
|
|
func WithPreShutdownPause(d time.Duration) ServerOptions {
|
|
|
|
|
return func(s *server) { s.preShutdownPause = d }
|
2020-10-14 22:29:51 +00:00
|
|
|
}
|
|
|
|
|
|
2020-11-18 21:30:27 +00:00
|
|
|
// New returns a BackgroundRoutine that serves the given server on the given listener.
|
2021-11-25 00:16:40 +00:00
|
|
|
func New(listener net.Listener, httpServer *http.Server, options ...ServerOptions) goroutine.BackgroundRoutine {
|
|
|
|
|
makeListener := func() (net.Listener, error) { return listener, nil }
|
|
|
|
|
return newServer(httpServer, makeListener, options...)
|
2020-10-14 22:29:51 +00:00
|
|
|
}
|
|
|
|
|
|
2022-07-01 11:42:03 +00:00
|
|
|
// NewFromAddr returns a BackgroundRoutine that serves the given handler on the given address.
|
2021-11-25 00:16:40 +00:00
|
|
|
func NewFromAddr(addr string, httpServer *http.Server, options ...ServerOptions) goroutine.BackgroundRoutine {
|
|
|
|
|
makeListener := func() (net.Listener, error) { return NewListener(addr) }
|
|
|
|
|
return newServer(httpServer, makeListener, options...)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func newServer(httpServer *http.Server, makeListener func() (net.Listener, error), options ...ServerOptions) goroutine.BackgroundRoutine {
|
|
|
|
|
s := &server{
|
2020-11-18 21:30:27 +00:00
|
|
|
server: httpServer,
|
2021-11-25 00:16:40 +00:00
|
|
|
makeListener: makeListener,
|
2020-11-02 16:31:41 +00:00
|
|
|
}
|
2021-11-25 00:16:40 +00:00
|
|
|
|
|
|
|
|
for _, option := range options {
|
|
|
|
|
option(s)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return s
|
2020-11-02 16:31:41 +00:00
|
|
|
}
|
|
|
|
|
|
2024-05-24 14:04:55 +00:00
|
|
|
func (s *server) Name() string {
|
|
|
|
|
return "HTTP server"
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-14 22:29:51 +00:00
|
|
|
func (s *server) Start() {
|
2020-11-18 21:30:27 +00:00
|
|
|
listener, err := s.makeListener()
|
|
|
|
|
if err != nil {
|
|
|
|
|
log15.Error("Failed to create listener", "error", err)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := s.server.Serve(listener); err != http.ErrServerClosed {
|
2020-10-14 22:29:51 +00:00
|
|
|
log15.Error("Failed to start server", "error", err)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-24 14:04:55 +00:00
|
|
|
func (s *server) Stop(ctx context.Context) error {
|
2020-10-14 22:29:51 +00:00
|
|
|
s.once.Do(func() {
|
2021-11-25 00:16:40 +00:00
|
|
|
if s.preShutdownPause > 0 {
|
|
|
|
|
time.Sleep(s.preShutdownPause)
|
2021-11-24 18:16:00 +00:00
|
|
|
}
|
|
|
|
|
|
2024-05-24 14:04:55 +00:00
|
|
|
ctx, cancel := context.WithTimeout(ctx, goroutine.GracefulShutdownTimeout)
|
2021-11-24 18:16:00 +00:00
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
|
|
if err := s.server.Shutdown(ctx); err != nil {
|
2020-10-14 22:29:51 +00:00
|
|
|
log15.Error("Failed to shutdown server", "error", err)
|
|
|
|
|
}
|
|
|
|
|
})
|
2024-05-24 14:04:55 +00:00
|
|
|
return nil
|
2020-10-14 22:29:51 +00:00
|
|
|
}
|