mirror of
https://github.com/kataras/iris.git
synced 2025-01-23 18:51:03 +01:00
3093d65363
Former-commit-id: cda69f08955cb0d594e98bf26197ee573cbba4b2
203 lines
6.3 KiB
Go
203 lines
6.3 KiB
Go
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 set 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(contents), 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 {
|
|
if w.ResponseWriter.Header()[ContentTypeHeaderKey] == 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 {
|
|
if w.ResponseWriter.Header()[ContentTypeHeaderKey] == 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)
|
|
}
|
|
|
|
AddGzipHeaders(w.ResponseWriter)
|
|
// 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)
|
|
}
|
|
|
|
// AddGzipHeaders just adds the headers "Vary" to "Accept-Encoding"
|
|
// and "Content-Encoding" to "gzip".
|
|
func AddGzipHeaders(w ResponseWriter) {
|
|
w.Header().Add(VaryHeaderKey, AcceptEncodingHeaderKey)
|
|
w.Header().Add(ContentEncodingHeaderKey, GzipHeaderValue)
|
|
}
|
|
|
|
// 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
|
|
}
|