From 51b185b55acf9fbefb5f9bcae225ca55687fdf50 Mon Sep 17 00:00:00 2001 From: Simone Basso Date: Tue, 23 Apr 2024 13:43:29 +0200 Subject: [PATCH] feat(testingx): add HTTPHandlerResetWhileReadingBody (#1563) Closes https://github.com/ooni/probe/issues/2714 Extracted from https://github.com/ooni/probe-cli/pull/1560 Part of https://github.com/ooni/probe/issues/2700 --- internal/testingx/httptestx.go | 48 ++++++++++++++++++++++++----- internal/testingx/httptestx_test.go | 40 ++++++++++++++++++++++++ 2 files changed, 80 insertions(+), 8 deletions(-) diff --git a/internal/testingx/httptestx.go b/internal/testingx/httptestx.go index 2129794914..5a8ce91290 100644 --- a/internal/testingx/httptestx.go +++ b/internal/testingx/httptestx.go @@ -6,8 +6,10 @@ import ( "net" "net/http" "net/url" + "time" "github.com/ooni/netem" + "github.com/ooni/probe-cli/v3/internal/randx" "github.com/ooni/probe-cli/v3/internal/runtimex" ) @@ -173,14 +175,7 @@ func HTTPHandlerTimeout() http.Handler { } func httpHandlerHijack(w http.ResponseWriter, r *http.Request, policy string) { - // Note: - // - // 1. we assume we can hihack the connection - // - // 2. Hijack won't fail the first time it's invoked - hijacker := w.(http.Hijacker) - conn, _ := runtimex.Try2(hijacker.Hijack()) - + conn := httpHijack(w) defer conn.Close() switch policy { @@ -194,3 +189,40 @@ func httpHandlerHijack(w http.ResponseWriter, r *http.Request, policy string) { // nothing } } + +// HTTPHandlerResetWhileReadingBody returns a handler that sends a +// connection reset by peer while the client is reading the body. +func HTTPHandlerResetWhileReadingBody() http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + conn := httpHijack(w) + defer conn.Close() + + // write the HTTP response headers + conn.Write([]byte("HTTP/1.1 200 Ok\r\n")) + conn.Write([]byte("Content-Type: text/html\r\n")) + conn.Write([]byte("Content-Length: 65535\r\n")) + conn.Write([]byte("\r\n")) + + // start writing the response + content := randx.Letters(32768) + conn.Write([]byte(content)) + + // sleep for half a second simulating something wrong + time.Sleep(500 * time.Millisecond) + + // finally issue reset for the conn + tcpMaybeResetNetConn(conn) + }) +} + +// httpHijack is a convenience function to hijack the underlying connection. +func httpHijack(w http.ResponseWriter) net.Conn { + // Note: + // + // 1. we assume we can hihack the connection + // + // 2. Hijack won't fail the first time it's invoked + hijacker := w.(http.Hijacker) + conn, _ := runtimex.Try2(hijacker.Hijack()) + return conn +} diff --git a/internal/testingx/httptestx_test.go b/internal/testingx/httptestx_test.go index 57211ce70d..54df90e9ea 100644 --- a/internal/testingx/httptestx_test.go +++ b/internal/testingx/httptestx_test.go @@ -498,3 +498,43 @@ func TestHTTPTestxWithNetem(t *testing.T) { }) } } + +func TestHTTPHandlerResetWhileReadingBody(t *testing.T) { + // create a server for testing the given handler + server := testingx.MustNewHTTPServer(testingx.HTTPHandlerResetWhileReadingBody()) + defer server.Close() + + // create a suitable HTTP transport using netxlite + netx := &netxlite.Netx{Underlying: nil} + dialer := netx.NewDialerWithoutResolver(log.Log) + handshaker := netx.NewTLSHandshakerStdlib(log.Log) + tlsDialer := netxlite.NewTLSDialer(dialer, handshaker) + txp := netxlite.NewHTTPTransportWithOptions(log.Log, dialer, tlsDialer) + + // create the request + req := runtimex.Try1(http.NewRequest("GET", server.URL, nil)) + + // perform the round trip + resp, err := txp.RoundTrip(req) + + // we do not expect an error during the round trip + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // start reading the response where we expect to see a RST + respbody, err := netxlite.ReadAllContext(req.Context(), resp.Body) + + // verify we received a connection reset + if !errors.Is(err, netxlite.ECONNRESET) { + t.Fatal("expected ECONNRESET, got", err) + } + + // make sure we've got no bytes + if len(respbody) != 0 { + t.Fatal("expected to see zero bytes here") + } +}