134 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package mucp
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"math/rand"
 | |
| 	"sync"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/golang/protobuf/proto"
 | |
| 	"github.com/micro/go-micro/v3/codec/json"
 | |
| 	protoCodec "github.com/micro/go-micro/v3/codec/proto"
 | |
| )
 | |
| 
 | |
| // protoStruct implements proto.Message
 | |
| type protoStruct struct {
 | |
| 	Payload string `protobuf:"bytes,1,opt,name=service,proto3" json:"service,omitempty"`
 | |
| }
 | |
| 
 | |
| func (m *protoStruct) Reset()         { *m = protoStruct{} }
 | |
| func (m *protoStruct) String() string { return proto.CompactTextString(m) }
 | |
| func (*protoStruct) ProtoMessage()    {}
 | |
| 
 | |
| // safeBuffer throws away everything and wont Read data back
 | |
| type safeBuffer struct {
 | |
| 	sync.RWMutex
 | |
| 	buf []byte
 | |
| 	off int
 | |
| }
 | |
| 
 | |
| func (b *safeBuffer) Write(p []byte) (n int, err error) {
 | |
| 	if len(p) == 0 {
 | |
| 		return 0, nil
 | |
| 	}
 | |
| 	// Cannot retain p, so we must copy it:
 | |
| 	p2 := make([]byte, len(p))
 | |
| 	copy(p2, p)
 | |
| 	b.Lock()
 | |
| 	b.buf = append(b.buf, p2...)
 | |
| 	b.Unlock()
 | |
| 	return len(p2), nil
 | |
| }
 | |
| 
 | |
| func (b *safeBuffer) Read(p []byte) (n int, err error) {
 | |
| 	if len(p) == 0 {
 | |
| 		return 0, nil
 | |
| 	}
 | |
| 	b.RLock()
 | |
| 	n = copy(p, b.buf[b.off:])
 | |
| 	b.RUnlock()
 | |
| 	if n == 0 {
 | |
| 		return 0, io.EOF
 | |
| 	}
 | |
| 	b.off += n
 | |
| 	return n, nil
 | |
| }
 | |
| 
 | |
| func (b *safeBuffer) Close() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func TestRPCStream_Sequence(t *testing.T) {
 | |
| 	buffer := new(bytes.Buffer)
 | |
| 	rwc := readWriteCloser{
 | |
| 		rbuf: buffer,
 | |
| 		wbuf: buffer,
 | |
| 	}
 | |
| 	codec := json.NewCodec(&rwc)
 | |
| 	streamServer := rpcStream{
 | |
| 		codec: codec,
 | |
| 		request: &rpcRequest{
 | |
| 			codec: codec,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// Check if sequence is correct
 | |
| 	for i := 0; i < 1000; i++ {
 | |
| 		if err := streamServer.Send(fmt.Sprintf(`{"test":"value %d"}`, i)); err != nil {
 | |
| 			t.Errorf("Unexpected Send error: %s", err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for i := 0; i < 1000; i++ {
 | |
| 		var msg string
 | |
| 		if err := streamServer.Recv(&msg); err != nil {
 | |
| 			t.Errorf("Unexpected Recv error: %s", err)
 | |
| 		}
 | |
| 		if msg != fmt.Sprintf(`{"test":"value %d"}`, i) {
 | |
| 			t.Errorf("Unexpected msg: %s", msg)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRPCStream_Concurrency(t *testing.T) {
 | |
| 	buffer := new(safeBuffer)
 | |
| 	codec := protoCodec.NewCodec(buffer)
 | |
| 	streamServer := rpcStream{
 | |
| 		codec: codec,
 | |
| 		request: &rpcRequest{
 | |
| 			codec: codec,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	var wg sync.WaitGroup
 | |
| 	// Check if race conditions happen
 | |
| 	for i := 0; i < 10; i++ {
 | |
| 		wg.Add(2)
 | |
| 
 | |
| 		go func() {
 | |
| 			for i := 0; i < 50; i++ {
 | |
| 				msg := protoStruct{Payload: "test"}
 | |
| 				<-time.After(time.Duration(rand.Intn(50)) * time.Millisecond)
 | |
| 				if err := streamServer.Send(msg); err != nil {
 | |
| 					t.Errorf("Unexpected Send error: %s", err)
 | |
| 				}
 | |
| 			}
 | |
| 			wg.Done()
 | |
| 		}()
 | |
| 
 | |
| 		go func() {
 | |
| 			for i := 0; i < 50; i++ {
 | |
| 				<-time.After(time.Duration(rand.Intn(50)) * time.Millisecond)
 | |
| 				if err := streamServer.Recv(&protoStruct{}); err != nil {
 | |
| 					t.Errorf("Unexpected Recv error: %s", err)
 | |
| 				}
 | |
| 			}
 | |
| 			wg.Done()
 | |
| 		}()
 | |
| 	}
 | |
| 	wg.Wait()
 | |
| }
 |