2016-02-09 21:43:00 +02:00
|
|
|
// +build !appengine
|
|
|
|
|
2016-01-29 09:46:11 +02:00
|
|
|
package fasthttp
|
|
|
|
|
|
|
|
import (
|
2016-02-22 05:58:19 +02:00
|
|
|
"sync"
|
2016-02-05 00:40:08 +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
|
|
|
"github.com/valyala/fasthttp"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2016-03-15 04:58:46 +02:00
|
|
|
// Server implements `engine.Engine`.
|
2016-01-29 09:46:11 +02:00
|
|
|
Server struct {
|
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-01-29 09:46:11 +02:00
|
|
|
}
|
2016-02-22 05:58:19 +02:00
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
pool struct {
|
2016-02-22 05:58:19 +02:00
|
|
|
request sync.Pool
|
|
|
|
response sync.Pool
|
|
|
|
requestHeader sync.Pool
|
|
|
|
responseHeader sync.Pool
|
|
|
|
url sync.Pool
|
|
|
|
}
|
2016-01-29 09:46:11 +02:00
|
|
|
)
|
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
// New returns an instance of `fasthttp.Server` with specified 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-03-09 06:30:35 +02:00
|
|
|
return NewFromConfig(c)
|
2016-02-09 23:37:38 +02:00
|
|
|
}
|
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
// NewFromTLS returns an instance of `fasthttp.Server` from TLS config.
|
2016-03-09 06:30:35 +02:00
|
|
|
func NewFromTLS(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-03-09 06:30:35 +02:00
|
|
|
return NewFromConfig(c)
|
2016-02-09 23:37:38 +02:00
|
|
|
}
|
|
|
|
|
2016-03-15 04:58:46 +02:00
|
|
|
// NewFromConfig returns an instance of `standard.Server` from config.
|
2016-03-12 13:54:54 +02:00
|
|
|
func NewFromConfig(c engine.Config) (s *Server) {
|
2016-02-10 03:16:46 +02:00
|
|
|
s = &Server{
|
|
|
|
config: c,
|
2016-03-15 04:58:46 +02:00
|
|
|
pool: &pool{
|
2016-02-22 05:58:19 +02:00
|
|
|
request: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &Request{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
response: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &Response{logger: s.logger}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
requestHeader: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &RequestHeader{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
responseHeader: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &ResponseHeader{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
url: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &URL{}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-03-08 18:14:25 +02:00
|
|
|
handler: engine.HandlerFunc(func(req engine.Request, res engine.Response) {
|
2016-03-06 19:52:32 +02:00
|
|
|
s.logger.Fatal("handler not set")
|
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-02-10 03:16:46 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-03-15 08:33:16 +02:00
|
|
|
// SetHandler implements `engine.Engine#SetHandler` method.
|
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-15 08:33:16 +02:00
|
|
|
// SetLogger implements `engine.Engine#SetLogger` method.
|
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-15 08:33:16 +02:00
|
|
|
// Start implements `engine.Engine#Start` method.
|
2016-01-29 09:46:11 +02:00
|
|
|
func (s *Server) Start() {
|
2016-02-23 08:24:56 +02:00
|
|
|
handler := func(c *fasthttp.RequestCtx) {
|
2016-02-22 05:58:19 +02:00
|
|
|
// Request
|
|
|
|
req := s.pool.request.Get().(*Request)
|
|
|
|
reqHdr := s.pool.requestHeader.Get().(*RequestHeader)
|
|
|
|
reqURL := s.pool.url.Get().(*URL)
|
2016-03-13 19:03:39 +02:00
|
|
|
reqHdr.reset(&c.Request.Header)
|
2016-02-22 05:58:19 +02:00
|
|
|
reqURL.reset(c.URI())
|
|
|
|
req.reset(c, reqHdr, reqURL)
|
|
|
|
|
|
|
|
// Response
|
|
|
|
res := s.pool.response.Get().(*Response)
|
|
|
|
resHdr := s.pool.responseHeader.Get().(*ResponseHeader)
|
2016-03-13 19:03:39 +02:00
|
|
|
resHdr.reset(&c.Response.Header)
|
2016-02-22 05:58:19 +02:00
|
|
|
res.reset(c, resHdr)
|
|
|
|
|
2016-03-08 18:14:25 +02:00
|
|
|
s.handler.ServeHTTP(req, res)
|
2016-02-22 05:58:19 +02:00
|
|
|
|
|
|
|
s.pool.request.Put(req)
|
|
|
|
s.pool.requestHeader.Put(reqHdr)
|
|
|
|
s.pool.url.Put(reqURL)
|
|
|
|
s.pool.response.Put(res)
|
|
|
|
s.pool.responseHeader.Put(resHdr)
|
2016-02-23 08:24:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
addr := s.config.Address
|
|
|
|
certfile := s.config.TLSCertfile
|
|
|
|
keyfile := s.config.TLSKeyfile
|
|
|
|
if certfile != "" && keyfile != "" {
|
|
|
|
s.logger.Fatal(fasthttp.ListenAndServeTLS(addr, certfile, keyfile, handler))
|
|
|
|
} else {
|
|
|
|
s.logger.Fatal(fasthttp.ListenAndServe(addr, handler))
|
|
|
|
}
|
2016-01-29 09:46:11 +02:00
|
|
|
}
|
2016-03-07 06:53:23 +02:00
|
|
|
|
2016-03-07 08:05:53 +02:00
|
|
|
// WrapHandler wraps `fasthttp.RequestHandler` into `echo.HandlerFunc`.
|
|
|
|
func WrapHandler(h fasthttp.RequestHandler) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) error {
|
2016-03-10 22:05:33 +02:00
|
|
|
ctx := c.Request().(*Request).RequestCtx
|
2016-03-07 06:53:23 +02:00
|
|
|
h(ctx)
|
|
|
|
return nil
|
2016-03-07 08:05:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WrapMiddleware wraps `fasthttp.RequestHandler` into `echo.MiddlewareFunc`
|
2016-03-07 09:24:51 +02:00
|
|
|
func WrapMiddleware(h fasthttp.RequestHandler) echo.MiddlewareFunc {
|
2016-03-07 08:05:53 +02:00
|
|
|
return func(next echo.Handler) echo.Handler {
|
|
|
|
return echo.HandlerFunc(func(c echo.Context) error {
|
2016-03-10 22:05:33 +02:00
|
|
|
ctx := c.Request().(*Request).RequestCtx
|
2016-03-10 09:24:45 +02:00
|
|
|
h(ctx)
|
2016-03-07 08:05:53 +02:00
|
|
|
return next.Handle(c)
|
|
|
|
})
|
|
|
|
}
|
2016-03-07 06:53:23 +02:00
|
|
|
}
|