add support for the Extended CONNECT method (#3357)

Extended CONNECT is used by WebTransport.
This commit is contained in:
Marten Seemann 2022-03-25 09:43:48 +01:00 committed by GitHub
parent 85b495445e
commit d065fb47e1
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 74 additions and 30 deletions

View file

@ -12,9 +12,9 @@ import (
) )
func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) { func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) {
var path, authority, method, contentLengthStr string var path, authority, method, protocol, scheme, contentLengthStr string
httpHeaders := http.Header{}
httpHeaders := http.Header{}
for _, h := range headers { for _, h := range headers {
switch h.Name { switch h.Name {
case ":path": case ":path":
@ -23,6 +23,10 @@ func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) {
method = h.Value method = h.Value
case ":authority": case ":authority":
authority = h.Value authority = h.Value
case ":protocol":
protocol = h.Value
case ":scheme":
scheme = h.Value
case "content-length": case "content-length":
contentLengthStr = h.Value contentLengthStr = h.Value
default: default:
@ -39,7 +43,12 @@ func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) {
isConnect := method == http.MethodConnect isConnect := method == http.MethodConnect
if isConnect { if isConnect {
if path != "" || authority == "" { // Extended CONNECT, see https://datatracker.ietf.org/doc/html/rfc8441#section-4
if protocol != "" {
if scheme == "" || path == "" || authority == "" {
return nil, errors.New("extended CONNECT: :scheme, :path and :authority must not be empty")
}
} else if path != "" || authority == "" { // normal CONNECT
return nil, errors.New(":path must be empty and :authority must not be empty") return nil, errors.New(":path must be empty and :authority must not be empty")
} }
} else if len(path) == 0 || len(authority) == 0 || len(method) == 0 { } else if len(path) == 0 || len(authority) == 0 || len(method) == 0 {
@ -51,9 +60,14 @@ func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) {
var err error var err error
if isConnect { if isConnect {
u = &url.URL{Host: authority} u = &url.URL{
Scheme: scheme,
Host: authority,
Path: path,
}
requestURI = authority requestURI = authority
} else { } else {
protocol = "HTTP/3"
u, err = url.ParseRequestURI(path) u, err = url.ParseRequestURI(path)
if err != nil { if err != nil {
return nil, err return nil, err
@ -72,7 +86,7 @@ func requestFromHeaders(headers []qpack.HeaderField) (*http.Request, error) {
return &http.Request{ return &http.Request{
Method: method, Method: method,
URL: u, URL: u,
Proto: "HTTP/3", Proto: protocol,
ProtoMajor: 3, ProtoMajor: 3,
ProtoMinor: 0, ProtoMinor: 0,
Header: httpHeaders, Header: httpHeaders,

View file

@ -81,17 +81,6 @@ var _ = Describe("Request", func() {
})) }))
}) })
It("handles CONNECT method", func() {
headers := []qpack.HeaderField{
{Name: ":authority", Value: "quic.clemente.io"},
{Name: ":method", Value: http.MethodConnect},
}
req, err := requestFromHeaders(headers)
Expect(err).NotTo(HaveOccurred())
Expect(req.Method).To(Equal(http.MethodConnect))
Expect(req.RequestURI).To(Equal("quic.clemente.io"))
})
It("errors with missing path", func() { It("errors with missing path", func() {
headers := []qpack.HeaderField{ headers := []qpack.HeaderField{
{Name: ":authority", Value: "quic.clemente.io"}, {Name: ":authority", Value: "quic.clemente.io"},
@ -119,22 +108,63 @@ var _ = Describe("Request", func() {
Expect(err).To(MatchError(":path, :authority and :method must not be empty")) Expect(err).To(MatchError(":path, :authority and :method must not be empty"))
}) })
It("errors with missing authority in CONNECT method", func() { Context("regular HTTP CONNECT", func() {
headers := []qpack.HeaderField{ It("handles CONNECT method", func() {
{Name: ":method", Value: http.MethodConnect}, headers := []qpack.HeaderField{
} {Name: ":authority", Value: "quic.clemente.io"},
_, err := requestFromHeaders(headers) {Name: ":method", Value: http.MethodConnect},
Expect(err).To(MatchError(":path must be empty and :authority must not be empty")) }
req, err := requestFromHeaders(headers)
Expect(err).NotTo(HaveOccurred())
Expect(req.Method).To(Equal(http.MethodConnect))
Expect(req.RequestURI).To(Equal("quic.clemente.io"))
})
It("errors with missing authority in CONNECT method", func() {
headers := []qpack.HeaderField{
{Name: ":method", Value: http.MethodConnect},
}
_, err := requestFromHeaders(headers)
Expect(err).To(MatchError(":path must be empty and :authority must not be empty"))
})
It("errors with extra path in CONNECT method", func() {
headers := []qpack.HeaderField{
{Name: ":path", Value: "/foo"},
{Name: ":authority", Value: "quic.clemente.io"},
{Name: ":method", Value: http.MethodConnect},
}
_, err := requestFromHeaders(headers)
Expect(err).To(MatchError(":path must be empty and :authority must not be empty"))
})
}) })
It("errors with extra path in CONNECT method", func() { Context("Extended CONNECT", func() {
headers := []qpack.HeaderField{ It("handles Extended CONNECT method", func() {
{Name: ":path", Value: "/foo"}, headers := []qpack.HeaderField{
{Name: ":authority", Value: "quic.clemente.io"}, {Name: ":protocol", Value: "webtransport"},
{Name: ":method", Value: http.MethodConnect}, {Name: ":scheme", Value: "ftp"},
} {Name: ":method", Value: http.MethodConnect},
_, err := requestFromHeaders(headers) {Name: ":authority", Value: "quic.clemente.io"},
Expect(err).To(MatchError(":path must be empty and :authority must not be empty")) {Name: ":path", Value: "/foo"},
}
req, err := requestFromHeaders(headers)
Expect(err).NotTo(HaveOccurred())
Expect(req.Method).To(Equal(http.MethodConnect))
Expect(req.Proto).To(Equal("webtransport"))
Expect(req.URL.String()).To(Equal("ftp://quic.clemente.io/foo"))
})
It("errors with missing scheme", func() {
headers := []qpack.HeaderField{
{Name: ":protocol", Value: "webtransport"},
{Name: ":method", Value: http.MethodConnect},
{Name: ":authority", Value: "quic.clemente.io"},
{Name: ":path", Value: "/foo"},
}
_, err := requestFromHeaders(headers)
Expect(err).To(MatchError("extended CONNECT: :scheme, :path and :authority must not be empty"))
})
}) })
Context("extracting the hostname from a request", func() { Context("extracting the hostname from a request", func() {