mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
add a http3.Hijacker that allows stream creation on a QUIC session from a http.Response.Body
This commit is contained in:
parent
332473668a
commit
57461e01b5
3 changed files with 38 additions and 8 deletions
|
@ -1,12 +1,28 @@
|
|||
package http3
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"io"
|
||||
|
||||
"github.com/lucas-clemente/quic-go"
|
||||
)
|
||||
|
||||
type StreamCreator interface {
|
||||
OpenStream() (quic.Stream, error)
|
||||
OpenStreamSync(context.Context) (quic.Stream, error)
|
||||
OpenUniStream() (quic.SendStream, error)
|
||||
OpenUniStreamSync(context.Context) (quic.SendStream, error)
|
||||
}
|
||||
|
||||
var _ StreamCreator = quic.Connection(nil)
|
||||
|
||||
// A Hijacker allows hijacking of the stream creating part of a quic.Session from a http.Response.Body.
|
||||
// It is used by WebTransport to create WebTransport streams after a session has been established.
|
||||
type Hijacker interface {
|
||||
StreamCreator() StreamCreator
|
||||
}
|
||||
|
||||
// The body of a http.Request or http.Response.
|
||||
type body struct {
|
||||
str quic.Stream
|
||||
|
@ -24,6 +40,13 @@ type body struct {
|
|||
|
||||
var _ io.ReadCloser = &body{}
|
||||
|
||||
type hijackableBody struct {
|
||||
body
|
||||
conn quic.Connection // only needed to implement Hijacker
|
||||
}
|
||||
|
||||
var _ Hijacker = &hijackableBody{}
|
||||
|
||||
func newRequestBody(str quic.Stream, onFrameError func()) *body {
|
||||
return &body{
|
||||
str: str,
|
||||
|
@ -31,14 +54,21 @@ func newRequestBody(str quic.Stream, onFrameError func()) *body {
|
|||
}
|
||||
}
|
||||
|
||||
func newResponseBody(str quic.Stream, done chan<- struct{}, onFrameError func()) *body {
|
||||
return &body{
|
||||
str: str,
|
||||
onFrameError: onFrameError,
|
||||
reqDone: done,
|
||||
func newResponseBody(str quic.Stream, conn quic.Connection, done chan<- struct{}, onFrameError func()) *hijackableBody {
|
||||
return &hijackableBody{
|
||||
body: body{
|
||||
str: str,
|
||||
onFrameError: onFrameError,
|
||||
reqDone: done,
|
||||
},
|
||||
conn: conn,
|
||||
}
|
||||
}
|
||||
|
||||
func (r *hijackableBody) StreamCreator() StreamCreator {
|
||||
return r.conn
|
||||
}
|
||||
|
||||
func (r *body) Read(b []byte) (int, error) {
|
||||
n, err := r.readImpl(b)
|
||||
if err != nil {
|
||||
|
|
|
@ -29,7 +29,7 @@ func (t bodyType) String() string {
|
|||
|
||||
var _ = Describe("Body", func() {
|
||||
var (
|
||||
rb *body
|
||||
rb io.ReadCloser
|
||||
str *mockquic.MockStream
|
||||
buf *bytes.Buffer
|
||||
reqDone chan struct{}
|
||||
|
@ -68,7 +68,7 @@ var _ = Describe("Body", func() {
|
|||
rb = newRequestBody(str, errorCb)
|
||||
case bodyTypeResponse:
|
||||
reqDone = make(chan struct{})
|
||||
rb = newResponseBody(str, reqDone, errorCb)
|
||||
rb = newResponseBody(str, nil, reqDone, errorCb)
|
||||
}
|
||||
})
|
||||
|
||||
|
|
|
@ -316,7 +316,7 @@ func (c *client) doRequest(
|
|||
res.Header.Add(hf.Name, hf.Value)
|
||||
}
|
||||
}
|
||||
respBody := newResponseBody(str, reqDone, func() {
|
||||
respBody := newResponseBody(str, c.conn, reqDone, func() {
|
||||
c.conn.CloseWithError(quic.ApplicationErrorCode(errorFrameUnexpected), "")
|
||||
})
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue