2021-05-11 22:59:05 +00:00
|
|
|
package httputil
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2022-11-24 11:36:49 +00:00
|
|
|
// HTTPSrvPrm groups the required parameters of the Server's constructor.
|
2021-05-11 22:59:05 +00:00
|
|
|
//
|
|
|
|
// All values must comply with the requirements imposed on them.
|
|
|
|
// Passing incorrect parameter values will result in constructor
|
|
|
|
// failure (error or panic depending on the implementation).
|
2022-11-24 11:36:49 +00:00
|
|
|
type HTTPSrvPrm struct {
|
2021-05-11 22:59:05 +00:00
|
|
|
// TCP address for the server to listen on.
|
|
|
|
//
|
|
|
|
// Must be a valid TCP address.
|
|
|
|
Address string
|
|
|
|
|
|
|
|
// Must not be nil.
|
|
|
|
Handler http.Handler
|
|
|
|
}
|
|
|
|
|
|
|
|
// Server represents a wrapper over http.Server
|
2022-04-21 11:28:05 +00:00
|
|
|
// that provides an interface to start and stop
|
2021-05-11 22:59:05 +00:00
|
|
|
// listening routine.
|
|
|
|
//
|
|
|
|
// For correct operation, Server must be created
|
|
|
|
// using the constructor (New) based on the required parameters
|
|
|
|
// and optional components. After successful creation,
|
|
|
|
// Server is immediately ready to work through API.
|
|
|
|
type Server struct {
|
|
|
|
shutdownTimeout time.Duration
|
|
|
|
|
|
|
|
srv *http.Server
|
|
|
|
}
|
|
|
|
|
|
|
|
const invalidValFmt = "invalid %s %s (%T): %v"
|
|
|
|
|
2023-02-21 11:42:45 +00:00
|
|
|
func panicOnPrmValue(n string, v any) {
|
2021-05-11 22:59:05 +00:00
|
|
|
panicOnValue("parameter", n, v)
|
|
|
|
}
|
|
|
|
|
2023-02-21 11:42:45 +00:00
|
|
|
func panicOnOptValue(n string, v any) {
|
2021-05-11 22:59:05 +00:00
|
|
|
panicOnValue("option", n, v)
|
|
|
|
}
|
|
|
|
|
2023-02-21 11:42:45 +00:00
|
|
|
func panicOnValue(t, n string, v any) {
|
2021-05-11 22:59:05 +00:00
|
|
|
panic(fmt.Sprintf(invalidValFmt, t, n, v, v))
|
|
|
|
}
|
|
|
|
|
2022-11-24 11:36:49 +00:00
|
|
|
func checkSrvPrm(addr string, handler http.Handler) {
|
|
|
|
switch {
|
|
|
|
case addr == "":
|
|
|
|
panicOnPrmValue("Address", addr)
|
|
|
|
case handler == nil:
|
|
|
|
panicOnPrmValue("Handler", handler)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-11 22:59:05 +00:00
|
|
|
// New creates a new instance of the Server.
|
|
|
|
//
|
|
|
|
// Panics if at least one value of the parameters is invalid.
|
|
|
|
//
|
2021-07-13 12:13:59 +00:00
|
|
|
// Panics if at least one of next optional parameters is invalid:
|
2022-08-15 16:20:20 +00:00
|
|
|
// - shutdown timeout is non-positive.
|
2021-05-11 22:59:05 +00:00
|
|
|
//
|
|
|
|
// The created Server does not require additional
|
|
|
|
// initialization and is completely ready for work.
|
2022-11-24 11:36:49 +00:00
|
|
|
func New(prm HTTPSrvPrm, opts ...Option) *Server {
|
|
|
|
checkSrvPrm(prm.Address, prm.Handler)
|
2021-05-11 22:59:05 +00:00
|
|
|
|
|
|
|
c := defaultCfg()
|
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
o(c)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case c.shutdownTimeout <= 0:
|
|
|
|
panicOnOptValue("shutdown timeout", c.shutdownTimeout)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Server{
|
|
|
|
shutdownTimeout: c.shutdownTimeout,
|
|
|
|
srv: &http.Server{
|
|
|
|
Addr: prm.Address,
|
|
|
|
Handler: prm.Handler,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
2022-11-24 11:36:49 +00:00
|
|
|
|
|
|
|
// NewHTTPSrvPrm creates a new instance of the HTTPSrvPrm.
|
|
|
|
//
|
|
|
|
// Panics if at least one value of the parameters is invalid.
|
|
|
|
func NewHTTPSrvPrm(addr string, handler http.Handler) *HTTPSrvPrm {
|
|
|
|
checkSrvPrm(addr, handler)
|
|
|
|
return &HTTPSrvPrm{
|
|
|
|
Address: addr,
|
|
|
|
Handler: handler,
|
|
|
|
}
|
|
|
|
}
|