官术网_书友最值得收藏!

Accept-Encoding - gzip, deflate

REST endpoints should always support gzip and deflate encoding, when applicable.

Implementing gzip support in Go is relatively straightforward; we showed how it is possible to implement middleware into your microservices in Chapter 1, Introduction to Microservices. In the following example, we will use this technique to create a gzip response writer.

The core of writing a response in a gzipped format is the compress/gzip package, which is part of the standard library. It allows you to create a Writer interface that implements ioWriteCloser wrapping an existing io.Writer, which writes to the given writer using the gzip compression:

func NewWriter(w io.Writer) *Writer 

To create our handler we are going to write the NewGzipHandler function, this returns a new http.Handler that will wrap our standard output handler.

The first thing we need to do is create our own ResponseWriter that embeds http.ResponseWriter.

Example 2.1 chapter2/gzip/gzip_deflate.go:

68 type GzipResponseWriter struct { 
69 gw *gzip.Writer
70 http.ResponseWriter
71}

The core method for this is the implementation of the Write method:

73 func (w GzipResponseWriter) Write(b []byte) (int, error) { 
74 if _, ok := w.Header()["Content-Type"] !ok {
75 // If content type is not set, infer it from the uncompressed body.
76 w.Header().Set("Content-Type", http.DetectContentType(b))
77 }
78 return w.gw.Write(b)
79 }

If you look at the implementation for Write in the standard http.Response struct there is a whole load of stuff going on in there that we neither want to lose or re-implement because the gzip.Writer object is created with a writer when we call Write on it, it then in turn calls write on http.Response and we lose none of the complexity.

Internally in our NewGzipHandler our handler checks to see if the client has sent the Accept-Encoding header and if so we will write the response using the GzipResponseWriter method if the client has requested uncompressed content then we only call ServeHttp with the standard ResponseWriter:

40 type GZipHandler struct { 
41 next http.Handler
42 }
43
44 func (h *GZipHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
45 encodings := r.Header.Get("Accept-Encoding")
46
47 if strings.Contains(encodings, "gzip") {
48 h.serveGzipped(w, r)
49 } else if strings.Contains(encodings, "deflate") {
50 panic("Deflate not implemented")
51 } else {
52 h.servePlain(w, r)
53 }
54 }
55
56 func (h *GZipHandler) serveGzipped(w http.ResponseWriter, r *http.Request) {
57 gzw := gzip.NewWriter(w)
58 defer gzw.Close()
59
60 w.Header().Set("Content-Encoding", "gzip")
61 h.next.ServeHTTP(GzipResponseWriter{gzw, w}, r)
62 }

63 func (h *GZipHandler) servePlain(w http.ResponseWriter, r *http.Request) 64 {
65 h.next.ServeHTTP(w, r)
66 }

This is by no means a comprehensive example and there are many open source packages like the one from the team at the NY Times (https://github.com/NYTimes/gziphandler), which manages this for you.

As a little programming test, why not try and modify this example to implement DEFLATE.

主站蜘蛛池模板: 三穗县| 大城县| 社旗县| 苍山县| 平陆县| 伽师县| 邓州市| 申扎县| 荣成市| 余庆县| 将乐县| 武定县| 灵寿县| 象山县| 平果县| 民乐县| 太原市| 滨州市| 阆中市| 兴海县| 太白县| 遵化市| 汾西县| 海门市| 滦南县| 昌吉市| 凤庆县| 安康市| 景泰县| 蒙自县| 桐梓县| 广丰县| 公安县| 商丘市| 敖汉旗| 澄城县| 微山县| 鄂托克旗| 台北县| 盘山县| 屯门区|