mirror of
https://github.com/boostorg/beast.git
synced 2025-07-31 05:17:26 +02:00
static_buffer coverage
This commit is contained in:
@ -1,3 +1,9 @@
|
|||||||
|
Version 54:
|
||||||
|
|
||||||
|
* static_buffer coverage
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
Version 53:
|
Version 53:
|
||||||
|
|
||||||
* Fix basic_parser::maybe_flatten
|
* Fix basic_parser::maybe_flatten
|
||||||
|
@ -44,9 +44,7 @@ read_size_helper(
|
|||||||
BOOST_ASSERT(max_size >= 1);
|
BOOST_ASSERT(max_size >= 1);
|
||||||
auto const size = buffer.size();
|
auto const size = buffer.size();
|
||||||
auto const limit = buffer.max_size() - size;
|
auto const limit = buffer.max_size() - size;
|
||||||
if(limit <= 0)
|
BOOST_ASSERT(size <= buffer.max_size());
|
||||||
BOOST_THROW_EXCEPTION(std::length_error{
|
|
||||||
"dynamic buffer overflow"});
|
|
||||||
return std::min<std::size_t>(
|
return std::min<std::size_t>(
|
||||||
std::max<std::size_t>(512, buffer.capacity() - size),
|
std::max<std::size_t>(512, buffer.capacity() - size),
|
||||||
std::min<std::size_t>(max_size, limit));
|
std::min<std::size_t>(max_size, limit));
|
||||||
|
@ -76,7 +76,7 @@ prepare_impl(std::size_t n) ->
|
|||||||
auto const len = size();
|
auto const len = size();
|
||||||
if(n > capacity() - len)
|
if(n > capacity() - len)
|
||||||
BOOST_THROW_EXCEPTION(std::length_error{
|
BOOST_THROW_EXCEPTION(std::length_error{
|
||||||
"static_buffer overflow"});
|
"buffer overflow"});
|
||||||
if(len > 0)
|
if(len > 0)
|
||||||
std::memmove(begin_, in_, len);
|
std::memmove(begin_, in_, len);
|
||||||
in_ = begin_;
|
in_ = begin_;
|
||||||
@ -99,6 +99,31 @@ consume_impl(std::size_t n)
|
|||||||
in_ += n;
|
in_ += n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
template<std::size_t N>
|
||||||
|
static_buffer_n<N>::
|
||||||
|
static_buffer_n(static_buffer_n const& other)
|
||||||
|
: static_buffer(buf_, N)
|
||||||
|
{
|
||||||
|
using boost::asio::buffer_copy;
|
||||||
|
this->commit(buffer_copy(
|
||||||
|
this->prepare(other.size()), other.data()));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<std::size_t N>
|
||||||
|
auto
|
||||||
|
static_buffer_n<N>::
|
||||||
|
operator=(static_buffer_n const& other) ->
|
||||||
|
static_buffer_n<N>&
|
||||||
|
{
|
||||||
|
using boost::asio::buffer_copy;
|
||||||
|
this->consume(this->size());
|
||||||
|
this->commit(buffer_copy(
|
||||||
|
this->prepare(other.size()), other.data()));
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
} // beast
|
} // beast
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -10,6 +10,8 @@
|
|||||||
|
|
||||||
#include <beast/config.hpp>
|
#include <beast/config.hpp>
|
||||||
#include <boost/asio/buffer.hpp>
|
#include <boost/asio/buffer.hpp>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <cstddef>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
namespace beast {
|
namespace beast {
|
||||||
@ -37,6 +39,9 @@ class static_buffer
|
|||||||
char* last_;
|
char* last_;
|
||||||
char* end_;
|
char* end_;
|
||||||
|
|
||||||
|
static_buffer(static_buffer const& other) = delete;
|
||||||
|
static_buffer& operator=(static_buffer const&) = delete;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// The type used to represent the input sequence as a list of buffers.
|
/// The type used to represent the input sequence as a list of buffers.
|
||||||
using const_buffers_type = boost::asio::const_buffers_1;
|
using const_buffers_type = boost::asio::const_buffers_1;
|
||||||
@ -44,12 +49,6 @@ public:
|
|||||||
/// The type used to represent the output sequence as a list of buffers.
|
/// The type used to represent the output sequence as a list of buffers.
|
||||||
using mutable_buffers_type = boost::asio::mutable_buffers_1;
|
using mutable_buffers_type = boost::asio::mutable_buffers_1;
|
||||||
|
|
||||||
static_buffer(
|
|
||||||
static_buffer const& other) noexcept = delete;
|
|
||||||
|
|
||||||
static_buffer& operator=(
|
|
||||||
static_buffer const&) noexcept = delete;
|
|
||||||
|
|
||||||
/** Constructor.
|
/** Constructor.
|
||||||
|
|
||||||
This creates a dynamic buffer using the provided storage area.
|
This creates a dynamic buffer using the provided storage area.
|
||||||
@ -185,17 +184,31 @@ class static_buffer_n : public static_buffer
|
|||||||
char buf_[N];
|
char buf_[N];
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// Copy constructor (disallowed).
|
/// Copy constructor
|
||||||
static_buffer_n(static_buffer_n const&) = delete;
|
static_buffer_n(static_buffer_n const&);
|
||||||
|
|
||||||
/// Copy assignment (disallowed).
|
/// Copy assignment
|
||||||
static_buffer_n& operator=(static_buffer_n const&) = delete;
|
static_buffer_n& operator=(static_buffer_n const&);
|
||||||
|
|
||||||
/// Construct a static buffer.
|
/// Construct a static buffer.
|
||||||
static_buffer_n()
|
static_buffer_n()
|
||||||
: static_buffer(buf_, N)
|
: static_buffer(buf_, N)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the @ref static_buffer portion of this object
|
||||||
|
static_buffer&
|
||||||
|
base()
|
||||||
|
{
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the @ref static_buffer portion of this object
|
||||||
|
static_buffer const&
|
||||||
|
base() const
|
||||||
|
{
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // beast
|
} // beast
|
||||||
|
@ -7,3 +7,32 @@
|
|||||||
|
|
||||||
// Test that header file is self-contained.
|
// Test that header file is self-contained.
|
||||||
#include <beast/core/async_result.hpp>
|
#include <beast/core/async_result.hpp>
|
||||||
|
|
||||||
|
#include <beast/core/error.hpp>
|
||||||
|
#include <beast/core/type_traits.hpp>
|
||||||
|
#include <cstdlib>
|
||||||
|
|
||||||
|
namespace beast {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
struct handler
|
||||||
|
{
|
||||||
|
void operator()(beast::error_code, std::size_t) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
static_assert(detail::is_invocable<
|
||||||
|
typename async_result<handler, void(error_code, std::size_t)>::completion_handler_type,
|
||||||
|
void(error_code, std::size_t)>::value, "");
|
||||||
|
|
||||||
|
static_assert(std::is_same<void,
|
||||||
|
typename async_result<handler, void(error_code, std::size_t)>::return_type>::value, "");
|
||||||
|
|
||||||
|
static_assert(std::is_constructible<
|
||||||
|
async_result<handler,
|
||||||
|
void(error_code, std::size_t)>,
|
||||||
|
typename async_result<handler,
|
||||||
|
void(error_code, std::size_t)
|
||||||
|
>::completion_handler_type&>::value, "");
|
||||||
|
|
||||||
|
} // (anon-ns)
|
||||||
|
} // beast
|
||||||
|
@ -8,16 +8,43 @@
|
|||||||
#ifndef BEAST_TEST_BUFFER_TEST_HPP
|
#ifndef BEAST_TEST_BUFFER_TEST_HPP
|
||||||
#define BEAST_TEST_BUFFER_TEST_HPP
|
#define BEAST_TEST_BUFFER_TEST_HPP
|
||||||
|
|
||||||
|
#include <beast/core/string_view.hpp>
|
||||||
#include <beast/core/type_traits.hpp>
|
#include <beast/core/type_traits.hpp>
|
||||||
#include <beast/core/detail/read_size_helper.hpp>
|
#include <beast/core/detail/read_size_helper.hpp>
|
||||||
#include <beast/core/detail/type_traits.hpp>
|
#include <beast/core/detail/type_traits.hpp>
|
||||||
#include <boost/asio/buffer.hpp>
|
#include <boost/asio/buffer.hpp>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
#include <string>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
|
|
||||||
namespace beast {
|
namespace beast {
|
||||||
namespace test {
|
namespace test {
|
||||||
|
|
||||||
|
template<class ConstBufferSequence>
|
||||||
|
typename std::enable_if<
|
||||||
|
is_const_buffer_sequence<ConstBufferSequence>::value,
|
||||||
|
std::string>::type
|
||||||
|
to_string(ConstBufferSequence const& bs)
|
||||||
|
{
|
||||||
|
using boost::asio::buffer_cast;
|
||||||
|
using boost::asio::buffer_size;
|
||||||
|
std::string s;
|
||||||
|
s.reserve(buffer_size(bs));
|
||||||
|
for(auto const& b : bs)
|
||||||
|
s.append(buffer_cast<char const*>(b),
|
||||||
|
buffer_size(b));
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<class DynamicBuffer>
|
||||||
|
void
|
||||||
|
write_buffer(DynamicBuffer& b, string_view s)
|
||||||
|
{
|
||||||
|
b.commit(boost::asio::buffer_copy(
|
||||||
|
b.prepare(s.size()), boost::asio::buffer(
|
||||||
|
s.data(), s.size())));
|
||||||
|
}
|
||||||
|
|
||||||
template<class ConstBufferSequence>
|
template<class ConstBufferSequence>
|
||||||
typename std::enable_if<
|
typename std::enable_if<
|
||||||
is_const_buffer_sequence<ConstBufferSequence>::value,
|
is_const_buffer_sequence<ConstBufferSequence>::value,
|
||||||
@ -101,6 +128,8 @@ namespace has_read_size_helper
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
// Make sure read_size_helper works
|
// Make sure read_size_helper works
|
||||||
template<class DynamicBuffer>
|
template<class DynamicBuffer>
|
||||||
inline
|
inline
|
||||||
@ -109,6 +138,7 @@ check_read_size_helper()
|
|||||||
{
|
{
|
||||||
static_assert(is_dynamic_buffer<DynamicBuffer>::value,
|
static_assert(is_dynamic_buffer<DynamicBuffer>::value,
|
||||||
"DynamicBuffer requirements not met ");
|
"DynamicBuffer requirements not met ");
|
||||||
|
|
||||||
static_assert(has_read_size_helper::trait<DynamicBuffer>::value,
|
static_assert(has_read_size_helper::trait<DynamicBuffer>::value,
|
||||||
"Missing read_size_helper for dynamic buffer");
|
"Missing read_size_helper for dynamic buffer");
|
||||||
}
|
}
|
||||||
|
@ -9,33 +9,25 @@
|
|||||||
#include <beast/core/static_buffer.hpp>
|
#include <beast/core/static_buffer.hpp>
|
||||||
|
|
||||||
#include "buffer_test.hpp"
|
#include "buffer_test.hpp"
|
||||||
|
|
||||||
|
#include <beast/core/ostream.hpp>
|
||||||
|
#include <beast/core/string_view.hpp>
|
||||||
#include <beast/unit_test/suite.hpp>
|
#include <beast/unit_test/suite.hpp>
|
||||||
#include <boost/asio/buffer.hpp>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace beast {
|
namespace beast {
|
||||||
|
|
||||||
|
static_assert(
|
||||||
|
is_dynamic_buffer<static_buffer>::value,
|
||||||
|
"DynamicBuffer requirements not met");
|
||||||
|
|
||||||
class static_buffer_test : public beast::unit_test::suite
|
class static_buffer_test : public beast::unit_test::suite
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
template<class ConstBufferSequence>
|
|
||||||
static
|
|
||||||
std::string
|
|
||||||
to_string(ConstBufferSequence const& bs)
|
|
||||||
{
|
|
||||||
using boost::asio::buffer_cast;
|
|
||||||
using boost::asio::buffer_size;
|
|
||||||
std::string s;
|
|
||||||
s.reserve(buffer_size(bs));
|
|
||||||
for(auto const& b : bs)
|
|
||||||
s.append(buffer_cast<char const*>(b),
|
|
||||||
buffer_size(b));
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
testStaticBuffer()
|
testStaticBuffer()
|
||||||
{
|
{
|
||||||
|
using namespace test;
|
||||||
using boost::asio::buffer;
|
using boost::asio::buffer;
|
||||||
using boost::asio::buffer_cast;
|
using boost::asio::buffer_cast;
|
||||||
using boost::asio::buffer_size;
|
using boost::asio::buffer_size;
|
||||||
@ -142,15 +134,100 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
testReadSizeHelper()
|
testBuffer()
|
||||||
{
|
{
|
||||||
|
using namespace test;
|
||||||
|
string_view const s = "Hello, world!";
|
||||||
|
|
||||||
|
// static_buffer
|
||||||
|
{
|
||||||
|
char buf[64];
|
||||||
|
static_buffer b{buf, sizeof(buf)};
|
||||||
|
ostream(b) << s;
|
||||||
|
BEAST_EXPECT(to_string(b.data()) == s);
|
||||||
|
b.consume(b.size());
|
||||||
|
BEAST_EXPECT(to_string(b.data()) == "");
|
||||||
|
}
|
||||||
|
|
||||||
|
// static_buffer_n
|
||||||
|
{
|
||||||
|
static_buffer_n<64> b1;
|
||||||
|
BEAST_EXPECT(b1.size() == 0);
|
||||||
|
BEAST_EXPECT(b1.max_size() == 64);
|
||||||
|
BEAST_EXPECT(b1.capacity() == 64);
|
||||||
|
ostream(b1) << s;
|
||||||
|
BEAST_EXPECT(to_string(b1.data()) == s);
|
||||||
|
{
|
||||||
|
static_buffer_n<64> b2{b1};
|
||||||
|
BEAST_EXPECT(to_string(b2.data()) == s);
|
||||||
|
b2.consume(7);
|
||||||
|
BEAST_EXPECT(to_string(b2.data()) == s.substr(7));
|
||||||
|
}
|
||||||
|
{
|
||||||
|
static_buffer_n<64> b2;
|
||||||
|
b2 = b1;
|
||||||
|
BEAST_EXPECT(to_string(b2.data()) == s);
|
||||||
|
b2.consume(7);
|
||||||
|
BEAST_EXPECT(to_string(b2.data()) == s.substr(7));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// cause memmove
|
||||||
|
{
|
||||||
|
static_buffer_n<10> b;
|
||||||
|
write_buffer(b, "12345");
|
||||||
|
b.consume(3);
|
||||||
|
write_buffer(b, "67890123");
|
||||||
|
BEAST_EXPECT(to_string(b.data()) == "4567890123");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
b.prepare(1);
|
||||||
|
fail("", __FILE__, __LINE__);
|
||||||
|
}
|
||||||
|
catch(std::length_error const&)
|
||||||
|
{
|
||||||
|
pass();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// read_size_helper
|
||||||
|
{
|
||||||
|
using detail::read_size_helper;
|
||||||
|
static_buffer_n<10> b;
|
||||||
|
BEAST_EXPECT(read_size_helper(b, 512) == 10);
|
||||||
|
b.prepare(4);
|
||||||
|
b.commit(4);
|
||||||
|
BEAST_EXPECT(read_size_helper(b, 512) == 6);
|
||||||
|
b.consume(2);
|
||||||
|
BEAST_EXPECT(read_size_helper(b, 512) == 8);
|
||||||
|
b.prepare(8);
|
||||||
|
b.commit(8);
|
||||||
|
BEAST_EXPECT(read_size_helper(b, 512) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// base
|
||||||
|
{
|
||||||
|
static_buffer_n<10> b;
|
||||||
|
[&](static_buffer& b)
|
||||||
|
{
|
||||||
|
BEAST_EXPECT(b.max_size() == b.capacity());
|
||||||
|
}
|
||||||
|
(b.base());
|
||||||
|
|
||||||
|
[&](static_buffer const&)
|
||||||
|
{
|
||||||
|
BEAST_EXPECT(b.max_size() == b.capacity());
|
||||||
|
}
|
||||||
|
(b.base());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void run() override
|
void run() override
|
||||||
{
|
{
|
||||||
test::check_read_size_helper<static_buffer_n<32>>();
|
test::check_read_size_helper<static_buffer_n<32>>();
|
||||||
|
|
||||||
testStaticBuffer();
|
testBuffer();
|
||||||
|
//testStaticBuffer();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user