From cc90dfadafdeff069b39b3a9d6d3bc8dda6e7ab6 Mon Sep 17 00:00:00 2001 From: Marten Seemann Date: Wed, 25 Sep 2019 07:21:28 +0700 Subject: [PATCH] fix leaking request cancelation go routine in case an error occurred --- http3/client.go | 37 ++++++++++++++++++++++--------------- 1 file changed, 22 insertions(+), 15 deletions(-) diff --git a/http3/client.go b/http3/client.go index a0732c60..68f9f92b 100644 --- a/http3/client.go +++ b/http3/client.go @@ -152,21 +152,6 @@ func (c *client) RoundTrip(req *http.Request) (*http.Response, error) { return nil, err } - rsp, rerr := c.doRequest(req, str) - if rerr.streamErr != 0 { - str.CancelWrite(quic.ErrorCode(rerr.streamErr)) - } - if rerr.connErr != 0 { - var reason string - if rerr.err != nil { - reason = rerr.err.Error() - } - c.session.CloseWithError(quic.ErrorCode(rerr.connErr), reason) - } - return rsp, rerr.err -} - -func (c *client) doRequest(req *http.Request, str quic.Stream) (*http.Response, requestError) { // Request Cancellation: // This go routine keeps running even after RoundTrip() returns. // It is shut down when the application is done processing the body. @@ -180,6 +165,28 @@ func (c *client) doRequest(req *http.Request, str quic.Stream) (*http.Response, } }() + rsp, rerr := c.doRequest(req, str, reqDone) + if rerr.err != nil { // if any error occurred + close(reqDone) + if rerr.streamErr != 0 { // if it was a stream error + str.CancelWrite(quic.ErrorCode(rerr.streamErr)) + } + if rerr.connErr != 0 { // if it was a connection error + var reason string + if rerr.err != nil { + reason = rerr.err.Error() + } + c.session.CloseWithError(quic.ErrorCode(rerr.connErr), reason) + } + } + return rsp, rerr.err +} + +func (c *client) doRequest( + req *http.Request, + str quic.Stream, + reqDone chan struct{}, +) (*http.Response, requestError) { var requestGzip bool if !c.opts.DisableCompression && req.Method != "HEAD" && req.Header.Get("Accept-Encoding") == "" && req.Header.Get("Range") == "" { requestGzip = true