grpc分层设计

grpc 客户端 go grpc 客户端主线程_ide


按照grpc的分层设计,结合源码,分析grpc客户端启动和调用流程。

grpc的ManagedChannel创建

ManagedChannel 是对 Transport 层 SocketChannel 的抽象,Transport 层负责协议消息的序列化和反序列化,以及协议消息的发送和读取。

ManagedChannel 将处理后的请求和响应传递给与之相关联的 ClientCall 进行上层处理,同时,ManagedChannel 提供了对 Channel 的生命周期管理(链路创建、空闲、关闭等)。
创建代码如下:

ManagedChannel channel = ManagedChannelBuilder
               .forAddress(server.getHost(), server.getPort())
               .defaultLoadBalancingPolicy("round_robin")
               .nameResolverFactory(new DnsNameResolverProvider())
               .idleTimeout(30, TimeUnit.SECONDS)
               .usePlaintext().build();

a、 forAddress()
这个方法中主要是设置默认的provider和绑定服务端,这里并没有去连接服务端,只是根据host和port拼接String字符串,并返回NettyChannelBuilder 对象
b、 usePlaintext()
不会再去尝试升级http1
a、 build()
channel是利用反射进行创建的,默认基于epoll,如果没有找到对应的类,则创建nio对应的channel
buildTransportFactory(): 新建NettyChannelBuilder.NettyTransportFactory对象,NettyTransportFactory主要用来创建NettyClientTransport对象。


ManagedChannelBuilder 
    
      NettyChannelBuilder 
    
      NettyTransportFactory 
    
      ManagedChannelImpl 
    
    forAddress 
  
    创建NettyTransportFactory实例 
  
    创建ManagedChannelImpl对象 
  
      ManagedChannelBuilder 
    
      NettyChannelBuilder 
    
      NettyTransportFactory 
    
      ManagedChannelImpl

stub

定义的.proto文件中生成的代码中会生成对应的stub
根据build()中返回的channel和需要创建对应的stub,生成的代码中默认提供了newStub(channel),newBlockingStub(channel),newFutureStub(channel) 三种stub。
newStub(channel): 创建异步stub支持所有类型的service调用
newBlockingStub(channel): 创建一个阻塞的stub,支持一元的和输出流的service调用
newFutureStub(channel): 创建一个listener future的stub,支持一元的和输出流的service调用

简单demo进行测试,分别使用50,100,200并发。持续5分钟,

测试机器为4核8g的windows系统

50线程并发5分钟测试结果:

grpc 客户端 go grpc 客户端主线程_ci_02


grpc 客户端 go grpc 客户端主线程_bc_03


grpc 客户端 go grpc 客户端主线程_bc_04

100线程并发5分钟测试结果:

grpc 客户端 go grpc 客户端主线程_ide_05

grpc 客户端 go grpc 客户端主线程_ide_06


grpc 客户端 go grpc 客户端主线程_bc_07

200线程并发5分钟测试结果:

grpc 客户端 go grpc 客户端主线程_ide_08

grpc 客户端 go grpc 客户端主线程_ide_09


grpc 客户端 go grpc 客户端主线程_bc_10

ClientCall

客户端直接调用.proto生成的代码

@java.lang.Override
    public io.grpc.stub.StreamObserver<com.why.grpc.OptionServiceOuterClass.OrdersQuery> query(
        io.grpc.stub.StreamObserver<com.why.grpc.OptionServiceOuterClass.OrdersAck> responseObserver) {
      return asyncBidiStreamingCall(
          getChannel().newCall(METHOD_QUERY, getCallOptions()), responseObserver);
    }

getChannel(): 这里会获取创建stub时传入的channel,上面build()创建的channel类型为ManagedChannelOrphanWrapper( extends ForwardingManagedChannel --> extends ManagedChannel -->> extends Channel)
newCall:根据channel的类型,调用ForwardingManagedChannel 中的方法,interceptorChannel的类型为ClientInterceptors类型,下面就会进入ClientInterceptors的newCall方法

ForwardingManagedChannel ------------------->
 this.interceptorChannel = ClientInterceptors.intercept(channel, interceptors);
  @Override
  public <RequestT, ResponseT> ClientCall<RequestT, ResponseT> newCall(
      MethodDescriptor<RequestT, ResponseT> methodDescriptor, CallOptions callOptions) {
    return delegate.newCall(methodDescriptor, callOptions);
  }

在build创建ManagedChannelOrphanWrapper时的代码如下,可知delegate时ManagedChannelImpl的类型,代码会进入到ManagedChannelImpl的方法中

AbstractManagedChannelImplBuilder ---->>
  @Override
  public ManagedChannel build() {
    return new ManagedChannelOrphanWrapper(new ManagedChannelImpl(
        this,
        buildTransportFactory(),
        // TODO(carl-mastrangelo): Allow clients to pass this in
        new ExponentialBackoffPolicy.Provider(),
        SharedResourcePool.forResource(GrpcUtil.SHARED_CHANNEL_EXECUTOR),
        GrpcUtil.STOPWATCH_SUPPLIER,
        getEffectiveInterceptors(),
        TimeProvider.SYSTEM_TIME_PROVIDER));
  }

