mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
implement compressed requests and responses
This commit is contained in:
parent
edb34b1765
commit
d028624f77
8 changed files with 187 additions and 15 deletions
|
@ -2,6 +2,7 @@ coverage:
|
||||||
round: nearest
|
round: nearest
|
||||||
ignore:
|
ignore:
|
||||||
- ackhandler/packet_linkedlist.go
|
- ackhandler/packet_linkedlist.go
|
||||||
|
- h2quic/gzipreader.go
|
||||||
- h2quic/response.go
|
- h2quic/response.go
|
||||||
- utils/byteinterval_linkedlist.go
|
- utils/byteinterval_linkedlist.go
|
||||||
- utils/packetinterval_linkedlist.go
|
- utils/packetinterval_linkedlist.go
|
||||||
|
|
|
@ -168,7 +168,12 @@ func (c *Client) Do(req *http.Request) (*http.Response, error) {
|
||||||
c.Close(err)
|
c.Close(err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
err = c.requestWriter.WriteRequest(req, dataStreamID)
|
|
||||||
|
var requestedGzip bool
|
||||||
|
if req.Header.Get("Accept-Encoding") == "" && req.Header.Get("Range") == "" && req.Method != "HEAD" {
|
||||||
|
requestedGzip = true
|
||||||
|
}
|
||||||
|
err = c.requestWriter.WriteRequest(req, dataStreamID, requestedGzip)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.Close(err)
|
c.Close(err)
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -199,6 +204,13 @@ func (c *Client) Do(req *http.Request) (*http.Response, error) {
|
||||||
res.Body = noBody
|
res.Body = noBody
|
||||||
} else {
|
} else {
|
||||||
res.Body = dataStream
|
res.Body = dataStream
|
||||||
|
if requestedGzip && res.Header.Get("Content-Encoding") == "gzip" {
|
||||||
|
res.Header.Del("Content-Encoding")
|
||||||
|
res.Header.Del("Content-Length")
|
||||||
|
res.ContentLength = -1
|
||||||
|
res.Body = &gzipReader{body: res.Body}
|
||||||
|
setUncompressed(res)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
res.Request = req
|
res.Request = req
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
package h2quic
|
package h2quic
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"compress/gzip"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"golang.org/x/net/http2"
|
"golang.org/x/net/http2"
|
||||||
|
"golang.org/x/net/http2/hpack"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/protocol"
|
"github.com/lucas-clemente/quic-go/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/qerr"
|
"github.com/lucas-clemente/quic-go/qerr"
|
||||||
|
@ -88,19 +91,41 @@ var _ = Describe("Client", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("Doing requests", func() {
|
Context("Doing requests", func() {
|
||||||
|
var request *http.Request
|
||||||
|
|
||||||
|
getRequest := func(data []byte) *http2.MetaHeadersFrame {
|
||||||
|
r := bytes.NewReader(data)
|
||||||
|
decoder := hpack.NewDecoder(4096, func(hf hpack.HeaderField) {})
|
||||||
|
h2framer := http2.NewFramer(nil, r)
|
||||||
|
frame, err := h2framer.ReadFrame()
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
mhframe := &http2.MetaHeadersFrame{HeadersFrame: frame.(*http2.HeadersFrame)}
|
||||||
|
mhframe.Fields, err = decoder.DecodeFull(mhframe.HeadersFrame.HeaderBlockFragment())
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
return mhframe
|
||||||
|
}
|
||||||
|
|
||||||
|
getHeaderFields := func(f *http2.MetaHeadersFrame) map[string]string {
|
||||||
|
fields := make(map[string]string)
|
||||||
|
for _, hf := range f.Fields {
|
||||||
|
fields[hf.Name] = hf.Value
|
||||||
|
}
|
||||||
|
return fields
|
||||||
|
}
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
|
var err error
|
||||||
client.encryptionLevel = protocol.EncryptionForwardSecure
|
client.encryptionLevel = protocol.EncryptionForwardSecure
|
||||||
|
request, err = http.NewRequest("https", "https://quic.clemente.io:1337/file1.dat", nil)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
})
|
})
|
||||||
|
|
||||||
It("does a request", func(done Done) {
|
It("does a request", func(done Done) {
|
||||||
req, err := http.NewRequest("https", "https://quic.clemente.io:1337/file1.dat", nil)
|
|
||||||
Expect(err).ToNot(HaveOccurred())
|
|
||||||
|
|
||||||
var doRsp *http.Response
|
var doRsp *http.Response
|
||||||
var doErr error
|
var doErr error
|
||||||
var doReturned bool
|
var doReturned bool
|
||||||
go func() {
|
go func() {
|
||||||
doRsp, doErr = client.Do(req)
|
doRsp, doErr = client.Do(request)
|
||||||
doReturned = true
|
doReturned = true
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
@ -118,14 +143,11 @@ var _ = Describe("Client", func() {
|
||||||
Expect(doRsp).To(Equal(rsp))
|
Expect(doRsp).To(Equal(rsp))
|
||||||
Expect(doRsp.Body).ToNot(BeNil())
|
Expect(doRsp.Body).ToNot(BeNil())
|
||||||
Expect(doRsp.ContentLength).To(BeEquivalentTo(-1))
|
Expect(doRsp.ContentLength).To(BeEquivalentTo(-1))
|
||||||
Expect(doRsp.Request).To(Equal(req))
|
Expect(doRsp.Request).To(Equal(request))
|
||||||
close(done)
|
close(done)
|
||||||
})
|
})
|
||||||
|
|
||||||
It("closes the quic client when encountering an error on the header stream", func() {
|
It("closes the quic client when encountering an error on the header stream", func() {
|
||||||
req, err := http.NewRequest("https", "https://quic.clemente.io:1337/file1.dat", nil)
|
|
||||||
Expect(err).ToNot(HaveOccurred())
|
|
||||||
|
|
||||||
headerStream.dataToRead.Write([]byte("invalid response"))
|
headerStream.dataToRead.Write([]byte("invalid response"))
|
||||||
go client.handleHeaderStream()
|
go client.handleHeaderStream()
|
||||||
|
|
||||||
|
@ -133,7 +155,7 @@ var _ = Describe("Client", func() {
|
||||||
var doErr error
|
var doErr error
|
||||||
var doReturned bool
|
var doReturned bool
|
||||||
go func() {
|
go func() {
|
||||||
doRsp, doErr = client.Do(req)
|
doRsp, doErr = client.Do(request)
|
||||||
doReturned = true
|
doReturned = true
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
@ -180,6 +202,81 @@ var _ = Describe("Client", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Context("gzip compression", func() {
|
||||||
|
var gzippedData []byte // a gzipped foobar
|
||||||
|
var response *http.Response
|
||||||
|
|
||||||
|
BeforeEach(func() {
|
||||||
|
var b bytes.Buffer
|
||||||
|
w := gzip.NewWriter(&b)
|
||||||
|
w.Write([]byte("foobar"))
|
||||||
|
w.Close()
|
||||||
|
gzippedData = b.Bytes()
|
||||||
|
response = &http.Response{
|
||||||
|
StatusCode: 200,
|
||||||
|
Header: http.Header{"Content-Length": []string{"1000"}},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
It("adds the gzip header to requests", func() {
|
||||||
|
var doRsp *http.Response
|
||||||
|
var doErr error
|
||||||
|
go func() { doRsp, doErr = client.Do(request) }()
|
||||||
|
|
||||||
|
Eventually(func() chan *http.Response { return client.responses[5] }).ShouldNot(BeNil())
|
||||||
|
qClient.streams[5].dataToRead.Write(gzippedData)
|
||||||
|
response.Header.Add("Content-Encoding", "gzip")
|
||||||
|
client.responses[5] <- response
|
||||||
|
Eventually(func() *http.Response { return doRsp }).ShouldNot(BeNil())
|
||||||
|
Expect(doErr).ToNot(HaveOccurred())
|
||||||
|
headers := getHeaderFields(getRequest(headerStream.dataWritten.Bytes()))
|
||||||
|
Expect(headers).To(HaveKeyWithValue("accept-encoding", "gzip"))
|
||||||
|
Expect(doRsp.ContentLength).To(BeEquivalentTo(-1))
|
||||||
|
Expect(doRsp.Header.Get("Content-Encoding")).To(BeEmpty())
|
||||||
|
Expect(doRsp.Header.Get("Content-Length")).To(BeEmpty())
|
||||||
|
data := make([]byte, 6)
|
||||||
|
doRsp.Body.Read(data)
|
||||||
|
Expect(data).To(Equal([]byte("foobar")))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("only decompresses the response if the response contains the right content-encoding header", func() {
|
||||||
|
var doRsp *http.Response
|
||||||
|
var doErr error
|
||||||
|
go func() { doRsp, doErr = client.Do(request) }()
|
||||||
|
|
||||||
|
Eventually(func() chan *http.Response { return client.responses[5] }).ShouldNot(BeNil())
|
||||||
|
qClient.streams[5].dataToRead.Write([]byte("not gzipped"))
|
||||||
|
client.responses[5] <- response
|
||||||
|
Eventually(func() *http.Response { return doRsp }).ShouldNot(BeNil())
|
||||||
|
Expect(doErr).ToNot(HaveOccurred())
|
||||||
|
headers := getHeaderFields(getRequest(headerStream.dataWritten.Bytes()))
|
||||||
|
Expect(headers).To(HaveKeyWithValue("accept-encoding", "gzip"))
|
||||||
|
data := make([]byte, 11)
|
||||||
|
doRsp.Body.Read(data)
|
||||||
|
Expect(doRsp.ContentLength).ToNot(BeEquivalentTo(-1))
|
||||||
|
Expect(data).To(Equal([]byte("not gzipped")))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("doesn't add the gzip header for requests that have the accept-enconding set", func() {
|
||||||
|
request.Header.Add("accept-encoding", "gzip")
|
||||||
|
var doRsp *http.Response
|
||||||
|
var doErr error
|
||||||
|
go func() { doRsp, doErr = client.Do(request) }()
|
||||||
|
|
||||||
|
Eventually(func() chan *http.Response { return client.responses[5] }).ShouldNot(BeNil())
|
||||||
|
qClient.streams[5].dataToRead.Write([]byte("gzipped data"))
|
||||||
|
client.responses[5] <- response
|
||||||
|
Eventually(func() *http.Response { return doRsp }).ShouldNot(BeNil())
|
||||||
|
Expect(doErr).ToNot(HaveOccurred())
|
||||||
|
headers := getHeaderFields(getRequest(headerStream.dataWritten.Bytes()))
|
||||||
|
Expect(headers).To(HaveKeyWithValue("accept-encoding", "gzip"))
|
||||||
|
data := make([]byte, 12)
|
||||||
|
doRsp.Body.Read(data)
|
||||||
|
Expect(doRsp.ContentLength).ToNot(BeEquivalentTo(-1))
|
||||||
|
Expect(data).To(Equal([]byte("gzipped data")))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
Context("handling the header stream", func() {
|
Context("handling the header stream", func() {
|
||||||
var h2framer *http2.Framer
|
var h2framer *http2.Framer
|
||||||
|
|
||||||
|
|
35
h2quic/gzipreader.go
Normal file
35
h2quic/gzipreader.go
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
package h2quic
|
||||||
|
|
||||||
|
// copied from net/transport.go
|
||||||
|
|
||||||
|
// gzipReader wraps a response body so it can lazily
|
||||||
|
// call gzip.NewReader on the first call to Read
|
||||||
|
import (
|
||||||
|
"compress/gzip"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// call gzip.NewReader on the first call to Read
|
||||||
|
type gzipReader struct {
|
||||||
|
body io.ReadCloser // underlying Response.Body
|
||||||
|
zr *gzip.Reader // lazily-initialized gzip reader
|
||||||
|
zerr error // sticky error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gz *gzipReader) Read(p []byte) (n int, err error) {
|
||||||
|
if gz.zerr != nil {
|
||||||
|
return 0, gz.zerr
|
||||||
|
}
|
||||||
|
if gz.zr == nil {
|
||||||
|
gz.zr, err = gzip.NewReader(gz.body)
|
||||||
|
if err != nil {
|
||||||
|
gz.zerr = err
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return gz.zr.Read(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gz *gzipReader) Close() error {
|
||||||
|
return gz.body.Close()
|
||||||
|
}
|
|
@ -34,7 +34,7 @@ func newRequestWriter(headerStream utils.Stream) *requestWriter {
|
||||||
return rw
|
return rw
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *requestWriter) WriteRequest(req *http.Request, dataStreamID protocol.StreamID) error {
|
func (w *requestWriter) WriteRequest(req *http.Request, dataStreamID protocol.StreamID, requestGzip bool) error {
|
||||||
// TODO: add support for trailers
|
// TODO: add support for trailers
|
||||||
// TODO: add support for gzip compression
|
// TODO: add support for gzip compression
|
||||||
// TODO: write continuation frames, if the header frame is too long
|
// TODO: write continuation frames, if the header frame is too long
|
||||||
|
@ -42,7 +42,7 @@ func (w *requestWriter) WriteRequest(req *http.Request, dataStreamID protocol.St
|
||||||
w.mutex.Lock()
|
w.mutex.Lock()
|
||||||
defer w.mutex.Unlock()
|
defer w.mutex.Unlock()
|
||||||
|
|
||||||
w.encodeHeaders(req, false, "", actualContentLength(req))
|
w.encodeHeaders(req, requestGzip, "", actualContentLength(req))
|
||||||
h2framer := http2.NewFramer(w.headerStream, nil)
|
h2framer := http2.NewFramer(w.headerStream, nil)
|
||||||
return h2framer.WriteHeaders(http2.HeadersFrameParam{
|
return h2framer.WriteHeaders(http2.HeadersFrameParam{
|
||||||
StreamID: uint32(dataStreamID),
|
StreamID: uint32(dataStreamID),
|
||||||
|
|
|
@ -44,7 +44,7 @@ var _ = Describe("Request", func() {
|
||||||
It("writes a GET request", func() {
|
It("writes a GET request", func() {
|
||||||
req, err := http.NewRequest("GET", "https://quic.clemente.io/index.html?foo=bar", nil)
|
req, err := http.NewRequest("GET", "https://quic.clemente.io/index.html?foo=bar", nil)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
rw.WriteRequest(req, 1337)
|
rw.WriteRequest(req, 1337, false)
|
||||||
headerFrame, headerFields := decode(headerStream.dataWritten.Bytes())
|
headerFrame, headerFields := decode(headerStream.dataWritten.Bytes())
|
||||||
Expect(headerFrame.StreamID).To(Equal(uint32(1337)))
|
Expect(headerFrame.StreamID).To(Equal(uint32(1337)))
|
||||||
Expect(headerFrame.HasPriority()).To(BeTrue())
|
Expect(headerFrame.HasPriority()).To(BeTrue())
|
||||||
|
@ -52,6 +52,15 @@ var _ = Describe("Request", func() {
|
||||||
Expect(headerFields).To(HaveKeyWithValue(":method", "GET"))
|
Expect(headerFields).To(HaveKeyWithValue(":method", "GET"))
|
||||||
Expect(headerFields).To(HaveKeyWithValue(":path", "/index.html?foo=bar"))
|
Expect(headerFields).To(HaveKeyWithValue(":path", "/index.html?foo=bar"))
|
||||||
Expect(headerFields).To(HaveKeyWithValue(":scheme", "https"))
|
Expect(headerFields).To(HaveKeyWithValue(":scheme", "https"))
|
||||||
|
Expect(headerFields).ToNot(HaveKey("accept-encoding"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("requests gzip compression, if requested", func() {
|
||||||
|
req, err := http.NewRequest("GET", "https://quic.clemente.io/index.html?foo=bar", nil)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
rw.WriteRequest(req, 1337, true)
|
||||||
|
_, headerFields := decode(headerStream.dataWritten.Bytes())
|
||||||
|
Expect(headerFields).To(HaveKeyWithValue("accept-encoding", "gzip"))
|
||||||
})
|
})
|
||||||
|
|
||||||
It("writes a POST request", func() {
|
It("writes a POST request", func() {
|
||||||
|
@ -59,7 +68,7 @@ var _ = Describe("Request", func() {
|
||||||
form.Add("foo", "bar")
|
form.Add("foo", "bar")
|
||||||
req, err := http.NewRequest("POST", "https://quic.clemente.io/upload.html", strings.NewReader(form.Encode()))
|
req, err := http.NewRequest("POST", "https://quic.clemente.io/upload.html", strings.NewReader(form.Encode()))
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
rw.WriteRequest(req, 5)
|
rw.WriteRequest(req, 5, false)
|
||||||
_, headerFields := decode(headerStream.dataWritten.Bytes())
|
_, headerFields := decode(headerStream.dataWritten.Bytes())
|
||||||
Expect(headerFields).To(HaveKeyWithValue(":method", "POST"))
|
Expect(headerFields).To(HaveKeyWithValue(":method", "POST"))
|
||||||
Expect(headerFields).To(HaveKey("content-length"))
|
Expect(headerFields).To(HaveKey("content-length"))
|
||||||
|
@ -81,7 +90,7 @@ var _ = Describe("Request", func() {
|
||||||
}
|
}
|
||||||
req.AddCookie(cookie1)
|
req.AddCookie(cookie1)
|
||||||
req.AddCookie(cookie2)
|
req.AddCookie(cookie2)
|
||||||
rw.WriteRequest(req, 11)
|
rw.WriteRequest(req, 11, false)
|
||||||
_, headerFields := decode(headerStream.dataWritten.Bytes())
|
_, headerFields := decode(headerStream.dataWritten.Bytes())
|
||||||
Expect(headerFields).To(HaveKeyWithValue("cookie", "Cookie #1=Value #1; Cookie #2=Value #2"))
|
Expect(headerFields).To(HaveKeyWithValue("cookie", "Cookie #1=Value #1; Cookie #2=Value #2"))
|
||||||
})
|
})
|
||||||
|
|
9
h2quic/response_setuncompressed.go
Normal file
9
h2quic/response_setuncompressed.go
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
// +build go1.7
|
||||||
|
|
||||||
|
package h2quic
|
||||||
|
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
|
func setUncompressed(res *http.Response) {
|
||||||
|
res.Uncompressed = true
|
||||||
|
}
|
9
h2quic/response_setuncompressed_go16.go
Normal file
9
h2quic/response_setuncompressed_go16.go
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
// +build !go1.7
|
||||||
|
|
||||||
|
package h2quic
|
||||||
|
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
|
func setUncompressed(res *http.Response) {
|
||||||
|
// http.Response.Uncompressed was introduced in go 1.7
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue