跳到主要内容

11、Kafka 实战 - 服务端:轮询处理网络请求

接收器采用 Round-Robin 也就是轮询的方式分配客户端的 SocketChannel 给多个处理器,每个处理器都会有多个 SocketChannel

  • 服务端的处理器会为 SocketChannel 创建一个 Kafka 通道,configureNewConnections() 会为 SocketChannel 注册读事件,创建 Kafka 通道,并绑定到选择键上
/**
 * Thread that processes all requests from a single connection. There are N of these running in parallel
 * 处理来自单个连接的所有请求的线程。有N个平行的
 * each of which has its own selector
 * 每一个都有自己的选择器
 */
private[kafka] class Processor(val id: Int,requestChannel: RequestChannel,...){
   
     
  private val newConnections = new ConcurrentLinkedQueue[SocketChannel]()
  private val inflightResponses = mutable.Map[String, RequestChannel.Response]()
  private val selector = new KSelector(..., "socket-server", ..., ChannelBuilders.create(...))
  /**
   * Queue up a new connection for reading
   * 为读取新连接排队
   * Acceptor 会把多个客户端的 SocketChannel 分配给一个 Processor,这里就需要一个队列来保存新的客户端连接通道
   */
  def accept(socketChannel: SocketChannel) {
   
     
    // 把获取到的 socketChannel 存放到队列里面
    newConnections.add(socketChannel)
    // 触发选择器开始轮询,原先的轮询因为没有事件到来而阻塞
    wakeup()
  }
  /**
   * Register any new connections that have been queued up
   * 注册已排队的任何新连接,往通道注册 OP_READ 事件
   */
  private def configureNewConnections() {
   
     
    while (!newConnections.isEmpty) {
   
     
      // 从队列里弹出 socketChannel,一个 SocketChannel 只会注册一次
      val channel = newConnections.poll()
      // 从通道中获取本地服务端和远程客户端的地址和端口,构造唯一的 ConnectionId
      val connectionId = ConnectionId(localHost, localPort, remoteHost, remotePort).toString
      // 往 selector 里面进行注册
      selector.register(connectionId, channel)
    }
  }
  // java 的 Selector 类
  // 选择器注册方法,为 SocketChannel 注册读事件,并创建 Kafka 通道
  public void register(String id, SocketChannel socketChannel) throws ClosedChannelException {
   
     
        // 往 selector 里面注册 OP_READ 事件,processor 线程就可以读取客户端发送过来的连接
        SelectionKey key = socketChannel.register(nioSelector, SelectionKey.OP_READ);
        // kafka 里面对 SocketChannel 进行来封装 KafkaChannel
        KafkaChannel channel = channelBuilder.buildChannel(id, key, maxReceiveSize);
        // key 和 channel
        key.attach(channel);
        // 服务端代码和客户端网络部分的代码是复用的,channels 里面维护来多个网络连接
        this.channels.put(id, channel);
    }
}

  • 客户端的 NetworkClient 和服务端的处理器都使用相同的选择器类(Selector)进行轮询,发送请求和响应请求都是通过选择器的轮询才会触发
  • 客户端和服务端在发送请求时,会将请求加入 inFlightRequests/inFlightResponse 队列
// Processor 的 run 方法
// 服务端处理器的运行方法的处理方式和客户端的轮询类似
override def run() {
   
     
    while (isRunning) {
   
     
        // setup any new connections that have been queued up
        // 设置已排队的任何新连接,读取每个 SocketChannel,都往 Selector 里面注册 OP_READ 事件
        configureNewConnections()
        // register any new responses for writing
        // 记录任何新的回复
        // TODO 处理响应
        processNewResponses()
        // 读取和发送请求的代码
        poll()
        // TODO 处理接收到的请求
        processCompletedReceives()
        // TODO 处理我们发送出去的响应
        processCompletedSends()
        processDisconnected()
    }  
    // 关闭处理器
    shutdownComplete()
}

private def processNewResponses() {
   
     
    // 有多个 processor 线程,一个线程对应一个响应队列,这个 id 就是一个 processor 线程编号
    // TODO Response 对象
    var curr = requestChannel.receiveResponse(id)
    // 只要当前处理器还有响应要发送
    while (curr != null) {
   
     
        curr.responseAction match {
   
     
          case RequestChannel.NoOpAction =>
            // There is no response to send to the client, we need to read more pipelined requests
            // 没有响应发送到客户端,我们需要读取更多的流水线请求
            // that are sitting in the server's socket buffer
            // 它们位于服务器的套接字缓冲区中
            curr.request.updateRequestMetrics
            trace("Socket server received empty response to send, registering for read: " + curr)
            // 没有响应需要发送给客户端,需要读取更多的请求,添加 OP_READ
            selector.unmute(curr.request.connectionId)
          case RequestChannel.SendAction =>
            // 有响应要发送给客户端,注册写事件,下次轮询时要把响应发送给客户端
            // 绑定 OP_WRITE
            sendResponse(curr)
          // 关闭 Socket 通信
          case RequestChannel.CloseConnectionAction =>
            curr.request.updateRequestMetrics
            trace("Closing socket connection actively according to the response code.")
            close(selector, curr.request.connectionId)
        }
    }
}

  • 选择器轮询操作最后会返回 completedReceives 和 completedSends,分别表示接收和发送
  // 服务端处理器的运行方法在调用选择器的轮询后,处理已经完成的请求接收和响应发送
  private def processCompletedReceives() {
   
     
    // 遍历每一个请求
    selector.completedReceives.asScala.foreach {
   
      receive =>
      try {
   
     
        val channel = selector.channel(receive.source)
        val session = RequestChannel.Session(new KafkaPrincipal(KafkaPrincipal.USER_TYPE, channel.principal.getName),
          channel.socketAddress)
        // 对于获取到到请求按照协议进行解析,解析出来的就是一个个 Request
        val req = RequestChannel.Request(processor = id, connectionId = receive.source, session = session, buffer = receive.payload, startTimeMs = time.milliseconds, securityProtocol = protocol)
        // request 请求放入队列
        requestChannel.sendRequest(req)
        // 移除 OP_READ 事件,接收到响应后就不需要再读了
        selector.mute(receive.source)
      } catch {
   
     
        case e @ (_: InvalidRequestException | _: SchemaException) =>
          // note that even though we got an exception, we can assume that receive.source is valid. Issues with constructing a valid receive object were handled earlier
          error(s"Closing socket for ${receive.source} because of error", e)
          close(selector, receive.source)
      }
    }
  }
  private def processCompletedSends() {
   
     
    selector.completedSends.asScala.foreach {
   
      send =>
      // 移除数据结构里面的信息
      val resp = inflightResponses.remove(send.destination).getOrElse {
   
     
        throw new IllegalStateException(s"Send for ${send.destination} completed, but not in inflightResponses")
      }
      resp.request.updateRequestMetrics()
      // 添加 OP_READ 事件,继续读取客户端请求
      selector.unmute(send.destination)
    }
  }

  • 客户端和服务端的交互都是通过各自的选择器轮询所驱动,完整的请求和响应过程串联起来的步骤

  • 客户端完成请求的发送,服务端轮询到客户端发送的请求

  • 服务端接受客户端发送的请求,进行业务处理,并准备好响应结果准备发送

  • 服务端完成响应的发送,客户端轮询到服务端发送的响应

  • 客户端接收完服务端发送的响应,整个流程结束