mirror of
https://github.com/kataras/iris.git
synced 2025-01-23 02:31:04 +01:00
fix: ctx.Record and then iris.Compression flow
This commit is contained in:
parent
53c6f46941
commit
eacbcea653
|
@ -6,6 +6,7 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/kataras/iris/v12"
|
||||||
"github.com/kataras/iris/v12/context"
|
"github.com/kataras/iris/v12/context"
|
||||||
"github.com/kataras/iris/v12/httptest"
|
"github.com/kataras/iris/v12/httptest"
|
||||||
)
|
)
|
||||||
|
@ -15,7 +16,49 @@ func TestCompression(t *testing.T) {
|
||||||
e := httptest.New(t, app)
|
e := httptest.New(t, app)
|
||||||
|
|
||||||
var expectedReply = payload{Username: "Makis"}
|
var expectedReply = payload{Username: "Makis"}
|
||||||
body := e.GET("/").WithHeader(context.AcceptEncodingHeaderKey, context.GZIP).Expect().
|
testBody(t, e.GET("/"), expectedReply)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCompressionAfterRecorder(t *testing.T) {
|
||||||
|
var expectedReply = payload{Username: "Makis"}
|
||||||
|
|
||||||
|
app := iris.New()
|
||||||
|
app.Use(func(ctx iris.Context) {
|
||||||
|
ctx.Record()
|
||||||
|
ctx.Next()
|
||||||
|
})
|
||||||
|
app.Use(iris.Compression)
|
||||||
|
|
||||||
|
app.Get("/", func(ctx iris.Context) {
|
||||||
|
ctx.JSON(expectedReply)
|
||||||
|
})
|
||||||
|
|
||||||
|
e := httptest.New(t, app)
|
||||||
|
testBody(t, e.GET("/"), expectedReply)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCompressionBeforeRecorder(t *testing.T) {
|
||||||
|
var expectedReply = payload{Username: "Makis"}
|
||||||
|
|
||||||
|
app := iris.New()
|
||||||
|
app.Use(iris.Compression)
|
||||||
|
app.Use(func(ctx iris.Context) {
|
||||||
|
ctx.Record()
|
||||||
|
ctx.Next()
|
||||||
|
})
|
||||||
|
|
||||||
|
app.Get("/", func(ctx iris.Context) {
|
||||||
|
ctx.JSON(expectedReply)
|
||||||
|
})
|
||||||
|
|
||||||
|
e := httptest.New(t, app)
|
||||||
|
testBody(t, e.GET("/"), expectedReply)
|
||||||
|
}
|
||||||
|
|
||||||
|
func testBody(t *testing.T, req *httptest.Request, expectedReply interface{}) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
body := req.WithHeader(context.AcceptEncodingHeaderKey, context.GZIP).Expect().
|
||||||
Status(httptest.StatusOK).
|
Status(httptest.StatusOK).
|
||||||
ContentEncoding(context.GZIP).
|
ContentEncoding(context.GZIP).
|
||||||
ContentType(context.ContentJSONHeaderValue).Body().Raw()
|
ContentType(context.ContentJSONHeaderValue).Body().Raw()
|
||||||
|
|
|
@ -243,12 +243,24 @@ func releaseCompressResponseWriter(w *CompressResponseWriter) {
|
||||||
func (w *CompressResponseWriter) FlushResponse() {
|
func (w *CompressResponseWriter) FlushResponse() {
|
||||||
w.FlushHeaders()
|
w.FlushHeaders()
|
||||||
|
|
||||||
|
/* this should NEVER happen, see `context.CompressWriter` method.
|
||||||
|
if rec, ok := w.ResponseWriter.(*ResponseRecorder); ok {
|
||||||
|
// Usecase: record, then compression.
|
||||||
|
w.CompressWriter.Close() // flushes and closes.
|
||||||
|
rec.FlushResponse()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
// write the status, after header set and before any flushed content sent.
|
// write the status, after header set and before any flushed content sent.
|
||||||
w.ResponseWriter.FlushResponse()
|
w.ResponseWriter.FlushResponse()
|
||||||
|
|
||||||
w.CompressWriter.Close() // flushes and closes.
|
w.CompressWriter.Close() // flushes and closes.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FlushHeaders deletes the encoding headers if
|
||||||
|
// the compressed writer was disabled otherwise
|
||||||
|
// removes the content-length so next callers can re-calculate the correct length.
|
||||||
func (w *CompressResponseWriter) FlushHeaders() {
|
func (w *CompressResponseWriter) FlushHeaders() {
|
||||||
if w.Disabled {
|
if w.Disabled {
|
||||||
w.Header().Del(VaryHeaderKey)
|
w.Header().Del(VaryHeaderKey)
|
||||||
|
@ -294,3 +306,18 @@ func (w *CompressResponseWriter) Flush() {
|
||||||
|
|
||||||
w.ResponseWriter.Flush()
|
w.ResponseWriter.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WriteTo writes the "p" to "dest" Writer using the compression that this compress writer was made of.
|
||||||
|
func (w *CompressResponseWriter) WriteTo(dest io.Writer, p []byte) (int, error) {
|
||||||
|
if w.Disabled {
|
||||||
|
return dest.Write(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
cw, err := NewCompressWriter(dest, w.Encoding, w.Level)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
n, err := cw.Write(p)
|
||||||
|
cw.Close()
|
||||||
|
return n, err
|
||||||
|
}
|
||||||
|
|
|
@ -2297,22 +2297,42 @@ func (ctx *Context) ClientSupportsEncoding(encodings ...string) bool {
|
||||||
// Sometimes, using additional compression doesn't reduce payload size and
|
// Sometimes, using additional compression doesn't reduce payload size and
|
||||||
// can even make the payload longer.
|
// can even make the payload longer.
|
||||||
func (ctx *Context) CompressWriter(enable bool) error {
|
func (ctx *Context) CompressWriter(enable bool) error {
|
||||||
cw, ok := ctx.writer.(*CompressResponseWriter)
|
switch w := ctx.writer.(type) {
|
||||||
if enable {
|
case *CompressResponseWriter:
|
||||||
if ok {
|
if enable {
|
||||||
// already a compress writer.
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
w, err := AcquireCompressResponseWriter(ctx.writer, ctx.request, -1)
|
w.Disabled = true
|
||||||
|
case *ResponseRecorder:
|
||||||
|
if enable {
|
||||||
|
// Keep the Recorder as ctx.writer.
|
||||||
|
// Wrap the existing net/http response writer
|
||||||
|
// with the compressed writer and
|
||||||
|
// replace the recorder's response writer
|
||||||
|
// reference with that compressed one.
|
||||||
|
// Fixes an issue when Record is called before CompressWriter.
|
||||||
|
cw, err := AcquireCompressResponseWriter(w.ResponseWriter, ctx.request, -1)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
w.ResponseWriter = cw
|
||||||
|
} else {
|
||||||
|
cw, ok := w.ResponseWriter.(*CompressResponseWriter)
|
||||||
|
if ok {
|
||||||
|
cw.Disabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
if !enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
cw, err := AcquireCompressResponseWriter(w, ctx.request, -1)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
ctx.writer = w
|
ctx.writer = cw
|
||||||
} else {
|
|
||||||
if ok {
|
|
||||||
cw.Disabled = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
@ -4341,7 +4361,7 @@ func (ctx *Context) BeginTransaction(pipe func(t *Transaction)) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// write the temp contents to the original writer
|
// write the temp contents to the original writer
|
||||||
t.Context().ResponseWriter().WriteTo(ctx.writer)
|
t.Context().ResponseWriter().CopyTo(ctx.writer)
|
||||||
// give back to the transaction the original writer (SetBeforeFlush works this way and only this way)
|
// give back to the transaction the original writer (SetBeforeFlush works this way and only this way)
|
||||||
// this is tricky but nessecery if we want ctx.FireStatusCode to work inside transactions
|
// this is tricky but nessecery if we want ctx.FireStatusCode to work inside transactions
|
||||||
t.Context().ResetResponseWriter(ctx.writer)
|
t.Context().ResetResponseWriter(ctx.writer)
|
||||||
|
|
|
@ -192,8 +192,8 @@ func (w *ResponseRecorder) Clone() ResponseWriter {
|
||||||
return wc
|
return wc
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteTo writes a response writer (temp: status code, headers and body) to another response writer
|
// CopyTo writes a response writer (temp: status code, headers and body) to another response writer
|
||||||
func (w *ResponseRecorder) WriteTo(res ResponseWriter) {
|
func (w *ResponseRecorder) CopyTo(res ResponseWriter) {
|
||||||
if to, ok := res.(*ResponseRecorder); ok {
|
if to, ok := res.(*ResponseRecorder); ok {
|
||||||
|
|
||||||
// set the status code, to is first ( probably an error? (context.StatusCodeNotSuccessful, defaults to >=400).
|
// set the status code, to is first ( probably an error? (context.StatusCodeNotSuccessful, defaults to >=400).
|
||||||
|
|
|
@ -3,6 +3,7 @@ package context
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"errors"
|
"errors"
|
||||||
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
|
@ -68,8 +69,8 @@ type ResponseWriter interface {
|
||||||
// it copies the header, status code, headers and the beforeFlush finally returns a new ResponseRecorder.
|
// it copies the header, status code, headers and the beforeFlush finally returns a new ResponseRecorder.
|
||||||
Clone() ResponseWriter
|
Clone() ResponseWriter
|
||||||
|
|
||||||
// WiteTo writes a response writer (temp: status code, headers and body) to another response writer
|
// CopyTo writes a response writer (temp: status code, headers and body) to another response writer
|
||||||
WriteTo(ResponseWriter)
|
CopyTo(ResponseWriter)
|
||||||
|
|
||||||
// Flusher indicates if `Flush` is supported by the client.
|
// Flusher indicates if `Flush` is supported by the client.
|
||||||
//
|
//
|
||||||
|
@ -112,6 +113,16 @@ type ResponseWriterReseter interface {
|
||||||
Reset() bool
|
Reset() bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ResponseWriterWriteTo can be implemented
|
||||||
|
// by response writers that needs a special
|
||||||
|
// encoding before writing to their buffers.
|
||||||
|
// E.g. a custom recorder that wraps a custom compressed one.
|
||||||
|
//
|
||||||
|
// Not used by the framework itself.
|
||||||
|
type ResponseWriterWriteTo interface {
|
||||||
|
WriteTo(dest io.Writer, p []byte)
|
||||||
|
}
|
||||||
|
|
||||||
// +------------------------------------------------------------+
|
// +------------------------------------------------------------+
|
||||||
// | Response Writer Implementation |
|
// | Response Writer Implementation |
|
||||||
// +------------------------------------------------------------+
|
// +------------------------------------------------------------+
|
||||||
|
@ -300,8 +311,8 @@ func (w *responseWriter) Clone() ResponseWriter {
|
||||||
return wc
|
return wc
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteTo writes a response writer (temp: status code, headers and body) to another response writer.
|
// CopyTo writes a response writer (temp: status code, headers and body) to another response writer.
|
||||||
func (w *responseWriter) WriteTo(to ResponseWriter) {
|
func (w *responseWriter) CopyTo(to ResponseWriter) {
|
||||||
// set the status code, failure status code are first class
|
// set the status code, failure status code are first class
|
||||||
if w.statusCode >= 400 {
|
if w.statusCode >= 400 {
|
||||||
to.WriteHeader(w.statusCode)
|
to.WriteHeader(w.statusCode)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user