"Connection closed" 使用 QNetworkAccessManager 和 QTcpServer 时

"Connection closed" When using QNetworkAccessManager and QTcpServer

本文关键字:QTcpServer 使用 Connection closed QNetworkAccessManager      更新时间:2023-10-16

我有一个简单的服务器与QTcpServer和简单的客户端与QNetworkAccessManager

  1. 当我通过curl或浏览器从服务器请求数据时,一切正常
  2. 当我通过QNetworkAccessManager从任何站点请求数据时,一切都ok
  3. 但是我不能通过QNetworkAccessManager从QTcpServer读取数据。所有请求都被重置。QNetworkAccessManager(客户端)在收到服务器的数据后立即发送RST(重置连接)。在客户端代码中,我们得到错误:"连接关闭"(RemoteHostClosedError)
  4. 此外,我尝试使用QNetworkAccessManager从DownloadManager的例子和QTcpServer从FortuneServer的例子在各种组合,但结果是相同的。

已测试Qt版本:

  • Mac Qt 5.7
  • Linux Qt 5.7Linux Qt 5.6.2Linux Qt 5.5.1

Wireshark截图:qt-wireshark.png
上面的部分(红线)是QNetworkAccessManager的结果,最新的数据包成功的结果是curl尝试从QTcpServer获取数据

也有一个简单的例子来重现错误:testNetwork.zip

下面是客户端的示例代码:

void test(quint16 port)
{
    QNetworkAccessManager *manager = new QNetworkAccessManager();
    QNetworkRequest request;
    request.setUrl(QUrl(QString("http://127.0.0.1:%1/").arg(port)));
    manager->connect(manager, &QNetworkAccessManager::finished,
                     [](QNetworkReply *reply) {
        qDebug() << QString("Finished. %1. %2").arg(reply->errorString()).arg(reply->error());
        qDebug() << "readed: " << reply->readAll();
    });
    QNetworkReply *reply = manager->get(request);
    reply->connect(reply, &QNetworkReply::readyRead, [reply]() {
        qDebug() << QString("readyRead: '%1'").arg(QString(reply->readAll()));
    });
}

对于服务器:

QTcpSocket socket;
...
if(socket.waitForReadyRead(5000))
{
    QByteArray request;
    request += socket.readAll();
    QByteArray responce("HELLO, WORLD! HELLO, WORLD! HELLO, WORLD! HELLO, WORLD!");
    socket.write(responce);
    if(!socket.waitForBytesWritten())
    {
        qWarning() << QString("Error occurred in waitForBytesWritten() method of the tcp socket. %1 (%2)")
                      .arg(socket.errorString())
                      .arg(socket.error());
    }
}
else
{
    qWarning() << QString("Error occurred in read method of the tcp socket. %1 (%2)")
                  .arg(socket.errorString())
                  .arg(socket.error());
}

我还在qt.io上创建了一个Bug报告(QTBUG-56631)

您的客户端正在发出HTTP请求,但您的服务器不是HTTP服务器-它没有发送回有效的HTTP请求。

你的客户端工作时,你指向一个web服务器,因为这是一个http服务器。

    QByteArray responce("HELLO, WORLD! HELLO, WORLD! HELLO, WORLD! HELLO, WORLD!");

不是有效的HTTP响应