mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
http3: add compatibility with net/http.ResponseController (#3790)
* feat: compatibility with "net/http".ResponseController * better deadline tests * don't run deadline tests on Go 1.19 * skip deadline tests on Go 1.19
This commit is contained in:
parent
4a2a5740b2
commit
172123c340
5 changed files with 129 additions and 0 deletions
|
@ -6,6 +6,7 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/quic-go/quic-go"
|
"github.com/quic-go/quic-go"
|
||||||
"github.com/quic-go/quic-go/internal/utils"
|
"github.com/quic-go/quic-go/internal/utils"
|
||||||
|
@ -15,6 +16,7 @@ import (
|
||||||
|
|
||||||
type responseWriter struct {
|
type responseWriter struct {
|
||||||
conn quic.Connection
|
conn quic.Connection
|
||||||
|
str quic.Stream
|
||||||
bufferedStr *bufio.Writer
|
bufferedStr *bufio.Writer
|
||||||
buf []byte
|
buf []byte
|
||||||
|
|
||||||
|
@ -36,6 +38,7 @@ func newResponseWriter(str quic.Stream, conn quic.Connection, logger utils.Logge
|
||||||
header: http.Header{},
|
header: http.Header{},
|
||||||
buf: make([]byte, 16),
|
buf: make([]byte, 16),
|
||||||
conn: conn,
|
conn: conn,
|
||||||
|
str: str,
|
||||||
bufferedStr: bufio.NewWriter(str),
|
bufferedStr: bufio.NewWriter(str),
|
||||||
logger: logger,
|
logger: logger,
|
||||||
}
|
}
|
||||||
|
@ -121,6 +124,14 @@ func (w *responseWriter) StreamCreator() StreamCreator {
|
||||||
return w.conn
|
return w.conn
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (w *responseWriter) SetReadDeadline(deadline time.Time) error {
|
||||||
|
return w.str.SetReadDeadline(deadline)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *responseWriter) SetWriteDeadline(deadline time.Time) error {
|
||||||
|
return w.str.SetWriteDeadline(deadline)
|
||||||
|
}
|
||||||
|
|
||||||
// copied from http2/http2.go
|
// copied from http2/http2.go
|
||||||
// bodyAllowedForStatus reports whether a given response status code
|
// bodyAllowedForStatus reports whether a given response status code
|
||||||
// permits a body. See RFC 2616, section 4.4.
|
// permits a body. See RFC 2616, section 4.4.
|
||||||
|
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
mockquic "github.com/quic-go/quic-go/internal/mocks/quic"
|
mockquic "github.com/quic-go/quic-go/internal/mocks/quic"
|
||||||
"github.com/quic-go/quic-go/internal/utils"
|
"github.com/quic-go/quic-go/internal/utils"
|
||||||
|
@ -25,6 +26,8 @@ var _ = Describe("Response Writer", func() {
|
||||||
strBuf = &bytes.Buffer{}
|
strBuf = &bytes.Buffer{}
|
||||||
str := mockquic.NewMockStream(mockCtrl)
|
str := mockquic.NewMockStream(mockCtrl)
|
||||||
str.EXPECT().Write(gomock.Any()).DoAndReturn(strBuf.Write).AnyTimes()
|
str.EXPECT().Write(gomock.Any()).DoAndReturn(strBuf.Write).AnyTimes()
|
||||||
|
str.EXPECT().SetReadDeadline(gomock.Any()).Return(nil).AnyTimes()
|
||||||
|
str.EXPECT().SetWriteDeadline(gomock.Any()).Return(nil).AnyTimes()
|
||||||
rw = newResponseWriter(str, nil, utils.DefaultLogger)
|
rw = newResponseWriter(str, nil, utils.DefaultLogger)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -156,4 +159,9 @@ var _ = Describe("Response Writer", func() {
|
||||||
fields := decodeHeader(strBuf)
|
fields := decodeHeader(strBuf)
|
||||||
Expect(fields).To(HaveKeyWithValue("content-type", []string{"text/html; charset=utf-8"}))
|
Expect(fields).To(HaveKeyWithValue("content-type", []string{"text/html; charset=utf-8"}))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It(`is compatible with "net/http".ResponseController`, func() {
|
||||||
|
Expect(rw.SetReadDeadline(time.Now().Add(1 * time.Second))).To(BeNil())
|
||||||
|
Expect(rw.SetWriteDeadline(time.Now().Add(1 * time.Second))).To(BeNil())
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
22
integrationtests/self/go119_test.go
Normal file
22
integrationtests/self/go119_test.go
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
//go:build go1.19 && !go1.20
|
||||||
|
|
||||||
|
package self_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
go120 = false
|
||||||
|
errNotSupported = errors.New("not supported")
|
||||||
|
)
|
||||||
|
|
||||||
|
func setReadDeadline(w http.ResponseWriter, deadline time.Time) error {
|
||||||
|
return errNotSupported
|
||||||
|
}
|
||||||
|
|
||||||
|
func setWriteDeadline(w http.ResponseWriter, deadline time.Time) error {
|
||||||
|
return errNotSupported
|
||||||
|
}
|
22
integrationtests/self/go120_test.go
Normal file
22
integrationtests/self/go120_test.go
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
//go:build go1.20
|
||||||
|
|
||||||
|
package self_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var go120 = true
|
||||||
|
|
||||||
|
func setReadDeadline(w http.ResponseWriter, deadline time.Time) error {
|
||||||
|
rc := http.NewResponseController(w)
|
||||||
|
|
||||||
|
return rc.SetReadDeadline(deadline)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setWriteDeadline(w http.ResponseWriter, deadline time.Time) error {
|
||||||
|
rc := http.NewResponseController(w)
|
||||||
|
|
||||||
|
return rc.SetWriteDeadline(deadline)
|
||||||
|
}
|
|
@ -11,6 +11,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -25,6 +26,17 @@ import (
|
||||||
"github.com/onsi/gomega/gbytes"
|
"github.com/onsi/gomega/gbytes"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type neverEnding byte
|
||||||
|
|
||||||
|
func (b neverEnding) Read(p []byte) (n int, err error) {
|
||||||
|
for i := range p {
|
||||||
|
p[i] = byte(b)
|
||||||
|
}
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
const deadlineDelay = 250 * time.Millisecond
|
||||||
|
|
||||||
var _ = Describe("HTTP tests", func() {
|
var _ = Describe("HTTP tests", func() {
|
||||||
var (
|
var (
|
||||||
mux *http.ServeMux
|
mux *http.ServeMux
|
||||||
|
@ -374,6 +386,60 @@ var _ = Describe("HTTP tests", func() {
|
||||||
Expect(repl).To(Equal(data))
|
Expect(repl).To(Equal(data))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("supports read deadlines", func() {
|
||||||
|
if !go120 {
|
||||||
|
Skip("This test requires Go 1.20+")
|
||||||
|
}
|
||||||
|
|
||||||
|
mux.HandleFunc("/read-deadline", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
defer GinkgoRecover()
|
||||||
|
err := setReadDeadline(w, time.Now().Add(deadlineDelay))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
|
||||||
|
body, err := io.ReadAll(r.Body)
|
||||||
|
Expect(err).To(MatchError(os.ErrDeadlineExceeded))
|
||||||
|
Expect(body).To(ContainSubstring("aa"))
|
||||||
|
|
||||||
|
w.Write([]byte("ok"))
|
||||||
|
})
|
||||||
|
|
||||||
|
expectedEnd := time.Now().Add(deadlineDelay)
|
||||||
|
resp, err := client.Post("https://localhost:"+port+"/read-deadline", "text/plain", neverEnding('a'))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(resp.StatusCode).To(Equal(200))
|
||||||
|
|
||||||
|
body, err := io.ReadAll(gbytes.TimeoutReader(resp.Body, 2*deadlineDelay))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(time.Now().After(expectedEnd)).To(BeTrue())
|
||||||
|
Expect(string(body)).To(Equal("ok"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("supports write deadlines", func() {
|
||||||
|
if !go120 {
|
||||||
|
Skip("This test requires Go 1.20+")
|
||||||
|
}
|
||||||
|
|
||||||
|
mux.HandleFunc("/write-deadline", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
defer GinkgoRecover()
|
||||||
|
err := setWriteDeadline(w, time.Now().Add(deadlineDelay))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
|
||||||
|
_, err = io.Copy(w, neverEnding('a'))
|
||||||
|
Expect(err).To(MatchError(os.ErrDeadlineExceeded))
|
||||||
|
})
|
||||||
|
|
||||||
|
expectedEnd := time.Now().Add(deadlineDelay)
|
||||||
|
|
||||||
|
resp, err := client.Get("https://localhost:" + port + "/write-deadline")
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(resp.StatusCode).To(Equal(200))
|
||||||
|
|
||||||
|
body, err := io.ReadAll(gbytes.TimeoutReader(resp.Body, 2*deadlineDelay))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(time.Now().After(expectedEnd)).To(BeTrue())
|
||||||
|
Expect(string(body)).To(ContainSubstring("aa"))
|
||||||
|
})
|
||||||
|
|
||||||
if version != protocol.VersionDraft29 {
|
if version != protocol.VersionDraft29 {
|
||||||
It("serves other QUIC connections", func() {
|
It("serves other QUIC connections", func() {
|
||||||
tlsConf := testdata.GetTLSConfig()
|
tlsConf := testdata.GetTLSConfig()
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue