2017-10-26 07:21:35 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"compress/flate"
|
|
|
|
"compress/gzip"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
"unicode"
|
|
|
|
|
2017-11-27 12:04:25 +00:00
|
|
|
"git.loafle.net/commons_go/rpc/codec"
|
2017-10-26 07:21:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// gzipWriter writes and closes the gzip writer.
|
|
|
|
type gzipWriter struct {
|
|
|
|
w *gzip.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gw *gzipWriter) Write(p []byte) (n int, err error) {
|
|
|
|
defer gw.w.Close()
|
|
|
|
return gw.w.Write(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
// gzipEncoder implements the gzip compressed http encoder.
|
|
|
|
type gzipEncoder struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (enc *gzipEncoder) Encode(w http.ResponseWriter) io.Writer {
|
|
|
|
w.Header().Set("Content-Encoding", "gzip")
|
|
|
|
return &gzipWriter{gzip.NewWriter(w)}
|
|
|
|
}
|
|
|
|
|
|
|
|
// flateWriter writes and closes the flate writer.
|
|
|
|
type flateWriter struct {
|
|
|
|
w *flate.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fw *flateWriter) Write(p []byte) (n int, err error) {
|
|
|
|
defer fw.w.Close()
|
|
|
|
return fw.w.Write(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
// flateEncoder implements the flate compressed http encoder.
|
|
|
|
type flateEncoder struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (enc *flateEncoder) Encode(w http.ResponseWriter) io.Writer {
|
|
|
|
fw, err := flate.NewWriter(w, flate.DefaultCompression)
|
|
|
|
if err != nil {
|
|
|
|
return w
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Encoding", "deflate")
|
|
|
|
return &flateWriter{fw}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CompressionSelector generates the compressed http encoder.
|
|
|
|
type CompressionSelector struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
// acceptedEnc returns the first compression type in "Accept-Encoding" header
|
|
|
|
// field of the request.
|
|
|
|
func acceptedEnc(req *http.Request) string {
|
|
|
|
encHeader := req.Header.Get("Accept-Encoding")
|
|
|
|
if encHeader == "" {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
encTypes := strings.FieldsFunc(encHeader, func(r rune) bool {
|
|
|
|
return unicode.IsSpace(r) || r == ','
|
|
|
|
})
|
|
|
|
for _, enc := range encTypes {
|
|
|
|
if enc == "gzip" || enc == "deflate" {
|
|
|
|
return enc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// Select method selects the correct compression encoder based on http HEADER.
|
2017-11-27 12:04:25 +00:00
|
|
|
func (_ *CompressionSelector) Select(r *http.Request) Codec {
|
2017-10-26 07:21:35 +00:00
|
|
|
switch acceptedEnc(r) {
|
|
|
|
case "gzip":
|
|
|
|
return &gzipEncoder{}
|
|
|
|
case "flate":
|
|
|
|
return &flateEncoder{}
|
|
|
|
}
|
2017-11-27 12:04:25 +00:00
|
|
|
return codec.DefaultCodec
|
2017-10-26 07:21:35 +00:00
|
|
|
}
|