|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 |
- package middleware
-
- import (
- "bytes"
- "compress/flate"
- "compress/gzip"
- "io"
- "net/http"
- "regexp"
- "strings"
- )
-
- var (
- regexAcceptEncoding = regexp.MustCompile(`(gzip|deflate|\*)(;q=(1(\.0)?|0(\.[0-9])?))?`)
- regexContentType = regexp.MustCompile(`((message|text)\/.+)|((application\/).*(javascript|json|xml))`)
- )
-
- type compression interface {
- io.WriteCloser
- }
-
- type compressor struct {
- http.ResponseWriter
- encodings []string
- }
-
- // AcceptEncodings fetches the requested encodings from client with priority.
- func (self *compressor) acceptEncodings(request *http.Request) (encodings []string) {
- // find all encodings supported by backend server.
- matches := regexAcceptEncoding.FindAllString(request.Header.Get("Accept-Encoding"), -1)
- for _, item := range matches {
- units := strings.SplitN(item, ";", 2)
- // top priority with q=1|q=1.0|Not Specified.
- if len(units) == 1 {
- encodings = append(encodings, units[0])
-
- } else {
- if strings.HasPrefix(units[1], "q=1") {
- // insert the specified top priority to the first.
- encodings = append([]string{units[0]}, encodings...)
-
- } else if strings.HasSuffix(units[1], "0") {
- // not acceptable at client side.
- continue
- } else {
- // lower priority encoding
- encodings = append(encodings, units[0])
- }
- }
- }
- return
- }
-
- func (self *compressor) filter(src []byte) ([]byte, string) {
- var mimetype = self.Header().Get("Content-Type")
- if mimetype == "" {
- mimetype = http.DetectContentType(src)
- self.Header().Set("Content-Type", mimetype)
- }
-
- if self.Header().Get("Content-Encoding") != "" {
- return src, ""
- }
-
- if !regexContentType.MatchString(strings.TrimSpace(strings.SplitN(mimetype, ";", 2)[0])) {
- return src, ""
- }
-
- // okay to start compressing.
- var e error
- var encoding string
- var writer compression
- var buffer *bytes.Buffer = new(bytes.Buffer)
- // try compress the data, if any error occrued, fallback to ResponseWriter.
- if self.encodings[0] == "deflate" {
- encoding = "deflate"
- writer, e = flate.NewWriter(buffer, flate.DefaultCompression)
- } else {
- encoding = "gzip"
- writer, e = gzip.NewWriterLevel(buffer, gzip.DefaultCompression)
- }
- _, e = writer.Write(src)
- writer.Close()
- if e == nil {
- return buffer.Bytes(), encoding
- }
- // fallback to standard http.ResponseWriter, nothing happened~ (~__~"")
- return src, ""
- }
-
- func (self *compressor) Write(data []byte) (size int, err error) {
- if bytes, encoding := self.filter(data); encoding != "" {
- self.Header().Set("Content-Encoding", encoding)
- self.Header().Add("Vary", "Accept-Encoding")
- self.Header().Del("Content-Length")
- return self.ResponseWriter.Write(bytes)
- }
- return self.ResponseWriter.Write(data)
- }
-
- // GZIP/Deflate compression supports.
- func Compress(next http.Handler) http.Handler {
- return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
- if r.Header.Get("Sec-WebSocket-Key") != "" || r.Method == "HEAD" {
- next.ServeHTTP(w, r)
- } else {
- compressor := new(compressor)
- compressor.ResponseWriter = w
-
- encodings := compressor.acceptEncodings(r)
- if len(encodings) == 0 {
- next.ServeHTTP(w, r)
- } else {
- compressor.encodings = encodings
- next.ServeHTTP(compressor, r)
- }
- }
- })
- }
|