|
|
@@ -8,8 +8,10 @@ package ctxhttp
|
|
|
|
|
|
import (
|
|
|
"io/ioutil"
|
|
|
+ "net"
|
|
|
"net/http"
|
|
|
"net/http/httptest"
|
|
|
+ "sync"
|
|
|
"testing"
|
|
|
"time"
|
|
|
|
|
|
@@ -111,3 +113,64 @@ func doRequest(ctx context.Context) (*http.Response, error) {
|
|
|
|
|
|
return Get(ctx, nil, serv.URL)
|
|
|
}
|
|
|
+
|
|
|
+// golang.org/issue/14065
|
|
|
+func TestClosesResponseBodyOnCancel(t *testing.T) {
|
|
|
+ defer func() { testHookContextDoneBeforeHeaders = nop }()
|
|
|
+ defer func() { testHookDoReturned = nop }()
|
|
|
+ defer func() { testHookDidBodyClose = nop }()
|
|
|
+
|
|
|
+ ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {}))
|
|
|
+ defer ts.Close()
|
|
|
+
|
|
|
+ ctx, cancel := context.WithCancel(context.Background())
|
|
|
+
|
|
|
+ // closed when Do enters select case <-ctx.Done()
|
|
|
+ enteredDonePath := make(chan struct{})
|
|
|
+
|
|
|
+ testHookContextDoneBeforeHeaders = func() {
|
|
|
+ close(enteredDonePath)
|
|
|
+ }
|
|
|
+
|
|
|
+ testHookDoReturned = func() {
|
|
|
+ // We now have the result (the Flush'd headers) at least,
|
|
|
+ // so we can cancel the request.
|
|
|
+ cancel()
|
|
|
+
|
|
|
+ // But block the client.Do goroutine from sending
|
|
|
+ // until Do enters into the <-ctx.Done() path, since
|
|
|
+ // otherwise if both channels are readable, select
|
|
|
+ // picks a random one.
|
|
|
+ <-enteredDonePath
|
|
|
+ }
|
|
|
+
|
|
|
+ sawBodyClose := make(chan struct{})
|
|
|
+ testHookDidBodyClose = func() { close(sawBodyClose) }
|
|
|
+
|
|
|
+ tr := &http.Transport{}
|
|
|
+ defer tr.CloseIdleConnections()
|
|
|
+ c := &http.Client{Transport: tr}
|
|
|
+ req, _ := http.NewRequest("GET", ts.URL, nil)
|
|
|
+ _, doErr := Do(ctx, c, req)
|
|
|
+
|
|
|
+ select {
|
|
|
+ case <-sawBodyClose:
|
|
|
+ case <-time.After(5 * time.Second):
|
|
|
+ t.Fatal("timeout waiting for body to close")
|
|
|
+ }
|
|
|
+
|
|
|
+ if doErr != ctx.Err() {
|
|
|
+ t.Errorf("Do error = %v; want %v", doErr, ctx.Err())
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+type noteCloseConn struct {
|
|
|
+ net.Conn
|
|
|
+ onceClose sync.Once
|
|
|
+ closefn func()
|
|
|
+}
|
|
|
+
|
|
|
+func (c *noteCloseConn) Close() error {
|
|
|
+ c.onceClose.Do(c.closefn)
|
|
|
+ return c.Conn.Close()
|
|
|
+}
|