server.go 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. package server
  2. import (
  3. "context"
  4. "fmt"
  5. golog "log"
  6. "net/http"
  7. log "github.com/sirupsen/logrus"
  8. "golang.org/x/net/netutil"
  9. "github.com/imgproxy/imgproxy/v3/config"
  10. "github.com/imgproxy/imgproxy/v3/reuseport"
  11. )
  12. const (
  13. // maxHeaderBytes represents max bytes in request header
  14. maxHeaderBytes = 1 << 20
  15. )
  16. // Server represents the HTTP server wrapper struct
  17. type Server struct {
  18. router *Router
  19. server *http.Server
  20. }
  21. // Start starts the http server. cancel is called in case server failed to start, but it happened
  22. // asynchronously. It should cancel the upstream context.
  23. func Start(cancel context.CancelFunc, router *Router) (*Server, error) {
  24. l, err := reuseport.Listen(router.config.Network, router.config.Bind, router.config.SocketReusePort)
  25. if err != nil {
  26. cancel()
  27. return nil, fmt.Errorf("can't start server: %s", err)
  28. }
  29. if router.config.MaxClients > 0 {
  30. l = netutil.LimitListener(l, router.config.MaxClients)
  31. }
  32. errLogger := golog.New(
  33. log.WithField("source", "http_server").WriterLevel(log.ErrorLevel),
  34. "", 0,
  35. )
  36. srv := &http.Server{
  37. Handler: router,
  38. ReadTimeout: router.config.ReadRequestTimeout,
  39. MaxHeaderBytes: maxHeaderBytes,
  40. ErrorLog: errLogger,
  41. }
  42. if config.KeepAliveTimeout > 0 {
  43. srv.IdleTimeout = router.config.KeepAliveTimeout
  44. } else {
  45. srv.SetKeepAlivesEnabled(false)
  46. }
  47. go func() {
  48. log.Infof("Starting server at %s", router.config.Bind)
  49. if err := srv.Serve(l); err != nil && err != http.ErrServerClosed {
  50. log.Error(err)
  51. }
  52. cancel()
  53. }()
  54. return &Server{
  55. router: router,
  56. server: srv,
  57. }, nil
  58. }
  59. // Shutdown gracefully shuts down the server
  60. func (s *Server) Shutdown(ctx context.Context) {
  61. log.Info("Shutting down the server...")
  62. ctx, close := context.WithTimeout(ctx, s.router.config.GracefulTimeout)
  63. defer close()
  64. s.server.Shutdown(ctx)
  65. }