2017-07-20 08:01:46 -07:00
|
|
|
//
|
2017-02-06 20:07:03 -05:00
|
|
|
// Copyright (c) 2013-2017 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)
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
// Test that header file is self-contained.
|
2016-05-07 14:57:15 -04:00
|
|
|
#include <beast/core/buffer_cat.hpp>
|
2017-07-20 08:01:46 -07:00
|
|
|
|
2016-05-06 19:14:17 -04:00
|
|
|
#include <beast/unit_test/suite.hpp>
|
2017-07-20 08:01:46 -07:00
|
|
|
#include <boost/asio/buffer.hpp>
|
|
|
|
|
#include <boost/asio/streambuf.hpp>
|
|
|
|
|
#include <iterator>
|
|
|
|
|
#include <list>
|
2016-11-08 19:00:40 -05:00
|
|
|
#include <type_traits>
|
2017-07-20 08:01:46 -07:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
namespace beast {
|
|
|
|
|
|
2016-05-06 19:14:17 -04:00
|
|
|
class buffer_cat_test : public unit_test::suite
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
|
|
|
|
public:
|
2016-11-08 19:00:40 -05:00
|
|
|
template<class Iterator>
|
2017-07-20 08:01:46 -07:00
|
|
|
static
|
|
|
|
|
std::reverse_iterator<Iterator>
|
2016-11-08 19:00:40 -05:00
|
|
|
make_reverse_iterator(Iterator i)
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
|
|
|
|
return std::reverse_iterator<Iterator>(i);
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-08 19:00:40 -05:00
|
|
|
template<class ConstBufferSequence>
|
|
|
|
|
static
|
|
|
|
|
std::size_t
|
|
|
|
|
bsize1(ConstBufferSequence const& bs)
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.begin(); it != bs.end(); ++it)
|
|
|
|
|
n += buffer_size(*it);
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class ConstBufferSequence>
|
|
|
|
|
static
|
|
|
|
|
std::size_t
|
|
|
|
|
bsize2(ConstBufferSequence const& bs)
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.begin(); it != bs.end(); it++)
|
|
|
|
|
n += buffer_size(*it);
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class ConstBufferSequence>
|
|
|
|
|
static
|
|
|
|
|
std::size_t
|
|
|
|
|
bsize3(ConstBufferSequence const& bs)
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.end(); it != bs.begin();)
|
|
|
|
|
n += buffer_size(*--it);
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class ConstBufferSequence>
|
|
|
|
|
static
|
|
|
|
|
std::size_t
|
|
|
|
|
bsize4(ConstBufferSequence const& bs)
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.end(); it != bs.begin();)
|
|
|
|
|
{
|
|
|
|
|
it--;
|
|
|
|
|
n += buffer_size(*it);
|
|
|
|
|
}
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 08:01:46 -07:00
|
|
|
void testBufferCat()
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
using boost::asio::const_buffer;
|
|
|
|
|
char buf[10];
|
|
|
|
|
std::list<const_buffer> b1;
|
|
|
|
|
std::vector<const_buffer> b2{
|
|
|
|
|
const_buffer{buf+0, 1},
|
|
|
|
|
const_buffer{buf+1, 2}};
|
|
|
|
|
std::list<const_buffer> b3;
|
|
|
|
|
std::array<const_buffer, 3> b4{{
|
|
|
|
|
const_buffer{buf+3, 1},
|
|
|
|
|
const_buffer{buf+4, 2},
|
|
|
|
|
const_buffer{buf+6, 3}}};
|
|
|
|
|
std::list<const_buffer> b5{
|
|
|
|
|
const_buffer{buf+9, 1}};
|
|
|
|
|
std::list<const_buffer> b6;
|
|
|
|
|
auto bs = buffer_cat(
|
|
|
|
|
b1, b2, b3, b4, b5, b6);
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(buffer_size(bs) == 10);
|
2016-11-08 19:00:40 -05:00
|
|
|
BEAST_EXPECT(bsize1(bs) == 10);
|
|
|
|
|
BEAST_EXPECT(bsize2(bs) == 10);
|
|
|
|
|
BEAST_EXPECT(bsize3(bs) == 10);
|
|
|
|
|
BEAST_EXPECT(bsize4(bs) == 10);
|
2017-07-20 08:01:46 -07:00
|
|
|
std::vector<const_buffer> v;
|
|
|
|
|
for(auto iter = make_reverse_iterator(bs.end());
|
|
|
|
|
iter != make_reverse_iterator(bs.begin()); ++iter)
|
|
|
|
|
v.emplace_back(*iter);
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(buffer_size(bs) == 10);
|
2017-07-20 08:01:46 -07:00
|
|
|
decltype(bs) bs2(bs);
|
|
|
|
|
auto bs3(std::move(bs));
|
|
|
|
|
{
|
|
|
|
|
boost::asio::streambuf sb1, sb2;
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(buffer_size(buffer_cat(
|
2017-07-20 08:01:46 -07:00
|
|
|
sb1.prepare(5), sb2.prepare(7))) == 12);
|
|
|
|
|
sb1.commit(5);
|
|
|
|
|
sb2.commit(7);
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(buffer_size(buffer_cat(
|
2017-07-20 08:01:46 -07:00
|
|
|
sb1.data(), sb2.data())) == 12);
|
|
|
|
|
}
|
2016-05-18 12:30:15 -04:00
|
|
|
for(auto it = bs.begin(); it != bs.end(); ++it)
|
|
|
|
|
{
|
|
|
|
|
decltype(bs)::const_iterator copy;
|
|
|
|
|
copy = it;
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(copy == it);
|
2016-05-18 12:30:15 -04:00
|
|
|
copy = copy;
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(copy == it);
|
2016-05-18 12:30:15 -04:00
|
|
|
}
|
2017-07-20 08:01:46 -07:00
|
|
|
}
|
|
|
|
|
|
2016-05-04 17:27:50 -04:00
|
|
|
void testIterators()
|
|
|
|
|
{
|
|
|
|
|
using boost::asio::buffer_size;
|
|
|
|
|
using boost::asio::const_buffer;
|
|
|
|
|
char buf[9];
|
|
|
|
|
std::vector<const_buffer> b1{
|
|
|
|
|
const_buffer{buf+0, 1},
|
|
|
|
|
const_buffer{buf+1, 2}};
|
|
|
|
|
std::array<const_buffer, 3> b2{{
|
|
|
|
|
const_buffer{buf+3, 1},
|
|
|
|
|
const_buffer{buf+4, 2},
|
|
|
|
|
const_buffer{buf+6, 3}}};
|
|
|
|
|
auto bs = buffer_cat(b1, b2);
|
2016-05-07 17:06:46 -04:00
|
|
|
for(int n = 0;
|
|
|
|
|
n <= std::distance(bs.begin(), bs.end()); ++n)
|
|
|
|
|
{
|
|
|
|
|
auto it = std::next(bs.begin(), n);
|
|
|
|
|
decltype(it) it2(std::move(it));
|
|
|
|
|
it = std::move(it2);
|
|
|
|
|
auto pit = ⁢
|
|
|
|
|
it = std::move(*pit);
|
|
|
|
|
}
|
2016-05-04 17:27:50 -04:00
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.begin(); n < 100; ++it)
|
|
|
|
|
++n;
|
|
|
|
|
fail();
|
|
|
|
|
}
|
|
|
|
|
catch(std::exception const&)
|
|
|
|
|
{
|
|
|
|
|
pass();
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-08 13:15:07 -05:00
|
|
|
// decrement iterator
|
2017-05-28 13:59:01 -07:00
|
|
|
/* VFALCO
|
|
|
|
|
This causes a mysterious "uninitialized variable"
|
|
|
|
|
warning related to this function (see comment)
|
|
|
|
|
https://code.woboq.org/qt5/include/c++/6.3.1/bits/stl_iterator.h.html#159
|
|
|
|
|
*/
|
|
|
|
|
#if 0
|
2016-11-08 13:15:07 -05:00
|
|
|
{
|
|
|
|
|
auto const rbegin =
|
|
|
|
|
make_reverse_iterator(bs.end());
|
|
|
|
|
auto const rend =
|
|
|
|
|
make_reverse_iterator(bs.begin());
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = rbegin; it != rend; ++it)
|
|
|
|
|
n += buffer_size(*it);
|
|
|
|
|
BEAST_EXPECT(n == 9);
|
|
|
|
|
}
|
2017-05-28 13:59:01 -07:00
|
|
|
#endif
|
2016-11-08 13:15:07 -05:00
|
|
|
|
2016-05-04 17:27:50 -04:00
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
std::size_t n = 0;
|
|
|
|
|
for(auto it = bs.end(); n < 100; --it)
|
|
|
|
|
++n;
|
|
|
|
|
fail();
|
|
|
|
|
}
|
|
|
|
|
catch(std::exception const&)
|
|
|
|
|
{
|
|
|
|
|
pass();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2016-04-30 10:29:39 -04:00
|
|
|
buffer_size(*bs.end());
|
|
|
|
|
fail();
|
2016-05-04 17:27:50 -04:00
|
|
|
}
|
|
|
|
|
catch(std::exception const&)
|
|
|
|
|
{
|
|
|
|
|
pass();
|
|
|
|
|
}
|
|
|
|
|
auto bs2 = bs;
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(bs.begin() != bs2.begin());
|
|
|
|
|
BEAST_EXPECT(bs.end() != bs2.end());
|
2016-05-04 17:27:50 -04:00
|
|
|
decltype(bs)::const_iterator it;
|
|
|
|
|
decltype(bs2)::const_iterator it2;
|
2016-08-03 15:34:23 -04:00
|
|
|
BEAST_EXPECT(it == it2);
|
2016-05-04 17:27:50 -04:00
|
|
|
}
|
|
|
|
|
|
2017-07-20 08:01:46 -07:00
|
|
|
void run() override
|
|
|
|
|
{
|
2016-11-08 19:00:40 -05:00
|
|
|
using boost::asio::const_buffer;
|
|
|
|
|
using boost::asio::const_buffers_1;
|
|
|
|
|
using boost::asio::mutable_buffer;
|
|
|
|
|
using boost::asio::mutable_buffers_1;
|
|
|
|
|
struct user_defined : mutable_buffer
|
|
|
|
|
{
|
|
|
|
|
};
|
|
|
|
|
|
2017-05-16 07:34:09 -07:00
|
|
|
// Check is_all_const_buffer_sequence
|
2017-05-23 12:33:31 -07:00
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
detail::is_all_const_buffer_sequence<const_buffers_1>::value);
|
|
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
detail::is_all_const_buffer_sequence<const_buffers_1, const_buffers_1>::value);
|
|
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
detail::is_all_const_buffer_sequence<mutable_buffers_1>::value);
|
|
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
detail::is_all_const_buffer_sequence<mutable_buffers_1, mutable_buffers_1>::value);
|
|
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
detail::is_all_const_buffer_sequence<const_buffers_1, mutable_buffers_1>::value);
|
|
|
|
|
BOOST_STATIC_ASSERT(
|
|
|
|
|
! detail::is_all_const_buffer_sequence<const_buffers_1, mutable_buffers_1, int>::value);
|
2017-02-06 20:07:03 -05:00
|
|
|
|
2016-11-08 19:00:40 -05:00
|
|
|
// Ensure that concatenating mutable buffer
|
|
|
|
|
// sequences results in a mutable buffer sequence
|
2017-05-23 12:33:31 -07:00
|
|
|
BOOST_STATIC_ASSERT(std::is_same<
|
2016-11-08 19:00:40 -05:00
|
|
|
mutable_buffer,
|
|
|
|
|
decltype(buffer_cat(
|
|
|
|
|
std::declval<mutable_buffer>(),
|
|
|
|
|
std::declval<user_defined>(),
|
|
|
|
|
std::declval<mutable_buffer>()
|
2017-05-23 12:33:31 -07:00
|
|
|
))::value_type>::value);
|
2016-11-08 19:00:40 -05:00
|
|
|
|
|
|
|
|
// Ensure that concatenating mixed buffer
|
|
|
|
|
// sequences results in a const buffer sequence.
|
2017-05-23 12:33:31 -07:00
|
|
|
BOOST_STATIC_ASSERT(std::is_same<
|
2016-11-08 19:00:40 -05:00
|
|
|
const_buffer,
|
|
|
|
|
decltype(buffer_cat(
|
|
|
|
|
std::declval<mutable_buffer>(),
|
|
|
|
|
std::declval<user_defined>(),
|
|
|
|
|
std::declval<const_buffer>()
|
2017-05-23 12:33:31 -07:00
|
|
|
))::value_type>::value);
|
2016-11-08 19:00:40 -05:00
|
|
|
|
2017-07-20 08:01:46 -07:00
|
|
|
testBufferCat();
|
2016-05-04 17:27:50 -04:00
|
|
|
testIterators();
|
2017-07-20 08:01:46 -07:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
BEAST_DEFINE_TESTSUITE(buffer_cat,core,beast);
|
|
|
|
|
|
|
|
|
|
} // beast
|