I am developing a SSL TCP Server in ruby and testing it against a multi threads client. When the number of threads on the client side is less than 190, there is no problem on the server, all the messages are received correctly. But once I increase the number of threads on the client side over 195, two problems pop up :
Problem 1 : Exception ECONNABORTED on the server side
/usr/local/rvm/rubies/ruby-2.1.5/lib/ruby/2.1.0/openssl/ssl.rb:232:in `accept': Software caused connection abort - accept(2) (Errno::ECONNABORTED)
from /usr/local/rvm/rubies/ruby-2.1.5/lib/ruby/2.1.0/openssl/ssl.rb:232:in `accept'
from server.rb:30:in `block (2 levels) in start_server'
I am able to workaround this by restarting the accept loop in the exception handler.
Problem 2 : Server stuck When I increase the number of thread on the client side (e.g. 250), after several seconds, the server is frozen, i.e. no exception and no new connection allowed. This one is really annoying because there is no way on the server side to know it is frozen.
OS : FreeBSD 10.1 Ruby version : 2.2.1 (tried 2.1.5 as well)
Server code :
loop do
server = TCPServer.new(ip_address, port)
sslContext = OpenSSL::SSL::SSLContext.new
sslContext.cert = OpenSSL::X509::Certificate.new(File.open("cert/cert.pem"))
sslContext.key = OpenSSL::PKey::RSA.new(File.open("cert/key.pem"), SSL_PASSWORD)
sslServer = OpenSSL::SSL::SSLServer.new(server, sslContext)
loop do
Thread.new(sslServer.accept) do |connection|
begin
messageIn = connection.gets
connection.close
rescue Exception => ex
puts "Exception in main loop : " + ex.message
puts "Backtrace : " + ex.backtrace.join("\n")
end
end
end
end
end
Client code :
def create_client(host, port)
begin
socket = TCPSocket.open(host,port)
ssl_context = OpenSSL::SSL::SSLContext.new()
ssl_context.cert = OpenSSL::X509::Certificate.new(File.open("lib/cert/cert.pem"))
ssl_context.key = OpenSSL::PKey::RSA.new(File.open("lib/cert/key.pem"), SSL_PASSWORD)
ssl_context.ssl_version = :SSLv3
ssl_context.ssl_timeout = 10
ssl_socket = OpenSSL::SSL::SSLSocket.new(socket, ssl_context)
ssl_socket.sync_close = true
ssl_socket.connect
rescue Exception => ex
puts "Exception in create_client"
sleep 1
return create_client(host, port )
end
return ssl_socket
end
........
for j in 1..10 do
threads = []
for i in 1..n.to_i do
threads << Thread.new do
begin
socket = create_client(ip, port)
socket.puts("hello")
socket.flush
socket.close
rescue Exception => ex
puts "Exception"
end
end
end
threads.each(&:join)
end
Sounds like you're running into networking limits on the server side. You probably need to increase the queue length with
sysctl kern.ipc.soacceptqueue=250
(or set it permanently in /etc/sysctl.conf). You can check the application connection queue withnetstat -Lan
.