在ManagedChannelImpl类中会继续调用ClientInterceptors.InterceptorChannel中的方法

ManagedChannelImpl----->>
 this.interceptorChannel = ClientInterceptors.intercept(channel, interceptors);
  /*
   * Creates a new outgoing call on the channel.
   */
  @Override
  public <ReqT, RespT> ClientCall<ReqT, RespT> newCall(MethodDescriptor<ReqT, RespT> method,
      CallOptions callOptions) {
    return interceptorChannel.newCall(method, callOptions);
  }

ClientInterceptors.InterceptorChannel中的newCall根据类型,在build()方法中的getEffectiveInterceptors()默认会设置StatsClientInterceptor和TracingClientInterceptor两个拦截器,在ClientInterceptors.intercept(channel, interceptors)方法调用时,会把这两个拦截器分别创建InterceptorChannel并返回,返回的channel包装了两层连接器,所以调用时先

ClientInterceptors.InterceptorChannel------------------->
private final ClientInterceptor interceptor;

    private InterceptorChannel(Channel channel, ClientInterceptor interceptor) {
      this.channel = channel;
      this.interceptor = Preconditions.checkNotNull(interceptor, "interceptor");
    }

    @Override
    public <ReqT, RespT> ClientCall<ReqT, RespT> newCall(
        MethodDescriptor<ReqT, RespT> method, CallOptions callOptions) {
      return interceptor.interceptCall(method, callOptions, channel);
    }
在build()中新建ManagedChannelImpl中,创建的channel中channel使用了ServiceConfigInterceptor拦截器,在ServiceConfigInterceptor中调用next.newCall(method, callOptions),build()方法中Channel channel = new RealChannel(nameResolver.getServiceAuthority());是创建的RealChannel对象接下来会进入RealChannel的newCall方法,最后创建ClientCallImpl对象。在CensusTracingModule中,创建SimpleForwardingClientCall(call)对象,在CensusStatsModule中创建SimpleForwardingClientCall(call)对象,最后返回到有.proto文件生成的代码getChannel().newCall中。

ClientCall newCall() 创建流程


OptionServiceStub 
    
      CensusStatsModule 
    
      CensusTracingModule 
    
      ServiceConfigInterceptor 
    
      ManagedChannelImpl.RealChannel 
    
    interceptCall 
  
    interceptCall 
  
    interceptCall 
  
    newCall 
  
    ClientCallImpl 
  
    SimpleForwardingClientCall 
  
    SimpleForwardingClientCall 
  
    SimpleForwardingClientCall 
  
      OptionServiceStub 
    
      CensusStatsModule 
    
      CensusTracingModule 
    
      ServiceConfigInterceptor 
    
      ManagedChannelImpl.RealChannel

clientcall组装好之后,在asyncBidiStreamingCall(clientcall,oberserver)中创建连接和stream并启动,
ManagedChannelImpl.ChannelTransportProvider: get 方法第一次调用时,异步进行域名解析和负载均衡,并返回DelayedClientTransport对象
DelayedStream:在transport可用之前的stream,当transport可用的时候会委托给真实的stream
这里的DelayedClientTransport和DelayedStream是把dns解析和发送消息等线程放入syncContext中进行异步执行,在异步执行ManagedChannelImpl的requestConnection方法时调用 InternalSubchannel的obtainActiveTransport方法判断是否存在activeTransport,若没有会新建NettyClientTransport并进行连接,在创建stream时会建立连接,在连接成功后更新InternalSubchannel的activeTransport的值和ManagedChannelImpl的subchannelPicker,下次直接从该transport创建stream。

ClientCalls 
    
      ClientCallImpl 
    
      ManagedChannelImpl.ChannelTransportProvider 
    
      DelayedClientTransport 
    
      DelayedStream 
    
    start 
  
    startInternal 
  
    get 
  
    DelayedClientTransport 
  
    newStream 
  
    createPendingStream 
  
    PendingStream,并放入syncContext中,异步执行 
  
    start 
  
      ClientCalls 
    
      ClientCallImpl 
    
      ManagedChannelImpl.ChannelTransportProvider 
    
      DelayedClientTransport 
    
      DelayedStream


grpc客户端创建连接和发送消息流程


Created with Raphaël 2.2.0 ClientCallImpl.start() getTransport (可能会返回 DelayedClientTransport和NettyClientTransport) NettyClientTransport? 创建NettyClientStream 同步创建stream,发送消息直接调用NettyClientStream的write方法,向channel中写入消息。 等待返回 创建PendingStream 要执行的任务放入队列中,异步执行 异步执行dns解析和创建netty客户端并连接, 连接成功后更新InternalSubchannel中的activeTransport等信息, 下次getTransport 时会返回NettyClientTransport yes no