|  | @@ -30,15 +30,41 @@
 | 
	
		
			
				|  |  |  require 'grpc'
 | 
	
		
			
				|  |  |  require 'xray/thread_dump_signal_handler'
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -NOOP = proc { |x| x }
 | 
	
		
			
				|  |  | -FAKE_HOST = 'localhost:0'
 | 
	
		
			
				|  |  | +# Notifier is useful high-level synchronization primitive.
 | 
	
		
			
				|  |  | +class Notifier
 | 
	
		
			
				|  |  | +  attr_reader :payload, :notified
 | 
	
		
			
				|  |  | +  alias_method :notified?, :notified
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def initialize
 | 
	
		
			
				|  |  | +    @mutex    = Mutex.new
 | 
	
		
			
				|  |  | +    @cvar     = ConditionVariable.new
 | 
	
		
			
				|  |  | +    @notified = false
 | 
	
		
			
				|  |  | +    @payload  = nil
 | 
	
		
			
				|  |  | +  end
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def wait
 | 
	
		
			
				|  |  | +    @mutex.synchronize do
 | 
	
		
			
				|  |  | +      @cvar.wait(@mutex) until notified?
 | 
	
		
			
				|  |  | +    end
 | 
	
		
			
				|  |  | +  end
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def notify(payload)
 | 
	
		
			
				|  |  | +    @mutex.synchronize do
 | 
	
		
			
				|  |  | +      return Error.new('already notified') if notified?
 | 
	
		
			
				|  |  | +      @payload  = payload
 | 
	
		
			
				|  |  | +      @notified = true
 | 
	
		
			
				|  |  | +      @cvar.signal
 | 
	
		
			
				|  |  | +      return nil
 | 
	
		
			
				|  |  | +    end
 | 
	
		
			
				|  |  | +  end
 | 
	
		
			
				|  |  | +end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  def wakey_thread(&blk)
 | 
	
		
			
				|  |  | -  awake_mutex, awake_cond = Mutex.new, ConditionVariable.new
 | 
	
		
			
				|  |  | +  n = Notifier.new
 | 
	
		
			
				|  |  |    t = Thread.new do
 | 
	
		
			
				|  |  | -    blk.call(awake_mutex, awake_cond)
 | 
	
		
			
				|  |  | +    blk.call(n)
 | 
	
		
			
				|  |  |    end
 | 
	
		
			
				|  |  | -  awake_mutex.synchronize { awake_cond.wait(awake_mutex) }
 | 
	
		
			
				|  |  | +  n.wait
 | 
	
		
			
				|  |  |    t
 | 
	
		
			
				|  |  |  end
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -50,8 +76,11 @@ end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  include GRPC::Core::StatusCodes
 | 
	
		
			
				|  |  |  include GRPC::Core::TimeConsts
 | 
	
		
			
				|  |  | +include GRPC::Core::CallOps
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  describe 'ClientStub' do
 | 
	
		
			
				|  |  | +  let(:noop) { proc { |x| x } }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    before(:each) do
 | 
	
		
			
				|  |  |      Thread.abort_on_exception = true
 | 
	
		
			
				|  |  |      @server = nil
 | 
	
	
		
			
				|  | @@ -66,61 +95,56 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |    end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    describe '#new' do
 | 
	
		
			
				|  |  | +    let(:fake_host) { 'localhost:0' }
 | 
	
		
			
				|  |  |      it 'can be created from a host and args' do
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        opts = { a_channel_arg: 'an_arg' }
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).not_to raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      it 'can be created with a default deadline' do
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        opts = { a_channel_arg: 'an_arg', deadline: 5 }
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).not_to raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      it 'can be created with an channel override' do
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        opts = { a_channel_arg: 'an_arg', channel_override: @ch }
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).not_to raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      it 'cannot be created with a bad channel override' do
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  |          opts = { a_channel_arg: 'an_arg', channel_override: Object.new }
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).to raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      it 'cannot be created with bad credentials' do
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  |          opts = { a_channel_arg: 'an_arg', creds: Object.new }
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).to raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      it 'can be created with test test credentials' do
 | 
	
		
			
				|  |  |        certs = load_test_certs
 | 
	
		
			
				|  |  | -      host = FAKE_HOST
 | 
	
		
			
				|  |  |        blk = proc do
 | 
	
		
			
				|  |  |          opts = {
 | 
	
		
			
				|  |  |            GRPC::Core::Channel::SSL_TARGET => 'foo.test.google.fr',
 | 
	
		
			
				|  |  |            a_channel_arg: 'an_arg',
 | 
	
		
			
				|  |  |            creds: GRPC::Core::Credentials.new(certs[0], nil, nil)
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  | -        GRPC::ClientStub.new(host, @cq, **opts)
 | 
	
		
			
				|  |  | +        GRPC::ClientStub.new(fake_host, @cq, **opts)
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |        expect(&blk).to_not raise_error
 | 
	
		
			
				|  |  |      end
 | 
	
	
		
			
				|  | @@ -187,7 +211,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'without a call operation' do
 | 
	
		
			
				|  |  |        def get_response(stub)
 | 
	
		
			
				|  |  | -        stub.request_response(@method, @sent_msg, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        stub.request_response(@method, @sent_msg, noop, noop,
 | 
	
		
			
				|  |  |                                k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -196,7 +220,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'via a call operation' do
 | 
	
		
			
				|  |  |        def get_response(stub)
 | 
	
		
			
				|  |  | -        op = stub.request_response(@method, @sent_msg, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        op = stub.request_response(@method, @sent_msg, noop, noop,
 | 
	
		
			
				|  |  |                                     return_op: true, k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |          expect(op).to be_a(GRPC::ActiveCall::Operation)
 | 
	
		
			
				|  |  |          op.execute
 | 
	
	
		
			
				|  | @@ -259,7 +283,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'without a call operation' do
 | 
	
		
			
				|  |  |        def get_response(stub)
 | 
	
		
			
				|  |  | -        stub.client_streamer(@method, @sent_msgs, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        stub.client_streamer(@method, @sent_msgs, noop, noop,
 | 
	
		
			
				|  |  |                               k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -268,7 +292,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'via a call operation' do
 | 
	
		
			
				|  |  |        def get_response(stub)
 | 
	
		
			
				|  |  | -        op = stub.client_streamer(@method, @sent_msgs, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        op = stub.client_streamer(@method, @sent_msgs, noop, noop,
 | 
	
		
			
				|  |  |                                    return_op: true, k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |          expect(op).to be_a(GRPC::ActiveCall::Operation)
 | 
	
		
			
				|  |  |          op.execute
 | 
	
	
		
			
				|  | @@ -333,7 +357,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'without a call operation' do
 | 
	
		
			
				|  |  |        def get_responses(stub)
 | 
	
		
			
				|  |  | -        e = stub.server_streamer(@method, @sent_msg, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        e = stub.server_streamer(@method, @sent_msg, noop, noop,
 | 
	
		
			
				|  |  |                                   k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |          expect(e).to be_a(Enumerator)
 | 
	
		
			
				|  |  |          e
 | 
	
	
		
			
				|  | @@ -344,7 +368,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'via a call operation' do
 | 
	
		
			
				|  |  |        def get_responses(stub)
 | 
	
		
			
				|  |  | -        op = stub.server_streamer(@method, @sent_msg, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        op = stub.server_streamer(@method, @sent_msg, noop, noop,
 | 
	
		
			
				|  |  |                                    return_op: true, k1: 'v1', k2: 'v2')
 | 
	
		
			
				|  |  |          expect(op).to be_a(GRPC::ActiveCall::Operation)
 | 
	
		
			
				|  |  |          e = op.execute
 | 
	
	
		
			
				|  | @@ -361,34 +385,30 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |        before(:each) do
 | 
	
		
			
				|  |  |          @sent_msgs = Array.new(3) { |i| 'msg_' + (i + 1).to_s }
 | 
	
		
			
				|  |  |          @replys = Array.new(3) { |i| 'reply_' + (i + 1).to_s }
 | 
	
		
			
				|  |  | +        server_port = create_test_server
 | 
	
		
			
				|  |  | +        @host = "localhost:#{server_port}"
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |        it 'supports sending all the requests first', bidi: true do
 | 
	
		
			
				|  |  | -        server_port = create_test_server
 | 
	
		
			
				|  |  | -        host = "localhost:#{server_port}"
 | 
	
		
			
				|  |  |          th = run_bidi_streamer_handle_inputs_first(@sent_msgs, @replys,
 | 
	
		
			
				|  |  |                                                     @pass)
 | 
	
		
			
				|  |  | -        stub = GRPC::ClientStub.new(host, @cq)
 | 
	
		
			
				|  |  | +        stub = GRPC::ClientStub.new(@host, @cq)
 | 
	
		
			
				|  |  |          e = get_responses(stub)
 | 
	
		
			
				|  |  |          expect(e.collect { |r| r }).to eq(@replys)
 | 
	
		
			
				|  |  |          th.join
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |        it 'supports client-initiated ping pong', bidi: true do
 | 
	
		
			
				|  |  | -        server_port = create_test_server
 | 
	
		
			
				|  |  | -        host = "localhost:#{server_port}"
 | 
	
		
			
				|  |  |          th = run_bidi_streamer_echo_ping_pong(@sent_msgs, @pass, true)
 | 
	
		
			
				|  |  | -        stub = GRPC::ClientStub.new(host, @cq)
 | 
	
		
			
				|  |  | +        stub = GRPC::ClientStub.new(@host, @cq)
 | 
	
		
			
				|  |  |          e = get_responses(stub)
 | 
	
		
			
				|  |  |          expect(e.collect { |r| r }).to eq(@sent_msgs)
 | 
	
		
			
				|  |  |          th.join
 | 
	
		
			
				|  |  |        end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |        it 'supports a server-initiated ping pong', bidi: true do
 | 
	
		
			
				|  |  | -        server_port = create_test_server
 | 
	
		
			
				|  |  | -        host = "localhost:#{server_port}"
 | 
	
		
			
				|  |  |          th = run_bidi_streamer_echo_ping_pong(@sent_msgs, @pass, false)
 | 
	
		
			
				|  |  | -        stub = GRPC::ClientStub.new(host, @cq)
 | 
	
		
			
				|  |  | +        stub = GRPC::ClientStub.new(@host, @cq)
 | 
	
		
			
				|  |  |          e = get_responses(stub)
 | 
	
		
			
				|  |  |          expect(e.collect { |r| r }).to eq(@sent_msgs)
 | 
	
		
			
				|  |  |          th.join
 | 
	
	
		
			
				|  | @@ -397,7 +417,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'without a call operation' do
 | 
	
		
			
				|  |  |        def get_responses(stub)
 | 
	
		
			
				|  |  | -        e = stub.bidi_streamer(@method, @sent_msgs, NOOP, NOOP)
 | 
	
		
			
				|  |  | +        e = stub.bidi_streamer(@method, @sent_msgs, noop, noop)
 | 
	
		
			
				|  |  |          expect(e).to be_a(Enumerator)
 | 
	
		
			
				|  |  |          e
 | 
	
		
			
				|  |  |        end
 | 
	
	
		
			
				|  | @@ -407,7 +427,7 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      describe 'via a call operation' do
 | 
	
		
			
				|  |  |        def get_responses(stub)
 | 
	
		
			
				|  |  | -        op = stub.bidi_streamer(@method, @sent_msgs, NOOP, NOOP,
 | 
	
		
			
				|  |  | +        op = stub.bidi_streamer(@method, @sent_msgs, noop, noop,
 | 
	
		
			
				|  |  |                                  return_op: true)
 | 
	
		
			
				|  |  |          expect(op).to be_a(GRPC::ActiveCall::Operation)
 | 
	
		
			
				|  |  |          e = op.execute
 | 
	
	
		
			
				|  | @@ -421,8 +441,8 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    def run_server_streamer(expected_input, replys, status, **kw)
 | 
	
		
			
				|  |  |      wanted_metadata = kw.clone
 | 
	
		
			
				|  |  | -    wakey_thread do |mtx, cnd|
 | 
	
		
			
				|  |  | -      c = expect_server_to_be_invoked(mtx, cnd)
 | 
	
		
			
				|  |  | +    wakey_thread do |notifier|
 | 
	
		
			
				|  |  | +      c = expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |        wanted_metadata.each do |k, v|
 | 
	
		
			
				|  |  |          expect(c.metadata[k.to_s]).to eq(v)
 | 
	
		
			
				|  |  |        end
 | 
	
	
		
			
				|  | @@ -434,8 +454,8 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    def run_bidi_streamer_handle_inputs_first(expected_inputs, replys,
 | 
	
		
			
				|  |  |                                              status)
 | 
	
		
			
				|  |  | -    wakey_thread do |mtx, cnd|
 | 
	
		
			
				|  |  | -      c = expect_server_to_be_invoked(mtx, cnd)
 | 
	
		
			
				|  |  | +    wakey_thread do |notifier|
 | 
	
		
			
				|  |  | +      c = expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |        expected_inputs.each { |i| expect(c.remote_read).to eq(i) }
 | 
	
		
			
				|  |  |        replys.each { |r| c.remote_send(r) }
 | 
	
		
			
				|  |  |        c.send_status(status, status == @pass ? 'OK' : 'NOK', true)
 | 
	
	
		
			
				|  | @@ -443,8 +463,8 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |    end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    def run_bidi_streamer_echo_ping_pong(expected_inputs, status, client_starts)
 | 
	
		
			
				|  |  | -    wakey_thread do |mtx, cnd|
 | 
	
		
			
				|  |  | -      c = expect_server_to_be_invoked(mtx, cnd)
 | 
	
		
			
				|  |  | +    wakey_thread do |notifier|
 | 
	
		
			
				|  |  | +      c = expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |        expected_inputs.each do |i|
 | 
	
		
			
				|  |  |          if client_starts
 | 
	
		
			
				|  |  |            expect(c.remote_read).to eq(i)
 | 
	
	
		
			
				|  | @@ -460,8 +480,8 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    def run_client_streamer(expected_inputs, resp, status, **kw)
 | 
	
		
			
				|  |  |      wanted_metadata = kw.clone
 | 
	
		
			
				|  |  | -    wakey_thread do |mtx, cnd|
 | 
	
		
			
				|  |  | -      c = expect_server_to_be_invoked(mtx, cnd)
 | 
	
		
			
				|  |  | +    wakey_thread do |notifier|
 | 
	
		
			
				|  |  | +      c = expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |        expected_inputs.each { |i| expect(c.remote_read).to eq(i) }
 | 
	
		
			
				|  |  |        wanted_metadata.each do |k, v|
 | 
	
		
			
				|  |  |          expect(c.metadata[k.to_s]).to eq(v)
 | 
	
	
		
			
				|  | @@ -473,8 +493,8 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    def run_request_response(expected_input, resp, status, **kw)
 | 
	
		
			
				|  |  |      wanted_metadata = kw.clone
 | 
	
		
			
				|  |  | -    wakey_thread do |mtx, cnd|
 | 
	
		
			
				|  |  | -      c = expect_server_to_be_invoked(mtx, cnd)
 | 
	
		
			
				|  |  | +    wakey_thread do |notifier|
 | 
	
		
			
				|  |  | +      c = expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |        expect(c.remote_read).to eq(expected_input)
 | 
	
		
			
				|  |  |        wanted_metadata.each do |k, v|
 | 
	
		
			
				|  |  |          expect(c.metadata[k.to_s]).to eq(v)
 | 
	
	
		
			
				|  | @@ -490,24 +510,16 @@ describe 'ClientStub' do
 | 
	
		
			
				|  |  |      @server.add_http2_port('0.0.0.0:0')
 | 
	
		
			
				|  |  |    end
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  def start_test_server(awake_mutex, awake_cond)
 | 
	
		
			
				|  |  | +  def expect_server_to_be_invoked(notifier)
 | 
	
		
			
				|  |  |      @server.start
 | 
	
		
			
				|  |  | -    @server_tag = Object.new
 | 
	
		
			
				|  |  | -    @server.request_call(@server_tag)
 | 
	
		
			
				|  |  | -    awake_mutex.synchronize { awake_cond.signal }
 | 
	
		
			
				|  |  | -  end
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  def expect_server_to_be_invoked(awake_mutex, awake_cond)
 | 
	
		
			
				|  |  | -    start_test_server(awake_mutex, awake_cond)
 | 
	
		
			
				|  |  | -    ev = @server_queue.pluck(@server_tag, INFINITE_FUTURE)
 | 
	
		
			
				|  |  | -    fail OutOfTime if ev.nil?
 | 
	
		
			
				|  |  | -    server_call = ev.call
 | 
	
		
			
				|  |  | -    server_call.metadata = ev.result.metadata
 | 
	
		
			
				|  |  | -    finished_tag = Object.new
 | 
	
		
			
				|  |  | -    server_call.server_accept(@server_queue, finished_tag)
 | 
	
		
			
				|  |  | -    server_call.server_end_initial_metadata
 | 
	
		
			
				|  |  | -    GRPC::ActiveCall.new(server_call, @server_queue, NOOP, NOOP,
 | 
	
		
			
				|  |  | -                         INFINITE_FUTURE,
 | 
	
		
			
				|  |  | -                         finished_tag: finished_tag)
 | 
	
		
			
				|  |  | +    notifier.notify(nil)
 | 
	
		
			
				|  |  | +    server_tag = Object.new
 | 
	
		
			
				|  |  | +    recvd_rpc = @server.request_call(@server_queue, server_tag,
 | 
	
		
			
				|  |  | +                                     INFINITE_FUTURE)
 | 
	
		
			
				|  |  | +    recvd_call = recvd_rpc.call
 | 
	
		
			
				|  |  | +    recvd_call.metadata = recvd_rpc.metadata
 | 
	
		
			
				|  |  | +    recvd_call.run_batch(@server_queue, server_tag, Time.now + 2,
 | 
	
		
			
				|  |  | +                         SEND_INITIAL_METADATA => nil)
 | 
	
		
			
				|  |  | +    GRPC::ActiveCall.new(recvd_call, @server_queue, noop, noop, INFINITE_FUTURE)
 | 
	
		
			
				|  |  |    end
 | 
	
		
			
				|  |  |  end
 |