2016-01-29 09:46:11 +02:00
|
|
|
package standard
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
2016-02-05 00:40:08 +02:00
|
|
|
"sync"
|
2016-01-29 09:46:11 +02:00
|
|
|
|
2016-03-07 06:53:23 +02:00
|
|
|
"github.com/labstack/echo"
|
2016-01-29 09:46:11 +02:00
|
|
|
"github.com/labstack/echo/engine"
|
2016-02-10 03:16:46 +02:00
|
|
|
"github.com/labstack/gommon/log"
|
2016-01-29 09:46:11 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2016-03-18 06:49:06 +02:00
|
|
|
// Server implements `engine.Server`.
|
2016-01-29 09:46:11 +02:00
|
|
|
Server struct {
|
2016-03-11 02:35:20 +02:00
|
|
|
*http.Server
|
2016-03-12 13:54:54 +02:00
|
|
|
config engine.Config
|
2016-03-08 18:14:25 +02:00
|
|
|
handler engine.Handler
|
2016-03-06 19:52:32 +02:00
|
|
|
logger *log.Logger
|
2016-03-15 04:58:46 +02:00
|
|
|
pool *pool
|
2016-02-05 00:40:08 +02:00
|
|
|
}
|
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
pool struct {
|
2016-04-01 17:51:18 +02:00
|
|
|
request sync.Pool
|
|
|
|
response sync.Pool
|
|
|
|
responseAdapter sync.Pool
|
|
|
|
header sync.Pool
|
|
|
|
url sync.Pool
|
2016-01-29 09:46:11 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2016-04-08 06:20:50 +02:00
|
|
|
// New returns `standard.Server` with provided listen address.
|
2016-02-10 03:16:46 +02:00
|
|
|
func New(addr string) *Server {
|
2016-03-12 13:54:54 +02:00
|
|
|
c := engine.Config{Address: addr}
|
2016-04-08 06:20:50 +02:00
|
|
|
return WithConfig(c)
|
2016-02-09 23:37:38 +02:00
|
|
|
}
|
|
|
|
|
2016-04-08 06:20:50 +02:00
|
|
|
// WithTLS returns `standard.Server` with TLS config.
|
|
|
|
func WithTLS(addr, certfile, keyfile string) *Server {
|
2016-03-12 13:54:54 +02:00
|
|
|
c := engine.Config{
|
2016-02-09 23:37:38 +02:00
|
|
|
Address: addr,
|
|
|
|
TLSCertfile: certfile,
|
|
|
|
TLSKeyfile: keyfile,
|
|
|
|
}
|
2016-04-08 06:20:50 +02:00
|
|
|
return WithConfig(c)
|
2016-02-09 23:37:38 +02:00
|
|
|
}
|
|
|
|
|
2016-04-08 06:20:50 +02:00
|
|
|
// WithConfig returns `standard.Server` with config.
|
|
|
|
func WithConfig(c engine.Config) (s *Server) {
|
2016-02-10 03:16:46 +02:00
|
|
|
s = &Server{
|
2016-03-11 02:35:20 +02:00
|
|
|
Server: new(http.Server),
|
2016-02-10 03:16:46 +02:00
|
|
|
config: c,
|
2016-03-15 04:58:46 +02:00
|
|
|
pool: &pool{
|
2016-02-05 00:40:08 +02:00
|
|
|
request: sync.Pool{
|
|
|
|
New: func() interface{} {
|
2016-03-23 18:10:22 +02:00
|
|
|
return &Request{logger: s.logger}
|
2016-02-05 00:40:08 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
response: sync.Pool{
|
|
|
|
New: func() interface{} {
|
2016-02-10 03:16:46 +02:00
|
|
|
return &Response{logger: s.logger}
|
2016-02-05 00:40:08 +02:00
|
|
|
},
|
|
|
|
},
|
2016-04-01 17:51:18 +02:00
|
|
|
responseAdapter: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &responseAdapter{}
|
|
|
|
},
|
|
|
|
},
|
2016-02-05 00:40:08 +02:00
|
|
|
header: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &Header{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
url: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &URL{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-03-22 02:27:14 +02:00
|
|
|
handler: engine.HandlerFunc(func(rq engine.Request, rs engine.Response) {
|
2016-03-18 21:13:37 +02:00
|
|
|
s.logger.Error("handler not set, use `SetHandler()` to set it.")
|
2016-03-08 18:14:25 +02:00
|
|
|
}),
|
2016-02-10 03:16:46 +02:00
|
|
|
logger: log.New("echo"),
|
2016-01-29 09:46:11 +02:00
|
|
|
}
|
2016-03-11 02:35:20 +02:00
|
|
|
s.Addr = c.Address
|
|
|
|
s.Handler = s
|
2016-02-10 03:16:46 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-03-20 00:47:20 +02:00
|
|
|
// SetHandler implements `engine.Server#SetHandler` function.
|
2016-03-08 18:14:25 +02:00
|
|
|
func (s *Server) SetHandler(h engine.Handler) {
|
2016-02-10 03:16:46 +02:00
|
|
|
s.handler = h
|
|
|
|
}
|
|
|
|
|
2016-03-20 00:47:20 +02:00
|
|
|
// SetLogger implements `engine.Server#SetLogger` function.
|
2016-03-06 19:52:32 +02:00
|
|
|
func (s *Server) SetLogger(l *log.Logger) {
|
2016-02-10 03:16:46 +02:00
|
|
|
s.logger = l
|
2016-01-29 09:46:11 +02:00
|
|
|
}
|
|
|
|
|
2016-03-20 00:47:20 +02:00
|
|
|
// Start implements `engine.Server#Start` function.
|
2016-03-18 21:13:37 +02:00
|
|
|
func (s *Server) Start() error {
|
2016-03-18 17:47:03 +02:00
|
|
|
if s.config.Listener == nil {
|
2016-03-18 21:13:37 +02:00
|
|
|
return s.startDefaultListener()
|
2016-03-17 16:15:42 +02:00
|
|
|
}
|
2016-03-18 21:13:37 +02:00
|
|
|
return s.startCustomListener()
|
2016-03-17 16:15:42 +02:00
|
|
|
}
|
|
|
|
|
2016-03-18 21:13:37 +02:00
|
|
|
func (s *Server) startDefaultListener() error {
|
|
|
|
c := s.config
|
|
|
|
if c.TLSCertfile != "" && c.TLSKeyfile != "" {
|
|
|
|
return s.ListenAndServeTLS(c.TLSCertfile, c.TLSKeyfile)
|
2016-02-23 08:24:56 +02:00
|
|
|
}
|
2016-03-18 21:13:37 +02:00
|
|
|
return s.ListenAndServe()
|
2016-01-29 09:46:11 +02:00
|
|
|
}
|
2016-03-05 06:19:59 +02:00
|
|
|
|
2016-03-18 21:13:37 +02:00
|
|
|
func (s *Server) startCustomListener() error {
|
|
|
|
return s.Serve(s.config.Listener)
|
2016-03-17 16:15:42 +02:00
|
|
|
}
|
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
// ServeHTTP implements `http.Handler` interface.
|
2016-03-05 06:19:59 +02:00
|
|
|
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// Request
|
2016-03-22 02:27:14 +02:00
|
|
|
rq := s.pool.request.Get().(*Request)
|
2016-04-01 17:51:18 +02:00
|
|
|
rqHdr := s.pool.header.Get().(*Header)
|
|
|
|
rqURL := s.pool.url.Get().(*URL)
|
|
|
|
rqHdr.reset(r.Header)
|
|
|
|
rqURL.reset(r.URL)
|
|
|
|
rq.reset(r, rqHdr, rqURL)
|
2016-03-05 06:19:59 +02:00
|
|
|
|
|
|
|
// Response
|
2016-03-22 02:27:14 +02:00
|
|
|
rs := s.pool.response.Get().(*Response)
|
2016-04-01 17:51:18 +02:00
|
|
|
rsAdpt := s.pool.responseAdapter.Get().(*responseAdapter)
|
|
|
|
rsAdpt.reset(w, rs)
|
|
|
|
rsHdr := s.pool.header.Get().(*Header)
|
|
|
|
rsHdr.reset(w.Header())
|
|
|
|
rs.reset(w, rsAdpt, rsHdr)
|
2016-03-05 06:19:59 +02:00
|
|
|
|
2016-03-22 02:27:14 +02:00
|
|
|
s.handler.ServeHTTP(rq, rs)
|
2016-03-05 06:19:59 +02:00
|
|
|
|
2016-04-01 17:51:18 +02:00
|
|
|
// Return to pool
|
2016-03-22 02:27:14 +02:00
|
|
|
s.pool.request.Put(rq)
|
2016-04-01 17:51:18 +02:00
|
|
|
s.pool.header.Put(rqHdr)
|
|
|
|
s.pool.url.Put(rqURL)
|
2016-03-22 02:27:14 +02:00
|
|
|
s.pool.response.Put(rs)
|
2016-04-01 17:51:18 +02:00
|
|
|
s.pool.header.Put(rsHdr)
|
2016-03-05 06:19:59 +02:00
|
|
|
}
|
2016-03-07 06:53:23 +02:00
|
|
|
|
2016-03-07 08:05:53 +02:00
|
|
|
// WrapHandler wraps `http.Handler` into `echo.HandlerFunc`.
|
|
|
|
func WrapHandler(h http.Handler) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) error {
|
2016-03-28 00:21:56 +02:00
|
|
|
rq := c.Request().(*Request)
|
|
|
|
rs := c.Response().(*Response)
|
2016-04-01 17:51:18 +02:00
|
|
|
h.ServeHTTP(rs.ResponseWriter, rq.Request)
|
2016-03-07 06:53:23 +02:00
|
|
|
return nil
|
2016-03-07 08:05:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-10 09:24:45 +02:00
|
|
|
// WrapMiddleware wraps `func(http.Handler) http.Handler` into `echo.MiddlewareFunc`
|
|
|
|
func WrapMiddleware(m func(http.Handler) http.Handler) echo.MiddlewareFunc {
|
2016-04-02 23:19:39 +02:00
|
|
|
return func(next echo.HandlerFunc) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) (err error) {
|
2016-03-22 02:27:14 +02:00
|
|
|
rq := c.Request().(*Request)
|
|
|
|
rs := c.Response().(*Response)
|
2016-03-10 09:24:45 +02:00
|
|
|
m(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2016-04-02 23:19:39 +02:00
|
|
|
err = next(c)
|
2016-04-01 17:51:18 +02:00
|
|
|
})).ServeHTTP(rs.ResponseWriter, rq.Request)
|
2016-03-10 09:24:45 +02:00
|
|
|
return
|
2016-04-02 23:19:39 +02:00
|
|
|
}
|
2016-03-07 08:05:53 +02:00
|
|
|
}
|
2016-03-07 06:53:23 +02:00
|
|
|
}
|