ruby - Ruby 中的非阻塞 SSL 套接字协商。可能的?

标签 ruby sockets ssl openssl connection-timeout

简介

我有一个客户端与第三方服务建立大量 SSL 连接。在某些情况下,第 3 方在套接字和 ssl 协商过程中停止响应。发生这种情况时,我当前的实现会在超时之前连续“等待”几个小时。

为了解决这个问题,我正在尝试实现以下流程:

require 'socket'
require 'openssl'

# variables

host = '....'
port = ...
p12  = #OpenSSL::PKCS12 object 

# set up socket

addr = Socket.getaddrinfo(host, nil)
sockaddr = Socket.pack_sockaddr_in(port, addr[0][3])

socket = Socket.new(Socket.const_get(addr[0][0]), Socket::SOCK_STREAM, 0)
socket.setsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY, 1)

begin
  socket.connect_nonblock(sockaddr)
rescue IO::WaitWritable
  if IO.select(nil, [socket], nil, timeout)
    begin
      socket.connect_nonblock(sockaddr)
    rescue Errno::EISCONN
      puts "socket connected"
    rescue
      puts "socket error"
      socket.close
      raise
    end
  else
    socket.close
    raise "Connection timeout"
  end
end

# negotiate ssl

context      = OpenSSL::SSL::SSLContext.new
context.cert = p12.certificate
context.key  = p12.key

ssl_socket = OpenSSL::SSL::SSLSocket.new(socket, context)
ssl_socket.sync_close = true

puts "ssl connecting"
ssl_socket.connect_nonblock
puts "ssl connected"

# cleanup

ssl_socket.close
puts "socket closed"

ssl_socket.connect_nonblock 最终将被包装在与 socket.connect_nonblock 类似的结构中。

问题

我遇到的问题是 ssl_socket.connect_nonblock 在运行时引发以下问题:

`connect_nonblock': read would block (OpenSSL::SSL::SSLError)

相反,我希望它像 socket.connect_nonblock 那样引发 IO::WaitWritable

我在互联网上搜索了有关此特定错误的信息,但找不到任何特殊用途。据我所知,其他人已经使用这种方法取得了成功,所以我不确定我错过了什么。为了完整起见,我在 ruby​​ 2.2.01.9.3 上发现了相同的结果。

非常感谢任何建议!

最佳答案

有同样的问题,我尝试了下面,它似乎适合我的情况。

ssl_socket = OpenSSL::SSL::SSLSocket.new socket, context
ssl_socket.sync = true

begin
  ssl_socket.connect_nonblock
rescue IO::WaitReadable
  if IO.select([ssl_socket], nil, nil, timeout)
    retry
  else
    # timeout
  end
rescue IO::WaitWritable
  if IO.select(nil, [ssl_socket], nil, timeout)
    retry
  else
    # timeout
  end
end

关于ruby - Ruby 中的非阻塞 SSL 套接字协商。可能的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30787466/

相关文章:

ruby-on-rails - Redmine:安装时出现 rake 错误

ios - Google App Engine 和 Socket.IO 上的移动聊天应用程序

java - 为什么 Java DatagramSocket 在确定连接时在 "isConnected"上返回 false?

线程启动失败的Java异常类

node.js - 如何创建安全(TLS/SSL)Websocket 服务器

ssl - 如何为两个子域使用一个 GCP 负载均衡器?

javascript - acts_as_votable javascript html 表

ruby - 获取 Rails 项目中包含的所有 gem 的变更日志

ios - [iOS]如何使用NSURLConnection自定义SSL验证?

Ruby 和 SystemVerilog DPI