mirror of https://github.com/grpc/grpc.git
Merge pull request #10670 from apolcyn/fix_ruby_require_before_fork
defer grpc_init and background threads until first grpc object initpull/10694/head
commit
023e39deb7
13 changed files with 380 additions and 24 deletions
@ -0,0 +1,69 @@ |
|||||||
|
#!/usr/bin/env ruby |
||||||
|
|
||||||
|
# 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. |
||||||
|
|
||||||
|
# Prompted by and minimal repro of https://github.com/grpc/grpc/issues/10658 |
||||||
|
|
||||||
|
require_relative './end2end_common' |
||||||
|
|
||||||
|
def main |
||||||
|
server_port = '' |
||||||
|
OptionParser.new do |opts| |
||||||
|
opts.on('--client_control_port=P', String) do |
||||||
|
STDERR.puts 'client control port not used' |
||||||
|
end |
||||||
|
opts.on('--server_port=P', String) do |p| |
||||||
|
server_port = p |
||||||
|
end |
||||||
|
end.parse! |
||||||
|
|
||||||
|
p = fork do |
||||||
|
stub = Echo::EchoServer::Stub.new("localhost:#{server_port}", |
||||||
|
:this_channel_is_insecure) |
||||||
|
stub.echo(Echo::EchoRequest.new(request: 'hello')) |
||||||
|
end |
||||||
|
|
||||||
|
begin |
||||||
|
Timeout.timeout(10) do |
||||||
|
Process.wait(p) |
||||||
|
end |
||||||
|
rescue Timeout::Error |
||||||
|
STDERR.puts "timeout waiting for forked process #{p}" |
||||||
|
Process.kill('SIGKILL', p) |
||||||
|
Process.wait(p) |
||||||
|
raise 'Timed out waiting for client process. ' \ |
||||||
|
'It likely hangs when using gRPC after loading it and then forking' |
||||||
|
end |
||||||
|
|
||||||
|
client_exit_code = $CHILD_STATUS |
||||||
|
fail "forked process failed #{client_exit_code}" if client_exit_code != 0 |
||||||
|
end |
||||||
|
|
||||||
|
main |
@ -0,0 +1,69 @@ |
|||||||
|
#!/usr/bin/env ruby |
||||||
|
|
||||||
|
# Copyright 2016, 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_relative './end2end_common' |
||||||
|
|
||||||
|
def main |
||||||
|
STDERR.puts 'start server' |
||||||
|
server_runner = ServerRunner.new(EchoServerImpl) |
||||||
|
server_port = server_runner.run |
||||||
|
|
||||||
|
# TODO(apolcyn) Can we get rid of this sleep? |
||||||
|
# Without it, an immediate call to the just started EchoServer |
||||||
|
# fails with UNAVAILABLE |
||||||
|
sleep 1 |
||||||
|
|
||||||
|
STDERR.puts 'start client' |
||||||
|
_, client_pid = start_client('forking_client_client.rb', |
||||||
|
server_port) |
||||||
|
|
||||||
|
begin |
||||||
|
Timeout.timeout(10) do |
||||||
|
Process.wait(client_pid) |
||||||
|
end |
||||||
|
rescue Timeout::Error |
||||||
|
STDERR.puts "timeout wait for client pid #{client_pid}" |
||||||
|
Process.kill('SIGKILL', client_pid) |
||||||
|
Process.wait(client_pid) |
||||||
|
STDERR.puts 'killed client child' |
||||||
|
raise 'Timed out waiting for client process. ' \ |
||||||
|
'It likely hangs when requiring grpc, then forking, then using grpc ' |
||||||
|
end |
||||||
|
|
||||||
|
client_exit_code = $CHILD_STATUS |
||||||
|
if client_exit_code != 0 |
||||||
|
fail "forking client client failed, exit code #{client_exit_code}" |
||||||
|
end |
||||||
|
|
||||||
|
server_runner.stop |
||||||
|
end |
||||||
|
|
||||||
|
main |
@ -0,0 +1,77 @@ |
|||||||
|
#!/usr/bin/env ruby |
||||||
|
|
||||||
|
# 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. |
||||||
|
|
||||||
|
# For GRPC::Core classes, which use the grpc c-core, object init |
||||||
|
# is interesting because it's related to overall library init. |
||||||
|
|
||||||
|
require_relative './end2end_common' |
||||||
|
|
||||||
|
def main |
||||||
|
grpc_class = '' |
||||||
|
OptionParser.new do |opts| |
||||||
|
opts.on('--grpc_class=P', String) do |p| |
||||||
|
grpc_class = p |
||||||
|
end |
||||||
|
end.parse! |
||||||
|
|
||||||
|
test_proc = nil |
||||||
|
|
||||||
|
case grpc_class |
||||||
|
when 'channel' |
||||||
|
test_proc = proc do |
||||||
|
GRPC::Core::Channel.new('dummy_host', nil, :this_channel_is_insecure) |
||||||
|
end |
||||||
|
when 'server' |
||||||
|
test_proc = proc do |
||||||
|
GRPC::Core::Server.new({}) |
||||||
|
end |
||||||
|
when 'channel_credentials' |
||||||
|
test_proc = proc do |
||||||
|
GRPC::Core::ChannelCredentials.new |
||||||
|
end |
||||||
|
when 'call_credentials' |
||||||
|
test_proc = proc do |
||||||
|
GRPC::Core::CallCredentials.new(proc { |noop| noop }) |
||||||
|
end |
||||||
|
when 'compression_options' |
||||||
|
test_proc = proc do |
||||||
|
GRPC::Core::CompressionOptions.new |
||||||
|
end |
||||||
|
else |
||||||
|
fail "bad --grpc_class=#{grpc_class} param" |
||||||
|
end |
||||||
|
|
||||||
|
th = Thread.new { test_proc.call } |
||||||
|
test_proc.call |
||||||
|
th.join |
||||||
|
end |
||||||
|
|
||||||
|
main |
@ -0,0 +1,67 @@ |
|||||||
|
#!/usr/bin/env ruby |
||||||
|
|
||||||
|
# Copyright 2016, 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_relative './end2end_common' |
||||||
|
|
||||||
|
def main |
||||||
|
native_grpc_classes = %w( channel |
||||||
|
server |
||||||
|
channel_credentials |
||||||
|
call_credentials |
||||||
|
compression_options ) |
||||||
|
|
||||||
|
native_grpc_classes.each do |grpc_class| |
||||||
|
STDERR.puts 'start client' |
||||||
|
this_dir = File.expand_path(File.dirname(__FILE__)) |
||||||
|
client_path = File.join(this_dir, 'grpc_class_init_client.rb') |
||||||
|
client_pid = Process.spawn(RbConfig.ruby, |
||||||
|
client_path, |
||||||
|
"--grpc_class=#{grpc_class}") |
||||||
|
begin |
||||||
|
Timeout.timeout(10) do |
||||||
|
Process.wait(client_pid) |
||||||
|
end |
||||||
|
rescue Timeout::Error |
||||||
|
STDERR.puts "timeout waiting for client pid #{client_pid}" |
||||||
|
Process.kill('SIGKILL', client_pid) |
||||||
|
Process.wait(client_pid) |
||||||
|
STDERR.puts 'killed client child' |
||||||
|
raise 'Timed out waiting for client process. ' \ |
||||||
|
'It likely hangs when the first constructed gRPC object has ' \ |
||||||
|
"type: #{grpc_class}" |
||||||
|
end |
||||||
|
|
||||||
|
client_exit_code = $CHILD_STATUS |
||||||
|
fail "client failed, exit code #{client_exit_code}" if client_exit_code != 0 |
||||||
|
end |
||||||
|
end |
||||||
|
|
||||||
|
main |
Loading…
Reference in new issue