From 9b073aad58f6b3f1428b7e784c231c73aeedab28 Mon Sep 17 00:00:00 2001 From: Tw Date: Tue, 25 Jul 2017 15:12:38 +0800 Subject: [PATCH] proxy: fix hang on chunked websocket server Signed-off-by: Tw --- caddyhttp/proxy/proxy_test.go | 48 +++++++++++++++++++++++++++++++++ caddyhttp/proxy/reverseproxy.go | 2 +- 2 files changed, 49 insertions(+), 1 deletion(-) diff --git a/caddyhttp/proxy/proxy_test.go b/caddyhttp/proxy/proxy_test.go index 64c2a7be..d7342560 100644 --- a/caddyhttp/proxy/proxy_test.go +++ b/caddyhttp/proxy/proxy_test.go @@ -1422,3 +1422,51 @@ func BenchmarkProxy(b *testing.B) { p.ServeHTTP(w, r) } } + +func TestChunkedWebSocketReverseProxy(t *testing.T) { + s := websocket.Server{ + Handler: websocket.Handler(func(ws *websocket.Conn) { + for { + select {} + } + }), + } + s.Config.Header = http.Header(make(map[string][]string)) + s.Config.Header.Set("Transfer-Encoding", "chunked") + + wsNop := httptest.NewServer(s) + defer wsNop.Close() + + // Get proxy to use for the test + p := newWebSocketTestProxy(wsNop.URL, false) + + // Create client request + r := httptest.NewRequest("GET", "/", nil) + + r.Header = http.Header{ + "Connection": {"Upgrade"}, + "Upgrade": {"websocket"}, + "Origin": {wsNop.URL}, + "Sec-WebSocket-Key": {"x3JJHMbDL1EzLkh9GBhXDw=="}, + "Sec-WebSocket-Version": {"13"}, + } + + // Capture the request + w := &recorderHijacker{httptest.NewRecorder(), new(fakeConn)} + + // Booya! Do the test. + _, err := p.ServeHTTP(w, r) + + // Make sure the backend accepted the WS connection. + // Mostly interested in the Upgrade and Connection response headers + // and the 101 status code. + expected := []byte("HTTP/1.1 101 Switching Protocols\r\nUpgrade: websocket\r\nConnection: Upgrade\r\nSec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk=\r\nTransfer-Encoding: chunked\r\n\r\n") + actual := w.fakeConn.writeBuf.Bytes() + if !bytes.Equal(actual, expected) { + t.Errorf("Expected backend to accept response:\n'%s'\nActually got:\n'%s'", expected, actual) + } + + if err != nil { + t.Error(err) + } +} diff --git a/caddyhttp/proxy/reverseproxy.go b/caddyhttp/proxy/reverseproxy.go index 08bfdd56..41687cc1 100644 --- a/caddyhttp/proxy/reverseproxy.go +++ b/caddyhttp/proxy/reverseproxy.go @@ -272,7 +272,7 @@ func (rp *ReverseProxy) ServeHTTP(rw http.ResponseWriter, outreq *http.Request, } if isWebsocket { - res.Body.Close() + defer res.Body.Close() hj, ok := rw.(http.Hijacker) if !ok { panic(httpserver.NonHijackerError{Underlying: rw})