require 'test_helper' require 'timeout' require 'socket' class TestPersistent < MiniTest::Unit::TestCase def setup @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" @http10_request = "GET / HTTP/1.0\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n" @keep_request = "GET / HTTP/1.0\r\nHost: test.com\r\nContent-Type: text/plain\r\nConnection: Keep-Alive\r\n\r\n" @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" @headers = { "X-Header" => "Works" } @body = ["Hello"] @inputs = [] @simple = lambda do |env| @inputs << env['rack.input'] status = Integer(env['HTTP_X_STATUS'] || 200) [status, @headers, @body] end @host = "127.0.0.1" @port = 3215 @server = Jubilee::Server.new @simple @server.start sleep 0.1 @client = TCPSocket.new @host, @port end def teardown @client.close sleep 0.1 # in case server shutdown before request is submitted @server.stop end def lines(count, s=@client) str = "" timeout(5) do count.times { str << s.gets } end str end def valid_response(size) Regexp.new("HTTP/1.1 200 OK\r\nX-Header: Works\r\nContent-Length: #{size}\r\n\r\n", true) end def test_one_with_content_length @client << @valid_request sz = @body[0].size.to_s assert_match valid_response(sz), lines(4) assert_equal "Hello", @client.read(5) end def test_two_back_to_back @client << @valid_request sz = @body[0].size.to_s assert_match valid_response(sz), lines(4) assert_equal "Hello", @client.read(5) @client << @valid_request sz = @body[0].size.to_s assert_match valid_response(sz), lines(4) assert_equal "Hello", @client.read(5) end def test_post_then_get @client << @valid_post sz = @body[0].size.to_s assert_match valid_response(sz), lines(4) assert_equal "Hello", @client.read(5) @client << @valid_request sz = @body[0].size.to_s assert_match valid_response(sz), lines(4) assert_equal "Hello", @client.read(5) end #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 def test_chunked @body << "Chunked" @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 def test_no_chunked_in_http10 @body << "Chunked" @client << @http10_request assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\n\r\n", lines(3) assert_equal "HelloChunked", @client.read end def test_hex str = "This is longer and will be in hex" @body << str @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\n#{str.size.to_s(16)}\r\n#{str}\r\n0\r\n\r\n", lines(10) end =begin def test_client11_close @client << @close_request sz = @body[0].size.to_s assert_equal "HTTP/1.1 200 OK\r\nConnection: Close\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(5) assert_equal "Hello", @client.read(5) end =end def test_client10_close @client << @http10_request sz = @body[0].size.to_s assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\n\r\n", lines(4) assert_equal "Hello", @client.read(5) end def test_one_with_keep_alive_header @client << @keep_request sz = @body[0].size.to_s assert_equal "HTTP/1.0 200 OK\r\nX-Header: Works\r\nContent-Length: #{sz}\r\nConnection: Keep-Alive\r\n\r\n", lines(5) assert_equal "Hello", @client.read(5) end =begin def test_persistent_timeout @server.persistent_timeout = 2 @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) sleep 3 assert_raises EOFError do @client.read_nonblock(1) end end def test_app_sets_content_length @body = ["hello", " world"] @headers['Content-Length'] = "11" @client << @valid_request assert_equal "HTTP/1.1 200 OK\r\ncontent-length: 11\r\nx-header: Works\r\n\r\n", lines(4) assert_equal "hello world", @client.read(11) end def test_allow_app_to_chunk_itself skip "vertx doesn't support chunk self yet" @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 def test_two_requests_in_one_chunk @server.persistent_timeout = 3 req = @valid_request.to_s req << "GET /second HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n" @client << req sz = @body[0].size.to_s assert_equal "HTTP/1.1 200 OK\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(4) assert_equal "Hello", @client.read(5) assert_equal "HTTP/1.1 200 OK\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(4) assert_equal "Hello", @client.read(5) end def test_second_request_not_in_first_req_body @server.persistent_timeout = 3 req = @valid_request.to_s req << "GET /second HTTP/1.1\r\nHost: test.com\r\nContent-Type: text/plain\r\n\r\n" @client << req sz = @body[0].size.to_s assert_equal "HTTP/1.1 200 OK\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(4) assert_equal "Hello", @client.read(5) assert_equal "HTTP/1.1 200 OK\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(4) assert_equal "Hello", @client.read(5) # Since rack process request before the body is ready, we cannot # utilize this optimization # assert_kind_of Jubilee::NullIO, @inputs[0] # assert_kind_of Jubilee::NullIO, @inputs[1] end def test_keepalive_doesnt_starve_clients sz = @body[0].size.to_s @client << @valid_request c2 = TCPSocket.new @host, @port c2 << @valid_request out = IO.select([c2], nil, nil, 1) assert out, "select returned nil" assert_equal c2, out.first.first assert_equal "HTTP/1.1 200 OK\r\ncontent-length: #{sz}\r\nx-header: Works\r\n\r\n", lines(4, c2) assert_equal "Hello", c2.read(5) end =end end