github.com/hxx258456/ccgo@v0.0.5-0.20230213014102-48b35f46f66f/grpc/interop/fake_grpclb/fake_grpclb.go (about)

     1  /*
     2   *
     3   * Copyright 2018 gRPC authors.
     4   *
     5   * Licensed under the Apache License, Version 2.0 (the "License");
     6   * you may not use this file except in compliance with the License.
     7   * You may obtain a copy of the License at
     8   *
     9   *     http://www.apache.org/licenses/LICENSE-2.0
    10   *
    11   * Unless required by applicable law or agreed to in writing, software
    12   * distributed under the License is distributed on an "AS IS" BASIS,
    13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    14   * See the License for the specific language governing permissions and
    15   * limitations under the License.
    16   *
    17   */
    18  
    19  // This file is for testing only. Runs a fake grpclb balancer server.
    20  // The name of the service to load balance for and the addresses
    21  // of that service are provided by command line flags.
    22  package main
    23  
    24  import (
    25  	"flag"
    26  	"net"
    27  	"strconv"
    28  	"strings"
    29  	"time"
    30  
    31  	grpc "github.com/hxx258456/ccgo/grpc"
    32  	lbpb "github.com/hxx258456/ccgo/grpc/balancer/grpclb/grpc_lb_v1"
    33  	"github.com/hxx258456/ccgo/grpc/codes"
    34  	"github.com/hxx258456/ccgo/grpc/credentials"
    35  	"github.com/hxx258456/ccgo/grpc/credentials/alts"
    36  	"github.com/hxx258456/ccgo/grpc/grpclog"
    37  	"github.com/hxx258456/ccgo/grpc/status"
    38  	"github.com/hxx258456/ccgo/grpc/testdata"
    39  )
    40  
    41  var (
    42  	port         = flag.Int("port", 10000, "Port to listen on.")
    43  	backendAddrs = flag.String("backend_addrs", "", "Comma separated list of backend IP/port addresses.")
    44  	useALTS      = flag.Bool("use_alts", false, "Listen on ALTS credentials.")
    45  	useTLS       = flag.Bool("use_tls", false, "Listen on TLS credentials, using a test certificate.")
    46  	shortStream  = flag.Bool("short_stream", false, "End the balancer stream immediately after sending the first server list.")
    47  	serviceName  = flag.String("service_name", "UNSET", "Name of the service being load balanced for.")
    48  
    49  	logger = grpclog.Component("interop")
    50  )
    51  
    52  type loadBalancerServer struct {
    53  	lbpb.UnimplementedLoadBalancerServer
    54  	serverListResponse *lbpb.LoadBalanceResponse
    55  }
    56  
    57  func (l *loadBalancerServer) BalanceLoad(stream lbpb.LoadBalancer_BalanceLoadServer) error {
    58  	logger.Info("Begin handling new BalancerLoad request.")
    59  	var lbReq *lbpb.LoadBalanceRequest
    60  	var err error
    61  	if lbReq, err = stream.Recv(); err != nil {
    62  		logger.Errorf("Error receiving LoadBalanceRequest: %v", err)
    63  		return err
    64  	}
    65  	logger.Info("LoadBalancerRequest received.")
    66  	initialReq := lbReq.GetInitialRequest()
    67  	if initialReq == nil {
    68  		logger.Info("Expected first request to be an InitialRequest. Got: %v", lbReq)
    69  		return status.Error(codes.Unknown, "First request not an InitialRequest")
    70  	}
    71  	// gRPC clients targeting foo.bar.com:443 can sometimes include the ":443" suffix in
    72  	// their requested names; handle this case. TODO: make 443 configurable?
    73  	var cleanedName string
    74  	var requestedNamePortNumber string
    75  	if cleanedName, requestedNamePortNumber, err = net.SplitHostPort(initialReq.Name); err != nil {
    76  		cleanedName = initialReq.Name
    77  	} else {
    78  		if requestedNamePortNumber != "443" {
    79  			logger.Info("Bad requested service name port number: %v.", requestedNamePortNumber)
    80  			return status.Error(codes.Unknown, "Bad requested service name port number")
    81  		}
    82  	}
    83  	if cleanedName != *serviceName {
    84  		logger.Info("Expected requested service name: %v. Got: %v", *serviceName, initialReq.Name)
    85  		return status.Error(codes.NotFound, "Bad requested service name")
    86  	}
    87  	if err := stream.Send(&lbpb.LoadBalanceResponse{
    88  		LoadBalanceResponseType: &lbpb.LoadBalanceResponse_InitialResponse{
    89  			InitialResponse: &lbpb.InitialLoadBalanceResponse{},
    90  		},
    91  	}); err != nil {
    92  		logger.Errorf("Error sending initial LB response: %v", err)
    93  		return status.Error(codes.Unknown, "Error sending initial response")
    94  	}
    95  	logger.Info("Send LoadBalanceResponse: %v", l.serverListResponse)
    96  	if err := stream.Send(l.serverListResponse); err != nil {
    97  		logger.Errorf("Error sending LB response: %v", err)
    98  		return status.Error(codes.Unknown, "Error sending response")
    99  	}
   100  	if *shortStream {
   101  		return nil
   102  	}
   103  	for {
   104  		logger.Info("Send LoadBalanceResponse: %v", l.serverListResponse)
   105  		if err := stream.Send(l.serverListResponse); err != nil {
   106  			logger.Errorf("Error sending LB response: %v", err)
   107  			return status.Error(codes.Unknown, "Error sending response")
   108  		}
   109  		time.Sleep(10 * time.Second)
   110  	}
   111  }
   112  
   113  func main() {
   114  	flag.Parse()
   115  	var opts []grpc.ServerOption
   116  	if *useTLS {
   117  		certFile := testdata.Path("server1.pem")
   118  		keyFile := testdata.Path("server1.key")
   119  		creds, err := credentials.NewServerTLSFromFile(certFile, keyFile)
   120  		if err != nil {
   121  			logger.Fatalf("Failed to generate credentials %v", err)
   122  		}
   123  		opts = append(opts, grpc.Creds(creds))
   124  	} else if *useALTS {
   125  		altsOpts := alts.DefaultServerOptions()
   126  		altsTC := alts.NewServerCreds(altsOpts)
   127  		opts = append(opts, grpc.Creds(altsTC))
   128  	}
   129  	var serverList []*lbpb.Server
   130  	if len(*backendAddrs) == 0 {
   131  		serverList = make([]*lbpb.Server, 0)
   132  	} else {
   133  		rawBackendAddrs := strings.Split(*backendAddrs, ",")
   134  		serverList = make([]*lbpb.Server, len(rawBackendAddrs))
   135  		for i := range rawBackendAddrs {
   136  			rawIP, rawPort, err := net.SplitHostPort(rawBackendAddrs[i])
   137  			if err != nil {
   138  				logger.Fatalf("Failed to parse --backend_addrs[%d]=%v, error: %v", i, rawBackendAddrs[i], err)
   139  			}
   140  			ip := net.ParseIP(rawIP)
   141  			if ip == nil {
   142  				logger.Fatalf("Failed to parse ip: %v", rawIP)
   143  			}
   144  			numericPort, err := strconv.Atoi(rawPort)
   145  			if err != nil {
   146  				logger.Fatalf("Failed to convert port %v to int", rawPort)
   147  			}
   148  			logger.Infof("Adding backend ip: %v, port: %d", ip.String(), numericPort)
   149  			serverList[i] = &lbpb.Server{
   150  				IpAddress: ip,
   151  				Port:      int32(numericPort),
   152  			}
   153  		}
   154  	}
   155  	serverListResponse := &lbpb.LoadBalanceResponse{
   156  		LoadBalanceResponseType: &lbpb.LoadBalanceResponse_ServerList{
   157  			ServerList: &lbpb.ServerList{
   158  				Servers: serverList,
   159  			},
   160  		},
   161  	}
   162  	server := grpc.NewServer(opts...)
   163  	logger.Infof("Begin listening on %d.", *port)
   164  	lis, err := net.Listen("tcp", ":"+strconv.Itoa(*port))
   165  	if err != nil {
   166  		logger.Fatalf("Failed to listen on port %v: %v", *port, err)
   167  	}
   168  	lbpb.RegisterLoadBalancerServer(server, &loadBalancerServer{
   169  		serverListResponse: serverListResponse,
   170  	})
   171  	server.Serve(lis)
   172  }