在C中使用OpenSSL时,我们在上下文中设置选项以删除弱受损和受伤的协议,如SSLv2和SSLv3。从ssl.h
开始,这里有一些有用的选项的位掩码:
#define SSL_OP_NO_SSLv2 0x01000000L
#define SSL_OP_NO_SSLv3 0x02000000L
#define SSL_OP_NO_TLSv1 0x04000000L
#define SSL_OP_NO_TLSv1_2 0x08000000L
#define SSL_OP_NO_TLSv1_1 0x10000000L
但是,我在Ruby中设置它们时遇到了麻烦:
if uri.scheme == "https"
http.use_ssl = true
http.verify_mode = OpenSSL::SSL::VERIFY_PEER
http.options = OpenSSL::SSL::SSL_OP_NO_SSLv2 | OpenSSL::SSL::OP_NO_SSLv3 |
OpenSSL::SSL::SSL_OP_NO_COMPRESSION
end
结果:
$ ./TestCert.rb
./TestCert.rb:12:in `<main>': uninitialized constant OpenSSL::SSL::SSL_OP_SSL2 (NameError)
Ruby docs for 1.9.3(和2.0.0)甚至不愿意提及它。
如何在Ruby中设置TLS上下文选项?
相关:setting SSLContext options in ruby。但是http
时无法将上下文附加到http.use_ssl = true
。
答案 0 :(得分:14)
在Ruby OpenSSL库中,选项常量不以&#39; SSL _&#39;为前缀。您可以通过在irb / console中运行类似的内容来查看选项常量列表:OpenSSL::SSL.constants.grep(/OP_/)
。以下是定义它们的相关ruby C源:https://github.com/ruby/ruby/blob/trunk/ext/openssl/ossl_ssl.c#L2225。
修改强> 开箱即用似乎不是一种为net http设置SSL选项的方法。 请参阅https://bugs.ruby-lang.org/issues/9450。
但是暂时你可以使用这个小黑客:
(Net::HTTP::SSL_IVNAMES << :@ssl_options).uniq!
(Net::HTTP::SSL_ATTRIBUTES << :options).uniq!
Net::HTTP.class_eval do
attr_accessor :ssl_options
end
现在只需在Net :: HTTP实例上设置ssl_options访问器即可。用法示例:
uri = URI('https://google.com:443')
options_mask = OpenSSL::SSL::OP_NO_SSLv2 + OpenSSL::SSL::OP_NO_SSLv3 +
OpenSSL::SSL::OP_NO_COMPRESSION
http = Net::HTTP.new(uri.host, uri.port)
request = Net::HTTP::Get.new(uri.request_uri)
if uri.scheme == "https"
http.use_ssl = true
http.verify_mode = OpenSSL::SSL::VERIFY_PEER
http.ssl_options = options_mask
end
response = http.request request
# To Test
ssl_context = http.instance_variable_get(:@ssl_context)
ssl_context.options == options_mask # => true
我正在使用ruby 2.1.2进行测试,因此您在其他版本的红宝石上的里程可能会有所不同。如果它不适用于您的首选版本,请告诉我。
对于那些感兴趣的人,我看到了ruby代码的相关部分来创建这个hack:https://github.com/ruby/ruby/blob/e9dce8d1b482200685996f64cc2c3bd6ba790110/lib/net/http.rb#L886