|
|
@@ -7,11 +7,11 @@
|
|
|
package ctxhttp
|
|
|
|
|
|
import (
|
|
|
+ "io"
|
|
|
"io/ioutil"
|
|
|
- "net"
|
|
|
"net/http"
|
|
|
"net/http/httptest"
|
|
|
- "sync"
|
|
|
+ "strings"
|
|
|
"testing"
|
|
|
"time"
|
|
|
|
|
|
@@ -23,59 +23,62 @@ const (
|
|
|
requestBody = "ok"
|
|
|
)
|
|
|
|
|
|
-func TestNoTimeout(t *testing.T) {
|
|
|
- ctx := context.Background()
|
|
|
- resp, err := doRequest(ctx)
|
|
|
-
|
|
|
- if resp == nil || err != nil {
|
|
|
- t.Fatalf("error received from client: %v %v", err, resp)
|
|
|
- }
|
|
|
+func okHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
+ time.Sleep(requestDuration)
|
|
|
+ io.WriteString(w, requestBody)
|
|
|
}
|
|
|
|
|
|
-func TestCancel(t *testing.T) {
|
|
|
- ctx, cancel := context.WithCancel(context.Background())
|
|
|
- go func() {
|
|
|
- time.Sleep(requestDuration / 2)
|
|
|
- cancel()
|
|
|
- }()
|
|
|
-
|
|
|
- resp, err := doRequest(ctx)
|
|
|
+func TestNoTimeout(t *testing.T) {
|
|
|
+ ts := httptest.NewServer(http.HandlerFunc(okHandler))
|
|
|
+ defer ts.Close()
|
|
|
|
|
|
- if resp != nil || err == nil {
|
|
|
- t.Fatalf("expected error, didn't get one. resp: %v", resp)
|
|
|
+ ctx := context.Background()
|
|
|
+ res, err := Get(ctx, nil, ts.URL)
|
|
|
+ if err != nil {
|
|
|
+ t.Fatal(err)
|
|
|
}
|
|
|
- if err != ctx.Err() {
|
|
|
- t.Fatalf("expected error from context but got: %v", err)
|
|
|
+ defer res.Body.Close()
|
|
|
+ slurp, err := ioutil.ReadAll(res.Body)
|
|
|
+ if err != nil {
|
|
|
+ t.Fatal(err)
|
|
|
+ }
|
|
|
+ if string(slurp) != requestBody {
|
|
|
+ t.Errorf("body = %q; want %q", slurp, requestBody)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-func TestCancelAfterRequest(t *testing.T) {
|
|
|
+func TestCancelBeforeHeaders(t *testing.T) {
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
|
|
- resp, err := doRequest(ctx)
|
|
|
-
|
|
|
- // Cancel before reading the body.
|
|
|
- // Request.Body should still be readable after the context is canceled.
|
|
|
- cancel()
|
|
|
+ blockServer := make(chan struct{})
|
|
|
+ ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
+ cancel()
|
|
|
+ <-blockServer
|
|
|
+ io.WriteString(w, requestBody)
|
|
|
+ }))
|
|
|
+ defer ts.Close()
|
|
|
+ defer close(blockServer)
|
|
|
|
|
|
- b, err := ioutil.ReadAll(resp.Body)
|
|
|
- if err != nil || string(b) != requestBody {
|
|
|
- t.Fatalf("could not read body: %q %v", b, err)
|
|
|
+ res, err := Get(ctx, nil, ts.URL)
|
|
|
+ if err == nil {
|
|
|
+ res.Body.Close()
|
|
|
+ t.Fatal("Get returned unexpected nil error")
|
|
|
+ }
|
|
|
+ if !strings.Contains(err.Error(), "canceled") {
|
|
|
+ t.Errorf("err = %v; want something with \"canceled\"", err)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
func TestCancelAfterHangingRequest(t *testing.T) {
|
|
|
- handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
+ ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
w.(http.Flusher).Flush()
|
|
|
<-w.(http.CloseNotifier).CloseNotify()
|
|
|
- })
|
|
|
-
|
|
|
- serv := httptest.NewServer(handler)
|
|
|
- defer serv.Close()
|
|
|
+ }))
|
|
|
+ defer ts.Close()
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
- resp, err := Get(ctx, nil, serv.URL)
|
|
|
+ resp, err := Get(ctx, nil, ts.URL)
|
|
|
if err != nil {
|
|
|
t.Fatalf("unexpected error in Get: %v", err)
|
|
|
}
|
|
|
@@ -101,76 +104,3 @@ func TestCancelAfterHangingRequest(t *testing.T) {
|
|
|
case <-done:
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
-func doRequest(ctx context.Context) (*http.Response, error) {
|
|
|
- var okHandler = http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
- time.Sleep(requestDuration)
|
|
|
- w.Write([]byte(requestBody))
|
|
|
- })
|
|
|
-
|
|
|
- serv := httptest.NewServer(okHandler)
|
|
|
- defer serv.Close()
|
|
|
-
|
|
|
- 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()
|
|
|
-}
|