From 517b08a28ccf47a0359f7d2b923cc6c133ba4d9e Mon Sep 17 00:00:00 2001 From: Mohammad Nejati Date: Fri, 23 Feb 2024 12:12:21 +0000 Subject: [PATCH] fix: buffer preparation in serializer --- src/serializer.cpp | 88 +++++++++++++------------ test/unit/serializer.cpp | 134 +++++++++++++++++++++++++++++++-------- 2 files changed, 151 insertions(+), 71 deletions(-) diff --git a/src/serializer.cpp b/src/serializer.cpp index c6571aa9..65b00831 100644 --- a/src/serializer.cpp +++ b/src/serializer.cpp @@ -10,6 +10,7 @@ #include #include #include +#include #include #include #include @@ -138,59 +139,55 @@ prepare() -> if(st_ == style::source) { - if(! is_chunked_) + if(more_) { - auto rv = src_->read( - tmp0_.prepare( - tmp0_.capacity() - - tmp0_.size())); - tmp0_.commit(rv.bytes); - if(rv.ec.failed()) - return rv.ec; - more_ = ! rv.finished; - } - else - { - if((tmp0_.capacity() - - tmp0_.size()) > - chunked_overhead_) + if(! is_chunked_) { - auto dest = tmp0_.prepare(18); - write_chunk_header(dest, 0); - tmp0_.commit(18); auto rv = src_->read( - tmp0_.prepare( - tmp0_.capacity() - - 2 - // CRLF - 5 - // final chunk - tmp0_.size())); + tmp0_.prepare(tmp0_.capacity())); tmp0_.commit(rv.bytes); - // VFALCO FIXME! - //if(rv.bytes == 0) - //tmp0_.uncommit(18); // undo if(rv.ec.failed()) return rv.ec; - if(rv.bytes > 0) - { - // rewrite with correct size - write_chunk_header( - dest, rv.bytes); - // terminate chunk - tmp0_.commit( - buffers::buffer_copy( - tmp0_.prepare(2), - buffers::const_buffer( - "\r\n", 2))); - } - if(rv.finished) + more_ = ! rv.finished; + } + else + { + if(tmp0_.capacity() > chunked_overhead_) { - tmp0_.commit( - buffers::buffer_copy( - tmp0_.prepare(5), - buffers::const_buffer( - "0\r\n\r\n", 5))); + auto dest = tmp0_.prepare( + tmp0_.capacity() - + 2 - // CRLF + 5); // final chunk + + auto rv = src_->read( + buffers::sans_prefix(dest, 18)); + + if(rv.ec.failed()) + return rv.ec; + + if(rv.bytes != 0) + { + write_chunk_header( + buffers::prefix(dest, 18), rv.bytes); + tmp0_.commit(rv.bytes + 18); + // terminate chunk + tmp0_.commit( + buffers::buffer_copy( + tmp0_.prepare(2), + buffers::const_buffer( + "\r\n", 2))); + } + + if(rv.finished) + { + tmp0_.commit( + buffers::buffer_copy( + tmp0_.prepare(5), + buffers::const_buffer( + "0\r\n\r\n", 5))); + more_ = false; + } } - more_ = ! rv.finished; } } @@ -474,6 +471,7 @@ start_source( hp_ = &out_[0]; *hp_ = { m.ph_->cbuf, m.ph_->size }; + more_ = true; } auto diff --git a/test/unit/serializer.cpp b/test/unit/serializer.cpp index 9b3e05e7..53ba72e0 100644 --- a/test/unit/serializer.cpp +++ b/test/unit/serializer.cpp @@ -33,8 +33,8 @@ struct serializer_test { struct test_source : source { - test_source() - : s_("12345") + test_source(core::string_view s) + : s_(s) { } @@ -42,6 +42,7 @@ struct serializer_test on_read( buffers::mutable_buffer b) { + BOOST_TEST(! is_done_); results rv; rv.bytes = buffers::buffer_copy( @@ -51,11 +52,13 @@ struct serializer_test s_.size())); s_ = s_.substr(rv.bytes); rv.finished = s_.empty(); + is_done_ = rv.finished; return rv; } private: core::string_view s_; + bool is_done_ = false; }; template< @@ -81,11 +84,18 @@ struct serializer_test { std::string s; auto cbs = sr.prepare().value(); - s.resize(buffers::buffer_size(cbs)); + // We limit buffer consumption to necessitate + // multiple calls to serializer::prepare() and + // serializer::consume(), allowing tests to cover + // state management within these functions + auto consumed = (std::min)( + std::size_t{256}, + buffers::buffer_size(cbs)); + s.resize(consumed); buffers::buffer_copy( buffers::mutable_buffer( - &s[0], s.size()), cbs); - sr.consume(s.size()); + &s[0], consumed), cbs); + sr.consume(consumed); return s; } @@ -107,6 +117,40 @@ struct serializer_test return t; } + static + void + check_chunked_body( + core::string_view body, + core::string_view expected) + { + for(;;) + { + auto n = body.find_first_of("\r\n"); + BOOST_TEST_NE(n, core::string_view::npos); + std::string tmp = body.substr(0, n); + body.remove_prefix(n + 2); + auto chunk_size = std::stoul(tmp, nullptr, 16); + + if( chunk_size == 0 ) // last chunk + { + BOOST_TEST_EQ(body, "\r\n"); + body.remove_prefix(2); + break; + } + + BOOST_TEST_GE(expected.size(), chunk_size); + BOOST_TEST(body.starts_with( + expected.substr(0, chunk_size))); + body.remove_prefix(chunk_size); + expected.remove_prefix(chunk_size); + + BOOST_TEST(body.starts_with("\r\n")); + body.remove_prefix(2); + } + BOOST_TEST(body.empty()); + BOOST_TEST(expected.empty()); + } + //-------------------------------------------- void @@ -118,10 +162,10 @@ struct serializer_test sr.start(res); sr.start(res, buffers::const_buffer{}); sr.start(res, buffers::mutable_buffer{}); - sr.start(res, test_source{}); + sr.start(res, test_source{"12345"}); sr.start(res, make_const(buffers::const_buffer{})); sr.start(res, make_const(buffers::mutable_buffer{})); - sr.start(res, make_const(test_source{})); + sr.start(res, make_const(test_source{"12345"})); serializer(65536); #ifdef BOOST_HTTP_PROTO_HAS_ZLIB @@ -189,19 +233,21 @@ struct serializer_test BOOST_TEST(s == expected); }; - template + template void check_src( core::string_view headers, Source&& src, - core::string_view expected) + F const& f) { response res(headers); - serializer sr; + // we limit the buffer size of the serializer, requiring + // it to make multiple calls to source::read + serializer sr(1024); sr.start(res, std::forward< Source>(src)); std::string s = read(sr); - BOOST_TEST(s == expected); + f(s); }; void @@ -256,15 +302,32 @@ struct serializer_test check_src( "HTTP/1.1 200 OK\r\n" "Server: test\r\n" - "Content-Length: 5\r\n" + "Content-Length: 2048\r\n" "\r\n", - test_source{}, - //-------------------------- + test_source{std::string(2048, '*')}, + [](core::string_view s){ + BOOST_TEST(s == + "HTTP/1.1 200 OK\r\n" + "Server: test\r\n" + "Content-Length: 2048\r\n" + "\r\n" + + std::string(2048, '*')); + }); + + // empty source + check_src( "HTTP/1.1 200 OK\r\n" "Server: test\r\n" - "Content-Length: 5\r\n" - "\r\n" - "12345"); + "Content-Length: 0\r\n" + "\r\n", + test_source{""}, + [](core::string_view s){ + BOOST_TEST(s == + "HTTP/1.1 200 OK\r\n" + "Server: test\r\n" + "Content-Length: 0\r\n" + "\r\n"); + }); // source chunked check_src( @@ -272,16 +335,35 @@ struct serializer_test "Server: test\r\n" "Transfer-Encoding: chunked\r\n" "\r\n", - test_source{}, - //-------------------------- + test_source{std::string(2048, '*')}, + [](core::string_view s){ + core::string_view expected_header = + "HTTP/1.1 200 OK\r\n" + "Server: test\r\n" + "Transfer-Encoding: chunked\r\n" + "\r\n"; + BOOST_TEST(s.starts_with(expected_header)); + s.remove_prefix(expected_header.size()); + check_chunked_body(s, std::string(2048, '*')); + }); + + // empty source chunked + check_src( "HTTP/1.1 200 OK\r\n" "Server: test\r\n" "Transfer-Encoding: chunked\r\n" - "\r\n" - "0000000000000005\r\n" - "12345" - "\r\n" - "0\r\n\r\n"); + "\r\n", + test_source{""}, + [](core::string_view s){ + core::string_view expected_header = + "HTTP/1.1 200 OK\r\n" + "Server: test\r\n" + "Transfer-Encoding: chunked\r\n" + "\r\n"; + BOOST_TEST(s.starts_with(expected_header)); + s.remove_prefix(expected_header.size()); + check_chunked_body(s, ""); + }); } void @@ -295,7 +377,7 @@ struct serializer_test "Expect: 100-continue\r\n" "Content-Length: 5\r\n" "\r\n"); - sr.start(req, test_source{}); + sr.start(req, test_source{"12345"}); std::string s; system::result< serializer::const_buffers_type> rv; @@ -375,7 +457,7 @@ struct serializer_test "\r\n"; serializer sr; response res(sv); - sr.start(res, test_source{}); + sr.start(res, test_source{"12345"}); auto s = read(sr); BOOST_TEST(s == "HTTP/1.1 200 OK\r\n"