2019-07-16 18:53:28 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-05-12 15:16:55 -04:00
|
|
|
require_relative "helper"
|
2020-07-05 21:57:35 -04:00
|
|
|
require_relative "helpers/ssl" if ::Puma::HAS_SSL
|
2020-07-12 13:00:29 -04:00
|
|
|
require_relative "helpers/tmp_path"
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2016-11-22 10:05:49 -05:00
|
|
|
require "puma/binder"
|
2020-03-07 08:34:35 -05:00
|
|
|
require "puma/events"
|
2020-06-01 03:17:46 -04:00
|
|
|
require "puma/configuration"
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
class TestBinderBase < Minitest::Test
|
2020-07-05 21:57:35 -04:00
|
|
|
include SSLHelper if ::Puma::HAS_SSL
|
2020-07-12 13:00:29 -04:00
|
|
|
include TmpPath
|
2019-10-20 21:35:15 -04:00
|
|
|
|
2016-11-21 09:40:56 -05:00
|
|
|
def setup
|
2022-02-05 12:06:22 -05:00
|
|
|
@log_writer = Puma::LogWriter.strings
|
|
|
|
@binder = Puma::Binder.new(@log_writer)
|
2016-11-21 09:40:56 -05:00
|
|
|
end
|
|
|
|
|
2020-05-15 11:45:06 -04:00
|
|
|
def teardown
|
|
|
|
@binder.ios.reject! { |io| Minitest::Mock === io || io.to_io.closed? }
|
|
|
|
@binder.close
|
|
|
|
@binder.unix_paths.select! { |path| File.exist? path }
|
|
|
|
@binder.close_listeners
|
|
|
|
end
|
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
private
|
|
|
|
|
2019-10-11 13:31:41 -04:00
|
|
|
def ssl_context_for_binder(binder = @binder)
|
|
|
|
binder.ios[0].instance_variable_get(:@ctx)
|
2019-09-11 08:01:24 -04:00
|
|
|
end
|
2019-07-16 18:53:28 -04:00
|
|
|
end
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2022-02-17 18:51:34 -05:00
|
|
|
class TestBinderParallel < TestBinderBase
|
2020-03-07 13:44:52 -05:00
|
|
|
parallelize_me!
|
|
|
|
|
2020-11-10 12:24:39 -05:00
|
|
|
def test_synthesize_binds_from_activated_fds_no_sockets
|
|
|
|
binds = ['tcp://0.0.0.0:3000']
|
|
|
|
result = @binder.synthesize_binds_from_activated_fs(binds, true)
|
|
|
|
|
|
|
|
assert_equal ['tcp://0.0.0.0:3000'], result
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_synthesize_binds_from_activated_fds_non_matching_together
|
|
|
|
binds = ['tcp://0.0.0.0:3000']
|
|
|
|
sockets = {['tcp', '0.0.0.0', '5000'] => nil}
|
|
|
|
@binder.instance_variable_set(:@activated_sockets, sockets)
|
|
|
|
result = @binder.synthesize_binds_from_activated_fs(binds, false)
|
|
|
|
|
|
|
|
assert_equal ['tcp://0.0.0.0:3000', 'tcp://0.0.0.0:5000'], result
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_synthesize_binds_from_activated_fds_non_matching_only
|
|
|
|
binds = ['tcp://0.0.0.0:3000']
|
|
|
|
sockets = {['tcp', '0.0.0.0', '5000'] => nil}
|
|
|
|
@binder.instance_variable_set(:@activated_sockets, sockets)
|
|
|
|
result = @binder.synthesize_binds_from_activated_fs(binds, true)
|
|
|
|
|
|
|
|
assert_equal ['tcp://0.0.0.0:5000'], result
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_synthesize_binds_from_activated_fds_complex_binds
|
|
|
|
binds = [
|
|
|
|
'tcp://0.0.0.0:3000',
|
|
|
|
'ssl://192.0.2.100:5000',
|
|
|
|
'ssl://192.0.2.101:5000?no_tlsv1=true',
|
|
|
|
'unix:///run/puma.sock'
|
|
|
|
]
|
|
|
|
sockets = {
|
|
|
|
['tcp', '0.0.0.0', '5000'] => nil,
|
|
|
|
['tcp', '192.0.2.100', '5000'] => nil,
|
|
|
|
['tcp', '192.0.2.101', '5000'] => nil,
|
|
|
|
['unix', '/run/puma.sock'] => nil
|
|
|
|
}
|
|
|
|
@binder.instance_variable_set(:@activated_sockets, sockets)
|
|
|
|
result = @binder.synthesize_binds_from_activated_fs(binds, false)
|
|
|
|
|
|
|
|
expected = ['tcp://0.0.0.0:3000', 'ssl://192.0.2.100:5000', 'ssl://192.0.2.101:5000?no_tlsv1=true', 'unix:///run/puma.sock', 'tcp://0.0.0.0:5000']
|
|
|
|
assert_equal expected, result
|
|
|
|
end
|
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
def test_localhost_addresses_dont_alter_listeners_for_tcp_addresses
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://localhost:0"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
assert_empty @binder.listeners
|
2019-10-11 13:31:41 -04:00
|
|
|
end
|
|
|
|
|
2020-03-07 13:44:52 -05:00
|
|
|
def test_home_alters_listeners_for_tcp_addresses
|
|
|
|
port = UniquePort.call
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:#{port}"], @log_writer
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
assert_equal "tcp://127.0.0.1:#{port}", @binder.listeners[0][0]
|
|
|
|
assert_kind_of TCPServer, @binder.listeners[0][1]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_connected_ports
|
|
|
|
ports = (1..3).map { |_| UniquePort.call }
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse(ports.map { |p| "tcp://localhost:#{p}" }, @log_writer)
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
assert_equal ports, @binder.connected_ports
|
|
|
|
end
|
|
|
|
|
2019-10-11 13:31:41 -04:00
|
|
|
def test_localhost_addresses_dont_alter_listeners_for_ssl_addresses
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://localhost:0?#{ssl_query}"], @log_writer
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
assert_empty @binder.listeners
|
2016-11-21 09:40:56 -05:00
|
|
|
end
|
2019-09-11 08:01:24 -04:00
|
|
|
|
2020-03-07 13:44:52 -05:00
|
|
|
def test_home_alters_listeners_for_ssl_addresses
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-03-07 13:44:52 -05:00
|
|
|
port = UniquePort.call
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://127.0.0.1:#{port}?#{ssl_query}"], @log_writer
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
assert_equal "ssl://127.0.0.1:#{port}?#{ssl_query}", @binder.listeners[0][0]
|
|
|
|
assert_kind_of TCPServer, @binder.listeners[0][1]
|
|
|
|
end
|
|
|
|
|
2019-09-11 08:01:24 -04:00
|
|
|
def test_correct_zero_port
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://localhost:0"], @log_writer
|
2019-09-11 08:01:24 -04:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
m = %r!http://127.0.0.1:(\d+)!.match(@log_writer.stdout.string)
|
2019-09-11 08:01:24 -04:00
|
|
|
port = m[1].to_i
|
|
|
|
|
|
|
|
refute_equal 0, port
|
|
|
|
end
|
|
|
|
|
2019-09-26 06:54:15 -04:00
|
|
|
def test_correct_zero_port_ssl
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-07-05 21:57:35 -04:00
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
ssl_regex = %r!ssl://127.0.0.1:(\d+)!
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://localhost:0?#{ssl_query}"], @log_writer
|
2019-09-26 06:54:15 -04:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
port = ssl_regex.match(@log_writer.stdout.string)[1].to_i
|
2019-09-26 06:54:15 -04:00
|
|
|
|
|
|
|
refute_equal 0, port
|
|
|
|
end
|
2019-09-26 06:55:23 -04:00
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
def test_logs_all_localhost_bindings
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://localhost:0"], @log_writer
|
2020-03-07 08:34:35 -05:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match %r!http://127.0.0.1:(\d+)!, @log_writer.stdout.string
|
2020-03-07 13:44:52 -05:00
|
|
|
if Socket.ip_address_list.any? {|i| i.ipv6_loopback? }
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match %r!http://\[::1\]:(\d+)!, @log_writer.stdout.string
|
2020-03-07 08:34:35 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
def test_logs_all_localhost_bindings_ssl
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-07-05 21:57:35 -04:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://localhost:0?#{ssl_query}"], @log_writer
|
2020-03-07 08:34:35 -05:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match %r!ssl://127.0.0.1:(\d+)!, @log_writer.stdout.string
|
2020-03-07 13:44:52 -05:00
|
|
|
if Socket.ip_address_list.any? {|i| i.ipv6_loopback? }
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match %r!ssl://\[::1\]:(\d+)!, @log_writer.stdout.string
|
2020-03-07 08:34:35 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-10-01 05:33:00 -04:00
|
|
|
def test_allows_both_ssl_and_tcp
|
|
|
|
assert_parsing_logs_uri [:ssl, :tcp]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_allows_both_unix_and_tcp
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_if :jruby # Undiagnosed thread race. TODO fix
|
2019-10-01 05:33:00 -04:00
|
|
|
assert_parsing_logs_uri [:unix, :tcp]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_allows_both_tcp_and_unix
|
|
|
|
assert_parsing_logs_uri [:tcp, :unix]
|
|
|
|
end
|
|
|
|
|
2019-10-01 10:01:45 -04:00
|
|
|
def test_pre_existing_unix
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :unix
|
2019-10-01 10:01:45 -04:00
|
|
|
|
2020-07-06 08:10:55 -04:00
|
|
|
unix_path = tmp_path('.sock')
|
2019-10-01 12:28:33 -04:00
|
|
|
File.open(unix_path, mode: 'wb') { |f| f.puts 'pre existing' }
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["unix://#{unix_path}"], @log_writer
|
2019-10-01 10:01:45 -04:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match %r!unix://#{unix_path}!, @log_writer.stdout.string
|
2019-10-01 10:01:45 -04:00
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
refute_includes @binder.unix_paths, unix_path
|
2019-10-01 10:01:45 -04:00
|
|
|
|
2020-03-03 15:09:08 -05:00
|
|
|
@binder.close_listeners
|
2019-10-01 12:28:33 -04:00
|
|
|
|
|
|
|
assert File.exist?(unix_path)
|
|
|
|
|
2019-10-01 10:01:45 -04:00
|
|
|
ensure
|
|
|
|
if UNIX_SKT_EXIST
|
|
|
|
File.unlink unix_path if File.exist? unix_path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-05-20 12:17:34 -04:00
|
|
|
def test_binder_parses_nil_low_latency
|
|
|
|
skip_if :jruby
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://0.0.0.0:0?low_latency"], @log_writer
|
2021-05-20 12:17:34 -04:00
|
|
|
|
|
|
|
socket = @binder.listeners.first.last
|
|
|
|
|
|
|
|
assert socket.getsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY).bool
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_true_low_latency
|
|
|
|
skip_if :jruby
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://0.0.0.0:0?low_latency=true"], @log_writer
|
2021-05-20 12:17:34 -04:00
|
|
|
|
|
|
|
socket = @binder.listeners.first.last
|
|
|
|
|
|
|
|
assert socket.getsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY).bool
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_false_low_latency
|
|
|
|
skip_if :jruby
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://0.0.0.0:0?low_latency=false"], @log_writer
|
2021-05-20 12:17:34 -04:00
|
|
|
|
|
|
|
socket = @binder.listeners.first.last
|
|
|
|
|
|
|
|
refute socket.getsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY).bool
|
|
|
|
end
|
|
|
|
|
2019-10-11 13:31:41 -04:00
|
|
|
def test_binder_parses_tlsv1_disabled
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:0?#{ssl_query}&no_tlsv1=true"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
|
|
|
assert ssl_context_for_binder.no_tlsv1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_tlsv1_enabled
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:0?#{ssl_query}&no_tlsv1=false"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
|
|
|
refute ssl_context_for_binder.no_tlsv1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_tlsv1_tlsv1_1_unspecified_defaults_to_enabled
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:0?#{ssl_query}"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
|
|
|
refute ssl_context_for_binder.no_tlsv1
|
|
|
|
refute ssl_context_for_binder.no_tlsv1_1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_tlsv1_1_disabled
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:0?#{ssl_query}&no_tlsv1_1=true"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
|
|
|
assert ssl_context_for_binder.no_tlsv1_1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_tlsv1_1_enabled
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:0?#{ssl_query}&no_tlsv1_1=false"], @log_writer
|
2019-10-11 13:31:41 -04:00
|
|
|
|
|
|
|
refute ssl_context_for_binder.no_tlsv1_1
|
|
|
|
end
|
|
|
|
|
2020-03-07 08:34:35 -05:00
|
|
|
def test_env_contains_protoenv
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://localhost:0?#{ssl_query}"], @log_writer
|
2020-03-07 08:34:35 -05:00
|
|
|
|
|
|
|
env_hash = @binder.envs[@binder.ios.first]
|
|
|
|
|
|
|
|
@binder.proto_env.each do |k,v|
|
2021-10-31 09:59:21 -04:00
|
|
|
assert env_hash[k] == v
|
2020-03-07 08:34:35 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_env_contains_stderr
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://localhost:0?#{ssl_query}"], @log_writer
|
2020-03-07 08:34:35 -05:00
|
|
|
|
|
|
|
env_hash = @binder.envs[@binder.ios.first]
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_equal @log_writer.stderr, env_hash["rack.errors"]
|
2020-03-07 08:34:35 -05:00
|
|
|
end
|
|
|
|
|
2020-03-07 13:44:52 -05:00
|
|
|
def test_close_calls_close_on_ios
|
|
|
|
@mocked_ios = [Minitest::Mock.new, Minitest::Mock.new]
|
|
|
|
@mocked_ios.each { |m| m.expect(:close, true) }
|
|
|
|
@binder.ios = @mocked_ios
|
|
|
|
|
|
|
|
@binder.close
|
|
|
|
|
2020-03-09 12:12:19 -04:00
|
|
|
assert @mocked_ios.map(&:verify).all?
|
2020-03-07 13:44:52 -05:00
|
|
|
end
|
|
|
|
|
2020-03-08 12:06:42 -04:00
|
|
|
def test_redirects_for_restart_creates_a_hash
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:0"], @log_writer
|
2020-03-08 12:06:42 -04:00
|
|
|
|
|
|
|
result = @binder.redirects_for_restart
|
|
|
|
ios = @binder.listeners.map { |_l, io| io.to_i }
|
|
|
|
|
|
|
|
ios.each { |int| assert_equal int, result[int] }
|
|
|
|
assert result[:close_others]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_redirects_for_restart_env
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:0"], @log_writer
|
2020-03-08 12:06:42 -04:00
|
|
|
|
|
|
|
result = @binder.redirects_for_restart_env
|
|
|
|
|
|
|
|
@binder.listeners.each_with_index do |l, i|
|
|
|
|
assert_equal "#{l[1].to_i}:#{l[0]}", result["PUMA_INHERIT_#{i}"]
|
|
|
|
end
|
|
|
|
end
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
def test_close_listeners_closes_ios
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:#{UniquePort.call}"], @log_writer
|
2020-03-07 13:44:52 -05:00
|
|
|
|
2021-10-31 09:59:21 -04:00
|
|
|
refute @binder.listeners.any? { |_l, io| io.closed? }
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
@binder.close_listeners
|
|
|
|
|
2021-10-31 09:59:21 -04:00
|
|
|
assert @binder.listeners.all? { |_l, io| io.closed? }
|
2020-03-07 13:44:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_close_listeners_closes_ios_unless_closed?
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:0"], @log_writer
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
bomb = @binder.listeners.first[1]
|
|
|
|
bomb.close
|
|
|
|
def bomb.close; raise "Boom!"; end # the bomb has been planted
|
|
|
|
|
2021-10-31 09:59:21 -04:00
|
|
|
assert @binder.listeners.any? { |_l, io| io.closed? }
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
@binder.close_listeners
|
|
|
|
|
2021-10-31 09:59:21 -04:00
|
|
|
assert @binder.listeners.all? { |_l, io| io.closed? }
|
2020-03-07 13:44:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_listeners_file_unlink_if_unix_listener
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :unix
|
2020-03-07 13:44:52 -05:00
|
|
|
|
2020-07-06 08:10:55 -04:00
|
|
|
unix_path = tmp_path('.sock')
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["unix://#{unix_path}"], @log_writer
|
2020-07-06 08:10:55 -04:00
|
|
|
assert File.socket?(unix_path)
|
2020-03-07 13:44:52 -05:00
|
|
|
|
|
|
|
@binder.close_listeners
|
2020-07-06 08:10:55 -04:00
|
|
|
refute File.socket?(unix_path)
|
2020-03-07 13:44:52 -05:00
|
|
|
end
|
|
|
|
|
2020-03-08 12:06:42 -04:00
|
|
|
def test_import_from_env_listen_inherit
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["tcp://127.0.0.1:0"], @log_writer
|
2020-03-11 14:50:44 -04:00
|
|
|
removals = @binder.create_inherited_fds(@binder.redirects_for_restart_env)
|
2020-03-08 12:06:42 -04:00
|
|
|
|
2021-10-31 09:59:21 -04:00
|
|
|
@binder.listeners.each do |l, io|
|
|
|
|
assert_equal io.to_i, @binder.inherited_fds[l]
|
2020-03-08 12:06:42 -04:00
|
|
|
end
|
|
|
|
assert_includes removals, "PUMA_INHERIT_0"
|
|
|
|
end
|
|
|
|
|
2020-03-11 14:50:44 -04:00
|
|
|
# Socket activation tests. We have to skip all of these on non-UNIX platforms
|
|
|
|
# because the check that we do in the code only works if you support UNIX sockets.
|
|
|
|
# This is OK, because systemd obviously only works on Linux.
|
|
|
|
def test_socket_activation_tcp
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :unix
|
2020-03-11 14:50:44 -04:00
|
|
|
url = "127.0.0.1"
|
|
|
|
port = UniquePort.call
|
|
|
|
sock = Addrinfo.tcp(url, port).listen
|
|
|
|
assert_activates_sockets(url: url, port: port, sock: sock)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_socket_activation_tcp_ipv6
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :unix
|
2020-03-11 14:50:44 -04:00
|
|
|
url = "::"
|
|
|
|
port = UniquePort.call
|
|
|
|
sock = Addrinfo.tcp(url, port).listen
|
|
|
|
assert_activates_sockets(url: url, port: port, sock: sock)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_socket_activation_unix
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_if :jruby # Failing with what I think is a JRuby bug
|
|
|
|
skip_unless :unix
|
2020-07-06 08:10:55 -04:00
|
|
|
|
|
|
|
state_path = tmp_path('.state')
|
|
|
|
sock = Addrinfo.unix(state_path).listen
|
|
|
|
assert_activates_sockets(path: state_path, sock: sock)
|
2020-03-11 14:50:44 -04:00
|
|
|
ensure
|
2020-07-06 08:10:55 -04:00
|
|
|
File.unlink(state_path) rescue nil # JRuby race?
|
2020-03-11 14:50:44 -04:00
|
|
|
end
|
2020-03-08 12:06:42 -04:00
|
|
|
|
2020-06-01 03:17:46 -04:00
|
|
|
def test_rack_multithread_default_configuration
|
2022-02-05 12:06:22 -05:00
|
|
|
binder = Puma::Binder.new(@log_writer)
|
2020-06-01 03:17:46 -04:00
|
|
|
|
|
|
|
assert binder.proto_env["rack.multithread"]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_rack_multithread_custom_configuration
|
|
|
|
conf = Puma::Configuration.new(max_threads: 1)
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
binder = Puma::Binder.new(@log_writer, conf)
|
2020-06-01 03:17:46 -04:00
|
|
|
|
|
|
|
refute binder.proto_env["rack.multithread"]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_rack_multiprocess_default_configuration
|
2022-02-05 12:06:22 -05:00
|
|
|
binder = Puma::Binder.new(@log_writer)
|
2020-06-01 03:17:46 -04:00
|
|
|
|
|
|
|
refute binder.proto_env["rack.multiprocess"]
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_rack_multiprocess_custom_configuration
|
|
|
|
conf = Puma::Configuration.new(workers: 1)
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
binder = Puma::Binder.new(@log_writer, conf)
|
2020-06-01 03:17:46 -04:00
|
|
|
|
|
|
|
assert binder.proto_env["rack.multiprocess"]
|
|
|
|
end
|
|
|
|
|
2019-10-01 05:33:00 -04:00
|
|
|
private
|
|
|
|
|
2020-03-11 14:50:44 -04:00
|
|
|
def assert_activates_sockets(path: nil, port: nil, url: nil, sock: nil)
|
|
|
|
hash = { "LISTEN_FDS" => 1, "LISTEN_PID" => $$ }
|
2022-02-05 12:06:22 -05:00
|
|
|
@log_writer.instance_variable_set(:@debug, true)
|
2020-03-11 14:50:44 -04:00
|
|
|
|
|
|
|
@binder.instance_variable_set(:@sock_fd, sock.fileno)
|
|
|
|
def @binder.socket_activation_fd(int); @sock_fd; end
|
|
|
|
@result = @binder.create_activated_fds(hash)
|
|
|
|
|
|
|
|
url = "[::]" if url == "::"
|
|
|
|
ary = path ? [:unix, path] : [:tcp, url, port]
|
|
|
|
|
|
|
|
assert_kind_of TCPServer, @binder.activated_sockets[ary]
|
2022-02-05 12:06:22 -05:00
|
|
|
assert_match "Registered #{ary.join(":")} for activation from LISTEN_FDS", @log_writer.stdout.string
|
2020-03-11 14:50:44 -04:00
|
|
|
assert_equal ["LISTEN_FDS", "LISTEN_PID"], @result
|
|
|
|
end
|
|
|
|
|
2019-10-01 05:33:00 -04:00
|
|
|
def assert_parsing_logs_uri(order = [:unix, :tcp])
|
2021-03-15 10:10:43 -04:00
|
|
|
skip MSG_UNIX if order.include?(:unix) && !UNIX_SKT_EXIST
|
|
|
|
skip_unless :ssl
|
2019-10-01 05:33:00 -04:00
|
|
|
|
2020-07-06 08:10:55 -04:00
|
|
|
unix_path = tmp_path('.sock')
|
2019-10-01 05:33:00 -04:00
|
|
|
prepared_paths = {
|
2019-10-11 13:31:41 -04:00
|
|
|
ssl: "ssl://127.0.0.1:#{UniquePort.call}?#{ssl_query}",
|
2019-10-01 05:33:00 -04:00
|
|
|
tcp: "tcp://127.0.0.1:#{UniquePort.call}",
|
2020-07-06 08:10:55 -04:00
|
|
|
unix: "unix://#{unix_path}"
|
2019-10-01 05:33:00 -04:00
|
|
|
}
|
|
|
|
|
2020-06-30 15:08:16 -04:00
|
|
|
expected_logs = prepared_paths.dup.tap do |logs|
|
|
|
|
logs[:tcp] = logs[:tcp].gsub('tcp://', 'http://')
|
|
|
|
end
|
|
|
|
|
2019-10-01 05:33:00 -04:00
|
|
|
tested_paths = [prepared_paths[order[0]], prepared_paths[order[1]]]
|
2019-09-26 06:55:23 -04:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse tested_paths, @log_writer
|
|
|
|
stdout = @log_writer.stdout.string
|
2019-09-26 06:55:23 -04:00
|
|
|
|
2020-03-07 08:42:38 -05:00
|
|
|
order.each do |prot|
|
2020-06-30 15:08:16 -04:00
|
|
|
assert_match expected_logs[prot], stdout
|
2020-03-07 08:42:38 -05:00
|
|
|
end
|
2019-10-01 05:33:00 -04:00
|
|
|
ensure
|
2020-03-03 15:09:08 -05:00
|
|
|
@binder.close_listeners if order.include?(:unix) && UNIX_SKT_EXIST
|
2019-09-26 06:55:23 -04:00
|
|
|
end
|
2019-07-16 18:53:28 -04:00
|
|
|
end
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2022-03-30 20:33:03 -04:00
|
|
|
class TestBinderSingle < TestBinderBase
|
2022-02-17 18:51:34 -05:00
|
|
|
def test_ssl_binder_sets_backlog
|
|
|
|
skip_unless :ssl
|
|
|
|
|
|
|
|
host = '127.0.0.1'
|
|
|
|
port = UniquePort.call
|
|
|
|
tcp_server = TCPServer.new(host, port)
|
|
|
|
tcp_server.define_singleton_method(:listen) do |backlog|
|
|
|
|
Thread.current[:backlog] = backlog
|
|
|
|
super(backlog)
|
|
|
|
end
|
|
|
|
|
|
|
|
TCPServer.stub(:new, tcp_server) do
|
|
|
|
@binder.parse ["ssl://#{host}:#{port}?#{ssl_query}&backlog=2048"], @log_writer
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal 2048, Thread.current[:backlog]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
class TestBinderJRuby < TestBinderBase
|
|
|
|
def test_binder_parses_jruby_ssl_options
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-07-05 21:57:35 -04:00
|
|
|
|
2022-07-04 10:16:31 -04:00
|
|
|
cipher_suites = ['TLS_DHE_RSA_WITH_AES_128_CBC_SHA', 'TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256']
|
|
|
|
|
|
|
|
@binder.parse ["ssl://0.0.0.0:8080?#{ssl_query}"], @log_writer
|
|
|
|
|
|
|
|
assert_equal @keystore, ssl_context_for_binder.keystore
|
|
|
|
assert_equal cipher_suites, ssl_context_for_binder.cipher_suites
|
|
|
|
assert_equal cipher_suites, ssl_context_for_binder.ssl_cipher_list
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_jruby_ssl_protocols_and_cipher_suites_options
|
|
|
|
skip_unless :ssl
|
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
keystore = File.expand_path "../../examples/puma/keystore.jks", __FILE__
|
2022-07-04 10:16:31 -04:00
|
|
|
cipher = "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256"
|
|
|
|
ssl_query = "keystore=#{keystore}&keystore-pass=jruby_puma&cipher_suites=#{cipher}&protocols=TLSv1.3,TLSv1.2"
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0:8080?#{ssl_query}"], @log_writer
|
2016-11-21 09:40:56 -05:00
|
|
|
|
2022-07-04 10:16:31 -04:00
|
|
|
assert_equal [ 'TLSv1.3', 'TLSv1.2' ], ssl_context_for_binder.protocols
|
|
|
|
assert_equal [ cipher ], ssl_context_for_binder.cipher_suites
|
2016-11-21 09:40:56 -05:00
|
|
|
end
|
2019-10-11 13:31:41 -04:00
|
|
|
end if ::Puma::IS_JRUBY
|
2017-12-01 13:56:48 -05:00
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
class TestBinderMRI < TestBinderBase
|
|
|
|
def test_binder_parses_ssl_cipher_filter
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-07-05 21:57:35 -04:00
|
|
|
|
2019-07-16 18:53:28 -04:00
|
|
|
ssl_cipher_filter = "AES@STRENGTH"
|
2017-12-01 14:06:33 -05:00
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0?#{ssl_query}&ssl_cipher_filter=#{ssl_cipher_filter}"], @log_writer
|
2019-08-03 15:56:30 -04:00
|
|
|
|
2019-10-11 13:31:41 -04:00
|
|
|
assert_equal ssl_cipher_filter, ssl_context_for_binder.ssl_cipher_filter
|
2018-04-17 09:25:22 -04:00
|
|
|
end
|
2020-12-07 18:21:23 -05:00
|
|
|
|
|
|
|
def test_binder_parses_ssl_verification_flags_one
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-12-07 18:21:23 -05:00
|
|
|
|
|
|
|
input = "&verification_flags=TRUSTED_FIRST"
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0?#{ssl_query}#{input}"], @log_writer
|
2020-12-07 18:21:23 -05:00
|
|
|
|
|
|
|
assert_equal 0x8000, ssl_context_for_binder.verification_flags
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binder_parses_ssl_verification_flags_multiple
|
2021-03-15 10:10:43 -04:00
|
|
|
skip_unless :ssl
|
2020-12-07 18:21:23 -05:00
|
|
|
|
|
|
|
input = "&verification_flags=TRUSTED_FIRST,NO_CHECK_TIME"
|
|
|
|
|
2022-02-05 12:06:22 -05:00
|
|
|
@binder.parse ["ssl://0.0.0.0?#{ssl_query}#{input}"], @log_writer
|
2020-12-07 18:21:23 -05:00
|
|
|
|
|
|
|
assert_equal 0x8000 | 0x200000, ssl_context_for_binder.verification_flags
|
|
|
|
end
|
2019-10-11 13:31:41 -04:00
|
|
|
end unless ::Puma::IS_JRUBY
|