mirror of https://github.com/grpc/grpc-go.git
154 lines
4.4 KiB
Go
154 lines
4.4 KiB
Go
/*
|
|
*
|
|
* Copyright 2018 gRPC authors.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*/
|
|
|
|
package health
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
"google.golang.org/protobuf/testing/protocmp"
|
|
|
|
healthpb "google.golang.org/grpc/health/grpc_health_v1"
|
|
"google.golang.org/grpc/internal/grpctest"
|
|
)
|
|
|
|
type s struct {
|
|
grpctest.Tester
|
|
}
|
|
|
|
func Test(t *testing.T) {
|
|
grpctest.RunSubTests(t, s{})
|
|
}
|
|
|
|
func (s) TestShutdown(t *testing.T) {
|
|
const testService = "tteesstt"
|
|
s := NewServer()
|
|
s.SetServingStatus(testService, healthpb.HealthCheckResponse_SERVING)
|
|
|
|
status := s.statusMap[testService]
|
|
if status != healthpb.HealthCheckResponse_SERVING {
|
|
t.Fatalf("status for %s is %v, want %v", testService, status, healthpb.HealthCheckResponse_SERVING)
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(2)
|
|
// Run SetServingStatus and Shutdown in parallel.
|
|
go func() {
|
|
for i := 0; i < 1000; i++ {
|
|
s.SetServingStatus(testService, healthpb.HealthCheckResponse_SERVING)
|
|
time.Sleep(time.Microsecond)
|
|
}
|
|
wg.Done()
|
|
}()
|
|
go func() {
|
|
time.Sleep(300 * time.Microsecond)
|
|
s.Shutdown()
|
|
wg.Done()
|
|
}()
|
|
wg.Wait()
|
|
|
|
s.mu.Lock()
|
|
status = s.statusMap[testService]
|
|
s.mu.Unlock()
|
|
if status != healthpb.HealthCheckResponse_NOT_SERVING {
|
|
t.Fatalf("status for %s is %v, want %v", testService, status, healthpb.HealthCheckResponse_NOT_SERVING)
|
|
}
|
|
|
|
s.Resume()
|
|
status = s.statusMap[testService]
|
|
if status != healthpb.HealthCheckResponse_SERVING {
|
|
t.Fatalf("status for %s is %v, want %v", testService, status, healthpb.HealthCheckResponse_SERVING)
|
|
}
|
|
|
|
s.SetServingStatus(testService, healthpb.HealthCheckResponse_NOT_SERVING)
|
|
status = s.statusMap[testService]
|
|
if status != healthpb.HealthCheckResponse_NOT_SERVING {
|
|
t.Fatalf("status for %s is %v, want %v", testService, status, healthpb.HealthCheckResponse_NOT_SERVING)
|
|
}
|
|
}
|
|
|
|
// TestList verifies that List() returns the health status of all the services if no. of services are within
|
|
// maxAllowedLimits.
|
|
func (s) TestList(t *testing.T) {
|
|
s := NewServer()
|
|
|
|
// Fill out status map with information
|
|
const length = 3
|
|
for i := 0; i < length; i++ {
|
|
s.SetServingStatus(fmt.Sprintf("%d", i),
|
|
healthpb.HealthCheckResponse_ServingStatus(i))
|
|
}
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
var in healthpb.HealthListRequest
|
|
got, err := s.List(ctx, &in)
|
|
|
|
if err != nil {
|
|
t.Fatalf("s.List(ctx, &in) returned err %v, want nil", err)
|
|
}
|
|
if len(got.GetStatuses()) != length+1 {
|
|
t.Fatalf("len(out.GetStatuses()) = %d, want %d",
|
|
len(got.GetStatuses()), length+1)
|
|
}
|
|
want := &healthpb.HealthListResponse{
|
|
Statuses: map[string]*healthpb.HealthCheckResponse{
|
|
"": {Status: healthpb.HealthCheckResponse_SERVING},
|
|
"0": {Status: healthpb.HealthCheckResponse_UNKNOWN},
|
|
"1": {Status: healthpb.HealthCheckResponse_SERVING},
|
|
"2": {Status: healthpb.HealthCheckResponse_NOT_SERVING},
|
|
},
|
|
}
|
|
if diff := cmp.Diff(got, want, protocmp.Transform()); diff != "" {
|
|
t.Fatalf("Health response did not match expectation. Diff (-got, +want): %s", diff)
|
|
}
|
|
}
|
|
|
|
// TestListResourceExhausted verifies that List()
|
|
// returns a ResourceExhausted error if no. of services are more than
|
|
// maxAllowedServices.
|
|
func (s) TestListResourceExhausted(t *testing.T) {
|
|
s := NewServer()
|
|
|
|
// Fill out status map with service information,
|
|
// 101 (100 + 1 existing) elements will trigger an error.
|
|
for i := 1; i <= maxAllowedServices; i++ {
|
|
s.SetServingStatus(fmt.Sprintf("%d", i),
|
|
healthpb.HealthCheckResponse_SERVING)
|
|
}
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
var in healthpb.HealthListRequest
|
|
_, err := s.List(ctx, &in)
|
|
|
|
want := status.Errorf(codes.ResourceExhausted,
|
|
"server health list exceeds maximum capacity: %d", maxAllowedServices)
|
|
if !errors.Is(err, want) {
|
|
t.Fatalf("s.List(ctx, &in) returned %v, want %v", err, want)
|
|
}
|
|
}
|