package grpcutil import ( "context" "crypto/tls" "crypto/x509" "fmt" "io/ioutil" "time" "github.com/c2h5oh/datasize" grpc_middleware "github.com/grpc-ecosystem/go-grpc-middleware" grpc_recovery "github.com/grpc-ecosystem/go-grpc-middleware/recovery" "google.golang.org/grpc" "google.golang.org/grpc/backoff" "google.golang.org/grpc/credentials" "google.golang.org/grpc/keepalive" "google.golang.org/grpc/reflection" ) func TLS(tlsCACert, tlsCertFile, tlsKeyFile string) (credentials.TransportCredentials, error) { // load peer cert/key, ca cert if tlsCACert == "" { if tlsCertFile == "" && tlsKeyFile == "" { return nil, nil } return credentials.NewServerTLSFromFile(tlsCertFile, tlsKeyFile) } var caCert []byte peerCert, err := tls.LoadX509KeyPair(tlsCertFile, tlsKeyFile) if err != nil { return nil, fmt.Errorf("load peer cert/key error:%w", err) } caCert, err = ioutil.ReadFile(tlsCACert) if err != nil { return nil, fmt.Errorf("read ca cert file error:%w", err) } caCertPool := x509.NewCertPool() caCertPool.AppendCertsFromPEM(caCert) return credentials.NewTLS(&tls.Config{ Certificates: []tls.Certificate{peerCert}, ClientCAs: caCertPool, ClientAuth: tls.RequireAndVerifyClientCert, MinVersion: tls.VersionTLS12, //nolint:gosec InsecureSkipVerify: true, // This is to make it work when Common Name does not match - remove when procedure is updated for common name }), nil } func NewServer(rateLimit uint32, creds credentials.TransportCredentials) *grpc.Server { var ( streamInterceptors []grpc.StreamServerInterceptor unaryInterceptors []grpc.UnaryServerInterceptor ) streamInterceptors = append(streamInterceptors, grpc_recovery.StreamServerInterceptor()) unaryInterceptors = append(unaryInterceptors, grpc_recovery.UnaryServerInterceptor()) //if metrics.Enabled { // streamInterceptors = append(streamInterceptors, grpc_prometheus.StreamServerInterceptor) // unaryInterceptors = append(unaryInterceptors, grpc_prometheus.UnaryServerInterceptor) //} var grpcServer *grpc.Server reflection.Register(grpcServer) //cpus := uint32(runtime.GOMAXPROCS(-1)) opts := []grpc.ServerOption{ //grpc.NumStreamWorkers(cpus), // reduce amount of goroutines //grpc.WriteBufferSize(1024), // reduce buffers to save mem //grpc.ReadBufferSize(1024), grpc.MaxConcurrentStreams(rateLimit), // to force clients reduce concurrency level // Don't drop the connection, settings accordign to this comment on GitHub // https://github.com/grpc/grpc-go/issues/3171#issuecomment-552796779 grpc.KeepaliveEnforcementPolicy(keepalive.EnforcementPolicy{ MinTime: 10 * time.Second, PermitWithoutStream: true, }), grpc.StreamInterceptor(grpc_middleware.ChainStreamServer(streamInterceptors...)), grpc.UnaryInterceptor(grpc_middleware.ChainUnaryServer(unaryInterceptors...)), grpc.Creds(creds), } grpcServer = grpc.NewServer(opts...) //if metrics.Enabled { // grpc_prometheus.Register(grpcServer) //} return grpcServer } func Connect(creds credentials.TransportCredentials, dialAddress string) (*grpc.ClientConn, error) { var dialOpts []grpc.DialOption backoffCfg := backoff.DefaultConfig backoffCfg.BaseDelay = 500 * time.Millisecond backoffCfg.MaxDelay = 10 * time.Second dialOpts = []grpc.DialOption{ grpc.WithConnectParams(grpc.ConnectParams{Backoff: backoffCfg, MinConnectTimeout: 10 * time.Minute}), grpc.WithDefaultCallOptions(grpc.MaxCallRecvMsgSize(int(15 * datasize.MB))), grpc.WithKeepaliveParams(keepalive.ClientParameters{}), } if creds == nil { dialOpts = append(dialOpts, grpc.WithInsecure()) } else { dialOpts = append(dialOpts, grpc.WithTransportCredentials(creds)) } //if opts.inMemConn != nil { // dialOpts = append(dialOpts, grpc.WithContextDialer(func(ctx context.Context, url string) (net.Conn, error) { // return opts.inMemConn.Dial() // })) //} ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() return grpc.DialContext(ctx, dialAddress, dialOpts...) }