mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
http3: deduplicate Content-Length headers (#3972)
This commit is contained in:
parent
bb296b8c17
commit
5d59c3059f
2 changed files with 35 additions and 2 deletions
|
@ -30,7 +30,7 @@ type header struct {
|
||||||
|
|
||||||
func parseHeaders(headers []qpack.HeaderField, isRequest bool) (header, error) {
|
func parseHeaders(headers []qpack.HeaderField, isRequest bool) (header, error) {
|
||||||
hdr := header{Headers: make(http.Header, len(headers))}
|
hdr := header{Headers: make(http.Header, len(headers))}
|
||||||
var readFirstRegularHeader bool
|
var readFirstRegularHeader, readContentLength bool
|
||||||
var contentLengthStr string
|
var contentLengthStr string
|
||||||
for _, h := range headers {
|
for _, h := range headers {
|
||||||
// field names need to be lowercase, see section 4.2 of RFC 9114
|
// field names need to be lowercase, see section 4.2 of RFC 9114
|
||||||
|
@ -74,7 +74,14 @@ func parseHeaders(headers []qpack.HeaderField, isRequest bool) (header, error) {
|
||||||
readFirstRegularHeader = true
|
readFirstRegularHeader = true
|
||||||
switch h.Name {
|
switch h.Name {
|
||||||
case "content-length":
|
case "content-length":
|
||||||
contentLengthStr = h.Value
|
// Ignore duplicate Content-Length headers.
|
||||||
|
// Fail if the duplicates differ.
|
||||||
|
if !readContentLength {
|
||||||
|
readContentLength = true
|
||||||
|
contentLengthStr = h.Value
|
||||||
|
} else if contentLengthStr != h.Value {
|
||||||
|
return header{}, fmt.Errorf("contradicting content lengths (%s and %s)", contentLengthStr, h.Value)
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
hdr.Headers.Add(h.Name, h.Value)
|
hdr.Headers.Add(h.Name, h.Value)
|
||||||
}
|
}
|
||||||
|
|
|
@ -89,6 +89,32 @@ var _ = Describe("Request", func() {
|
||||||
Expect(err.Error()).To(ContainSubstring("invalid content length"))
|
Expect(err.Error()).To(ContainSubstring("invalid content length"))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("rejects multiple Content-Length headers, if they differ", func() {
|
||||||
|
headers := []qpack.HeaderField{
|
||||||
|
{Name: ":path", Value: "/foo"},
|
||||||
|
{Name: ":authority", Value: "quic.clemente.io"},
|
||||||
|
{Name: ":method", Value: "GET"},
|
||||||
|
{Name: "content-length", Value: "42"},
|
||||||
|
{Name: "content-length", Value: "1337"},
|
||||||
|
}
|
||||||
|
_, err := requestFromHeaders(headers)
|
||||||
|
Expect(err).To(MatchError("contradicting content lengths (42 and 1337)"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("deduplicates multiple Content-Length headers, if they're the same", func() {
|
||||||
|
headers := []qpack.HeaderField{
|
||||||
|
{Name: ":path", Value: "/foo"},
|
||||||
|
{Name: ":authority", Value: "quic.clemente.io"},
|
||||||
|
{Name: ":method", Value: "GET"},
|
||||||
|
{Name: "content-length", Value: "42"},
|
||||||
|
{Name: "content-length", Value: "42"},
|
||||||
|
}
|
||||||
|
req, err := requestFromHeaders(headers)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(req.ContentLength).To(Equal(int64(42)))
|
||||||
|
Expect(req.Header.Get("Content-Length")).To(Equal("42"))
|
||||||
|
})
|
||||||
|
|
||||||
It("rejects pseudo header fields defined for responses", func() {
|
It("rejects pseudo header fields defined for responses", func() {
|
||||||
headers := []qpack.HeaderField{
|
headers := []qpack.HeaderField{
|
||||||
{Name: ":path", Value: "/foo"},
|
{Name: ":path", Value: "/foo"},
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue