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.
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:
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.