2014-05-05 17:30:15 -04:00
|
|
|
require "rbconfig"
|
|
|
|
require 'test/unit'
|
|
|
|
require 'socket'
|
|
|
|
require 'openssl'
|
|
|
|
|
|
|
|
require 'puma/minissl'
|
|
|
|
require 'puma/server'
|
|
|
|
|
|
|
|
require 'net/https'
|
|
|
|
|
2015-01-13 23:11:26 -05:00
|
|
|
class SSLEventsHelper < ::Puma::Events
|
|
|
|
attr_accessor :addr, :cert, :error
|
|
|
|
|
|
|
|
def ssl_error(server, peeraddr, peercert, error)
|
|
|
|
self.addr = peeraddr
|
|
|
|
self.cert = peercert
|
|
|
|
self.error = error
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-11-06 14:00:58 -05:00
|
|
|
DISABLE_SSL = begin
|
|
|
|
Puma::MiniSSL.check
|
|
|
|
rescue
|
|
|
|
true
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2014-05-05 17:30:15 -04:00
|
|
|
class TestPumaServerSSL < Test::Unit::TestCase
|
|
|
|
|
|
|
|
def setup
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
2014-05-05 17:30:15 -04:00
|
|
|
@port = 3212
|
|
|
|
@host = "127.0.0.1"
|
|
|
|
|
|
|
|
@app = lambda { |env| [200, {}, [env['rack.url_scheme']]] }
|
|
|
|
|
2014-10-15 21:39:35 -04:00
|
|
|
@ctx = Puma::MiniSSL::Context.new
|
2014-05-05 17:30:15 -04:00
|
|
|
|
|
|
|
if defined?(JRUBY_VERSION)
|
2014-10-15 21:39:35 -04:00
|
|
|
@ctx.keystore = File.expand_path "../../examples/puma/keystore.jks", __FILE__
|
|
|
|
@ctx.keystore_pass = 'blahblah'
|
2014-05-05 17:30:15 -04:00
|
|
|
else
|
2014-10-15 21:39:35 -04:00
|
|
|
@ctx.key = File.expand_path "../../examples/puma/puma_keypair.pem", __FILE__
|
|
|
|
@ctx.cert = File.expand_path "../../examples/puma/cert_puma.pem", __FILE__
|
2014-05-05 17:30:15 -04:00
|
|
|
end
|
|
|
|
|
2014-10-15 21:39:35 -04:00
|
|
|
@ctx.verify_mode = Puma::MiniSSL::VERIFY_NONE
|
2014-05-05 17:30:15 -04:00
|
|
|
|
2015-01-13 23:11:26 -05:00
|
|
|
@events = SSLEventsHelper.new STDOUT, STDERR
|
2014-05-05 17:30:15 -04:00
|
|
|
@server = Puma::Server.new @app, @events
|
2014-10-15 21:39:35 -04:00
|
|
|
@server.add_ssl_listener @host, @port, @ctx
|
2014-05-05 17:30:15 -04:00
|
|
|
@server.run
|
|
|
|
|
|
|
|
@http = Net::HTTP.new @host, @port
|
|
|
|
@http.use_ssl = true
|
|
|
|
@http.verify_mode = OpenSSL::SSL::VERIFY_NONE
|
|
|
|
end
|
|
|
|
|
|
|
|
def teardown
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
2014-05-05 17:30:15 -04:00
|
|
|
@server.stop(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_url_scheme_for_https
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
|
|
|
|
2014-05-05 17:30:15 -04:00
|
|
|
body = nil
|
|
|
|
@http.start do
|
|
|
|
req = Net::HTTP::Get.new "/", {}
|
|
|
|
|
|
|
|
@http.request(req) do |rep|
|
|
|
|
body = rep.body
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal "https", body
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_very_large_return
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
|
|
|
|
2014-05-05 17:30:15 -04:00
|
|
|
giant = "x" * 2056610
|
|
|
|
|
|
|
|
@server.app = proc do
|
|
|
|
[200, {}, [giant]]
|
|
|
|
end
|
|
|
|
|
|
|
|
body = nil
|
|
|
|
@http.start do
|
|
|
|
req = Net::HTTP::Get.new "/"
|
|
|
|
@http.request(req) do |rep|
|
|
|
|
body = rep.body
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal giant.bytesize, body.bytesize
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_form_submit
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
|
|
|
|
2014-05-05 17:30:15 -04:00
|
|
|
body = nil
|
|
|
|
@http.start do
|
|
|
|
req = Net::HTTP::Post.new '/'
|
|
|
|
req.set_form_data('a' => '1', 'b' => '2')
|
|
|
|
|
|
|
|
@http.request(req) do |rep|
|
|
|
|
body = rep.body
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal "https", body
|
|
|
|
end
|
|
|
|
|
2015-07-31 15:51:07 -04:00
|
|
|
unless RUBY_VERSION =~ /^1.8/
|
|
|
|
def test_ssl_v3_rejection
|
2015-11-06 14:00:58 -05:00
|
|
|
return if DISABLE_SSL
|
2015-07-31 15:51:07 -04:00
|
|
|
@http.ssl_version='SSLv3'
|
|
|
|
assert_raises(OpenSSL::SSL::SSLError) do
|
|
|
|
@http.start do
|
|
|
|
Net::HTTP::Get.new '/'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
unless defined?(JRUBY_VERSION)
|
|
|
|
assert_match("wrong version number", @events.error.message) if @events.error
|
2014-10-15 21:39:35 -04:00
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
2015-05-01 19:39:22 -04:00
|
|
|
end
|
2014-10-15 21:39:35 -04:00
|
|
|
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
# client-side TLS authentication tests
|
2015-11-28 21:17:01 -05:00
|
|
|
class TestPumaServerSSLClient < Test::Unit::TestCase
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
def assert_ssl_client_error_match(error, subject=nil, &blk)
|
|
|
|
@port = 3212
|
|
|
|
@host = "127.0.0.1"
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
@app = lambda { |env| [200, {}, [env['rack.url_scheme']]] }
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
@ctx = Puma::MiniSSL::Context.new
|
|
|
|
if defined?(JRUBY_VERSION)
|
|
|
|
@ctx.keystore = File.expand_path "../../examples/puma/client-certs/keystore.jks", __FILE__
|
|
|
|
@ctx.keystore_pass = 'blahblah'
|
|
|
|
else
|
2015-01-13 23:11:26 -05:00
|
|
|
@ctx.key = File.expand_path "../../examples/puma/client-certs/server.key", __FILE__
|
|
|
|
@ctx.cert = File.expand_path "../../examples/puma/client-certs/server.crt", __FILE__
|
|
|
|
@ctx.ca = File.expand_path "../../examples/puma/client-certs/ca.crt", __FILE__
|
2015-11-28 21:17:01 -05:00
|
|
|
end
|
|
|
|
@ctx.verify_mode = Puma::MiniSSL::VERIFY_PEER | Puma::MiniSSL::VERIFY_FAIL_IF_NO_PEER_CERT
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
events = SSLEventsHelper.new STDOUT, STDERR
|
|
|
|
@server = Puma::Server.new @app, events
|
|
|
|
@server.add_ssl_listener @host, @port, @ctx
|
|
|
|
@server.run
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
@http = Net::HTTP.new @host, @port
|
|
|
|
@http.use_ssl = true
|
|
|
|
@http.verify_mode = OpenSSL::SSL::VERIFY_NONE
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
blk.call(@http)
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
client_error = false
|
|
|
|
begin
|
|
|
|
@http.start do
|
|
|
|
req = Net::HTTP::Get.new "/", {}
|
|
|
|
@http.request(req)
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
2015-11-28 21:17:01 -05:00
|
|
|
rescue OpenSSL::SSL::SSLError
|
|
|
|
client_error = true
|
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
sleep 0.1
|
|
|
|
assert_equal !!error, client_error
|
|
|
|
# The JRuby MiniSSL implementation lacks error capturing currently, so we can't inspect the
|
|
|
|
# messages here
|
|
|
|
unless defined?(JRUBY_VERSION)
|
2015-01-13 23:11:26 -05:00
|
|
|
assert_match error, events.error.message if error
|
|
|
|
assert_equal @host, events.addr if error
|
|
|
|
assert_equal subject, events.cert.subject.to_s if subject
|
|
|
|
end
|
2015-11-28 21:17:01 -05:00
|
|
|
ensure
|
|
|
|
@server.stop(true)
|
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
def test_verify_fail_if_no_client_cert
|
|
|
|
return if DISABLE_SSL
|
2015-11-06 14:00:58 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
assert_ssl_client_error_match 'peer did not return a certificate' do |http|
|
|
|
|
# nothing
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
2015-11-28 21:17:01 -05:00
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
def test_verify_fail_if_client_unknown_ca
|
|
|
|
return if DISABLE_SSL
|
2015-11-06 14:00:58 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
assert_ssl_client_error_match('self signed certificate in certificate chain', '/DC=net/DC=puma/CN=ca-unknown') do |http|
|
|
|
|
key = File.expand_path "../../examples/puma/client-certs/client_unknown.key", __FILE__
|
|
|
|
crt = File.expand_path "../../examples/puma/client-certs/client_unknown.crt", __FILE__
|
|
|
|
http.key = OpenSSL::PKey::RSA.new File.read(key)
|
|
|
|
http.cert = OpenSSL::X509::Certificate.new File.read(crt)
|
|
|
|
http.ca_file = File.expand_path "../../examples/puma/client-certs/unknown_ca.crt", __FILE__
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
2015-11-28 21:17:01 -05:00
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
def test_verify_fail_if_client_expired_cert
|
|
|
|
return if DISABLE_SSL
|
|
|
|
assert_ssl_client_error_match('certificate has expired', '/DC=net/DC=puma/CN=client-expired') do |http|
|
|
|
|
key = File.expand_path "../../examples/puma/client-certs/client_expired.key", __FILE__
|
|
|
|
crt = File.expand_path "../../examples/puma/client-certs/client_expired.crt", __FILE__
|
|
|
|
http.key = OpenSSL::PKey::RSA.new File.read(key)
|
|
|
|
http.cert = OpenSSL::X509::Certificate.new File.read(crt)
|
|
|
|
http.ca_file = File.expand_path "../../examples/puma/client-certs/ca.crt", __FILE__
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
2015-11-28 21:17:01 -05:00
|
|
|
end
|
2015-01-13 23:11:26 -05:00
|
|
|
|
2015-11-28 21:17:01 -05:00
|
|
|
def test_verify_client_cert
|
|
|
|
return if DISABLE_SSL
|
|
|
|
assert_ssl_client_error_match(nil) do |http|
|
|
|
|
key = File.expand_path "../../examples/puma/client-certs/client.key", __FILE__
|
|
|
|
crt = File.expand_path "../../examples/puma/client-certs/client.crt", __FILE__
|
|
|
|
http.key = OpenSSL::PKey::RSA.new File.read(key)
|
|
|
|
http.cert = OpenSSL::X509::Certificate.new File.read(crt)
|
|
|
|
http.ca_file = File.expand_path "../../examples/puma/client-certs/ca.crt", __FILE__
|
|
|
|
http.verify_mode = OpenSSL::SSL::VERIFY_PEER
|
2015-01-13 23:11:26 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|