2017-05-12 12:16:55 -07:00
|
|
|
require_relative "helper"
|
2011-09-30 10:30:37 -05:00
|
|
|
|
2016-11-22 16:05:49 +01:00
|
|
|
class TestPersistent < Minitest::Test
|
2019-09-11 14:12:57 +00:00
|
|
|
|
|
|
|
HOST = "127.0.0.1"
|
|
|
|
|
2011-09-30 10:30:37 -05:00
|
|
|
def setup
|
2019-09-11 14:12:57 +00:00
|
|
|
@valid_request = "GET / HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n"
|
|
|
|
@close_request = "GET / HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\nConnection: close\r\n\r\n"
|
2011-10-03 14:28:46 -07:00
|
|
|
@http10_request = "GET / HTTP/1.0\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n"
|
2019-09-11 14:12:57 +00:00
|
|
|
@keep_request = "GET / HTTP/1.0\r\nHost: test.com\r\nContent-Type: text/plain\r\nConnection: Keep-Alive\r\n\r\n"
|
2011-10-03 14:04:57 -07:00
|
|
|
|
2019-09-11 14:12:57 +00:00
|
|
|
@valid_post = "POST / HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\nContent-Length: 5\r\n\r\nhello"
|
|
|
|
@valid_no_body = "GET / HTTP/1.1\r\nHost: test.com\r\nX-Status: 204\r\nContent-Type: text/plain\r\n\r\n"
|
2011-11-22 10:45:58 -08:00
|
|
|
|
2011-09-30 10:30:37 -05:00
|
|
|
@headers = { "X-Header" => "Works" }
|
|
|
|
@body = ["Hello"]
|
2011-11-22 11:47:50 -08:00
|
|
|
@inputs = []
|
|
|
|
|
|
|
|
@simple = lambda do |env|
|
|
|
|
@inputs << env['rack.input']
|
2011-12-13 14:45:12 +01:00
|
|
|
status = Integer(env['HTTP_X_STATUS'] || 200)
|
|
|
|
[status, @headers, @body]
|
2011-11-22 11:47:50 -08:00
|
|
|
end
|
|
|
|
|
2022-09-15 14:47:52 +09:00
|
|
|
opts = {max_threads: 1}
|
2022-09-20 21:11:13 -05:00
|
|
|
@server = Puma::Server.new @simple, nil, opts
|
2020-07-22 10:04:36 -05:00
|
|
|
@port = (@server.add_tcp_listener HOST, 0).addr[1]
|
2011-09-30 10:30:37 -05:00
|
|
|
@server.run
|
2020-10-19 10:22:28 -05:00
|
|
|
sleep 0.15 if Puma.jruby?
|
2019-09-11 14:12:57 +00:00
|
|
|
@client = TCPSocket.new HOST, @port
|
2011-09-30 10:30:37 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def teardown
|
|
|
|
@client.close
|
|
|
|
@server.stop(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
def lines(count, s=@client)
|
2019-06-14 14:00:54 -05:00
|
|
|
str = "".dup
|
2016-01-14 20:45:20 -06:00
|
|
|
Timeout.timeout(5) do
|
2020-08-28 09:35:09 -05:00
|
|
|
count.times { str << (s.gets || "") }
|
2011-11-22 10:45:58 -08:00
|
|
|
end
|
2011-09-30 10:30:37 -05:00
|
|
|
str
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_one_with_content_length
|
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-09-30 10:30:37 -05:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_two_back_to_back
|
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-09-30 10:30:37 -05:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-11-22 10:45:58 -08:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_post_then_get
|
|
|
|
@client << @valid_post
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-09-30 10:30:37 -05:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
2011-12-13 14:45:12 +01:00
|
|
|
def test_no_body_then_get
|
|
|
|
@client << @valid_no_body
|
|
|
|
assert_equal "HTTP/1.1 204 No Content\r\nX-Header: Works\r\n\r\n", lines(3)
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
2011-09-30 10:30:37 -05:00
|
|
|
def test_chunked
|
|
|
|
@body << "Chunked"
|
2022-09-13 11:55:07 -05:00
|
|
|
@body = @body.to_enum
|
2011-09-30 10:30:37 -05:00
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
2011-10-20 23:04:24 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nTransfer-Encoding: chunked\r\n\r\n5\r\nHello\r\n7\r\nChunked\r\n0\r\n\r\n", lines(10)
|
2011-09-30 10:30:37 -05:00
|
|
|
end
|
|
|
|
|
2015-07-18 09:01:33 +09:30
|
|
|
def test_chunked_with_empty_part
|
|
|
|
@body << ""
|
|
|
|
@body << "Chunked"
|
2022-09-13 11:55:07 -05:00
|
|
|
@body = @body.to_enum
|
2015-07-18 09:01:33 +09:30
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nTransfer-Encoding: chunked\r\n\r\n5\r\nHello\r\n7\r\nChunked\r\n0\r\n\r\n", lines(10)
|
|
|
|
end
|
|
|
|
|
2011-10-17 23:08:09 -07:00
|
|
|
def test_no_chunked_in_http10
|
|
|
|
@body << "Chunked"
|
2022-09-13 11:55:07 -05:00
|
|
|
@body = @body.to_enum
|
2011-10-17 23:08:09 -07:00
|
|
|
|
|
|
|
@client << @http10_request
|
|
|
|
|
2015-02-19 14:43:17 +01:00
|
|
|
assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\n\r\n", lines(3)
|
2011-10-17 23:08:09 -07:00
|
|
|
assert_equal "HelloChunked", @client.read
|
|
|
|
end
|
|
|
|
|
2011-09-30 11:05:44 -05:00
|
|
|
def test_hex
|
|
|
|
str = "This is longer and will be in hex"
|
|
|
|
@body << str
|
2022-09-13 11:55:07 -05:00
|
|
|
@body = @body.to_enum
|
2011-09-30 11:05:44 -05:00
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
2011-10-20 23:04:24 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nTransfer-Encoding: chunked\r\n\r\n5\r\nHello\r\n#{str.size.to_s(16)}\r\n#{str}\r\n0\r\n\r\n", lines(10)
|
2011-09-30 11:05:44 -05:00
|
|
|
|
|
|
|
end
|
|
|
|
|
2011-10-03 14:44:18 -07:00
|
|
|
def test_client11_close
|
2011-10-03 14:04:57 -07:00
|
|
|
@client << @close_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nConnection: close\r\nContent-Length: #{sz}\r\n\r\n", lines(5)
|
2011-10-03 14:04:57 -07:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
2011-10-03 14:44:18 -07:00
|
|
|
def test_client10_close
|
2011-10-03 14:28:46 -07:00
|
|
|
@client << @http10_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2015-02-19 14:43:17 +01:00
|
|
|
assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-10-03 14:28:46 -07:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_one_with_keep_alive_header
|
|
|
|
@client << @keep_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-11-30 15:07:23 -08:00
|
|
|
assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\nConnection: Keep-Alive\r\nContent-Length: #{sz}\r\n\r\n", lines(5)
|
2011-10-03 14:28:46 -07:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
2011-10-04 21:11:10 -07:00
|
|
|
def test_persistent_timeout
|
2022-09-15 14:47:52 +09:00
|
|
|
@server.instance_variable_set(:@persistent_timeout, 1)
|
2011-10-04 21:11:10 -07:00
|
|
|
@client << @valid_request
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
2011-10-04 21:11:10 -07:00
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
2019-07-27 09:47:19 -07:00
|
|
|
sleep 2
|
2011-10-04 21:11:10 -07:00
|
|
|
|
|
|
|
assert_raises EOFError do
|
|
|
|
@client.read_nonblock(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2011-10-17 13:56:45 -07:00
|
|
|
def test_app_sets_content_length
|
|
|
|
@body = ["hello", " world"]
|
|
|
|
@headers['Content-Length'] = "11"
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: 11\r\n\r\n",
|
|
|
|
lines(4)
|
|
|
|
assert_equal "hello world", @client.read(11)
|
|
|
|
end
|
|
|
|
|
2011-10-22 23:06:27 -07:00
|
|
|
def test_allow_app_to_chunk_itself
|
|
|
|
@headers = {'Transfer-Encoding' => "chunked" }
|
|
|
|
|
|
|
|
@body = ["5\r\nhello\r\n0\r\n\r\n"]
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nTransfer-Encoding: chunked\r\n\r\n5\r\nhello\r\n0\r\n\r\n", lines(7)
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2011-10-17 22:47:35 -07:00
|
|
|
def test_two_requests_in_one_chunk
|
2022-09-15 14:47:52 +09:00
|
|
|
@server.instance_variable_set(:@persistent_timeout, 3)
|
2011-10-17 22:47:35 -07:00
|
|
|
|
|
|
|
req = @valid_request.to_s
|
2017-08-03 11:02:40 +10:00
|
|
|
req += "GET /second HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n"
|
2011-10-17 22:47:35 -07:00
|
|
|
|
|
|
|
@client << req
|
|
|
|
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
end
|
|
|
|
|
2011-11-22 11:47:50 -08:00
|
|
|
def test_second_request_not_in_first_req_body
|
2022-09-15 14:47:52 +09:00
|
|
|
@server.instance_variable_set(:@persistent_timeout, 3)
|
2011-11-22 11:47:50 -08:00
|
|
|
|
|
|
|
req = @valid_request.to_s
|
2017-08-03 11:02:40 +10:00
|
|
|
req += "GET /second HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n"
|
2011-11-22 11:47:50 -08:00
|
|
|
|
|
|
|
@client << req
|
|
|
|
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4)
|
|
|
|
assert_equal "Hello", @client.read(5)
|
|
|
|
|
2011-12-01 14:33:34 -08:00
|
|
|
assert_kind_of Puma::NullIO, @inputs[0]
|
|
|
|
assert_kind_of Puma::NullIO, @inputs[1]
|
2012-07-23 11:24:22 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_keepalive_doesnt_starve_clients
|
|
|
|
sz = @body[0].size.to_s
|
|
|
|
|
|
|
|
@client << @valid_request
|
|
|
|
|
2019-09-11 14:12:57 +00:00
|
|
|
c2 = TCPSocket.new HOST, @port
|
2012-07-23 11:24:22 -07:00
|
|
|
c2 << @valid_request
|
|
|
|
|
|
|
|
out = IO.select([c2], nil, nil, 1)
|
|
|
|
|
|
|
|
assert out, "select returned nil"
|
|
|
|
assert_equal c2, out.first.first
|
2011-11-22 11:47:50 -08:00
|
|
|
|
2012-07-23 11:24:22 -07:00
|
|
|
assert_equal "HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4, c2)
|
|
|
|
assert_equal "Hello", c2.read(5)
|
2019-09-11 14:12:57 +00:00
|
|
|
ensure
|
|
|
|
c2.close
|
2011-11-22 11:47:50 -08:00
|
|
|
end
|
|
|
|
|
2011-09-30 10:30:37 -05:00
|
|
|
end
|