Files

755 lines
22 KiB
C++
Raw Permalink Normal View History

2017-07-20 08:01:46 -07:00
//
2019-02-21 07:00:31 -08:00
// Copyright (c) 2016-2019 Vinnie Falco (vinnie dot falco at gmail dot com)
2017-07-20 08:01:46 -07:00
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//
2017-07-20 13:40:34 -07:00
// Official repository: https://github.com/boostorg/beast
//
2017-07-20 08:01:46 -07:00
// Test that header file is self-contained.
2017-07-20 13:40:34 -07:00
#include <boost/beast/http/read.hpp>
#include "test_parser.hpp"
2017-07-20 13:40:34 -07:00
#include <boost/beast/core/ostream.hpp>
#include <boost/beast/core/flat_static_buffer.hpp>
#include <boost/beast/http/fields.hpp>
#include <boost/beast/http/dynamic_body.hpp>
#include <boost/beast/http/parser.hpp>
#include <boost/beast/http/string_body.hpp>
2018-11-11 14:07:55 -08:00
#include <boost/beast/_experimental/test/stream.hpp>
#include <boost/beast/_experimental/unit_test/suite.hpp>
#include <boost/beast/test/yield_to.hpp>
#include <boost/asio/io_context.hpp>
#include <boost/asio/ip/tcp.hpp>
#include <boost/asio/strand.hpp>
#include <boost/asio/write.hpp>
#include <atomic>
#if BOOST_ASIO_HAS_CO_AWAIT
#include <boost/asio/use_awaitable.hpp>
#endif
2017-07-20 13:40:34 -07:00
namespace boost {
namespace beast {
namespace http {
class read_test
: public beast::unit_test::suite
, public test::enable_yield_to
{
public:
template<bool isRequest>
void
failMatrix(char const* s, yield_context do_yield)
{
static std::size_t constexpr limit = 100;
std::size_t n;
auto const len = strlen(s);
for(n = 0; n < limit; ++n)
{
multi_buffer b;
b.commit(net::buffer_copy(
b.prepare(len), net::buffer(s, len)));
2018-05-01 14:19:35 -07:00
test::fail_count fc(n);
test::stream ts{ioc_, fc};
test_parser<isRequest> p(fc);
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
read(ts, b, p, ec);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
static std::size_t constexpr pre = 10;
multi_buffer b;
b.commit(net::buffer_copy(
b.prepare(pre), net::buffer(s, pre)));
2018-05-01 14:19:35 -07:00
test::fail_count fc(n);
test::stream ts{ioc_, fc,
2017-08-15 22:20:13 -07:00
std::string(s + pre, len - pre)};
test_parser<isRequest> p(fc);
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
read(ts, b, p, ec);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
multi_buffer b;
b.commit(net::buffer_copy(
b.prepare(len), net::buffer(s, len)));
2018-05-01 14:19:35 -07:00
test::fail_count fc(n);
test::stream ts{ioc_, fc};
test_parser<isRequest> p(fc);
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
async_read(ts, b, p, do_yield[ec]);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
multi_buffer b;
b.commit(net::buffer_copy(
b.prepare(len), net::buffer(s, len)));
2018-05-01 14:19:35 -07:00
test::fail_count fc(n);
test::stream ts{ioc_, fc};
test_parser<isRequest> p(fc);
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
ts.close_remote();
async_read_header(ts, b, p, do_yield[ec]);
if(! ec)
break;
}
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
static std::size_t constexpr pre = 10;
multi_buffer b;
b.commit(net::buffer_copy(
b.prepare(pre), net::buffer(s, pre)));
2018-05-01 14:19:35 -07:00
test::fail_count fc(n);
test::stream ts(ioc_, fc,
2017-08-15 22:20:13 -07:00
std::string{s + pre, len - pre});
test_parser<isRequest> p(fc);
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
async_read(ts, b, p, do_yield[ec]);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
}
void testThrow()
{
try
{
multi_buffer b;
test::stream c{ioc_, "GET / X"};
2017-08-21 14:45:58 -07:00
c.close_remote();
request_parser<dynamic_body> p;
2017-07-29 17:47:04 -07:00
read(c, b, p);
fail();
}
catch(std::exception const&)
{
pass();
}
}
void
testBufferOverflow()
{
{
test::stream c{ioc_};
2017-07-29 17:47:04 -07:00
ostream(c.buffer()) <<
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n"
"10\r\n"
"****************\r\n"
"0\r\n\r\n";
flat_static_buffer<1024> b;
request<string_body> req;
try
{
2017-07-29 17:47:04 -07:00
read(c, b, req);
pass();
}
catch(std::exception const& e)
{
fail(e.what(), __FILE__, __LINE__);
}
}
{
test::stream c{ioc_};
2017-07-29 17:47:04 -07:00
ostream(c.buffer()) <<
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n"
"10\r\n"
"****************\r\n"
"0\r\n\r\n";
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
flat_static_buffer<10> b;
request<string_body> req;
2017-07-29 17:47:04 -07:00
read(c, b, req, ec);
2017-07-25 12:35:54 -07:00
BEAST_EXPECTS(ec == error::buffer_overflow,
ec.message());
}
}
void testFailures(yield_context do_yield)
{
char const* req[] = {
"GET / HTTP/1.0\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Empty:\r\n"
"\r\n"
,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Content-Length: 2\r\n"
"\r\n"
"**"
,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n"
"10\r\n"
"****************\r\n"
"0\r\n\r\n"
,
nullptr
};
char const* res[] = {
"HTTP/1.0 200 OK\r\n"
"Server: test\r\n"
"\r\n"
,
"HTTP/1.0 200 OK\r\n"
"Server: test\r\n"
"\r\n"
"***"
,
"HTTP/1.1 200 OK\r\n"
"Server: test\r\n"
"Content-Length: 3\r\n"
"\r\n"
"***"
,
"HTTP/1.1 200 OK\r\n"
"Server: test\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n"
"10\r\n"
"****************\r\n"
"0\r\n\r\n"
,
nullptr
};
for(std::size_t i = 0; req[i]; ++i)
failMatrix<true>(req[i], do_yield);
for(std::size_t i = 0; res[i]; ++i)
failMatrix<false>(res[i], do_yield);
}
void testRead(yield_context do_yield)
{
static std::size_t constexpr limit = 100;
std::size_t n;
for(n = 0; n < limit; ++n)
{
2018-05-01 14:19:35 -07:00
test::fail_count fc{n};
test::stream c{ioc_, fc,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Content-Length: 0\r\n"
"\r\n"
2017-07-29 17:47:04 -07:00
};
request<dynamic_body> m;
try
{
multi_buffer b;
2017-07-29 17:47:04 -07:00
read(c, b, m);
break;
}
catch(std::exception const&)
{
}
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
2018-05-01 14:19:35 -07:00
test::fail_count fc{n};
test::stream ts{ioc_, fc,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Content-Length: 0\r\n"
"\r\n"
2017-08-15 22:20:13 -07:00
};
request<dynamic_body> m;
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
multi_buffer b;
2017-08-15 22:20:13 -07:00
read(ts, b, m, ec);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
2018-05-01 14:19:35 -07:00
test::fail_count fc{n};
test::stream c{ioc_, fc,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Content-Length: 0\r\n"
"\r\n"
2017-07-29 17:47:04 -07:00
};
request<dynamic_body> m;
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
multi_buffer b;
2017-07-29 17:47:04 -07:00
async_read(c, b, m, do_yield[ec]);
if(! ec)
break;
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(n < limit);
for(n = 0; n < limit; ++n)
{
2018-05-01 14:19:35 -07:00
test::fail_count fc{n};
test::stream c{ioc_, fc,
"GET / HTTP/1.1\r\n"
"Host: localhost\r\n"
"User-Agent: test\r\n"
"Content-Length: 0\r\n"
"\r\n"
};
request_parser<dynamic_body> m;
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
multi_buffer b;
async_read_some(c, b, m, do_yield[ec]);
if(! ec)
break;
}
BEAST_EXPECT(n < limit);
}
void
testEof(yield_context do_yield)
{
{
multi_buffer b;
test::stream ts{ioc_};
request_parser<dynamic_body> p;
error_code ec;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
read(ts, b, p, ec);
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(ec == http::error::end_of_stream);
}
{
multi_buffer b;
test::stream ts{ioc_};
request_parser<dynamic_body> p;
error_code ec;
2017-08-21 14:45:58 -07:00
ts.close_remote();
2017-08-15 22:20:13 -07:00
async_read(ts, b, p, do_yield[ec]);
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(ec == http::error::end_of_stream);
}
}
// Ensure completion handlers are not leaked
struct handler
{
static std::atomic<std::size_t>&
count() { static std::atomic<std::size_t> n; return n; }
handler() { ++count(); }
~handler() { --count(); }
handler(handler const&) { ++count(); }
void operator()(error_code const&, std::size_t) const {}
};
void
testIoService()
{
{
// Make sure handlers are not destroyed
// after calling io_context::stop
net::io_context ioc;
test::stream ts{ioc,
"GET / HTTP/1.1\r\n\r\n"};
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() == 0);
multi_buffer b;
2017-06-07 16:30:49 -07:00
request<dynamic_body> m;
2017-08-15 22:20:13 -07:00
async_read(ts, b, m, handler{});
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() > 0);
ioc.stop();
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() > 0);
ioc.restart();
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() > 0);
ioc.run_one();
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() == 0);
}
{
// Make sure uninvoked handlers are
// destroyed when calling ~io_context
{
net::io_context ioc;
test::stream ts{ioc,
"GET / HTTP/1.1\r\n\r\n"};
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() == 0);
multi_buffer b;
2017-06-07 16:30:49 -07:00
request<dynamic_body> m;
2017-08-15 22:20:13 -07:00
async_read(ts, b, m, handler{});
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() > 0);
}
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(handler::count() == 0);
}
}
// https://github.com/boostorg/beast/issues/430
void
testRegression430()
{
test::stream ts{ioc_};
2017-08-15 22:20:13 -07:00
ts.read_size(1);
ostream(ts.buffer()) <<
"HTTP/1.1 200 OK\r\n"
"Transfer-Encoding: chunked\r\n"
"Content-Type: application/octet-stream\r\n"
"\r\n"
"4\r\nabcd\r\n"
"0\r\n\r\n";
error_code ec;
flat_buffer fb;
response_parser<dynamic_body> p;
2017-08-15 22:20:13 -07:00
read(ts, fb, p, ec);
2017-07-25 12:35:54 -07:00
BEAST_EXPECTS(! ec, ec.message());
}
//--------------------------------------------------------------------------
template<class Parser, class Pred>
void
readgrind(string_view s, Pred&& pred)
{
for(std::size_t n = 1; n < s.size() - 1; ++n)
{
Parser p;
2018-05-01 12:39:47 -07:00
error_code ec = test::error::test_failure;
flat_buffer b;
test::stream ts{ioc_};
2017-08-15 22:20:13 -07:00
ostream(ts.buffer()) << s;
ts.read_size(n);
read(ts, b, p, ec);
2017-07-25 12:35:54 -07:00
if(! BEAST_EXPECTS(! ec, ec.message()))
continue;
pred(p);
}
}
void
testReadGrind()
{
readgrind<test_parser<false>>(
"HTTP/1.1 200 OK\r\n"
"Transfer-Encoding: chunked\r\n"
"Content-Type: application/octet-stream\r\n"
"\r\n"
"4\r\nabcd\r\n"
"0\r\n\r\n"
,[&](test_parser<false> const& p)
{
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(p.body == "abcd");
});
readgrind<test_parser<false>>(
"HTTP/1.1 200 OK\r\n"
"Server: test\r\n"
"Expect: Expires, MD5-Fingerprint\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n"
"5\r\n"
"*****\r\n"
"2;a;b=1;c=\"2\"\r\n"
"--\r\n"
"0;d;e=3;f=\"4\"\r\n"
"Expires: never\r\n"
"MD5-Fingerprint: -\r\n"
"\r\n"
,[&](test_parser<false> const& p)
{
2017-07-25 12:35:54 -07:00
BEAST_EXPECT(p.body == "*****--");
});
}
struct copyable_handler
{
template<class... Args>
void
operator()(Args&&...) const
{
}
};
void
testAsioHandlerInvoke()
{
using strand = net::strand<
net::io_context::executor_type>;
// make sure things compile, also can set a
// breakpoint in asio_handler_invoke to make sure
// it is instantiated.
{
net::io_context ioc;
strand s{ioc.get_executor()};
test::stream ts{ioc};
flat_buffer b;
request_parser<dynamic_body> p;
async_read_some(ts, b, p,
net::bind_executor(
s, copyable_handler{}));
}
{
net::io_context ioc;
strand s{ioc.get_executor()};
test::stream ts{ioc};
flat_buffer b;
request_parser<dynamic_body> p;
async_read(ts, b, p,
net::bind_executor(
s, copyable_handler{}));
}
{
net::io_context ioc;
strand s{ioc.get_executor()};
test::stream ts{ioc};
flat_buffer b;
request<dynamic_body> m;
async_read(ts, b, m,
net::bind_executor(
s, copyable_handler{}));
}
}
#if BOOST_ASIO_HAS_CO_AWAIT
void testAwaitableCompiles(
test::stream& stream,
flat_buffer& dynbuf,
parser<true, string_body>& request_parser,
request<http::string_body>& request,
parser<false, string_body>& response_parser,
response<http::string_body>& response)
{
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read(stream, dynbuf, request, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read(stream, dynbuf, request_parser, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read(stream, dynbuf, response, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read(stream, dynbuf, response_parser, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read_some(stream, dynbuf, request_parser, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read_some(stream, dynbuf, response_parser, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read_header(stream, dynbuf, request_parser, net::use_awaitable))>);
static_assert(std::is_same_v<
net::awaitable<std::size_t>, decltype(
http::async_read_header(stream, dynbuf, response_parser, net::use_awaitable))>);
}
#endif
void testReadSomeHeader(net::yield_context yield)
{
std::string hdr =
"GET /foo HTTP/1.1" "\r\n"
"Connection: Keep-Alive" "\r\n"
"Content-Length: 6"
"\r\n"
"\r\n";
std::string body =
"Hello!";
{
// bytes_transferred returns length of header
request_parser<string_body> p;
test::stream s(ioc_);
s.append(string_view(hdr));
s.append(string_view(body));
flat_buffer fb;
error_code ec;
auto bt = async_read_header(s, fb, p, yield[ec]);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECT(bt == hdr.size());
// next read should be zero-size, success
bt = async_read_header(s, fb, p, yield[ec]);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECTS(bt == 0, std::to_string(0));
}
{
// incomplete header consumes all parsable header bytes
request_parser<string_body> p;
test::stream s(ioc_);
s.append(hdr.substr(0, hdr.size() - 1));
s.close();
flat_buffer fb;
error_code ec;
auto bt = async_read_header(s, fb, p, yield[ec]);
BEAST_EXPECTS(ec == error::partial_message, ec.message());
BEAST_EXPECTS(bt + fb.size() == hdr.size() - 1,
std::to_string(bt + fb.size()) +
" expected " +
std::to_string(hdr.size() - 1));
}
{
// read consumes and reports correct number of bytes
request_parser<string_body> p;
test::stream s(ioc_);
s.append(hdr);
s.append(body);
s.append(hdr);
s.append(body);
s.append(hdr);
s.append(body);
flat_buffer fb;
error_code ec;
auto bt = async_read_header(s, fb, p, yield[ec]);
BEAST_EXPECTS("ec", ec.message());
BEAST_EXPECT(bt == hdr.size());
auto bt2 = async_read_some(s, fb, p, yield[ec]);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECT(bt2 == body.size());
BEAST_EXPECTS(fb.size() / 2 == hdr.size() + body.size(),
std::to_string(fb.size() / 2) + " != " + std::to_string(hdr.size() + body.size()));
request_parser<string_body> p2;
bt = async_read(s, fb, p2, yield[ec]);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECTS(bt == hdr.size() + body.size(),
std::to_string(bt) +
" expected " +
std::to_string(hdr.size() + body.size()));
BEAST_EXPECTS(fb.size() == hdr.size() + body.size(),
std::to_string(fb.size()) + " != " + std::to_string(hdr.size() + body.size()));
}
}
void testReadSomeHeader()
{
net::io_context ioc;
std::string hdr =
"GET /foo HTTP/1.1" "\r\n"
"Connection: Keep-Alive" "\r\n"
"Content-Length: 6"
"\r\n"
"\r\n";
std::string body =
"Hello!";
{
// bytes_transferred returns length of header
request_parser<string_body> p;
test::stream s(ioc);
s.append(string_view(hdr));
s.append(string_view(body));
flat_buffer fb;
error_code ec;
auto bt = read_header(s, fb, p, ec);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECT(bt == hdr.size());
// next read should be zero-size, success
bt = read_header(s, fb, p, ec);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECTS(bt == 0, std::to_string(0));
}
{
// incomplete header consumes all parsable header bytes
request_parser<string_body> p;
test::stream s(ioc);
s.append(hdr.substr(0, hdr.size() - 1));
s.close();
flat_buffer fb;
error_code ec;
auto bt = read_header(s, fb, p, ec);
BEAST_EXPECTS(ec == error::partial_message, ec.message());
BEAST_EXPECTS(bt + fb.size() == hdr.size() - 1,
std::to_string(bt + fb.size()) +
" expected " +
std::to_string(hdr.size() - 1));
}
}
void
run() override
{
testThrow();
testBufferOverflow();
2017-08-15 22:20:13 -07:00
yield_to([&](yield_context yield)
{
testFailures(yield);
});
yield_to([&](yield_context yield)
{
testRead(yield);
});
yield_to([&](yield_context yield)
{
testEof(yield);
});
testIoService();
testRegression430();
testReadGrind();
testAsioHandlerInvoke();
#if BOOST_ASIO_HAS_CO_AWAIT
boost::ignore_unused(&read_test::testAwaitableCompiles);
#endif
yield_to([&](yield_context yield)
{
testReadSomeHeader(yield);
});
testReadSomeHeader();
}
};
2017-08-01 17:01:57 -07:00
BEAST_DEFINE_TESTSUITE(beast,http,read);
} // http
} // beast
2017-07-20 13:40:34 -07:00
} // boost