1
0
mirror of https://github.com/go-micro/go-micro.git synced 2025-04-17 11:06:19 +02:00

fix http_transport Recv and Close race condition on buff (#2374)

fix rpc_stream panic override with the "Unlock of unlocked RWMutex" panic

Co-authored-by: Hunyadvári Péter <peter.hunyadvari@vcc.live>
This commit is contained in:
Ak-Army 2021-12-05 12:56:17 +01:00 committed by GitHub
parent 81244a41f1
commit 97f169c424
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 94 additions and 6 deletions

View File

@ -75,10 +75,10 @@ func (r *rpcStream) Send(msg interface{}) error {
func (r *rpcStream) Recv(msg interface{}) error { func (r *rpcStream) Recv(msg interface{}) error {
r.Lock() r.Lock()
defer r.Unlock()
if r.isClosed() { if r.isClosed() {
r.err = errShutdown r.err = errShutdown
r.Unlock()
return errShutdown return errShutdown
} }
@ -90,9 +90,12 @@ func (r *rpcStream) Recv(msg interface{}) error {
if err != nil { if err != nil {
if err == io.EOF && !r.isClosed() { if err == io.EOF && !r.isClosed() {
r.err = io.ErrUnexpectedEOF r.err = io.ErrUnexpectedEOF
r.Unlock()
return io.ErrUnexpectedEOF return io.ErrUnexpectedEOF
} }
r.err = err r.err = err
r.Unlock()
return err return err
} }
@ -121,6 +124,7 @@ func (r *rpcStream) Recv(msg interface{}) error {
} }
} }
r.Unlock()
return r.err return r.err
} }

View File

@ -12,12 +12,13 @@ import (
"sync" "sync"
"time" "time"
"golang.org/x/net/http2"
"golang.org/x/net/http2/h2c"
maddr "go-micro.dev/v4/util/addr" maddr "go-micro.dev/v4/util/addr"
"go-micro.dev/v4/util/buf" "go-micro.dev/v4/util/buf"
mnet "go-micro.dev/v4/util/net" mnet "go-micro.dev/v4/util/net"
mls "go-micro.dev/v4/util/tls" mls "go-micro.dev/v4/util/tls"
"golang.org/x/net/http2"
"golang.org/x/net/http2/h2c"
) )
type httpTransport struct { type httpTransport struct {
@ -34,9 +35,10 @@ type httpTransportClient struct {
sync.RWMutex sync.RWMutex
// request must be stored for response processing // request must be stored for response processing
r chan *http.Request r chan *http.Request
bl []*http.Request bl []*http.Request
buff *bufio.Reader buff *bufio.Reader
closed bool
// local/remote ip // local/remote ip
local string local string
@ -137,7 +139,12 @@ func (h *httpTransportClient) Recv(m *Message) error {
h.conn.SetDeadline(time.Now().Add(h.ht.opts.Timeout)) h.conn.SetDeadline(time.Now().Add(h.ht.opts.Timeout))
} }
h.Lock()
if h.closed {
return io.EOF
}
rsp, err := http.ReadResponse(h.buff, r) rsp, err := http.ReadResponse(h.buff, r)
h.Unlock()
if err != nil { if err != nil {
return err return err
} }
@ -173,6 +180,7 @@ func (h *httpTransportClient) Close() error {
h.once.Do(func() { h.once.Do(func() {
h.Lock() h.Lock()
h.buff.Reset(nil) h.buff.Reset(nil)
h.closed = true
h.Unlock() h.Unlock()
close(h.r) close(h.r)
}) })

View File

@ -1,8 +1,10 @@
package transport package transport
import ( import (
"fmt"
"io" "io"
"net" "net"
"sync"
"testing" "testing"
"time" "time"
) )
@ -244,3 +246,77 @@ func TestHTTPTransportTimeout(t *testing.T) {
<-done <-done
} }
func TestHTTPTransportCloseWhenRecv(t *testing.T) {
tr := NewHTTPTransport()
l, err := tr.Listen("127.0.0.1:0")
if err != nil {
t.Errorf("Unexpected listen err: %v", err)
}
defer l.Close()
fn := func(sock Socket) {
defer sock.Close()
for {
var m Message
if err := sock.Recv(&m); err != nil {
return
}
if err := sock.Send(&m); err != nil {
return
}
}
}
done := make(chan bool)
go func() {
if err := l.Accept(fn); err != nil {
select {
case <-done:
default:
t.Errorf("Unexpected accept err: %v", err)
}
}
}()
c, err := tr.Dial(l.Addr())
if err != nil {
t.Errorf("Unexpected dial err: %v", err)
}
defer c.Close()
m := Message{
Header: map[string]string{
"Content-Type": "application/json",
},
Body: []byte(`{"message": "Hello World"}`),
}
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
for {
var rm Message
if err := c.Recv(&rm); err != nil {
if err == io.EOF {
return
}
t.Errorf("Unexpected recv err: %v", err)
}
fmt.Println("aa")
}
}()
for i := 1; i < 3; i++ {
if err := c.Send(&m); err != nil {
t.Errorf("Unexpected send err: %v", err)
}
}
close(done)
c.Close()
wg.Wait()
}