package http_test import ( "context" "sync" "testing" "time" "github.com/google/uuid" http "github.com/unistack-org/micro-broker-http/v3" jsoncodec "github.com/unistack-org/micro-codec-json/v3" "github.com/unistack-org/micro/v3/broker" "github.com/unistack-org/micro/v3/register" ) var ( // mock data testData = map[string][]*register.Service{ "foo": { { Name: "foo", Version: "1.0.0", Nodes: []*register.Node{ { ID: "foo-1.0.0-123", Address: "localhost:9999", }, { ID: "foo-1.0.0-321", Address: "localhost:9999", }, }, }, { Name: "foo", Version: "1.0.1", Nodes: []*register.Node{ { ID: "foo-1.0.1-321", Address: "localhost:6666", }, }, }, { Name: "foo", Version: "1.0.3", Nodes: []*register.Node{ { ID: "foo-1.0.3-345", Address: "localhost:8888", }, }, }, }, } ) func newTestRegister() register.Register { return register.NewRegister() } func sub(be *testing.B, c int) { be.StopTimer() m := newTestRegister() b := http.NewBroker(broker.Codec(jsoncodec.NewCodec()), broker.Register(m)) topic := uuid.New().String() if err := b.Init(); err != nil { be.Fatalf("Unexpected init error: %v", err) } if err := b.Connect(context.TODO()); err != nil { be.Fatalf("Unexpected connect error: %v", err) } msg := &broker.Message{ Header: map[string]string{ "Content-Type": "application/json", }, Body: []byte(`{"message": "Hello World"}`), } var subs []broker.Subscriber done := make(chan bool, c) for i := 0; i < c; i++ { sub, err := b.Subscribe(context.TODO(), topic, func(p broker.Event) error { done <- true m := p.Message() if string(m.Body) != string(msg.Body) { be.Fatalf("Unexpected msg %s, expected %s", string(m.Body), string(msg.Body)) } return nil }, broker.SubscribeGroup("shared")) if err != nil { be.Fatalf("Unexpected subscribe error: %v", err) } subs = append(subs, sub) } for i := 0; i < be.N; i++ { be.StartTimer() if err := b.Publish(context.TODO(), topic, msg); err != nil { be.Fatalf("Unexpected publish error: %v", err) } <-done be.StopTimer() } for _, sub := range subs { sub.Unsubscribe(context.TODO()) } if err := b.Disconnect(context.TODO()); err != nil { be.Fatalf("Unexpected disconnect error: %v", err) } } func pub(be *testing.B, c int) { be.StopTimer() m := newTestRegister() b := http.NewBroker(broker.Codec(jsoncodec.NewCodec()), broker.Register(m)) topic := uuid.New().String() if err := b.Init(); err != nil { be.Fatalf("Unexpected init error: %v", err) } if err := b.Connect(context.TODO()); err != nil { be.Fatalf("Unexpected connect error: %v", err) } msg := &broker.Message{ Header: map[string]string{ "Content-Type": "application/json", }, Body: []byte(`{"message": "Hello World"}`), } done := make(chan bool, c*4) sub, err := b.Subscribe(context.TODO(), topic, func(p broker.Event) error { done <- true m := p.Message() if string(m.Body) != string(msg.Body) { be.Fatalf("Unexpected msg %s, expected %s", string(m.Body), string(msg.Body)) } return nil }, broker.SubscribeGroup("shared")) if err != nil { be.Fatalf("Unexpected subscribe error: %v", err) } var wg sync.WaitGroup ch := make(chan int, c*4) be.StartTimer() for i := 0; i < c; i++ { go func() { for range ch { if err := b.Publish(context.TODO(), topic, msg); err != nil { be.Fatalf("Unexpected publish error: %v", err) } select { case <-done: case <-time.After(time.Second): } wg.Done() } }() } for i := 0; i < be.N; i++ { wg.Add(1) ch <- i } wg.Wait() be.StopTimer() sub.Unsubscribe(context.TODO()) close(ch) close(done) if err := b.Disconnect(context.TODO()); err != nil { be.Fatalf("Unexpected disconnect error: %v", err) } } func TestBroker(t *testing.T) { m := newTestRegister() b := http.NewBroker(broker.Codec(jsoncodec.NewCodec()), broker.Register(m)) if err := b.Init(); err != nil { t.Fatalf("Unexpected init error: %v", err) } if err := b.Connect(context.TODO()); err != nil { t.Fatalf("Unexpected connect error: %v", err) } msg := &broker.Message{ Header: map[string]string{ "Content-Type": "application/json", }, Body: []byte(`{"message":"Hello World"}`), } done := make(chan struct{}) sub, err := b.Subscribe(context.TODO(), "test", func(p broker.Event) error { m := p.Message() if string(m.Body) != string(msg.Body) { t.Errorf("Unexpected msg %s, expected %s", string(m.Body), string(msg.Body)) } close(done) return nil }) if err != nil { t.Fatalf("Unexpected subscribe error: %v", err) } if err := b.Publish(context.TODO(), "test", msg); err != nil { t.Fatalf("Unexpected publish error: %v", err) } <-done if err != nil { t.Error(err) } sub.Unsubscribe(context.TODO()) if err := b.Disconnect(context.TODO()); err != nil { t.Fatalf("Unexpected disconnect error: %v", err) } } func TestConcurrentSubBroker(t *testing.T) { m := newTestRegister() b := http.NewBroker(broker.Codec(jsoncodec.NewCodec()), broker.Register(m)) if err := b.Init(); err != nil { t.Fatalf("Unexpected init error: %v", err) } if err := b.Connect(context.TODO()); err != nil { t.Fatalf("Unexpected connect error: %v", err) } msg := &broker.Message{ Header: map[string]string{ "Content-Type": "application/json", }, Body: []byte(`{"message":"Hello World"}`), } var subs []broker.Subscriber var wg sync.WaitGroup for i := 0; i < 10; i++ { sub, err := b.Subscribe(context.TODO(), "test", func(p broker.Event) error { defer wg.Done() m := p.Message() if string(m.Body) != string(msg.Body) { t.Errorf("Unexpected msg %s, expected %s", string(m.Body), string(msg.Body)) } return nil }) if err != nil { t.Fatalf("Unexpected subscribe error: %v", err) } wg.Add(1) subs = append(subs, sub) } if err := b.Publish(context.TODO(), "test", msg); err != nil { t.Fatalf("Unexpected publish error: %v", err) } wg.Wait() for _, sub := range subs { sub.Unsubscribe(context.TODO()) } if err := b.Disconnect(context.TODO()); err != nil { t.Fatalf("Unexpected disconnect error: %v", err) } } func TestConcurrentPubBroker(t *testing.T) { m := newTestRegister() b := http.NewBroker(broker.Codec(jsoncodec.NewCodec()), broker.Register(m)) if err := b.Init(); err != nil { t.Fatalf("Unexpected init error: %v", err) } if err := b.Connect(context.TODO()); err != nil { t.Fatalf("Unexpected connect error: %v", err) } msg := &broker.Message{ Header: map[string]string{ "Content-Type": "application/json", }, Body: []byte(`{"message":"Hello World"}`), } var wg sync.WaitGroup sub, err := b.Subscribe(context.TODO(), "test", func(p broker.Event) error { defer wg.Done() m := p.Message() if string(m.Body) != string(msg.Body) { t.Fatalf("Unexpected msg %s, expected %s", string(m.Body), string(msg.Body)) } return nil }) if err != nil { t.Fatalf("Unexpected subscribe error: %v", err) } for i := 0; i < 10; i++ { wg.Add(1) if err := b.Publish(context.TODO(), "test", msg); err != nil { t.Fatalf("Unexpected publish error: %v", err) } } wg.Wait() sub.Unsubscribe(context.TODO()) if err := b.Disconnect(context.TODO()); err != nil { t.Fatalf("Unexpected disconnect error: %v", err) } } func BenchmarkSub1(b *testing.B) { sub(b, 1) } func BenchmarkSub8(b *testing.B) { sub(b, 8) } func BenchmarkSub32(b *testing.B) { sub(b, 32) } func BenchmarkSub64(b *testing.B) { sub(b, 64) } func BenchmarkSub128(b *testing.B) { sub(b, 128) } func BenchmarkPub1(b *testing.B) { pub(b, 1) } func BenchmarkPub8(b *testing.B) { pub(b, 8) } func BenchmarkPub32(b *testing.B) { pub(b, 32) } func BenchmarkPub64(b *testing.B) { pub(b, 64) } func BenchmarkPub128(b *testing.B) { pub(b, 128) }