package context import ( "fmt" "io" "sync" "github.com/klauspost/compress/gzip" ) // compressionPool is a wrapper of sync.Pool, to initialize a new compression writer pool type compressionPool struct { sync.Pool Level int } // +------------------------------------------------------------+ // |GZIP raw io.writer, our gzip response writer will use that. | // +------------------------------------------------------------+ // default writer pool with Compressor's level setted to -1 var gzipPool = &compressionPool{Level: -1} // acquireGzipWriter prepares a gzip writer and returns it. // // see releaseGzipWriter too. func acquireGzipWriter(w io.Writer) *gzip.Writer { v := gzipPool.Get() if v == nil { gzipWriter, err := gzip.NewWriterLevel(w, gzipPool.Level) if err != nil { return nil } return gzipWriter } gzipWriter := v.(*gzip.Writer) gzipWriter.Reset(w) return gzipWriter } // releaseGzipWriter called when flush/close and put the gzip writer back to the pool. // // see acquireGzipWriter too. func releaseGzipWriter(gzipWriter *gzip.Writer) { gzipWriter.Close() gzipPool.Put(gzipWriter) } // writeGzip writes a compressed form of p to the underlying io.Writer. The // compressed bytes are not necessarily flushed until the Writer is closed. func writeGzip(w io.Writer, b []byte) (int, error) { gzipWriter := acquireGzipWriter(w) n, err := gzipWriter.Write(b) if err != nil { releaseGzipWriter(gzipWriter) return -1, err } err = gzipWriter.Flush() releaseGzipWriter(gzipWriter) return n, err } var gzpool = sync.Pool{New: func() interface{} { return &GzipResponseWriter{} }} // AcquireGzipResponseWriter returns a new *GzipResponseWriter from the pool. // Releasing is done automatically when request and response is done. func AcquireGzipResponseWriter() *GzipResponseWriter { w := gzpool.Get().(*GzipResponseWriter) return w } func releaseGzipResponseWriter(w *GzipResponseWriter) { gzpool.Put(w) } // GzipResponseWriter is an upgraded response writer which writes compressed data to the underline ResponseWriter. // // It's a separate response writer because iris gives you the ability to "fallback" and "roll-back" the gzip encoding if something // went wrong with the response, and write http errors in plain form instead. type GzipResponseWriter struct { ResponseWriter chunks []byte disabled bool } var _ ResponseWriter = (*GzipResponseWriter)(nil) // BeginGzipResponse accepts a ResponseWriter // and prepares the new gzip response writer. // It's being called per-handler, when caller decide // to change the response writer type. func (w *GzipResponseWriter) BeginGzipResponse(underline ResponseWriter) { w.ResponseWriter = underline w.chunks = w.chunks[0:0] w.disabled = false } // EndResponse called right before the contents of this // response writer are flushed to the client. func (w *GzipResponseWriter) EndResponse() { releaseGzipResponseWriter(w) w.ResponseWriter.EndResponse() } // Write prepares the data write to the gzip writer and finally to its // underline response writer, returns the uncompressed len(contents). func (w *GzipResponseWriter) Write(contents []byte) (int, error) { // save the contents to serve them (only gzip data here) w.chunks = append(w.chunks, contents...) return len(w.chunks), nil } // Writef formats according to a format specifier and writes to the response. // // Returns the number of bytes written and any write error encountered. func (w *GzipResponseWriter) Writef(format string, a ...interface{}) (n int, err error) { n, err = fmt.Fprintf(w, format, a...) if err == nil { w.ResponseWriter.Header().Set(contentTypeHeaderKey, ContentTextHeaderValue) } return } // WriteString prepares the string data write to the gzip writer and finally to its // underline response writer, returns the uncompressed len(contents). func (w *GzipResponseWriter) WriteString(s string) (n int, err error) { n, err = w.Write([]byte(s)) if err == nil { w.ResponseWriter.Header().Set(contentTypeHeaderKey, ContentTextHeaderValue) } return } // WriteNow compresses and writes that data to the underline response writer, // returns the compressed written len. // // Use `WriteNow` instead of `Write` // when you need to know the compressed written size before // the `FlushResponse`, note that you can't post any new headers // after that, so that information is not closed to the handler anymore. func (w *GzipResponseWriter) WriteNow(contents []byte) (int, error) { if w.disabled { // type noOp struct{} // // func (n noOp) Write([]byte) (int, error) { // return 0, nil // } // // var noop = noOp{} // problem solved with w.gzipWriter.Reset(noop): // // the below Write called multiple times but not from here, // the gzip writer does something to the writer, even if we don't call the // w.gzipWriter.Write it does call the underline http.ResponseWriter // multiple times, and therefore it changes the content-length // the problem that results to the #723. // // Or a better idea, acquire and adapt the gzip writer on-time when is not disabled. // So that is not needed any more: // w.gzipWriter.Reset(noop) return w.ResponseWriter.Write(contents) } w.ResponseWriter.Header().Add(varyHeaderKey, "Accept-Encoding") w.ResponseWriter.Header().Add(contentEncodingHeaderKey, "gzip") // if not `WriteNow` but "Content-Length" header // is exists, then delete it before `.Write` // Content-Length should not be there. // no, for now at least: w.ResponseWriter.Header().Del(contentLengthHeaderKey) return writeGzip(w.ResponseWriter, contents) } // FlushResponse validates the response headers in order to be compatible with the gzip written data // and writes the data to the underline ResponseWriter. func (w *GzipResponseWriter) FlushResponse() { w.WriteNow(w.chunks) w.ResponseWriter.FlushResponse() } // ResetBody resets the response body. func (w *GzipResponseWriter) ResetBody() { w.chunks = w.chunks[0:0] } // Disable turns off the gzip compression for the next .Write's data, // if called then the contents are being written in plain form. func (w *GzipResponseWriter) Disable() { w.disabled = true }