123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- /*
- *
- * Copyright 2017 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 provides a service that exposes server's health and it must be
- // imported to enable support for client-side health checks.
- package health
- import (
- "context"
- "sync"
- "google.golang.org/grpc/codes"
- healthgrpc "google.golang.org/grpc/health/grpc_health_v1"
- healthpb "google.golang.org/grpc/health/grpc_health_v1"
- "google.golang.org/grpc/status"
- )
- // Server implements `service Health`.
- type Server struct {
- healthgrpc.UnimplementedHealthServer
- mu sync.RWMutex
- // If shutdown is true, it's expected all serving status is NOT_SERVING, and
- // will stay in NOT_SERVING.
- shutdown bool
- // statusMap stores the serving status of the services this Server monitors.
- statusMap map[string]healthpb.HealthCheckResponse_ServingStatus
- updates map[string]map[healthgrpc.Health_WatchServer]chan healthpb.HealthCheckResponse_ServingStatus
- }
- // NewServer returns a new Server.
- func NewServer() *Server {
- return &Server{
- statusMap: map[string]healthpb.HealthCheckResponse_ServingStatus{"": healthpb.HealthCheckResponse_SERVING},
- updates: make(map[string]map[healthgrpc.Health_WatchServer]chan healthpb.HealthCheckResponse_ServingStatus),
- }
- }
- // Check implements `service Health`.
- func (s *Server) Check(ctx context.Context, in *healthpb.HealthCheckRequest) (*healthpb.HealthCheckResponse, error) {
- s.mu.RLock()
- defer s.mu.RUnlock()
- if servingStatus, ok := s.statusMap[in.Service]; ok {
- return &healthpb.HealthCheckResponse{
- Status: servingStatus,
- }, nil
- }
- return nil, status.Error(codes.NotFound, "unknown service")
- }
- // Watch implements `service Health`.
- func (s *Server) Watch(in *healthpb.HealthCheckRequest, stream healthgrpc.Health_WatchServer) error {
- service := in.Service
- // update channel is used for getting service status updates.
- update := make(chan healthpb.HealthCheckResponse_ServingStatus, 1)
- s.mu.Lock()
- // Puts the initial status to the channel.
- if servingStatus, ok := s.statusMap[service]; ok {
- update <- servingStatus
- } else {
- update <- healthpb.HealthCheckResponse_SERVICE_UNKNOWN
- }
- // Registers the update channel to the correct place in the updates map.
- if _, ok := s.updates[service]; !ok {
- s.updates[service] = make(map[healthgrpc.Health_WatchServer]chan healthpb.HealthCheckResponse_ServingStatus)
- }
- s.updates[service][stream] = update
- defer func() {
- s.mu.Lock()
- delete(s.updates[service], stream)
- s.mu.Unlock()
- }()
- s.mu.Unlock()
- var lastSentStatus healthpb.HealthCheckResponse_ServingStatus = -1
- for {
- select {
- // Status updated. Sends the up-to-date status to the client.
- case servingStatus := <-update:
- if lastSentStatus == servingStatus {
- continue
- }
- lastSentStatus = servingStatus
- err := stream.Send(&healthpb.HealthCheckResponse{Status: servingStatus})
- if err != nil {
- return status.Error(codes.Canceled, "Stream has ended.")
- }
- // Context done. Removes the update channel from the updates map.
- case <-stream.Context().Done():
- return status.Error(codes.Canceled, "Stream has ended.")
- }
- }
- }
- // SetServingStatus is called when need to reset the serving status of a service
- // or insert a new service entry into the statusMap.
- func (s *Server) SetServingStatus(service string, servingStatus healthpb.HealthCheckResponse_ServingStatus) {
- s.mu.Lock()
- defer s.mu.Unlock()
- if s.shutdown {
- logger.Infof("health: status changing for %s to %v is ignored because health service is shutdown", service, servingStatus)
- return
- }
- s.setServingStatusLocked(service, servingStatus)
- }
- func (s *Server) setServingStatusLocked(service string, servingStatus healthpb.HealthCheckResponse_ServingStatus) {
- s.statusMap[service] = servingStatus
- for _, update := range s.updates[service] {
- // Clears previous updates, that are not sent to the client, from the channel.
- // This can happen if the client is not reading and the server gets flow control limited.
- select {
- case <-update:
- default:
- }
- // Puts the most recent update to the channel.
- update <- servingStatus
- }
- }
- // Shutdown sets all serving status to NOT_SERVING, and configures the server to
- // ignore all future status changes.
- //
- // This changes serving status for all services. To set status for a particular
- // services, call SetServingStatus().
- func (s *Server) Shutdown() {
- s.mu.Lock()
- defer s.mu.Unlock()
- s.shutdown = true
- for service := range s.statusMap {
- s.setServingStatusLocked(service, healthpb.HealthCheckResponse_NOT_SERVING)
- }
- }
- // Resume sets all serving status to SERVING, and configures the server to
- // accept all future status changes.
- //
- // This changes serving status for all services. To set status for a particular
- // services, call SetServingStatus().
- func (s *Server) Resume() {
- s.mu.Lock()
- defer s.mu.Unlock()
- s.shutdown = false
- for service := range s.statusMap {
- s.setServingStatusLocked(service, healthpb.HealthCheckResponse_SERVING)
- }
- }
|