I want to send two files to the client, the first file is img.jpg and the second file is message.txt
The first file img.jpg is received correctly, but the file message.txt is received with size zero
client output is:
img.jpg: 49152
message.txt: 4096
read: End of file [asio.misc:2]
here are my client and server codes:
server.cpp:
#include <iostream>
#include <string>
#include <fstream>
#include <vector>
#include <boost/asio.hpp>
int main(int argc, char* argv[])
{
try
{
boost::asio::io_context io;
std::cout << "Server Start\n";
boost::asio::ip::tcp::acceptor acc(io,
boost::asio::ip::tcp::endpoint(
boost::asio::ip::tcp::v4(), 6666));
for (;;) {
boost::asio::ip::tcp::socket sock(io);
acc.accept(sock);
std::vector<std::string> names{ "img.jpg" , "message.txt"};
std::vector<int> sizes{ 49152 , 4096 };
for (int i = 0; i < 2; ++i) {
//Send Header
boost::asio::streambuf reply;
std::ostream header(&reply);
header << names[i] << " ";
header << std::to_string(sizes[i]) << " ";
header << "\r\n";
boost::asio::write(sock, reply);
//Send Bytes
std::ifstream input(names[i], std::ifstream::binary);
std::vector<char> vec(sizes[i]);
input.read(&vec[i], sizes[i]);
boost::asio::write(sock, boost::asio::buffer(vec, sizes[i]));
}
sock.close();
}
acc.close();
}
catch (std::exception& e)
{
std::cerr << e.what() << std::endl;
}
return 0;
}
client.cpp:
#include <iostream>
#include <string>
#include <fstream>
#include <vector>
#include <boost/asio.hpp>
int main(int argc, char* argv[])
{
try
{
boost::asio::io_context io;
boost::asio::ip::tcp::resolver resolv(io);
boost::asio::ip::tcp::resolver::query q("127.0.0.1", "6666");
boost::asio::ip::tcp::resolver::iterator ep = resolv.resolve(q);
boost::asio::ip::tcp::socket sock(io);
boost::asio::connect(sock, ep);
//Get Files
for (int i = 0; i < 2; ++i) {
//Read Header
boost::asio::streambuf reply;
boost::asio::read_until(sock, reply, "\r\n");
std::istream header(&reply);
std::string fileName;
int fileSize;
header >> fileName;
header >> fileSize;
std::cout << fileName << ": " << fileSize << '\n';
//Read File Data
std::ofstream output(fileName, std::ofstream::binary | std::ofstream::app);
std::vector<char> vec(fileSize);
boost::asio::read(sock, boost::asio::buffer(vec, vec.size()));
output.write(&vec[0], vec.size());
output.close();
}
sock.close();
}
catch (const std::exception& e) {
std::cerr << e.what() << std::endl;
}
return 0;
}
There are a number of issues.
&vec[i]
is a bug in the server side here:If
i>0
(which it always is except on the first run) you will addressvec
out-of-bounds because size is notsizes[i]+i
your header format is sloppy: filenames with spaces will cause UB
os << to_string(n)
should just beos << n
server uses a delimiter
"\r\n"
that the client... completely ignores. All your files begin at least with\r\n
that wasn't consumed, and what's worse, the last two characters of the file will then be read as part of the next file's header. This, at best, will fail, but can lead to UBIn fact, it will always lead to UB because there's a complete lack of error handling on the client side
I notice now that you ALMOST skirt the issue by using a separate buffer for the header (
streambuf header;
) and the contents (directly intovec
). However,read_until
documents that it may read past the delimiter¹. So, you should have written any remaining data fromstreambuf
and subtract the length from the amount to still read.In short, recommend to use separate, exact size buffers OR one
DynamicBuffer
(likestreambuf
) per stream.The same issue is with the client using a new buffer each iteration through the loop:
It should at least be outside the loop so any excess data received will correctly be used on the next iteration
You usually want to deal with partial success of reads (i.e. accept data received together with EOF condition). Here it should not affect correctness because you are precisely limiting the body read to the expected size, but it is still a good habit
You specify redundant buffer size in
asio::buffer(vec, vec.size())
, this merely invites bugs. Leave them away to get the same behaviour without the risk of getting the wrong size:asio::buffer(vec)
(e.g. it would avoid the UB mentioned earlier)Demonstrating The Buffer Issues
A combined server/client with halfway fixes: https://coliru.stacked-crooked.com/a/03bee101ff6e8a7a
The client side adds a lot error handling
This allows us to demonstrate the problem with the
streambuf
reading beyond the delimiter:Or my local test:
The clunky/naive way to fix might seem to be something like:
And while it might appear to work correctly:
It just invites new problems with small files, where the entire following files are "accidentally" used as the contents for the current file. Instead, just SayWhatYouMean(TM):
Full Fixes
Also getting a file list from the command line instead of hardcoding files/sizes, and writing to an output directory for safety.
Note that it now uses
std::filesystem::path
wich already usesstd::quoted
under the hood to protect against problems with filenames with spaces: https://en.cppreference.com/w/cpp/filesystem/path/operator_ltltgtgtLive On Coliru
With the live test of
Printed:
BONUS
Instead of doing text-based IO, consider sending binary file size and name information. See for examples: https://stackoverflow.com/search?tab=newest&q=user%3a85371%20endian%20file&searchOn=3
UPDATE: Turns out bonus take removes ALL the complexity: no more streambuf, unbounded reads, partial success, parsing, error handling.
Given
You can now send evrything in one go:
And on the receiving side:
See it Live On Coliru
¹ because of how TCP packet delivery works; this is how all libraries behave