mirror of
https://github.com/soheilhy/cmux.git
synced 2025-01-19 03:06:07 +08:00
89dd8ce1fd
As reported in #4, when the root listener is closed, calling Accept
on mux'd listeners would block forever, instead of returning an error.
This is because of a bug introduced in b90740d
.
This commit fixes #4 by selecting on both donec and connc of muxed
listeners.
Added a test case to guard against this issue.
188 lines
3.2 KiB
Go
188 lines
3.2 KiB
Go
package cmux
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net"
|
|
"net/http"
|
|
"net/rpc"
|
|
"testing"
|
|
)
|
|
|
|
const (
|
|
testHTTP1Resp = "http1"
|
|
rpcVal = 1234
|
|
)
|
|
|
|
var testPort = 5125
|
|
|
|
func testAddr() string {
|
|
testPort++
|
|
return fmt.Sprintf("127.0.0.1:%d", testPort)
|
|
}
|
|
|
|
func testListener(t *testing.T) (net.Listener, string) {
|
|
addr := testAddr()
|
|
l, err := net.Listen("tcp", addr)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return l, addr
|
|
}
|
|
|
|
type testHTTP1Handler struct{}
|
|
|
|
func (h *testHTTP1Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
fmt.Fprintf(w, testHTTP1Resp)
|
|
}
|
|
|
|
func runTestHTTPServer(l net.Listener) {
|
|
s := &http.Server{
|
|
Handler: &testHTTP1Handler{},
|
|
}
|
|
s.Serve(l)
|
|
}
|
|
|
|
func runTestHTTP1Client(t *testing.T, addr string) {
|
|
r, err := http.Get("http://" + addr)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
defer r.Body.Close()
|
|
b, err := ioutil.ReadAll(r.Body)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
if string(b) != testHTTP1Resp {
|
|
t.Errorf("invalid response: want=%s got=%s", testHTTP1Resp, b)
|
|
}
|
|
}
|
|
|
|
type TestRPCRcvr struct{}
|
|
|
|
func (r TestRPCRcvr) Test(i int, j *int) error {
|
|
*j = i
|
|
return nil
|
|
}
|
|
|
|
func runTestRPCServer(l net.Listener) {
|
|
s := rpc.NewServer()
|
|
s.Register(TestRPCRcvr{})
|
|
|
|
for {
|
|
c, err := l.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
s.ServeConn(c)
|
|
}
|
|
}
|
|
|
|
func runTestRPCClient(t *testing.T, addr string) {
|
|
c, err := rpc.Dial("tcp", addr)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
var num int
|
|
if err := c.Call("TestRPCRcvr.Test", rpcVal, &num); err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
if num != rpcVal {
|
|
t.Errorf("wrong rpc response: want=%d got=%v", rpcVal, num)
|
|
}
|
|
}
|
|
|
|
func TestAny(t *testing.T) {
|
|
l, addr := testListener(t)
|
|
defer l.Close()
|
|
|
|
muxl := New(l)
|
|
httpl := muxl.Match(Any())
|
|
|
|
go runTestHTTPServer(httpl)
|
|
go muxl.Serve()
|
|
|
|
r, err := http.Get("http://" + addr)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
defer r.Body.Close()
|
|
b, err := ioutil.ReadAll(r.Body)
|
|
if string(b) != testHTTP1Resp {
|
|
t.Errorf("invalid response: want=%s got=%s", testHTTP1Resp, b)
|
|
}
|
|
}
|
|
|
|
func TestHTTPGoRPC(t *testing.T) {
|
|
l, addr := testListener(t)
|
|
defer l.Close()
|
|
|
|
muxl := New(l)
|
|
httpl := muxl.Match(HTTP2(), HTTP1Fast())
|
|
rpcl := muxl.Match(Any())
|
|
|
|
go runTestHTTPServer(httpl)
|
|
go runTestRPCServer(rpcl)
|
|
go muxl.Serve()
|
|
|
|
runTestHTTP1Client(t, addr)
|
|
runTestRPCClient(t, addr)
|
|
}
|
|
|
|
func TestErrorHandler(t *testing.T) {
|
|
l, addr := testListener(t)
|
|
defer l.Close()
|
|
|
|
muxl := New(l)
|
|
httpl := muxl.Match(HTTP2(), HTTP1Fast())
|
|
|
|
go runTestHTTPServer(httpl)
|
|
go muxl.Serve()
|
|
|
|
firstErr := true
|
|
muxl.HandleError(func(err error) bool {
|
|
if !firstErr {
|
|
return true
|
|
}
|
|
if _, ok := err.(ErrNotMatched); !ok {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
firstErr = false
|
|
return true
|
|
})
|
|
|
|
c, err := rpc.Dial("tcp", addr)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
var num int
|
|
if err := c.Call("TestRPCRcvr.Test", rpcVal, &num); err == nil {
|
|
t.Error("rpc got a response")
|
|
}
|
|
}
|
|
|
|
type closerConn struct {
|
|
net.Conn
|
|
}
|
|
|
|
func (c closerConn) Close() error { return nil }
|
|
|
|
func TestClosed(t *testing.T) {
|
|
mux := &cMux{}
|
|
lis := mux.Match(Any()).(muxListener)
|
|
close(lis.donec)
|
|
mux.serve(closerConn{})
|
|
_, err := lis.Accept()
|
|
if _, ok := err.(errListenerClosed); !ok {
|
|
t.Errorf("expected errListenerClosed got %v", err)
|
|
}
|
|
}
|