| Index: third_party/grpc/src/ruby/spec/generic/client_stub_spec.rb
|
| diff --git a/third_party/grpc/src/ruby/spec/generic/client_stub_spec.rb b/third_party/grpc/src/ruby/spec/generic/client_stub_spec.rb
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..5e13c25fcf3e6d1e70f416006a2609cc1af2363c
|
| --- /dev/null
|
| +++ b/third_party/grpc/src/ruby/spec/generic/client_stub_spec.rb
|
| @@ -0,0 +1,476 @@
|
| +# Copyright 2015, Google Inc.
|
| +# All rights reserved.
|
| +#
|
| +# Redistribution and use in source and binary forms, with or without
|
| +# modification, are permitted provided that the following conditions are
|
| +# met:
|
| +#
|
| +# * Redistributions of source code must retain the above copyright
|
| +# notice, this list of conditions and the following disclaimer.
|
| +# * Redistributions in binary form must reproduce the above
|
| +# copyright notice, this list of conditions and the following disclaimer
|
| +# in the documentation and/or other materials provided with the
|
| +# distribution.
|
| +# * Neither the name of Google Inc. nor the names of its
|
| +# contributors may be used to endorse or promote products derived from
|
| +# this software without specific prior written permission.
|
| +#
|
| +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
| +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
| +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
| +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
| +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
| +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
| +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
| +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
| +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
| +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
| +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
| +
|
| +require 'grpc'
|
| +
|
| +def wakey_thread(&blk)
|
| + n = GRPC::Notifier.new
|
| + t = Thread.new do
|
| + blk.call(n)
|
| + end
|
| + n.wait
|
| + t
|
| +end
|
| +
|
| +def load_test_certs
|
| + test_root = File.join(File.dirname(File.dirname(__FILE__)), 'testdata')
|
| + files = ['ca.pem', 'server1.key', 'server1.pem']
|
| + files.map { |f| File.open(File.join(test_root, f)).read }
|
| +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
|
| + @server_queue = nil
|
| + @method = 'an_rpc_method'
|
| + @pass = OK
|
| + @fail = INTERNAL
|
| + @cq = GRPC::Core::CompletionQueue.new
|
| + end
|
| +
|
| + after(:each) do
|
| + @server.close(@server_queue) unless @server_queue.nil?
|
| + end
|
| +
|
| + describe '#new' do
|
| + let(:fake_host) { 'localhost:0' }
|
| + it 'can be created from a host and args' do
|
| + opts = { a_channel_arg: 'an_arg' }
|
| + blk = proc do
|
| + GRPC::ClientStub.new(fake_host, @cq, :this_channel_is_insecure, **opts)
|
| + end
|
| + expect(&blk).not_to raise_error
|
| + end
|
| +
|
| + it 'can be created with a default deadline' do
|
| + opts = { a_channel_arg: 'an_arg', deadline: 5 }
|
| + blk = proc do
|
| + GRPC::ClientStub.new(fake_host, @cq, :this_channel_is_insecure, **opts)
|
| + end
|
| + expect(&blk).not_to raise_error
|
| + end
|
| +
|
| + it 'can be created with an channel override' do
|
| + opts = { a_channel_arg: 'an_arg', channel_override: @ch }
|
| + blk = proc do
|
| + GRPC::ClientStub.new(fake_host, @cq, :this_channel_is_insecure, **opts)
|
| + end
|
| + expect(&blk).not_to raise_error
|
| + end
|
| +
|
| + it 'cannot be created with a bad channel override' do
|
| + blk = proc do
|
| + opts = { a_channel_arg: 'an_arg', channel_override: Object.new }
|
| + GRPC::ClientStub.new(fake_host, @cq, :this_channel_is_insecure, **opts)
|
| + end
|
| + expect(&blk).to raise_error
|
| + end
|
| +
|
| + it 'cannot be created with bad credentials' do
|
| + blk = proc do
|
| + opts = { a_channel_arg: 'an_arg' }
|
| + GRPC::ClientStub.new(fake_host, @cq, Object.new, **opts)
|
| + end
|
| + expect(&blk).to raise_error
|
| + end
|
| +
|
| + it 'can be created with test test credentials' do
|
| + certs = load_test_certs
|
| + blk = proc do
|
| + opts = {
|
| + GRPC::Core::Channel::SSL_TARGET => 'foo.test.google.fr',
|
| + a_channel_arg: 'an_arg'
|
| + }
|
| + creds = GRPC::Core::ChannelCredentials.new(certs[0], nil, nil)
|
| + GRPC::ClientStub.new(fake_host, @cq, creds, **opts)
|
| + end
|
| + expect(&blk).to_not raise_error
|
| + end
|
| + end
|
| +
|
| + describe '#request_response' do
|
| + before(:each) do
|
| + @sent_msg, @resp = 'a_msg', 'a_reply'
|
| + end
|
| +
|
| + shared_examples 'request response' do
|
| + it 'should send a request to/receive a reply from a server' do
|
| + server_port = create_test_server
|
| + th = run_request_response(@sent_msg, @resp, @pass)
|
| + stub = GRPC::ClientStub.new("localhost:#{server_port}", @cq,
|
| + :this_channel_is_insecure)
|
| + expect(get_response(stub)).to eq(@resp)
|
| + th.join
|
| + end
|
| +
|
| + it 'should send metadata to the server ok' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_request_response(@sent_msg, @resp, @pass,
|
| + k1: 'v1', k2: 'v2')
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + expect(get_response(stub)).to eq(@resp)
|
| + th.join
|
| + end
|
| +
|
| + it 'should send a request when configured using an override channel' do
|
| + server_port = create_test_server
|
| + alt_host = "localhost:#{server_port}"
|
| + th = run_request_response(@sent_msg, @resp, @pass)
|
| + ch = GRPC::Core::Channel.new(alt_host, nil, :this_channel_is_insecure)
|
| + stub = GRPC::ClientStub.new('ignored-host', @cq,
|
| + :this_channel_is_insecure,
|
| + channel_override: ch)
|
| + expect(get_response(stub)).to eq(@resp)
|
| + th.join
|
| + end
|
| +
|
| + it 'should raise an error if the status is not OK' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_request_response(@sent_msg, @resp, @fail)
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + blk = proc { get_response(stub) }
|
| + expect(&blk).to raise_error(GRPC::BadStatus)
|
| + th.join
|
| + end
|
| + end
|
| +
|
| + describe 'without a call operation' do
|
| + def get_response(stub)
|
| + stub.request_response(@method, @sent_msg, noop, noop,
|
| + k1: 'v1', k2: 'v2')
|
| + end
|
| +
|
| + it_behaves_like 'request response'
|
| + end
|
| +
|
| + describe 'via a call operation' do
|
| + def get_response(stub)
|
| + 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
|
| + end
|
| +
|
| + it_behaves_like 'request response'
|
| + end
|
| + end
|
| +
|
| + describe '#client_streamer' do
|
| + shared_examples 'client streaming' do
|
| + before(:each) do
|
| + @sent_msgs = Array.new(3) { |i| 'msg_' + (i + 1).to_s }
|
| + @resp = 'a_reply'
|
| + end
|
| +
|
| + it 'should send requests to/receive a reply from a server' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_client_streamer(@sent_msgs, @resp, @pass)
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + expect(get_response(stub)).to eq(@resp)
|
| + th.join
|
| + end
|
| +
|
| + it 'should send metadata to the server ok' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_client_streamer(@sent_msgs, @resp, @pass,
|
| + k1: 'v1', k2: 'v2')
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + expect(get_response(stub)).to eq(@resp)
|
| + th.join
|
| + end
|
| +
|
| + it 'should raise an error if the status is not ok' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_client_streamer(@sent_msgs, @resp, @fail)
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + blk = proc { get_response(stub) }
|
| + expect(&blk).to raise_error(GRPC::BadStatus)
|
| + th.join
|
| + end
|
| + end
|
| +
|
| + describe 'without a call operation' do
|
| + def get_response(stub)
|
| + stub.client_streamer(@method, @sent_msgs, noop, noop,
|
| + k1: 'v1', k2: 'v2')
|
| + end
|
| +
|
| + it_behaves_like 'client streaming'
|
| + end
|
| +
|
| + describe 'via a call operation' do
|
| + def get_response(stub)
|
| + 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
|
| + end
|
| +
|
| + it_behaves_like 'client streaming'
|
| + end
|
| + end
|
| +
|
| + describe '#server_streamer' do
|
| + shared_examples 'server streaming' do
|
| + before(:each) do
|
| + @sent_msg = 'a_msg'
|
| + @replys = Array.new(3) { |i| 'reply_' + (i + 1).to_s }
|
| + end
|
| +
|
| + it 'should send a request to/receive replies from a server' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_server_streamer(@sent_msg, @replys, @pass)
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + expect(get_responses(stub).collect { |r| r }).to eq(@replys)
|
| + th.join
|
| + end
|
| +
|
| + it 'should raise an error if the status is not ok' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_server_streamer(@sent_msg, @replys, @fail)
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + e = get_responses(stub)
|
| + expect { e.collect { |r| r } }.to raise_error(GRPC::BadStatus)
|
| + th.join
|
| + end
|
| +
|
| + it 'should send metadata to the server ok' do
|
| + server_port = create_test_server
|
| + host = "localhost:#{server_port}"
|
| + th = run_server_streamer(@sent_msg, @replys, @fail,
|
| + k1: 'v1', k2: 'v2')
|
| + stub = GRPC::ClientStub.new(host, @cq, :this_channel_is_insecure)
|
| + e = get_responses(stub)
|
| + expect { e.collect { |r| r } }.to raise_error(GRPC::BadStatus)
|
| + th.join
|
| + end
|
| + end
|
| +
|
| + describe 'without a call operation' do
|
| + def get_responses(stub)
|
| + e = stub.server_streamer(@method, @sent_msg, noop, noop,
|
| + k1: 'v1', k2: 'v2')
|
| + expect(e).to be_a(Enumerator)
|
| + e
|
| + end
|
| +
|
| + it_behaves_like 'server streaming'
|
| + end
|
| +
|
| + describe 'via a call operation' do
|
| + def get_responses(stub)
|
| + 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
|
| + expect(e).to be_a(Enumerator)
|
| + e
|
| + end
|
| +
|
| + it_behaves_like 'server streaming'
|
| + end
|
| + end
|
| +
|
| + describe '#bidi_streamer' do
|
| + shared_examples 'bidi streaming' 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
|
| + th = run_bidi_streamer_handle_inputs_first(@sent_msgs, @replys,
|
| + @pass)
|
| + stub = GRPC::ClientStub.new(@host, @cq, :this_channel_is_insecure)
|
| + e = get_responses(stub)
|
| + expect(e.collect { |r| r }).to eq(@replys)
|
| + th.join
|
| + end
|
| +
|
| + it 'supports client-initiated ping pong', bidi: true do
|
| + th = run_bidi_streamer_echo_ping_pong(@sent_msgs, @pass, true)
|
| + stub = GRPC::ClientStub.new(@host, @cq, :this_channel_is_insecure)
|
| + 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
|
| + th = run_bidi_streamer_echo_ping_pong(@sent_msgs, @pass, false)
|
| + stub = GRPC::ClientStub.new(@host, @cq, :this_channel_is_insecure)
|
| + e = get_responses(stub)
|
| + expect(e.collect { |r| r }).to eq(@sent_msgs)
|
| + th.join
|
| + end
|
| + end
|
| +
|
| + describe 'without a call operation' do
|
| + def get_responses(stub)
|
| + e = stub.bidi_streamer(@method, @sent_msgs, noop, noop)
|
| + expect(e).to be_a(Enumerator)
|
| + e
|
| + end
|
| +
|
| + it_behaves_like 'bidi streaming'
|
| + end
|
| +
|
| + describe 'via a call operation' do
|
| + def get_responses(stub)
|
| + op = stub.bidi_streamer(@method, @sent_msgs, noop, noop,
|
| + return_op: true)
|
| + expect(op).to be_a(GRPC::ActiveCall::Operation)
|
| + e = op.execute
|
| + expect(e).to be_a(Enumerator)
|
| + e
|
| + end
|
| +
|
| + it_behaves_like 'bidi streaming'
|
| + end
|
| +
|
| + describe 'without enough time to run' 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 'should fail with DeadlineExceeded', bidi: true do
|
| + @server.start
|
| + stub = GRPC::ClientStub.new(@host, @cq, :this_channel_is_insecure)
|
| + blk = proc do
|
| + e = stub.bidi_streamer(@method, @sent_msgs, noop, noop,
|
| + timeout: 0.001)
|
| + e.collect { |r| r }
|
| + end
|
| + expect(&blk).to raise_error GRPC::BadStatus, /Deadline Exceeded/
|
| + end
|
| + end
|
| + end
|
| +
|
| + def run_server_streamer(expected_input, replys, status, **kw)
|
| + wanted_metadata = kw.clone
|
| + 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
|
| + expect(c.remote_read).to eq(expected_input)
|
| + replys.each { |r| c.remote_send(r) }
|
| + c.send_status(status, status == @pass ? 'OK' : 'NOK', true)
|
| + end
|
| + end
|
| +
|
| + def run_bidi_streamer_handle_inputs_first(expected_inputs, replys,
|
| + status)
|
| + 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)
|
| + end
|
| + end
|
| +
|
| + def run_bidi_streamer_echo_ping_pong(expected_inputs, status, client_starts)
|
| + 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)
|
| + c.remote_send(i)
|
| + else
|
| + c.remote_send(i)
|
| + expect(c.remote_read).to eq(i)
|
| + end
|
| + end
|
| + c.send_status(status, status == @pass ? 'OK' : 'NOK', true)
|
| + end
|
| + end
|
| +
|
| + def run_client_streamer(expected_inputs, resp, status, **kw)
|
| + wanted_metadata = kw.clone
|
| + 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)
|
| + end
|
| + c.remote_send(resp)
|
| + c.send_status(status, status == @pass ? 'OK' : 'NOK', true)
|
| + end
|
| + end
|
| +
|
| + def run_request_response(expected_input, resp, status, **kw)
|
| + wanted_metadata = kw.clone
|
| + 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)
|
| + end
|
| + c.remote_send(resp)
|
| + c.send_status(status, status == @pass ? 'OK' : 'NOK', true)
|
| + end
|
| + end
|
| +
|
| + def create_test_server
|
| + @server_queue = GRPC::Core::CompletionQueue.new
|
| + @server = GRPC::Core::Server.new(@server_queue, nil)
|
| + @server.add_http2_port('0.0.0.0:0', :this_port_is_insecure)
|
| + end
|
| +
|
| + def expect_server_to_be_invoked(notifier)
|
| + @server.start
|
| + 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
|
|
|