2020-11-19 20:44:00 -08:00
|
|
|
package middleware
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
2020-11-27 03:01:04 +00:00
|
|
|
"net/http"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/labstack/echo/v4"
|
2020-11-19 20:44:00 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
// DecompressConfig defines the config for Decompress middleware.
|
|
|
|
DecompressConfig struct {
|
|
|
|
// Skipper defines a function to skip middleware.
|
|
|
|
Skipper Skipper
|
2020-11-28 02:03:54 +00:00
|
|
|
|
|
|
|
// GzipDecompressPool defines an interface to provide the sync.Pool used to create/store Gzip readers
|
|
|
|
GzipDecompressPool Decompressor
|
2020-11-19 20:44:00 -08:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
//GZIPEncoding content-encoding header if set to "gzip", decompress body contents.
|
|
|
|
const GZIPEncoding string = "gzip"
|
|
|
|
|
2020-11-28 02:03:54 +00:00
|
|
|
// Decompressor is used to get the sync.Pool used by the middleware to get Gzip readers
|
|
|
|
type Decompressor interface {
|
|
|
|
gzipDecompressPool() sync.Pool
|
|
|
|
}
|
|
|
|
|
2020-11-19 20:44:00 -08:00
|
|
|
var (
|
|
|
|
//DefaultDecompressConfig defines the config for decompress middleware
|
2020-11-28 02:03:54 +00:00
|
|
|
DefaultDecompressConfig = DecompressConfig{
|
|
|
|
Skipper: DefaultSkipper,
|
|
|
|
GzipDecompressPool: &DefaultGzipDecompressPool{},
|
|
|
|
}
|
2020-11-19 20:44:00 -08:00
|
|
|
)
|
|
|
|
|
2020-11-28 02:03:54 +00:00
|
|
|
// DefaultGzipDecompressPool is the default implementation of Decompressor interface
|
|
|
|
type DefaultGzipDecompressPool struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefaultGzipDecompressPool) gzipDecompressPool() sync.Pool {
|
|
|
|
return sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
// create with an empty reader (but with GZIP header)
|
|
|
|
w, err := gzip.NewWriterLevel(ioutil.Discard, gzip.BestSpeed)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
b := new(bytes.Buffer)
|
|
|
|
w.Reset(b)
|
|
|
|
w.Flush()
|
|
|
|
w.Close()
|
|
|
|
|
|
|
|
r, err := gzip.NewReader(bytes.NewReader(b.Bytes()))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-19 20:44:00 -08:00
|
|
|
//Decompress decompresses request body based if content encoding type is set to "gzip" with default config
|
|
|
|
func Decompress() echo.MiddlewareFunc {
|
|
|
|
return DecompressWithConfig(DefaultDecompressConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
//DecompressWithConfig decompresses request body based if content encoding type is set to "gzip" with config
|
|
|
|
func DecompressWithConfig(config DecompressConfig) echo.MiddlewareFunc {
|
2020-11-28 02:03:54 +00:00
|
|
|
// Defaults
|
|
|
|
if config.Skipper == nil {
|
|
|
|
config.Skipper = DefaultGzipConfig.Skipper
|
|
|
|
}
|
|
|
|
if config.GzipDecompressPool == nil {
|
|
|
|
config.GzipDecompressPool = DefaultDecompressConfig.GzipDecompressPool
|
|
|
|
}
|
|
|
|
|
2020-11-19 20:44:00 -08:00
|
|
|
return func(next echo.HandlerFunc) echo.HandlerFunc {
|
2020-11-28 02:03:54 +00:00
|
|
|
pool := config.GzipDecompressPool.gzipDecompressPool()
|
2020-11-19 20:44:00 -08:00
|
|
|
return func(c echo.Context) error {
|
|
|
|
if config.Skipper(c) {
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
switch c.Request().Header.Get(echo.HeaderContentEncoding) {
|
|
|
|
case GZIPEncoding:
|
2020-11-27 03:01:04 +00:00
|
|
|
b := c.Request().Body
|
|
|
|
|
|
|
|
i := pool.Get()
|
|
|
|
gr, ok := i.(*gzip.Reader)
|
|
|
|
if !ok {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, i.(error).Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := gr.Reset(b); err != nil {
|
|
|
|
pool.Put(gr)
|
2020-11-19 20:44:00 -08:00
|
|
|
if err == io.EOF { //ignore if body is empty
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
|
|
io.Copy(&buf, gr)
|
2020-11-27 03:01:04 +00:00
|
|
|
|
|
|
|
gr.Close()
|
|
|
|
pool.Put(gr)
|
|
|
|
|
|
|
|
b.Close() // http.Request.Body is closed by the Server, but because we are replacing it, it must be closed here
|
|
|
|
|
2020-11-19 20:44:00 -08:00
|
|
|
r := ioutil.NopCloser(&buf)
|
|
|
|
c.Request().Body = r
|
|
|
|
}
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|