Files
boost_beast/test/beast/http/file_body.cpp
2022-04-17 12:30:52 -04:00

223 lines
6.4 KiB
C++

//
// Copyright (c) 2016-2019 Vinnie Falco (vinnie dot falco at gmail dot com)
//
// 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)
//
// Official repository: https://github.com/boostorg/beast
//
// Test that header file is self-contained.
#include <boost/beast/http/file_body.hpp>
#include <boost/beast/core/buffer_traits.hpp>
#include <boost/beast/core/buffers_prefix.hpp>
#include <boost/beast/core/file_stdio.hpp>
#include <boost/beast/core/flat_buffer.hpp>
#include <boost/beast/core/static_string.hpp>
#include <boost/beast/http/parser.hpp>
#include <boost/beast/http/serializer.hpp>
#include <boost/beast/_experimental/unit_test/suite.hpp>
#include <boost/filesystem.hpp>
#include <boost/asio/error.hpp>
#include <fstream>
namespace boost {
namespace beast {
namespace http {
class file_body_test : public beast::unit_test::suite
{
public:
struct lambda
{
flat_buffer buffer;
template<class ConstBufferSequence>
void
operator()(error_code&, ConstBufferSequence const& buffers)
{
buffer.commit(net::buffer_copy(
buffer.prepare(buffer_bytes(buffers)),
buffers));
}
};
struct temp_file
{
temp_file(std::ostream& logger)
: path_(boost::filesystem::unique_path())
, log_(logger)
{}
~temp_file()
{
if (!path_.empty())
{
auto ec = error_code();
boost::filesystem::remove(path_, ec);
if (ec)
{
log_ << "warning: failed to remove temporary file: " << path_ << "\n";
}
}
}
temp_file(temp_file&&) = default;
temp_file(temp_file const&) = delete;
temp_file& operator=(temp_file&&) = delete;
temp_file& operator=(temp_file const&) = delete;
boost::filesystem::path const& path() const
{
return path_;
}
private:
boost::filesystem::path path_;
std::ostream& log_;
};
template<class File>
void
doTestFileBody()
{
error_code ec;
string_view const s =
"HTTP/1.1 200 OK\r\n"
"Server: test\r\n"
"Content-Length: 3\r\n"
"\r\n"
"xyz";
auto const temp = boost::filesystem::unique_path();
{
response_parser<basic_file_body<File>> p;
p.eager(true);
p.get().body().open(
temp.string<std::string>().c_str(), file_mode::write, ec);
BEAST_EXPECTS(! ec, ec.message());
p.put(net::buffer(s.data(), s.size()), ec);
BEAST_EXPECTS(! ec, ec.message());
}
{
File f;
f.open(temp.string<std::string>().c_str(), file_mode::read, ec);
auto size = f.size(ec);
BEAST_EXPECTS(! ec, ec.message());
BEAST_EXPECT(size == 3);
std::string s1;
s1.resize(3);
f.read(&s1[0], s1.size(), ec);
BEAST_EXPECTS(! ec, ec.message());
BEAST_EXPECTS(s1 == "xyz", s);
}
{
lambda visit;
{
response<basic_file_body<File>> res{status::ok, 11};
res.set(field::server, "test");
res.body().open(temp.string<std::string>().c_str(),
file_mode::scan, ec);
BEAST_EXPECTS(! ec, ec.message());
res.prepare_payload();
serializer<false, basic_file_body<File>, fields> sr{res};
sr.next(ec, visit);
BEAST_EXPECTS(! ec, ec.message());
auto const b = buffers_front(visit.buffer.data());
string_view const s1{
reinterpret_cast<char const*>(b.data()),
b.size()};
BEAST_EXPECTS(s1 == s, s1);
}
}
boost::filesystem::remove(temp, ec);
BEAST_EXPECTS(! ec, ec.message());
}
template<class File>
void
fileBodyUnexpectedEofOnGet()
{
auto temp = temp_file(log);
error_code ec;
string_view const ten =
"0123456789"; // 40
// create the temporary file
{
std::ofstream fstemp(temp.path().native());
std::size_t written = 0;
std::size_t to_write = 4097;
while (written < to_write)
{
auto size = (std::min)(ten.size(), to_write - written);
fstemp << ten.substr(0, size);
BEAST_EXPECT(fstemp.good());
written += size;
}
fstemp.close();
}
// open the file and read the header
{
using file_body_type = basic_file_body<File>;
typename file_body_type::value_type value;
// opened in write mode so we can truncate later
value.open(temp.path().string<std::string>().c_str(), file_mode::read, ec);
BEAST_EXPECTS(! ec, ec.message());
response_header<> header;
header.version(11);
header.result(status::accepted);
header.set(field::server, "test");
header.set(field::content_length, "4097");
typename file_body_type::writer w(header, value);
auto maybe_range = w.get(ec);
BEAST_EXPECTS(!ec, ec.message());
BEAST_EXPECTS(maybe_range.has_value(), "no range returned");
BEAST_EXPECT(maybe_range.value().second);
value.file().seek(4097, ec);
BEAST_EXPECTS(!ec, ec.message());
maybe_range = w.get(ec);
BEAST_EXPECTS(ec == error::short_read, ec.message());
BEAST_EXPECTS(!maybe_range.has_value(), "range returned on error");
}
}
void
run() override
{
doTestFileBody<file_stdio>();
#if BOOST_BEAST_USE_WIN32_FILE
doTestFileBody<file_win32>();
#endif
#if BOOST_BEAST_USE_POSIX_FILE
doTestFileBody<file_posix>();
#endif
fileBodyUnexpectedEofOnGet<file_stdio>();
#if BOOST_BEAST_USE_POSIX_FILE
fileBodyUnexpectedEofOnGet<file_posix>();
#endif
#if BOOST_BEAST_USE_WIN32_FILE
fileBodyUnexpectedEofOnGet<file_win32>();
#endif
}
};
BEAST_DEFINE_TESTSUITE(beast,http,file_body);
} // http
} // beast
} // boost