1
0
mirror of https://github.com/go-micro/go-micro.git synced 2025-01-05 10:20:53 +02:00

HTTP Transport make streaming truly bidirectional (#2528)

* [fix] http transport can send when wait for recv

* [fix] http transport can send multiple message and recevie them. Do not block send and receive on stream mode

* [fix] http transport can close the connection when recv is in progress, add tests

Co-authored-by: Hunyadvári Péter <peter.hunyadvari@vcc.live>
This commit is contained in:
Ak-Army 2022-07-15 12:00:13 +02:00 committed by GitHub
parent 1219d57f58
commit ec6a47c894
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 122 additions and 15 deletions

View File

@ -13,12 +13,13 @@ import (
"sync" "sync"
"time" "time"
"golang.org/x/net/http2"
"golang.org/x/net/http2/h2c"
maddr "github.com/asim/go-micro/v3/util/addr" maddr "github.com/asim/go-micro/v3/util/addr"
"github.com/asim/go-micro/v3/util/buf" "github.com/asim/go-micro/v3/util/buf"
mnet "github.com/asim/go-micro/v3/util/net" mnet "github.com/asim/go-micro/v3/util/net"
mls "github.com/asim/go-micro/v3/util/tls" mls "github.com/asim/go-micro/v3/util/tls"
"golang.org/x/net/http2"
"golang.org/x/net/http2/h2c"
) )
type httpTransport struct { type httpTransport struct {
@ -103,14 +104,20 @@ func (h *httpTransportClient) Send(m *Message) error {
Host: h.addr, Host: h.addr,
} }
h.Lock() if !h.dialOpts.Stream {
h.bl = append(h.bl, req) h.Lock()
select { if h.closed {
case h.r <- h.bl[0]: h.Unlock()
h.bl = h.bl[1:] return io.EOF
default: }
h.bl = append(h.bl, req)
select {
case h.r <- h.bl[0]:
h.bl = h.bl[1:]
default:
}
h.Unlock()
} }
h.Unlock()
// set timeout if its greater than 0 // set timeout if its greater than 0
if h.ht.opts.Timeout > time.Duration(0) { if h.ht.opts.Timeout > time.Duration(0) {
@ -129,7 +136,14 @@ func (h *httpTransportClient) Recv(m *Message) error {
if !h.dialOpts.Stream { if !h.dialOpts.Stream {
rc, ok := <-h.r rc, ok := <-h.r
if !ok { if !ok {
return io.EOF h.Lock()
if len(h.bl) == 0 {
h.Unlock()
return io.EOF
}
rc = h.bl[0]
h.bl = h.bl[1:]
h.Unlock()
} }
r = rc r = rc
} }
@ -141,6 +155,7 @@ func (h *httpTransportClient) Recv(m *Message) error {
h.Lock() h.Lock()
if h.closed { if h.closed {
h.Unlock()
return io.EOF return io.EOF
} }
rsp, err := http.ReadResponse(h.buff, r) rsp, err := http.ReadResponse(h.buff, r)
@ -177,6 +192,17 @@ func (h *httpTransportClient) Recv(m *Message) error {
} }
func (h *httpTransportClient) Close() error { func (h *httpTransportClient) Close() error {
if !h.dialOpts.Stream {
h.once.Do(func() {
h.Lock()
h.buff.Reset(nil)
h.closed = true
h.Unlock()
close(h.r)
})
return h.conn.Close()
}
err := h.conn.Close()
h.once.Do(func() { h.once.Do(func() {
h.Lock() h.Lock()
h.buff.Reset(nil) h.buff.Reset(nil)
@ -184,7 +210,7 @@ func (h *httpTransportClient) Close() error {
h.Unlock() h.Unlock()
close(h.r) close(h.r)
}) })
return h.conn.Close() return err
} }
func (h *httpTransportSocket) Local() string { func (h *httpTransportSocket) Local() string {
@ -523,7 +549,7 @@ func (h *httpTransport) Dial(addr string, opts ...DialOption) (Client, error) {
conn: conn, conn: conn,
buff: bufio.NewReader(conn), buff: bufio.NewReader(conn),
dialOpts: dopts, dialOpts: dopts,
r: make(chan *http.Request, 1), r: make(chan *http.Request, 100),
local: conn.LocalAddr().String(), local: conn.LocalAddr().String(),
remote: conn.RemoteAddr().String(), remote: conn.RemoteAddr().String(),
}, nil }, nil

View File

@ -1,7 +1,6 @@
package transport package transport
import ( import (
"fmt"
"io" "io"
"net" "net"
"sync" "sync"
@ -305,9 +304,7 @@ func TestHTTPTransportCloseWhenRecv(t *testing.T) {
if err == io.EOF { if err == io.EOF {
return return
} }
t.Errorf("Unexpected recv err: %v", err)
} }
fmt.Println("aa")
} }
}() }()
for i := 1; i < 3; i++ { for i := 1; i < 3; i++ {
@ -320,3 +317,87 @@ func TestHTTPTransportCloseWhenRecv(t *testing.T) {
c.Close() c.Close()
wg.Wait() wg.Wait()
} }
func TestHTTPTransportMultipleSendWhenRecv(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()
readyToSend := make(chan struct{})
m := Message{
Header: map[string]string{
"Content-Type": "application/json",
},
Body: []byte(`{"message": "Hello World"}`),
}
wgSend := sync.WaitGroup{}
fn := func(sock Socket) {
defer sock.Close()
for {
var mr Message
if err := sock.Recv(&mr); err != nil {
return
}
wgSend.Add(1)
go func() {
defer wgSend.Done()
<-readyToSend
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(), WithStream())
if err != nil {
t.Errorf("Unexpected dial err: %v", err)
}
defer c.Close()
var wg sync.WaitGroup
wg.Add(1)
readyForRecv := make(chan struct{})
go func() {
defer wg.Done()
close(readyForRecv)
for {
var rm Message
if err := c.Recv(&rm); err != nil {
if err == io.EOF {
return
}
}
}
}()
<-readyForRecv
for i := 0; i < 3; i++ {
if err := c.Send(&m); err != nil {
t.Errorf("Unexpected send err: %v", err)
}
}
close(readyToSend)
wgSend.Wait()
close(done)
c.Close()
wg.Wait()
}