2017-01-07 16:52:23 +02:00
|
|
|
package micro
|
|
|
|
|
|
|
|
import (
|
2018-03-03 13:53:52 +02:00
|
|
|
"context"
|
2019-01-16 20:54:43 +02:00
|
|
|
"errors"
|
2017-01-07 16:52:23 +02:00
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
2021-10-12 13:55:53 +02:00
|
|
|
"go-micro.dev/v4/client"
|
|
|
|
proto "go-micro.dev/v4/debug/proto"
|
|
|
|
"go-micro.dev/v4/registry"
|
|
|
|
"go-micro.dev/v4/util/test"
|
2017-01-07 16:52:23 +02:00
|
|
|
)
|
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
func testShutdown(wg *sync.WaitGroup, cancel func()) {
|
|
|
|
// add 1
|
2017-01-07 16:52:23 +02:00
|
|
|
wg.Add(1)
|
2019-01-16 20:54:43 +02:00
|
|
|
// shutdown the service
|
|
|
|
cancel()
|
|
|
|
// wait for stop
|
|
|
|
wg.Wait()
|
|
|
|
}
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
func testService(ctx context.Context, wg *sync.WaitGroup, name string) Service {
|
|
|
|
// add self
|
|
|
|
wg.Add(1)
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2020-12-29 17:49:26 +02:00
|
|
|
r := registry.NewMemoryRegistry(registry.Services(test.Data))
|
2019-01-14 17:27:25 +02:00
|
|
|
|
2017-01-07 16:52:23 +02:00
|
|
|
// create service
|
2019-01-16 20:54:43 +02:00
|
|
|
return NewService(
|
|
|
|
Name(name),
|
2017-01-07 16:52:23 +02:00
|
|
|
Context(ctx),
|
2019-01-14 17:27:25 +02:00
|
|
|
Registry(r),
|
2017-01-07 16:52:23 +02:00
|
|
|
AfterStart(func() error {
|
|
|
|
wg.Done()
|
|
|
|
return nil
|
|
|
|
}),
|
2019-01-16 20:54:43 +02:00
|
|
|
AfterStop(func() error {
|
|
|
|
wg.Done()
|
|
|
|
return nil
|
|
|
|
}),
|
2017-01-07 16:52:23 +02:00
|
|
|
)
|
2019-01-16 20:54:43 +02:00
|
|
|
}
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
func testRequest(ctx context.Context, c client.Client, name string) error {
|
|
|
|
// test call debug
|
|
|
|
req := c.NewRequest(
|
|
|
|
name,
|
|
|
|
"Debug.Health",
|
|
|
|
new(proto.HealthRequest),
|
|
|
|
)
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
rsp := new(proto.HealthResponse)
|
2018-03-13 13:12:42 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
err := c.Call(context.TODO(), req, rsp)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-03-13 13:12:42 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
if rsp.Status != "ok" {
|
|
|
|
return errors.New("service response: " + rsp.Status)
|
|
|
|
}
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestService tests running and calling a service
|
|
|
|
func TestService(t *testing.T) {
|
|
|
|
// waitgroup for server start
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
// cancellation context
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
// start test server
|
|
|
|
service := testService(ctx, &wg, "test.service")
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
// wait for service to start
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
// make a test call
|
|
|
|
if err := testRequest(ctx, service.Client(), "test.service"); err != nil {
|
|
|
|
t.Fatal(err)
|
2018-05-25 15:39:50 +02:00
|
|
|
}
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2018-05-25 15:39:50 +02:00
|
|
|
// shutdown the service
|
2019-01-16 20:54:43 +02:00
|
|
|
testShutdown(&wg, cancel)
|
2018-05-25 15:39:50 +02:00
|
|
|
}()
|
2017-01-07 16:52:23 +02:00
|
|
|
|
2019-01-16 20:54:43 +02:00
|
|
|
// start service
|
2018-05-25 15:39:50 +02:00
|
|
|
if err := service.Run(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2017-01-07 16:52:23 +02:00
|
|
|
}
|
2019-01-16 20:54:43 +02:00
|
|
|
|
|
|
|
func benchmarkService(b *testing.B, n int, name string) {
|
|
|
|
// stop the timer
|
|
|
|
b.StopTimer()
|
|
|
|
|
|
|
|
// waitgroup for server start
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
// cancellation context
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
|
|
|
|
// create test server
|
|
|
|
service := testService(ctx, &wg, name)
|
|
|
|
|
|
|
|
// start the server
|
|
|
|
go func() {
|
|
|
|
if err := service.Run(); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// wait for service to start
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
// make a test call to warm the cache
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
if err := testRequest(ctx, service.Client(), name); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// start the timer
|
|
|
|
b.StartTimer()
|
|
|
|
|
|
|
|
// number of iterations
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
// for concurrency
|
|
|
|
for j := 0; j < n; j++ {
|
|
|
|
wg.Add(1)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
err := testRequest(ctx, service.Client(), name)
|
|
|
|
wg.Done()
|
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait for test completion
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
// stop the timer
|
|
|
|
b.StopTimer()
|
|
|
|
|
|
|
|
// shutdown service
|
|
|
|
testShutdown(&wg, cancel)
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkService1(b *testing.B) {
|
|
|
|
benchmarkService(b, 1, "test.service.1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkService8(b *testing.B) {
|
|
|
|
benchmarkService(b, 8, "test.service.8")
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkService16(b *testing.B) {
|
|
|
|
benchmarkService(b, 16, "test.service.16")
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkService32(b *testing.B) {
|
|
|
|
benchmarkService(b, 32, "test.service.32")
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkService64(b *testing.B) {
|
|
|
|
benchmarkService(b, 64, "test.service.64")
|
|
|
|
}
|