mirror of
https://github.com/boostorg/beast.git
synced 2025-07-31 13:27:33 +02:00
Remove header_parser (API Change)
This commit is contained in:
@@ -2,6 +2,10 @@ Version 50
|
||||
|
||||
* parser is constructible from other body types
|
||||
|
||||
API Changes:
|
||||
|
||||
* Remove header_parser
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
Version 49
|
||||
|
@@ -63,7 +63,6 @@
|
||||
[def __fields__ [link beast.ref.http__fields `fields`]]
|
||||
[def __flat_buffer__ [link beast.ref.flat_buffer `flat_buffer`]]
|
||||
[def __header__ [link beast.ref.http__header `header`]]
|
||||
[def __header_parser__ [link beast.ref.http__header_parser `header_parser`]]
|
||||
[def __message__ [link beast.ref.http__message `message`]]
|
||||
[def __multi_buffer__ [link beast.ref.multi_buffer `multi_buffer`]]
|
||||
[def __parser__ [link beast.ref.http__parser `parser`]]
|
||||
|
@@ -38,18 +38,6 @@ defined types deriving from the basic parser are possible:
|
||||
: public basic_parser<...>;
|
||||
```
|
||||
]]
|
||||
[[
|
||||
__header_parser__
|
||||
][
|
||||
```
|
||||
/// An HTTP/1 parser for producing a header.
|
||||
template<
|
||||
bool isRequest, // `true` to parse an HTTP request
|
||||
class Fields> // The type of container representing the fields
|
||||
class header_parser
|
||||
: public basic_parser<...>;
|
||||
```
|
||||
]]
|
||||
[[
|
||||
[link beast.ref.http__request_parser `request_parser`]
|
||||
][
|
||||
|
@@ -7,11 +7,11 @@
|
||||
|
||||
[section:parser_buffers Buffer-Oriented Parsing]
|
||||
|
||||
In extreme cases, users may wish to create an instance of __parser__,
|
||||
__header_parser__, or a user-defined type derived from __basic_parser__ and
|
||||
invoke its methods directly instead of using the provided stream algorithms.
|
||||
This could be useful for implementing algorithms on streams whose interface
|
||||
does not conform to any __Stream__. For example, a
|
||||
In some cases, users may wish to create an instance of __parser__, or a
|
||||
user-defined type derived from __basic_parser__ and invoke its methods
|
||||
directly instead of using the provided stream algorithms. This could be
|
||||
useful for implementing algorithms on streams whose interface does not
|
||||
conform to any __Stream__. For example, a
|
||||
[@http://zeromq.org/ *ZeroMQ* socket].
|
||||
The basic parser interface is interactive; the caller invokes the function
|
||||
[link beast.ref.http__basic_parser.put `basic_parser::put`]
|
||||
|
@@ -37,7 +37,6 @@
|
||||
<member><link linkend="beast.ref.http__empty_body">empty_body</link></member>
|
||||
<member><link linkend="beast.ref.http__fields">fields</link></member>
|
||||
<member><link linkend="beast.ref.http__header">header</link></member>
|
||||
<member><link linkend="beast.ref.http__header_parser">header_parser</link></member>
|
||||
<member><link linkend="beast.ref.http__message">message</link></member>
|
||||
<member><link linkend="beast.ref.http__parser">parser</link></member>
|
||||
<member><link linkend="beast.ref.http__no_chunk_decorator">no_chunk_decorator</link></member>
|
||||
|
@@ -19,6 +19,9 @@ struct header;
|
||||
template<bool, class, class>
|
||||
struct message;
|
||||
|
||||
template<bool isRequest,class Body, class Fields>
|
||||
class parser;
|
||||
|
||||
namespace detail {
|
||||
|
||||
template<class T>
|
||||
@@ -35,6 +38,12 @@ public:
|
||||
template<class T>
|
||||
using is_header = typename is_header_impl<T>::type;
|
||||
|
||||
template<class T>
|
||||
struct is_parser : std::false_type {};
|
||||
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
struct is_parser<parser<isRequest, Body, Fields>> : std::true_type {};
|
||||
|
||||
struct fields_model
|
||||
{
|
||||
string_view method() const;
|
||||
|
@@ -14,15 +14,6 @@
|
||||
namespace beast {
|
||||
namespace http {
|
||||
|
||||
template<bool isRequest, class Fields>
|
||||
template<class Arg0, class... ArgN, class>
|
||||
header_parser<isRequest, Fields>::
|
||||
header_parser(Arg0&& arg0, ArgN&&... argn)
|
||||
: h_(std::forward<Arg0>(arg0),
|
||||
std::forward<ArgN>(argn)...)
|
||||
{
|
||||
}
|
||||
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
template<class Arg1, class... ArgN, class>
|
||||
parser<isRequest, Body, Fields>::
|
||||
@@ -33,7 +24,7 @@ parser(Arg1&& arg1, ArgN&&... argn)
|
||||
}
|
||||
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
template<class OtherBody, class... Args>
|
||||
template<class OtherBody, class... Args, class>
|
||||
parser<isRequest, Body, Fields>::
|
||||
parser(parser<isRequest, OtherBody, Fields>&& parser,
|
||||
Args&&... args)
|
||||
@@ -46,17 +37,6 @@ parser(parser<isRequest, OtherBody, Fields>&& parser,
|
||||
"moved-from parser has a body"});
|
||||
}
|
||||
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
template<class... Args>
|
||||
parser<isRequest, Body, Fields>::
|
||||
parser(header_parser<
|
||||
isRequest, Fields>&& parser, Args&&... args)
|
||||
: base_type(std::move(static_cast<basic_parser<
|
||||
isRequest, header_parser<isRequest, Fields>>&>(parser)))
|
||||
, m_(parser.release(), std::forward<Args>(args)...)
|
||||
{
|
||||
}
|
||||
|
||||
} // http
|
||||
} // beast
|
||||
|
||||
|
@@ -20,199 +20,6 @@
|
||||
namespace beast {
|
||||
namespace http {
|
||||
|
||||
/** An HTTP/1 parser for producing a header.
|
||||
|
||||
This class uses the basic HTTP/1 wire format parser to convert
|
||||
a series of octets into a @ref header.
|
||||
|
||||
@note A new instance of the parser is required for each message.
|
||||
|
||||
@tparam isRequest Indicates whether a request or response
|
||||
will be parsed.
|
||||
|
||||
@tparam Fields The type of container used to represent the fields.
|
||||
*/
|
||||
template<bool isRequest, class Fields>
|
||||
class header_parser
|
||||
: public basic_parser<isRequest,
|
||||
header_parser<isRequest, Fields>>
|
||||
{
|
||||
header<isRequest, Fields> h_;
|
||||
string_view body_;
|
||||
|
||||
public:
|
||||
/// The type of @ref header this object produces.
|
||||
using value_type = header<isRequest, Fields>;
|
||||
|
||||
/// Default constructor.
|
||||
header_parser() = default;
|
||||
|
||||
/// Copy constructor.
|
||||
header_parser(header_parser const&) = default;
|
||||
|
||||
/// Copy assignment.
|
||||
header_parser& operator=(header_parser const&) = default;
|
||||
|
||||
/** Move constructor.
|
||||
|
||||
After the move, the only valid operation
|
||||
on the moved-from object is destruction.
|
||||
*/
|
||||
header_parser(header_parser&&) = default;
|
||||
|
||||
/** Move assignment
|
||||
|
||||
After the move, the only valid operation
|
||||
on the moved-from object is destruction.
|
||||
*/
|
||||
header_parser& operator=(header_parser&&) = default;
|
||||
|
||||
/** Constructor
|
||||
|
||||
@param args Optional arguments forwarded
|
||||
forwarded to the @ref http::header constructor.
|
||||
*/
|
||||
#if BEAST_DOXYGEN
|
||||
template<class... Args>
|
||||
explicit
|
||||
header_parser(Args&&... args);
|
||||
#else
|
||||
template<class Arg0, class... ArgN,
|
||||
class = typename std::enable_if<
|
||||
! std::is_convertible<typename
|
||||
std::decay<Arg0>::type,
|
||||
header_parser>::value>>
|
||||
explicit
|
||||
header_parser(Arg0&& arg0, ArgN&&... argn);
|
||||
#endif
|
||||
|
||||
/** Returns parsed body octets.
|
||||
|
||||
This function will return the most recent buffer
|
||||
of octets corresponding to the parsed body. This
|
||||
buffer will become invalidated on any subsequent
|
||||
call to @ref put or @ref put_eof
|
||||
*/
|
||||
string_view
|
||||
body() const
|
||||
{
|
||||
return body_;
|
||||
}
|
||||
|
||||
/** Returns the parsed header
|
||||
|
||||
@note The return value is undefined unless
|
||||
@ref is_header_done would return `true`.
|
||||
*/
|
||||
value_type const&
|
||||
get() const
|
||||
{
|
||||
return h_;
|
||||
}
|
||||
|
||||
/** Returns the parsed header.
|
||||
|
||||
@note The return value is undefined unless
|
||||
@ref is_header_done would return `true`.
|
||||
*/
|
||||
value_type&
|
||||
get()
|
||||
{
|
||||
return h_;
|
||||
}
|
||||
|
||||
/** Returns ownership of the parsed header.
|
||||
|
||||
Ownership is transferred to the caller.
|
||||
|
||||
@note The return value is undefined unless
|
||||
@ref is_header_done would return `true`.
|
||||
|
||||
Requires:
|
||||
@ref value_type is @b MoveConstructible
|
||||
*/
|
||||
value_type
|
||||
release()
|
||||
{
|
||||
static_assert(
|
||||
std::is_move_constructible<decltype(h_)>::value,
|
||||
"MoveConstructible requirements not met");
|
||||
return std::move(h_);
|
||||
}
|
||||
|
||||
private:
|
||||
friend class basic_parser<isRequest, header_parser>;
|
||||
|
||||
void
|
||||
on_request(string_view method,
|
||||
string_view path, int version, error_code&)
|
||||
{
|
||||
h_.target(path);
|
||||
h_.method(method);
|
||||
h_.version = version;
|
||||
}
|
||||
|
||||
void
|
||||
on_response(int code, string_view reason,
|
||||
int version, error_code&)
|
||||
{
|
||||
h_.result(code);
|
||||
h_.version = version;
|
||||
h_.reason(reason);
|
||||
}
|
||||
|
||||
void
|
||||
on_field(string_view name,
|
||||
string_view value, error_code&)
|
||||
{
|
||||
h_.fields.insert(name, value);
|
||||
}
|
||||
|
||||
void
|
||||
on_header(error_code&)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
on_body(boost::optional<std::
|
||||
uint64_t> const&, error_code&)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
on_data(string_view s, error_code&)
|
||||
{
|
||||
body_ = s;
|
||||
}
|
||||
|
||||
void
|
||||
on_chunk(std::uint64_t,
|
||||
string_view, error_code&)
|
||||
{
|
||||
body_ = {};
|
||||
}
|
||||
|
||||
void
|
||||
on_complete(error_code&)
|
||||
{
|
||||
body_ = {};
|
||||
}
|
||||
};
|
||||
|
||||
template<
|
||||
bool isRequest,class Body, class Fields = fields>
|
||||
class parser;
|
||||
|
||||
namespace detail {
|
||||
|
||||
template<class T>
|
||||
struct is_parser : std::false_type {};
|
||||
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
struct is_parser<parser<isRequest, Body, Fields>> : std::true_type {};
|
||||
|
||||
} // detail
|
||||
|
||||
/** An HTTP/1 parser for producing a message.
|
||||
|
||||
This class uses the basic HTTP/1 wire format parser to convert
|
||||
@@ -227,7 +34,10 @@ struct is_parser<parser<isRequest, Body, Fields>> : std::true_type {};
|
||||
|
||||
@note A new instance of the parser is required for each message.
|
||||
*/
|
||||
template<bool isRequest, class Body, class Fields>
|
||||
template<
|
||||
bool isRequest,
|
||||
class Body,
|
||||
class Fields = fields>
|
||||
class parser
|
||||
: public basic_parser<isRequest,
|
||||
parser<isRequest, Body, Fields>>
|
||||
@@ -265,7 +75,7 @@ public:
|
||||
After the move, the only valid operation
|
||||
on the moved-from object is destruction.
|
||||
*/
|
||||
parser(parser&& other);
|
||||
parser(parser&& other) = default;
|
||||
|
||||
/** Constructor
|
||||
|
||||
@@ -320,22 +130,14 @@ public:
|
||||
#if BEAST_DOXYGEN
|
||||
template<class OtherBody, class... Args>
|
||||
#else
|
||||
template<class OtherBody, class... Args>
|
||||
template<class OtherBody, class... Args,
|
||||
class = typename std::enable_if<
|
||||
! std::is_same<Body, OtherBody>::value>::type>
|
||||
#endif
|
||||
explicit
|
||||
parser(parser<isRequest, OtherBody, Fields>&& parser,
|
||||
Args&&... args);
|
||||
|
||||
/** Construct a parser from a @ref header_parser.
|
||||
@param parser The header parser to construct from.
|
||||
@param args Optional arguments forwarded to the message
|
||||
constructor.
|
||||
*/
|
||||
template<class... Args>
|
||||
explicit
|
||||
parser(header_parser<
|
||||
isRequest, Fields>&& parser, Args&&... args);
|
||||
|
||||
/** Returns the parsed message.
|
||||
|
||||
Depending on the progress of the parser, portions
|
||||
|
@@ -173,19 +173,15 @@ class stream<NextLayer>::accept_op
|
||||
{
|
||||
struct data
|
||||
{
|
||||
bool cont;
|
||||
stream<NextLayer>& ws;
|
||||
Decorator decorator;
|
||||
http::header_parser<true, http::fields> p;
|
||||
int state = 0;
|
||||
http::request_parser<http::empty_body> p;
|
||||
|
||||
data(Handler& handler, stream<NextLayer>& ws_,
|
||||
Decorator const& decorator_)
|
||||
: ws(ws_)
|
||||
, decorator(decorator_)
|
||||
{
|
||||
using boost::asio::asio_handler_is_continuation;
|
||||
cont = asio_handler_is_continuation(std::addressof(handler));
|
||||
}
|
||||
|
||||
template<class Buffers>
|
||||
@@ -195,8 +191,6 @@ class stream<NextLayer>::accept_op
|
||||
: ws(ws_)
|
||||
, decorator(decorator_)
|
||||
{
|
||||
using boost::asio::asio_handler_is_continuation;
|
||||
cont = asio_handler_is_continuation(std::addressof(handler));
|
||||
using boost::asio::buffer_copy;
|
||||
using boost::asio::buffer_size;
|
||||
// VFALCO What about catch(std::length_error const&)?
|
||||
@@ -218,11 +212,11 @@ public:
|
||||
: d_(std::forward<DeducedHandler>(h),
|
||||
ws, std::forward<Args>(args)...)
|
||||
{
|
||||
(*this)(error_code{}, 0, false);
|
||||
}
|
||||
|
||||
void operator()(error_code ec,
|
||||
std::size_t bytes_used, bool again = true);
|
||||
void operator()();
|
||||
|
||||
void operator()(error_code ec);
|
||||
|
||||
friend
|
||||
void* asio_handler_allocate(
|
||||
@@ -245,7 +239,9 @@ public:
|
||||
friend
|
||||
bool asio_handler_is_continuation(accept_op* op)
|
||||
{
|
||||
return op->d_->cont;
|
||||
using boost::asio::asio_handler_is_continuation;
|
||||
return asio_handler_is_continuation(
|
||||
std::addressof(op->d_.handler()));
|
||||
}
|
||||
|
||||
template<class Function>
|
||||
@@ -262,27 +258,24 @@ template<class NextLayer>
|
||||
template<class Decorator, class Handler>
|
||||
void
|
||||
stream<NextLayer>::accept_op<Decorator, Handler>::
|
||||
operator()(error_code ec,
|
||||
std::size_t bytes_used, bool again)
|
||||
operator()()
|
||||
{
|
||||
auto& d = *d_;
|
||||
d.cont = d.cont || again;
|
||||
if(ec)
|
||||
goto upcall;
|
||||
switch(d.state)
|
||||
{
|
||||
case 0:
|
||||
// read message
|
||||
d.state = 1;
|
||||
http::async_read_some(d.ws.next_layer(),
|
||||
d.ws.stream_.buffer(), d.p,
|
||||
std::move(*this));
|
||||
return;
|
||||
http::async_read_header(d.ws.next_layer(),
|
||||
d.ws.stream_.buffer(), d.p,
|
||||
std::move(*this));
|
||||
}
|
||||
|
||||
case 1:
|
||||
template<class NextLayer>
|
||||
template<class Decorator, class Handler>
|
||||
void
|
||||
stream<NextLayer>::accept_op<Decorator, Handler>::
|
||||
operator()(error_code ec)
|
||||
{
|
||||
auto& d = *d_;
|
||||
if(! ec)
|
||||
{
|
||||
BOOST_ASSERT(d.p.is_header_done());
|
||||
d.ws.stream_.buffer().consume(bytes_used);
|
||||
// Arguments from our state must be
|
||||
// moved to the stack before releasing
|
||||
// the handler.
|
||||
@@ -302,8 +295,6 @@ operator()(error_code ec,
|
||||
#endif
|
||||
return;
|
||||
}
|
||||
}
|
||||
upcall:
|
||||
d_.invoke(ec);
|
||||
}
|
||||
|
||||
@@ -619,7 +610,7 @@ async_accept(AcceptHandler&& handler)
|
||||
reset();
|
||||
accept_op<decltype(&default_decorate_res),
|
||||
handler_type<AcceptHandler, void(error_code)>>{
|
||||
init.completion_handler, *this, &default_decorate_res};
|
||||
init.completion_handler, *this, &default_decorate_res}();
|
||||
return init.result.get();
|
||||
}
|
||||
|
||||
@@ -641,7 +632,7 @@ async_accept_ex(ResponseDecorator const& decorator,
|
||||
reset();
|
||||
accept_op<ResponseDecorator, handler_type<
|
||||
AcceptHandler, void(error_code)>>{
|
||||
init.completion_handler, *this, decorator};
|
||||
init.completion_handler, *this, decorator}();
|
||||
return init.result.get();
|
||||
}
|
||||
|
||||
@@ -664,7 +655,8 @@ async_accept(ConstBufferSequence const& buffers,
|
||||
reset();
|
||||
accept_op<decltype(&default_decorate_res),
|
||||
handler_type<AcceptHandler, void(error_code)>>{
|
||||
init.completion_handler, *this, buffers, &default_decorate_res};
|
||||
init.completion_handler, *this, buffers,
|
||||
&default_decorate_res}();
|
||||
return init.result.get();
|
||||
}
|
||||
|
||||
@@ -692,7 +684,8 @@ async_accept_ex(ConstBufferSequence const& buffers,
|
||||
reset();
|
||||
accept_op<ResponseDecorator, handler_type<
|
||||
AcceptHandler, void(error_code)>>{
|
||||
init.completion_handler, *this, buffers, decorator};
|
||||
init.completion_handler, *this, buffers,
|
||||
decorator}();
|
||||
return init.result.get();
|
||||
}
|
||||
|
||||
|
@@ -91,13 +91,11 @@ stream<NextLayer>::
|
||||
do_accept(
|
||||
Decorator const& decorator, error_code& ec)
|
||||
{
|
||||
http::header_parser<true, http::fields> p;
|
||||
auto const bytes_used = http::read_some(
|
||||
next_layer(), stream_.buffer(), p, ec);
|
||||
http::request_parser<http::empty_body> p;
|
||||
http::read_header(next_layer(),
|
||||
stream_.buffer(), p, ec);
|
||||
if(ec)
|
||||
return;
|
||||
BOOST_ASSERT(p.is_header_done());
|
||||
stream_.buffer().consume(bytes_used);
|
||||
do_accept(p.get(), decorator, ec);
|
||||
}
|
||||
|
||||
|
@@ -30,12 +30,10 @@ namespace beast {
|
||||
namespace websocket {
|
||||
|
||||
/// The type of object holding HTTP Upgrade requests
|
||||
using request_type =
|
||||
http::message<true, http::empty_body, http::fields>;
|
||||
using request_type = http::request<http::empty_body>;
|
||||
|
||||
/// The type of object holding HTTP Upgrade responses
|
||||
using response_type =
|
||||
http::message<false, http::string_body, http::fields>;
|
||||
using response_type = http::response<http::string_body>;
|
||||
|
||||
/** Information about a WebSocket frame.
|
||||
|
||||
|
@@ -256,134 +256,6 @@ public:
|
||||
}
|
||||
|
||||
//--------------------------------------------------------------------------
|
||||
#if 0
|
||||
// VFALCO This is broken
|
||||
/*
|
||||
Efficiently relay a message from one stream to another
|
||||
*/
|
||||
template<
|
||||
bool isRequest,
|
||||
class SyncWriteStream,
|
||||
class DynamicBuffer,
|
||||
class SyncReadStream>
|
||||
void
|
||||
relay(
|
||||
SyncWriteStream& out,
|
||||
DynamicBuffer& b,
|
||||
SyncReadStream& in)
|
||||
{
|
||||
flat_buffer buffer{4096}; // 4K limit
|
||||
header_parser<isRequest, fields> parser;
|
||||
serializer<isRequest, buffer_body<
|
||||
typename flat_buffer::const_buffers_type>,
|
||||
fields> ws{parser.get()};
|
||||
error_code ec;
|
||||
do
|
||||
{
|
||||
auto const state0 = parser.state();
|
||||
auto const bytes_used =
|
||||
read_some(in, buffer, parser, ec);
|
||||
BEAST_EXPECTS(! ec, ec.message());
|
||||
switch(state0)
|
||||
{
|
||||
case parse_state::header:
|
||||
{
|
||||
BEAST_EXPECT(parser.is_header_done());
|
||||
for(;;)
|
||||
{
|
||||
ws.write_some(out, ec);
|
||||
if(ec == http::error::need_more)
|
||||
{
|
||||
ec = {};
|
||||
break;
|
||||
}
|
||||
if(ec)
|
||||
BOOST_THROW_EXCEPTION(system_error{ec});
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case parse_state::chunk_header:
|
||||
{
|
||||
// inspect parser.chunk_extension() here
|
||||
if(parser.is_done())
|
||||
boost::asio::write(out,
|
||||
chunk_encode_final());
|
||||
break;
|
||||
}
|
||||
|
||||
case parse_state::body:
|
||||
case parse_state::body_to_eof:
|
||||
case parse_state::chunk_body:
|
||||
{
|
||||
if(! parser.is_done())
|
||||
{
|
||||
auto const body = parser.body();
|
||||
boost::asio::write(out, chunk_encode(
|
||||
false, boost::asio::buffer(
|
||||
body.data(), body.size())));
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case parse_state::complete:
|
||||
break;
|
||||
}
|
||||
buffer.consume(bytes_used);
|
||||
}
|
||||
while(! parser.is_done());
|
||||
}
|
||||
|
||||
void
|
||||
testRelay()
|
||||
{
|
||||
// Content-Length
|
||||
{
|
||||
test::string_istream is{ios_,
|
||||
"GET / HTTP/1.1\r\n"
|
||||
"Content-Length: 5\r\n"
|
||||
"\r\n" // 37 byte header
|
||||
"*****",
|
||||
3 // max_read
|
||||
};
|
||||
test::string_ostream os{ios_};
|
||||
flat_buffer b{16};
|
||||
relay<true>(os, b, is);
|
||||
}
|
||||
|
||||
// end of file
|
||||
{
|
||||
test::string_istream is{ios_,
|
||||
"HTTP/1.1 200 OK\r\n"
|
||||
"\r\n" // 19 byte header
|
||||
"*****",
|
||||
3 // max_read
|
||||
};
|
||||
test::string_ostream os{ios_};
|
||||
flat_buffer b{16};
|
||||
relay<false>(os, b, is);
|
||||
}
|
||||
|
||||
// chunked
|
||||
{
|
||||
test::string_istream is{ios_,
|
||||
"GET / HTTP/1.1\r\n"
|
||||
"Transfer-Encoding: chunked\r\n"
|
||||
"\r\n"
|
||||
"5;x;y=1;z=\"-\"\r\n*****\r\n"
|
||||
"3\r\n---\r\n"
|
||||
"1\r\n+\r\n"
|
||||
"0\r\n\r\n",
|
||||
2 // max_read
|
||||
};
|
||||
test::string_ostream os{ios_};
|
||||
flat_buffer b{16};
|
||||
relay<true>(os, b, is);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
//--------------------------------------------------------------------------
|
||||
|
||||
void
|
||||
run()
|
||||
@@ -395,7 +267,6 @@ public:
|
||||
doWriteStdStream();
|
||||
doCustomParser();
|
||||
doHEAD();
|
||||
|
||||
doDeferredBody();
|
||||
}
|
||||
};
|
||||
|
@@ -25,57 +25,6 @@
|
||||
namespace beast {
|
||||
namespace http {
|
||||
|
||||
class header_parser_test
|
||||
: public beast::unit_test::suite
|
||||
, public test::enable_yield_to
|
||||
{
|
||||
public:
|
||||
static
|
||||
boost::asio::const_buffers_1
|
||||
buf(string_view s)
|
||||
{
|
||||
return {s.data(), s.size()};
|
||||
}
|
||||
|
||||
void
|
||||
testParse()
|
||||
{
|
||||
{
|
||||
test::string_istream is{ios_,
|
||||
"GET / HTTP/1.1\r\n"
|
||||
"User-Agent: test\r\n"
|
||||
"\r\n"
|
||||
};
|
||||
flat_buffer db{1024};
|
||||
header_parser<true, fields> p;
|
||||
read_some(is, db, p);
|
||||
BEAST_EXPECT(p.is_header_done());
|
||||
}
|
||||
{
|
||||
test::string_istream is{ios_,
|
||||
"POST / HTTP/1.1\r\n"
|
||||
"User-Agent: test\r\n"
|
||||
"Content-Length: 1\r\n"
|
||||
"\r\n"
|
||||
"*"
|
||||
};
|
||||
flat_buffer db{1024};
|
||||
header_parser<true, fields> p;
|
||||
read_some(is, db, p);
|
||||
BEAST_EXPECT(p.is_header_done());
|
||||
BEAST_EXPECT(! p.is_done());
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
run() override
|
||||
{
|
||||
testParse();
|
||||
}
|
||||
};
|
||||
|
||||
BEAST_DEFINE_TESTSUITE(header_parser,http,beast);
|
||||
|
||||
class parser_test
|
||||
: public beast::unit_test::suite
|
||||
, public beast::test::enable_yield_to
|
||||
@@ -335,30 +284,6 @@ public:
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
testExpect100Continue()
|
||||
{
|
||||
test::string_istream ss{ios_,
|
||||
"POST / HTTP/1.1\r\n"
|
||||
"Expect: 100-continue\r\n"
|
||||
"Content-Length: 5\r\n"
|
||||
"\r\n"
|
||||
"*****"};
|
||||
multi_buffer b;
|
||||
error_code ec;
|
||||
header_parser<true, fields> p0;
|
||||
auto const bytes_used =
|
||||
read_some(ss, b, p0, ec);
|
||||
b.consume(bytes_used);
|
||||
BEAST_EXPECTS(! ec, ec.message());
|
||||
BEAST_EXPECT(p0.is_header_done());
|
||||
BEAST_EXPECT(! p0.is_done());
|
||||
request_parser<string_body> p1{std::move(p0)};
|
||||
read(ss, b, p1, ec);
|
||||
BEAST_EXPECTS(! ec, ec.message());
|
||||
BEAST_EXPECT(p1.get().body == "*****");
|
||||
}
|
||||
|
||||
//--------------------------------------------------------------------------
|
||||
|
||||
template<class DynamicBuffer>
|
||||
|
Reference in New Issue
Block a user