我有一个客户端应用程序,我需要从服务器接收 http“长时间运行的请求”。我发送一个命令,在获得响应的标头后,我必须接收分隔的 json 数据,\r\n
直到连接终止。
我设法调整boost beast 客户端示例来发送消息并接收标头并解析它并接收来自服务器的响应。但是,我未能找到一种方法来序列化数据以便处理 json 消息。
可以在这个继电器示例中找到最接近该问题的演示。在那个例子中(p
是解析器,sr
是序列化器,input
是套接字输入流,output
是套接字输出流),在读取 http 标头后,我们有一个从服务器连续读取的循环:
do
{
if(! p.is_done())
{
// Set up the body for writing into our small buffer
p.get().body().data = buf;
p.get().body().size = sizeof(buf);
// Read as much as we can
read(input, buffer, p, ec);
// This error is returned when buffer_body uses up the buffer
if(ec == error::need_buffer)
ec = {};
if(ec)
return;
// Set up the body for reading.
// This is how much was parsed:
p.get().body().size = sizeof(buf) - p.get().body().size;
p.get().body().data = buf;
p.get().body().more = ! p.is_done();
}
else
{
p.get().body().data = nullptr;
p.get().body().size = 0;
}
// Write everything in the buffer (which might be empty)
write(output, sr, ec);
// This error is returned when buffer_body uses up the buffer
if(ec == error::need_buffer)
ec = {};
if(ec)
return;
}
while(! p.is_done() && ! sr.is_done());
这里有一些我不明白的事情:
- 我们已经读完了标题。为什么我们需要 boost beast 而不是 boost asio 来读取原始 tcp 消息?当我尝试这样做时(同时使用 async_read/async_read_some),我得到了一个零大小的无限读取。
- 解析器的文档说(在页面末尾)每条消息都需要一个新实例,但我在示例中没有看到。
- 由于 tcp 消息读取不起作用,有没有办法将解析器/序列化器数据转换为某种字符串?甚至以先进先出的方式将其写入文本文件,以便我可以使用一些 json 库来处理它?我不想像示例那样使用另一个套接字。
该函数boost::beast::buffers()
无法为解析器和序列化程序编译,并且解析器没有消费函数,序列化程序的消费似乎是针对消息的特定 http 部分,如果我为body()
.
除此之外,我还未能从解析器和缓冲区中获取一致的数据块 old-school std::copy
。我似乎不明白如何将数据组合在一起以获得数据流。在接收数据的任何时候使用缓冲区.consume()
都会导致need buffer
错误。
我真的很感谢有人解释所有这些应该如何协同工作的逻辑。