protocol/grpc/client.go (121 lines of code) (raw):

/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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 grpc import ( "reflect" "sync" "time" ) import ( "github.com/dubbogo/gost/log/logger" "github.com/dustin/go-humanize" "github.com/grpc-ecosystem/grpc-opentracing/go/otgrpc" "github.com/opentracing/opentracing-go" "google.golang.org/grpc" "google.golang.org/grpc/credentials" "google.golang.org/grpc/credentials/insecure" "gopkg.in/yaml.v2" ) import ( "dubbo.apache.org/dubbo-go/v3/common" "dubbo.apache.org/dubbo-go/v3/common/constant" "dubbo.apache.org/dubbo-go/v3/config" ) var clientConf *ClientConfig var clientConfInitOnce sync.Once // Client is gRPC client include client connection and invoker type Client struct { *grpc.ClientConn invoker reflect.Value } // NewClient creates a new gRPC client. func NewClient(url *common.URL) (*Client, error) { clientConfInitOnce.Do(clientInit) // If global trace instance was set, it means trace function enabled. // If not, will return NoopTracer. tracer := opentracing.GlobalTracer() dialOpts := make([]grpc.DialOption, 0, 4) // set max send and recv msg size maxCallRecvMsgSize := constant.DefaultMaxCallRecvMsgSize if recvMsgSize, err := humanize.ParseBytes(url.GetParam(constant.MaxCallRecvMsgSize, "")); err == nil && recvMsgSize > 0 { maxCallRecvMsgSize = int(recvMsgSize) } maxCallSendMsgSize := constant.DefaultMaxCallSendMsgSize if sendMsgSize, err := humanize.ParseBytes(url.GetParam(constant.MaxCallSendMsgSize, "")); err == nil && sendMsgSize > 0 { maxCallSendMsgSize = int(sendMsgSize) } // consumer config client connectTimeout //connectTimeout := config.GetConsumerConfig().ConnectTimeout dialOpts = append(dialOpts, grpc.WithBlock(), // todo config network timeout grpc.WithTimeout(time.Second*3), grpc.WithUnaryInterceptor(otgrpc.OpenTracingClientInterceptor(tracer, otgrpc.LogPayloads())), grpc.WithStreamInterceptor(otgrpc.OpenTracingStreamClientInterceptor(tracer, otgrpc.LogPayloads())), grpc.WithDefaultCallOptions( grpc.CallContentSubtype(clientConf.ContentSubType), grpc.MaxCallRecvMsgSize(maxCallRecvMsgSize), grpc.MaxCallSendMsgSize(maxCallSendMsgSize), ), ) tlsConfig := config.GetRootConfig().TLSConfig if tlsConfig != nil { cfg, err := config.GetClientTlsConfig(&config.TLSConfig{ CACertFile: tlsConfig.CACertFile, TLSCertFile: tlsConfig.TLSCertFile, TLSKeyFile: tlsConfig.TLSKeyFile, TLSServerName: tlsConfig.TLSServerName, }) logger.Infof("Grpc Client initialized the TLSConfig configuration") if err != nil { return nil, err } dialOpts = append(dialOpts, grpc.WithTransportCredentials(credentials.NewTLS(cfg))) } else { dialOpts = append(dialOpts, grpc.WithTransportCredentials(insecure.NewCredentials())) } conn, err := grpc.Dial(url.Location, dialOpts...) if err != nil { logger.Errorf("grpc dial error: %v", err) return nil, err } key := url.GetParam(constant.InterfaceKey, "") impl := config.GetConsumerServiceByInterfaceName(key) invoker := getInvoker(impl, conn) return &Client{ ClientConn: conn, invoker: reflect.ValueOf(invoker), }, nil } func clientInit() { // load rootConfig from runtime rootConfig := config.GetRootConfig() clientConfig := GetClientConfig() clientConf = &clientConfig // check client config and decide whether to use the default config defer func() { if clientConf == nil || len(clientConf.ContentSubType) == 0 { defaultClientConfig := GetDefaultClientConfig() clientConf = &defaultClientConfig } if err := clientConf.Validate(); err != nil { panic(err) } }() if rootConfig.Application == nil { return } protocolConf := config.GetRootConfig().Protocols if protocolConf == nil { logger.Info("protocol_conf default use dubbo config") } else { grpcConf := protocolConf[GRPC] if grpcConf == nil { logger.Warnf("grpcConf is nil") return } grpcConfByte, err := yaml.Marshal(grpcConf) if err != nil { panic(err) } err = yaml.Unmarshal(grpcConfByte, clientConf) if err != nil { panic(err) } } } func getInvoker(impl any, conn *grpc.ClientConn) any { var in []reflect.Value in = append(in, reflect.ValueOf(conn)) method := reflect.ValueOf(impl).MethodByName("GetDubboStub") res := method.Call(in) return res[0].Interface() }