Add metrics server to stress client and add metrics client
This commit is contained in:
@ -4,15 +4,19 @@ import (
|
|||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
|
"net"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/net/context"
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
"google.golang.org/grpc/grpclog"
|
"google.golang.org/grpc/grpclog"
|
||||||
"google.golang.org/grpc/interop"
|
"google.golang.org/grpc/interop"
|
||||||
testpb "google.golang.org/grpc/interop/grpc_testing"
|
testpb "google.golang.org/grpc/interop/grpc_testing"
|
||||||
|
metricspb "google.golang.org/grpc/stress/grpc_testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -105,23 +109,23 @@ func parseTestCases(testCaseString string) ([]testCaseWithWeight, error) {
|
|||||||
return testCases, nil
|
return testCases, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// WeightedRandomTestSelector defines a weighted random selector for test case types.
|
// weightedRandomTestSelector defines a weighted random selector for test case types.
|
||||||
type WeightedRandomTestSelector struct {
|
type weightedRandomTestSelector struct {
|
||||||
tests []testCaseWithWeight
|
tests []testCaseWithWeight
|
||||||
totalWeight int
|
totalWeight int
|
||||||
}
|
}
|
||||||
|
|
||||||
// newWeightedRandomTestSelector constructs a WeightedRandomTestSelector with the given list of testCaseWithWeight.
|
// newWeightedRandomTestSelector constructs a weightedRandomTestSelector with the given list of testCaseWithWeight.
|
||||||
func newWeightedRandomTestSelector(tests []testCaseWithWeight) *WeightedRandomTestSelector {
|
func newWeightedRandomTestSelector(tests []testCaseWithWeight) *weightedRandomTestSelector {
|
||||||
var totalWeight int
|
var totalWeight int
|
||||||
for _, t := range tests {
|
for _, t := range tests {
|
||||||
totalWeight += t.weight
|
totalWeight += t.weight
|
||||||
}
|
}
|
||||||
rand.Seed(time.Now().UnixNano())
|
rand.Seed(time.Now().UnixNano())
|
||||||
return &WeightedRandomTestSelector{tests, totalWeight}
|
return &weightedRandomTestSelector{tests, totalWeight}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (selector WeightedRandomTestSelector) getNextTest() (testCaseType, error) {
|
func (selector weightedRandomTestSelector) getNextTest() (testCaseType, error) {
|
||||||
random := rand.Intn(selector.totalWeight)
|
random := rand.Intn(selector.totalWeight)
|
||||||
var weightSofar int
|
var weightSofar int
|
||||||
for _, test := range selector.tests {
|
for _, test := range selector.tests {
|
||||||
@ -130,26 +134,110 @@ func (selector WeightedRandomTestSelector) getNextTest() (testCaseType, error) {
|
|||||||
return test.testCase, nil
|
return test.testCase, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return unknownTest, fmt.Errorf("no test case selected by WeightedRandomTestSelector")
|
return unknownTest, fmt.Errorf("no test case selected by weightedRandomTestSelector")
|
||||||
}
|
}
|
||||||
|
|
||||||
// StressClient defines client for stress test.
|
// gauge defines type for gauge.
|
||||||
type StressClient struct {
|
type gauge struct {
|
||||||
|
mutex sync.RWMutex
|
||||||
|
val int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set updates the gauge value
|
||||||
|
func (g *gauge) set(v int64) {
|
||||||
|
g.mutex.Lock()
|
||||||
|
defer g.mutex.Unlock()
|
||||||
|
g.val = v
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *gauge) get() int64 {
|
||||||
|
g.mutex.RLock()
|
||||||
|
defer g.mutex.RUnlock()
|
||||||
|
return g.val
|
||||||
|
}
|
||||||
|
|
||||||
|
// Server implements metrics server functions.
|
||||||
|
type server struct {
|
||||||
|
mutex sync.RWMutex
|
||||||
|
gauges map[string]*gauge
|
||||||
|
}
|
||||||
|
|
||||||
|
// newMetricsServer returns a new metrics server.
|
||||||
|
func newMetricsServer() *server {
|
||||||
|
return &server{gauges: make(map[string]*gauge)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAllGauges returns all gauges.
|
||||||
|
func (s *server) GetAllGauges(in *metricspb.EmptyMessage, stream metricspb.MetricsService_GetAllGaugesServer) error {
|
||||||
|
s.mutex.RLock()
|
||||||
|
defer s.mutex.RUnlock()
|
||||||
|
|
||||||
|
for name, gauge := range s.gauges {
|
||||||
|
err := stream.Send(&metricspb.GaugeResponse{Name: name, Value: &metricspb.GaugeResponse_LongValue{gauge.get()}})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetGauge returns the gauge for the given name.
|
||||||
|
func (s *server) GetGauge(ctx context.Context, in *metricspb.GaugeRequest) (*metricspb.GaugeResponse, error) {
|
||||||
|
s.mutex.RLock()
|
||||||
|
defer s.mutex.RUnlock()
|
||||||
|
|
||||||
|
if g, ok := s.gauges[in.Name]; ok {
|
||||||
|
return &metricspb.GaugeResponse{Name: in.Name, Value: &metricspb.GaugeResponse_LongValue{g.get()}}, nil
|
||||||
|
}
|
||||||
|
return nil, grpc.Errorf(codes.InvalidArgument, "gauge with name %s not found", in.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateGauge creates a guage using the given name in metrics server
|
||||||
|
func (s *server) createGauge(name string) (*gauge, error) {
|
||||||
|
s.mutex.Lock()
|
||||||
|
defer s.mutex.Unlock()
|
||||||
|
|
||||||
|
grpclog.Printf("create gauge: %s", name)
|
||||||
|
if _, ok := s.gauges[name]; ok {
|
||||||
|
// gauge already exists.
|
||||||
|
return nil, fmt.Errorf("gauge %s already exists", name)
|
||||||
|
}
|
||||||
|
var g gauge
|
||||||
|
s.gauges[name] = &g
|
||||||
|
return &g, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func startServer(server *server, port int) {
|
||||||
|
lis, err := net.Listen("tcp", ":"+strconv.Itoa(port))
|
||||||
|
if err != nil {
|
||||||
|
grpclog.Fatalf("failed to listen: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
s := grpc.NewServer()
|
||||||
|
metricspb.RegisterMetricsServiceServer(s, server)
|
||||||
|
s.Serve(lis)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// stressClient defines client for stress test.
|
||||||
|
type stressClient struct {
|
||||||
testID int
|
testID int
|
||||||
address string
|
address string
|
||||||
testDurationSecs int
|
testDurationSecs int
|
||||||
selector *WeightedRandomTestSelector
|
selector *weightedRandomTestSelector
|
||||||
interopClient testpb.TestServiceClient
|
interopClient testpb.TestServiceClient
|
||||||
}
|
}
|
||||||
|
|
||||||
// newStressClient construct a new StressClient.
|
// newStressClient construct a new stressClient.
|
||||||
func newStressClient(id int, addr string, conn *grpc.ClientConn, selector *WeightedRandomTestSelector, testDurSecs int) *StressClient {
|
func newStressClient(id int, addr string, conn *grpc.ClientConn, selector *weightedRandomTestSelector, testDurSecs int) *stressClient {
|
||||||
client := testpb.NewTestServiceClient(conn)
|
client := testpb.NewTestServiceClient(conn)
|
||||||
return &StressClient{testID: id, address: addr, selector: selector, testDurationSecs: testDurSecs, interopClient: client}
|
return &stressClient{testID: id, address: addr, selector: selector, testDurationSecs: testDurSecs, interopClient: client}
|
||||||
}
|
}
|
||||||
|
|
||||||
// MainLoop uses WeightedRandomTestSelector to select test case and runs the tests.
|
// mainLoop uses weightedRandomTestSelector to select test case and runs the tests.
|
||||||
func (c *StressClient) MainLoop(buf string) {
|
func (c *stressClient) mainLoop(gauge *gauge) {
|
||||||
|
var numCalls int64
|
||||||
|
timeStarted := time.Now()
|
||||||
for testEndTime := time.Now().Add(time.Duration(c.testDurationSecs) * time.Second); c.testDurationSecs < 0 || time.Now().Before(testEndTime); {
|
for testEndTime := time.Now().Add(time.Duration(c.testDurationSecs) * time.Second); c.testDurationSecs < 0 || time.Now().Before(testEndTime); {
|
||||||
test, err := c.selector.getNextTest()
|
test, err := c.selector.getNextTest()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -170,6 +258,8 @@ func (c *StressClient) MainLoop(buf string) {
|
|||||||
default:
|
default:
|
||||||
grpclog.Fatal("Unsupported test case: %d", test)
|
grpclog.Fatal("Unsupported test case: %d", test)
|
||||||
}
|
}
|
||||||
|
numCalls++
|
||||||
|
gauge.set(int64(float64(numCalls) / time.Since(timeStarted).Seconds()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -198,6 +288,8 @@ func main() {
|
|||||||
}
|
}
|
||||||
logParameterInfo(serverAddresses, testCases)
|
logParameterInfo(serverAddresses, testCases)
|
||||||
testSelector := newWeightedRandomTestSelector(testCases)
|
testSelector := newWeightedRandomTestSelector(testCases)
|
||||||
|
metricsServer := newMetricsServer()
|
||||||
|
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
wg.Add(len(serverAddresses) * *numChannelsPerServerPtr * *numStubsPerChannelPtr)
|
wg.Add(len(serverAddresses) * *numChannelsPerServerPtr * *numStubsPerChannelPtr)
|
||||||
var clientIndex int
|
var clientIndex int
|
||||||
@ -214,12 +306,18 @@ func main() {
|
|||||||
buf := fmt.Sprintf("/stress_test/server_%d/channel_%d/stub_%d/qps", serverIndex+1, connIndex+1, stubIndex+1)
|
buf := fmt.Sprintf("/stress_test/server_%d/channel_%d/stub_%d/qps", serverIndex+1, connIndex+1, stubIndex+1)
|
||||||
go func() {
|
go func() {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
client.MainLoop(buf)
|
if g, err := metricsServer.createGauge(buf); err != nil {
|
||||||
|
grpclog.Fatalf("%v", err)
|
||||||
|
} else {
|
||||||
|
client.mainLoop(g)
|
||||||
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
go startServer(metricsServer, *metricsPortPtr)
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
grpclog.Printf(" ===== ALL DONE ===== ")
|
grpclog.Printf(" ===== ALL DONE ===== ")
|
||||||
|
|
||||||
}
|
}
|
||||||
|
65
stress/metrics_client/main.go
Normal file
65
stress/metrics_client/main.go
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"io"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/net/context"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
metricspb "google.golang.org/grpc/stress/grpc_testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
metricsServerAddressPtr = flag.String("metrics_server_address", "", "The metrics server addresses in the fomrat <hostname>:<port>")
|
||||||
|
totalOnlyPtr = flag.Bool("total_only", false, "If true, this prints only the total value of all gauges")
|
||||||
|
)
|
||||||
|
|
||||||
|
const timeoutSeconds = 10
|
||||||
|
|
||||||
|
func printMetrics(client metricspb.MetricsServiceClient, totalOnly bool) {
|
||||||
|
ctx, _ := context.WithTimeout(context.Background(), timeoutSeconds*time.Second)
|
||||||
|
stream, err := client.GetAllGauges(ctx, &metricspb.EmptyMessage{})
|
||||||
|
if err != nil {
|
||||||
|
grpclog.Fatalf("failed to call GetAllGuages: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var overallQPS int64
|
||||||
|
var rpcStatus error
|
||||||
|
for {
|
||||||
|
gaugeResponse, err := stream.Recv()
|
||||||
|
if err != nil {
|
||||||
|
rpcStatus = err
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if _, ok := gaugeResponse.GetValue().(*metricspb.GaugeResponse_LongValue); ok {
|
||||||
|
if !totalOnly {
|
||||||
|
grpclog.Printf("%s: %d", gaugeResponse.Name, gaugeResponse.GetLongValue())
|
||||||
|
}
|
||||||
|
overallQPS += gaugeResponse.GetLongValue()
|
||||||
|
} else {
|
||||||
|
grpclog.Printf("gauge %s is not a long value", gaugeResponse.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
grpclog.Printf("overall qps: %d", overallQPS)
|
||||||
|
if rpcStatus != io.EOF {
|
||||||
|
grpclog.Fatalf("failed to finish server streaming: %v", rpcStatus)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
flag.Parse()
|
||||||
|
if len(*metricsServerAddressPtr) == 0 {
|
||||||
|
grpclog.Fatalf("Cannot connect to the Metrics server. Please pass the address of the metrics server to connect to via the 'metrics_server_address' flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
conn, err := grpc.Dial(*metricsServerAddressPtr, grpc.WithInsecure())
|
||||||
|
if err != nil {
|
||||||
|
grpclog.Fatalf("cannot connect to metrics server: %v", err)
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
|
||||||
|
c := metricspb.NewMetricsServiceClient(conn)
|
||||||
|
printMetrics(c, *totalOnlyPtr)
|
||||||
|
}
|
Reference in New Issue
Block a user