05fc3aef53
Replace the current license disclaimer in files by a SPDX-License-Identifier. Files that have to be modified by hand are modified. License files are organized under LICENSES directory. Task-number: QTBUG-67283 Change-Id: Id880c92784c40f3bbde861c0d93f58151c18b9f1 Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Lars Knoll <lars.knoll@qt.io> Reviewed-by: Jörg Bornemann <joerg.bornemann@qt.io>
746 lines
25 KiB
C++
746 lines
25 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
|
|
#include <QtTest/QTest>
|
|
#include <QtTest/QTestEventLoop>
|
|
#include <QtCore/QQueue>
|
|
#include <QtCore/QString>
|
|
#include <QtCore/QCoreApplication>
|
|
|
|
#include <private/qhttpsocketengine_p.h>
|
|
#include <qhostinfo.h>
|
|
#include <qhostaddress.h>
|
|
#include <qtcpsocket.h>
|
|
#include <qdebug.h>
|
|
#include <qelapsedtimer.h>
|
|
#include <qtcpserver.h>
|
|
|
|
#include "../../../network-settings.h"
|
|
|
|
class tst_QHttpSocketEngine : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
private slots:
|
|
void initTestCase();
|
|
void init();
|
|
void construction();
|
|
void errorTest_data();
|
|
void errorTest();
|
|
void simpleConnectToIMAP();
|
|
void simpleErrorsAndStates();
|
|
|
|
void tcpSocketBlockingTest();
|
|
void tcpSocketNonBlockingTest();
|
|
void downloadBigFile();
|
|
// void tcpLoopbackPerformance();
|
|
void passwordAuth();
|
|
void ensureEofTriggersNotification();
|
|
|
|
protected slots:
|
|
void tcpSocketNonBlocking_hostFound();
|
|
void tcpSocketNonBlocking_connected();
|
|
void tcpSocketNonBlocking_closed();
|
|
void tcpSocketNonBlocking_readyRead();
|
|
void tcpSocketNonBlocking_bytesWritten(qint64);
|
|
void exitLoopSlot();
|
|
void downloadBigFileSlot();
|
|
|
|
private:
|
|
QTcpSocket *tcpSocketNonBlocking_socket;
|
|
QStringList tcpSocketNonBlocking_data;
|
|
qint64 tcpSocketNonBlocking_totalWritten;
|
|
QTcpSocket *tmpSocket;
|
|
qint64 bytesAvailable;
|
|
};
|
|
|
|
class MiniHttpServer: public QTcpServer
|
|
{
|
|
Q_OBJECT
|
|
QTcpSocket *client;
|
|
QList<QByteArray> dataToTransmit;
|
|
|
|
public:
|
|
QByteArray receivedData;
|
|
|
|
MiniHttpServer(const QList<QByteArray> &data) : client(0), dataToTransmit(data)
|
|
{
|
|
listen();
|
|
connect(this, SIGNAL(newConnection()), this, SLOT(doAccept()));
|
|
}
|
|
|
|
public slots:
|
|
void doAccept()
|
|
{
|
|
client = nextPendingConnection();
|
|
connect(client, SIGNAL(readyRead()), this, SLOT(sendData()));
|
|
}
|
|
|
|
void sendData()
|
|
{
|
|
receivedData += client->readAll();
|
|
int idx = client->property("dataTransmitionIdx").toInt();
|
|
if (receivedData.contains("\r\n\r\n") ||
|
|
receivedData.contains("\n\n")) {
|
|
if (idx < dataToTransmit.length())
|
|
client->write(dataToTransmit.at(idx++));
|
|
if (idx == dataToTransmit.length()) {
|
|
client->disconnectFromHost();
|
|
disconnect(client, 0, this, 0);
|
|
client = 0;
|
|
} else {
|
|
client->setProperty("dataTransmitionIdx", idx);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
void tst_QHttpSocketEngine::initTestCase()
|
|
{
|
|
#ifdef QT_TEST_SERVER
|
|
QVERIFY(QtNetworkSettings::verifyConnection(QtNetworkSettings::httpProxyServerName(), 3128));
|
|
QVERIFY(QtNetworkSettings::verifyConnection(QtNetworkSettings::socksProxyServerName(), 1080));
|
|
QVERIFY(QtNetworkSettings::verifyConnection(QtNetworkSettings::httpServerName(), 80));
|
|
QVERIFY(QtNetworkSettings::verifyConnection(QtNetworkSettings::imapServerName(), 143));
|
|
#else
|
|
if (!QtNetworkSettings::verifyTestNetworkSettings())
|
|
QSKIP("No network test server available");
|
|
#endif
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::init()
|
|
{
|
|
tmpSocket = 0;
|
|
bytesAvailable = 0;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
void tst_QHttpSocketEngine::construction()
|
|
{
|
|
QHttpSocketEngine socketDevice;
|
|
|
|
QVERIFY(!socketDevice.isValid());
|
|
|
|
// Initialize device
|
|
QVERIFY(socketDevice.initialize(QAbstractSocket::TcpSocket, QAbstractSocket::IPv4Protocol));
|
|
QVERIFY(socketDevice.isValid());
|
|
QCOMPARE(socketDevice.protocol(), QAbstractSocket::IPv4Protocol);
|
|
QCOMPARE(socketDevice.socketType(), QAbstractSocket::TcpSocket);
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
// QVERIFY(socketDevice.socketDescriptor() != -1);
|
|
QCOMPARE(socketDevice.localAddress(), QHostAddress());
|
|
QCOMPARE(socketDevice.localPort(), quint16(0));
|
|
QCOMPARE(socketDevice.peerAddress(), QHostAddress());
|
|
QCOMPARE(socketDevice.peerPort(), quint16(0));
|
|
QCOMPARE(socketDevice.error(), QAbstractSocket::UnknownSocketError);
|
|
|
|
//QTest::ignoreMessage(QtWarningMsg, "QSocketLayer::bytesAvailable() was called in QAbstractSocket::UnconnectedState");
|
|
QCOMPARE(socketDevice.bytesAvailable(), 0);
|
|
|
|
//QTest::ignoreMessage(QtWarningMsg, "QSocketLayer::hasPendingDatagrams() was called in QAbstractSocket::UnconnectedState");
|
|
QVERIFY(!socketDevice.hasPendingDatagrams());
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
void tst_QHttpSocketEngine::errorTest_data()
|
|
{
|
|
QTest::addColumn<QString>("hostname");
|
|
QTest::addColumn<int>("port");
|
|
QTest::addColumn<QString>("username");
|
|
QTest::addColumn<QString>("response");
|
|
QTest::addColumn<int>("expectedError");
|
|
|
|
QQueue<QByteArray> responses;
|
|
QTest::newRow("proxy-host-not-found") << "this-host-does-not-exist." << 1080 << QString()
|
|
<< QString()
|
|
<< int(QAbstractSocket::ProxyNotFoundError);
|
|
QTest::newRow("proxy-connection-refused") << QtNetworkSettings::socksProxyServerName() << 2 << QString()
|
|
<< QString()
|
|
<< int(QAbstractSocket::ProxyConnectionRefusedError);
|
|
|
|
QTest::newRow("garbage1") << QString() << 0 << QString()
|
|
<< "This is not HTTP\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyProtocolError);
|
|
|
|
QTest::newRow("garbage2") << QString() << 0 << QString()
|
|
<< "This is not HTTP"
|
|
<< int(QAbstractSocket::ProxyProtocolError);
|
|
|
|
QTest::newRow("garbage3") << QString() << 0 << QString()
|
|
<< ""
|
|
<< int(QAbstractSocket::ProxyConnectionClosedError);
|
|
|
|
QTest::newRow("forbidden") << QString() << 0 << QString()
|
|
<< "HTTP/1.0 403 Forbidden\r\n\r\n"
|
|
<< int(QAbstractSocket::SocketAccessError);
|
|
|
|
QTest::newRow("method-not-allowed") << QString() << 0 << QString()
|
|
<< "HTTP/1.0 405 Method Not Allowed\r\n\r\n"
|
|
<< int(QAbstractSocket::SocketAccessError);
|
|
|
|
QTest::newRow("proxy-authentication-too-short")
|
|
<< QString() << 0 << "foo"
|
|
<< "HTTP/1.0 407 Proxy Authentication Required\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyProtocolError);
|
|
|
|
QTest::newRow("proxy-authentication-invalid-method")
|
|
<< QString() << 0 << "foo"
|
|
<< "HTTP/1.0 407 Proxy Authentication Required\r\n"
|
|
"Proxy-Authenticate: Frobnicator\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyProtocolError);
|
|
|
|
QTest::newRow("proxy-authentication-required")
|
|
<< QString() << 0 << "foo"
|
|
<< "HTTP/1.0 407 Proxy Authentication Required\r\n"
|
|
"Proxy-Connection: close\r\n"
|
|
"Proxy-Authenticate: Basic, realm=wonderland\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyAuthenticationRequiredError);
|
|
|
|
QTest::newRow("proxy-authentication-required2")
|
|
<< QString() << 0 << "foo"
|
|
<< "HTTP/1.0 407 Proxy Authentication Required\r\n"
|
|
"Proxy-Connection: keep-alive\r\n"
|
|
"Proxy-Authenticate: Basic, realm=wonderland\r\n\r\n"
|
|
"\1"
|
|
"HTTP/1.0 407 Proxy Authentication Required\r\n"
|
|
"Proxy-Authenticate: Basic, realm=wonderland\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyAuthenticationRequiredError);
|
|
|
|
QTest::newRow("proxy-authentication-required-noclose")
|
|
<< QString() << 0 << "foo"
|
|
<< "HTTP/1.0 407 Proxy Authentication Required\r\n"
|
|
"Proxy-Authenticate: Basic\r\n"
|
|
"\r\n"
|
|
<< int(QAbstractSocket::ProxyAuthenticationRequiredError);
|
|
|
|
QTest::newRow("connection-refused") << QString() << 0 << QString()
|
|
<< "HTTP/1.0 503 Service Unavailable\r\n\r\n"
|
|
<< int(QAbstractSocket::ConnectionRefusedError);
|
|
|
|
QTest::newRow("host-not-found") << QString() << 0 << QString()
|
|
<< "HTTP/1.0 404 Not Found\r\n\r\n"
|
|
<< int(QAbstractSocket::HostNotFoundError);
|
|
|
|
QTest::newRow("weird-http-reply") << QString() << 0 << QString()
|
|
<< "HTTP/1.0 206 Partial Content\r\n\r\n"
|
|
<< int(QAbstractSocket::ProxyProtocolError);
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::errorTest()
|
|
{
|
|
QFETCH(QString, hostname);
|
|
QFETCH(int, port);
|
|
QFETCH(QString, username);
|
|
QFETCH(QString, response);
|
|
QFETCH(int, expectedError);
|
|
|
|
MiniHttpServer server(response.toLatin1().split('\1'));
|
|
|
|
if (hostname.isEmpty()) {
|
|
hostname = "127.0.0.1";
|
|
port = server.serverPort();
|
|
}
|
|
QTcpSocket socket;
|
|
socket.setProxy(QNetworkProxy(QNetworkProxy::HttpProxy, hostname, port, username, username));
|
|
socket.connectToHost("0.1.2.3", 12345);
|
|
|
|
connect(&socket, SIGNAL(errorOccurred(QAbstractSocket::SocketError)),
|
|
&QTestEventLoop::instance(), SLOT(exitLoop()));
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
QVERIFY(!QTestEventLoop::instance().timeout());
|
|
|
|
QCOMPARE(int(socket.error()), expectedError);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
void tst_QHttpSocketEngine::simpleConnectToIMAP()
|
|
{
|
|
QHttpSocketEngine socketDevice;
|
|
|
|
// Initialize device
|
|
QVERIFY(socketDevice.initialize(QAbstractSocket::TcpSocket, QAbstractSocket::IPv4Protocol));
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
|
|
socketDevice.setProxy(QNetworkProxy(QNetworkProxy::HttpProxy, QtNetworkSettings::httpProxyServerName(), 3128));
|
|
QVERIFY(!socketDevice.connectToHost(QtNetworkSettings::imapServerIp(), 143));
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::ConnectingState);
|
|
QVERIFY(socketDevice.waitForWrite());
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::ConnectedState);
|
|
QCOMPARE(socketDevice.peerAddress(), QtNetworkSettings::imapServerIp());
|
|
QVERIFY(!socketDevice.localAddress().isNull());
|
|
QVERIFY(socketDevice.localPort() > 0);
|
|
|
|
// Wait for the greeting
|
|
QVERIFY(socketDevice.waitForRead());
|
|
|
|
// Read the greeting
|
|
qint64 available = int(socketDevice.bytesAvailable());
|
|
QVERIFY(available > 0);
|
|
QByteArray array;
|
|
array.resize(int(available));
|
|
QVERIFY(socketDevice.read(array.data(), array.size()) == available);
|
|
|
|
// Check that the greeting is what we expect it to be
|
|
QVERIFY2(QtNetworkSettings::compareReplyIMAP(array), array.constData());
|
|
|
|
|
|
// Write a logout message
|
|
QByteArray array2 = "XXXX LOGOUT\r\n";
|
|
QVERIFY(socketDevice.write(array2.data(),
|
|
array2.size()) == array2.size());
|
|
|
|
// Wait for the response
|
|
QVERIFY(socketDevice.waitForRead());
|
|
|
|
available = int(socketDevice.bytesAvailable());
|
|
QVERIFY(available > 0);
|
|
array.resize(int(available));
|
|
QVERIFY(socketDevice.read(array.data(), array.size()) == available);
|
|
|
|
// Check that the greeting is what we expect it to be
|
|
QCOMPARE(array.constData(), "* BYE LOGOUT received\r\nXXXX OK Completed\r\n");
|
|
|
|
// Wait for the response
|
|
QVERIFY(socketDevice.waitForRead());
|
|
char c;
|
|
QCOMPARE(socketDevice.read(&c, sizeof(c)), qint64(-1));
|
|
QCOMPARE(socketDevice.error(), QAbstractSocket::RemoteHostClosedError);
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
void tst_QHttpSocketEngine::simpleErrorsAndStates()
|
|
{
|
|
{
|
|
QHttpSocketEngine socketDevice;
|
|
|
|
// Initialize device
|
|
QVERIFY(socketDevice.initialize(QAbstractSocket::TcpSocket, QAbstractSocket::IPv4Protocol));
|
|
|
|
socketDevice.setProxy(QNetworkProxy(QNetworkProxy::HttpProxy, QtNetworkSettings::httpProxyServerName(), 3128));
|
|
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
QVERIFY(!socketDevice.connectToHost(QHostAddress(QtNetworkSettings::socksProxyServerName()), 8088));
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::ConnectingState);
|
|
if (socketDevice.waitForWrite(30000)) {
|
|
QVERIFY(socketDevice.state() == QAbstractSocket::ConnectedState ||
|
|
socketDevice.state() == QAbstractSocket::UnconnectedState);
|
|
} else {
|
|
QCOMPARE(socketDevice.error(), QAbstractSocket::SocketTimeoutError);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
//---------------------------------------------------------------------------
|
|
void tst_QHttpSocketEngine::tcpLoopbackPerformance()
|
|
{
|
|
QTcpServer server;
|
|
|
|
// Bind to any port on all interfaces
|
|
QVERIFY(server.bind(QHostAddress("0.0.0.0"), 0));
|
|
QCOMPARE(server.state(), QAbstractSocket::BoundState);
|
|
quint16 port = server.localPort();
|
|
|
|
// Listen for incoming connections
|
|
QVERIFY(server.listen());
|
|
QCOMPARE(server.state(), QAbstractSocket::ListeningState);
|
|
|
|
// Initialize a Tcp socket
|
|
QHttpSocketEngine client;
|
|
QVERIFY(client.initialize(QAbstractSocket::TcpSocket));
|
|
|
|
client.setProxy(QHostAddress("80.232.37.158"), 1081);
|
|
|
|
// Connect to our server
|
|
if (!client.connectToHost(QHostAddress("127.0.0.1"), port)) {
|
|
QVERIFY(client.waitForWrite());
|
|
QVERIFY(client.connectToHost(QHostAddress("127.0.0.1"), port));
|
|
}
|
|
|
|
// The server accepts the connectio
|
|
int socketDescriptor = server.accept();
|
|
QVERIFY(socketDescriptor > 0);
|
|
|
|
// A socket device is initialized on the server side, passing the
|
|
// socket descriptor from accept(). It's pre-connected.
|
|
QSocketLayer serverSocket;
|
|
QVERIFY(serverSocket.initialize(socketDescriptor));
|
|
QCOMPARE(serverSocket.state(), QAbstractSocket::ConnectedState);
|
|
|
|
const int messageSize = 1024 * 256;
|
|
QByteArray message1(messageSize, '@');
|
|
QByteArray answer(messageSize, '@');
|
|
|
|
QElapsedTimer timer;
|
|
timer.start();
|
|
qlonglong readBytes = 0;
|
|
while (timer.elapsed() < 30000) {
|
|
qlonglong written = serverSocket.write(message1.data(), message1.size());
|
|
while (written > 0) {
|
|
client.waitForRead();
|
|
if (client.bytesAvailable() > 0) {
|
|
qlonglong readNow = client.read(answer.data(), answer.size());
|
|
written -= readNow;
|
|
readBytes += readNow;
|
|
}
|
|
}
|
|
}
|
|
|
|
qDebug("\t\t%.1fMB/%.1fs: %.1fMB/s",
|
|
readBytes / (1024.0 * 1024.0),
|
|
timer.elapsed() / 1024.0,
|
|
(readBytes / (timer.elapsed() / 1000.0)) / (1024 * 1024));
|
|
}
|
|
*/
|
|
|
|
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketBlockingTest()
|
|
{
|
|
QHttpSocketEngineHandler http;
|
|
|
|
QTcpSocket socket;
|
|
|
|
// Connect
|
|
socket.connectToHost(QtNetworkSettings::imapServerName(), 143);
|
|
QVERIFY(socket.waitForConnected());
|
|
QCOMPARE(socket.state(), QTcpSocket::ConnectedState);
|
|
|
|
// Read greeting
|
|
QVERIFY(socket.waitForReadyRead(30000));
|
|
QString s = socket.readLine();
|
|
QVERIFY2(QtNetworkSettings::compareReplyIMAP(s.toLatin1()), qPrintable(s));
|
|
|
|
// Write NOOP
|
|
QCOMPARE((int) socket.write("1 NOOP\r\n", 8), 8);
|
|
|
|
if (!socket.canReadLine())
|
|
QVERIFY(socket.waitForReadyRead(30000));
|
|
|
|
// Read response
|
|
s = socket.readLine();
|
|
QCOMPARE(s.toLatin1().constData(), "1 OK Completed\r\n");
|
|
|
|
// Write LOGOUT
|
|
QCOMPARE((int) socket.write("2 LOGOUT\r\n", 10), 10);
|
|
|
|
if (!socket.canReadLine())
|
|
QVERIFY(socket.waitForReadyRead(30000));
|
|
|
|
// Read two lines of respose
|
|
s = socket.readLine();
|
|
QCOMPARE(s.toLatin1().constData(), "* BYE LOGOUT received\r\n");
|
|
|
|
if (!socket.canReadLine())
|
|
QVERIFY(socket.waitForReadyRead(30000));
|
|
|
|
s = socket.readLine();
|
|
QCOMPARE(s.toLatin1().constData(), "2 OK Completed\r\n");
|
|
|
|
// Close the socket
|
|
socket.close();
|
|
|
|
// Check that it's closed
|
|
QCOMPARE(socket.state(), QTcpSocket::UnconnectedState);
|
|
}
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlockingTest()
|
|
{
|
|
QHttpSocketEngineHandler http;
|
|
|
|
QTcpSocket socket;
|
|
connect(&socket, SIGNAL(hostFound()), SLOT(tcpSocketNonBlocking_hostFound()));
|
|
connect(&socket, SIGNAL(connected()), SLOT(tcpSocketNonBlocking_connected()));
|
|
connect(&socket, SIGNAL(disconnected()), SLOT(tcpSocketNonBlocking_closed()));
|
|
connect(&socket, SIGNAL(bytesWritten(qint64)), SLOT(tcpSocketNonBlocking_bytesWritten(qint64)));
|
|
connect(&socket, SIGNAL(readyRead()), SLOT(tcpSocketNonBlocking_readyRead()));
|
|
tcpSocketNonBlocking_socket = &socket;
|
|
|
|
// Connect
|
|
socket.connectToHost(QtNetworkSettings::imapServerName(), 143);
|
|
QVERIFY(socket.state() == QTcpSocket::HostLookupState ||
|
|
socket.state() == QTcpSocket::ConnectingState);
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
if (socket.state() == QTcpSocket::ConnectingState) {
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
}
|
|
|
|
QCOMPARE(socket.state(), QTcpSocket::ConnectedState);
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
// Read greeting
|
|
QVERIFY(!tcpSocketNonBlocking_data.isEmpty());
|
|
QByteArray data = tcpSocketNonBlocking_data.at(0).toLatin1();
|
|
QVERIFY2(QtNetworkSettings::compareReplyIMAP(data), data.constData());
|
|
|
|
|
|
tcpSocketNonBlocking_data.clear();
|
|
|
|
tcpSocketNonBlocking_totalWritten = 0;
|
|
|
|
// Write NOOP
|
|
QCOMPARE((int) socket.write("1 NOOP\r\n", 8), 8);
|
|
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
QCOMPARE(tcpSocketNonBlocking_totalWritten, 8);
|
|
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
|
|
// Read response
|
|
QVERIFY(!tcpSocketNonBlocking_data.isEmpty());
|
|
QCOMPARE(tcpSocketNonBlocking_data.at(0).toLatin1().constData(), "1 OK Completed\r\n");
|
|
tcpSocketNonBlocking_data.clear();
|
|
|
|
|
|
tcpSocketNonBlocking_totalWritten = 0;
|
|
|
|
// Write LOGOUT
|
|
QCOMPARE((int) socket.write("2 LOGOUT\r\n", 10), 10);
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
QCOMPARE(tcpSocketNonBlocking_totalWritten, 10);
|
|
|
|
// Wait for greeting
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout()) {
|
|
QFAIL("Timed out");
|
|
}
|
|
|
|
// Read two lines of respose
|
|
QCOMPARE(tcpSocketNonBlocking_data.at(0).toLatin1().constData(), "* BYE LOGOUT received\r\n");
|
|
QCOMPARE(tcpSocketNonBlocking_data.at(1).toLatin1().constData(), "2 OK Completed\r\n");
|
|
tcpSocketNonBlocking_data.clear();
|
|
|
|
// Close the socket
|
|
socket.close();
|
|
|
|
// Check that it's closed
|
|
QCOMPARE(socket.state(), QTcpSocket::UnconnectedState);
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlocking_hostFound()
|
|
{
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlocking_connected()
|
|
{
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlocking_readyRead()
|
|
{
|
|
while (tcpSocketNonBlocking_socket->canReadLine())
|
|
tcpSocketNonBlocking_data.append(tcpSocketNonBlocking_socket->readLine());
|
|
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlocking_bytesWritten(qint64 written)
|
|
{
|
|
tcpSocketNonBlocking_totalWritten += written;
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::tcpSocketNonBlocking_closed()
|
|
{
|
|
}
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
void tst_QHttpSocketEngine::downloadBigFile()
|
|
{
|
|
QHttpSocketEngineHandler http;
|
|
|
|
if (tmpSocket)
|
|
delete tmpSocket;
|
|
tmpSocket = new QTcpSocket;
|
|
|
|
connect(tmpSocket, SIGNAL(connected()), SLOT(exitLoopSlot()));
|
|
connect(tmpSocket, SIGNAL(readyRead()), SLOT(downloadBigFileSlot()));
|
|
|
|
tmpSocket->connectToHost(QtNetworkSettings::httpServerName(), 80);
|
|
|
|
QTestEventLoop::instance().enterLoop(30);
|
|
if (QTestEventLoop::instance().timeout())
|
|
QFAIL("Network operation timed out");
|
|
|
|
QByteArray hostName = QtNetworkSettings::httpServerName().toLatin1();
|
|
QCOMPARE(tmpSocket->state(), QAbstractSocket::ConnectedState);
|
|
QVERIFY(tmpSocket->write("GET /qtest/mediumfile HTTP/1.0\r\n") > 0);
|
|
QVERIFY(tmpSocket->write("Host: ") > 0);
|
|
QVERIFY(tmpSocket->write(hostName.data()) > 0);
|
|
QVERIFY(tmpSocket->write("\r\n") > 0);
|
|
QVERIFY(tmpSocket->write("\r\n") > 0);
|
|
|
|
bytesAvailable = 0;
|
|
|
|
QElapsedTimer stopWatch;
|
|
stopWatch.start();
|
|
|
|
QTestEventLoop::instance().enterLoop(60);
|
|
if (QTestEventLoop::instance().timeout())
|
|
QFAIL("Network operation timed out");
|
|
|
|
QVERIFY(bytesAvailable >= 10000000);
|
|
|
|
QCOMPARE(tmpSocket->state(), QAbstractSocket::ConnectedState);
|
|
|
|
qDebug("\t\t%.1fMB/%.1fs: %.1fMB/s",
|
|
bytesAvailable / (1024.0 * 1024.0),
|
|
stopWatch.elapsed() / 1024.0,
|
|
(bytesAvailable / (stopWatch.elapsed() / 1000.0)) / (1024 * 1024));
|
|
|
|
delete tmpSocket;
|
|
tmpSocket = 0;
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::exitLoopSlot()
|
|
{
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
|
|
void tst_QHttpSocketEngine::downloadBigFileSlot()
|
|
{
|
|
bytesAvailable += tmpSocket->readAll().size();
|
|
if (bytesAvailable >= 10000000)
|
|
QTestEventLoop::instance().exitLoop();
|
|
}
|
|
|
|
void tst_QHttpSocketEngine::passwordAuth()
|
|
{
|
|
QHttpSocketEngine socketDevice;
|
|
|
|
// Initialize device
|
|
QVERIFY(socketDevice.initialize(QAbstractSocket::TcpSocket, QAbstractSocket::IPv4Protocol));
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
|
|
socketDevice.setProxy(QNetworkProxy(QNetworkProxy::HttpProxy, QtNetworkSettings::httpProxyServerName(), 3128, "qsockstest", "password"));
|
|
|
|
QVERIFY(!socketDevice.connectToHost(QtNetworkSettings::imapServerIp(), 143));
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::ConnectingState);
|
|
QVERIFY(socketDevice.waitForWrite());
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::ConnectedState);
|
|
QCOMPARE(socketDevice.peerAddress(), QtNetworkSettings::imapServerIp());
|
|
|
|
// Wait for the greeting
|
|
QVERIFY(socketDevice.waitForRead());
|
|
|
|
// Read the greeting
|
|
qint64 available = socketDevice.bytesAvailable();
|
|
QVERIFY(available > 0);
|
|
QByteArray array;
|
|
array.resize(available);
|
|
QVERIFY(socketDevice.read(array.data(), array.size()) == available);
|
|
|
|
// Check that the greeting is what we expect it to be
|
|
QVERIFY2(QtNetworkSettings::compareReplyIMAP(array), array.constData());
|
|
|
|
|
|
// Write a logout message
|
|
QByteArray array2 = "XXXX LOGOUT\r\n";
|
|
QVERIFY(socketDevice.write(array2.data(),
|
|
array2.size()) == array2.size());
|
|
|
|
// Wait for the response
|
|
QVERIFY(socketDevice.waitForRead());
|
|
|
|
available = socketDevice.bytesAvailable();
|
|
QVERIFY(available > 0);
|
|
array.resize(available);
|
|
QVERIFY(socketDevice.read(array.data(), array.size()) == available);
|
|
|
|
// Check that the greeting is what we expect it to be
|
|
QCOMPARE(array.constData(), "* BYE LOGOUT received\r\nXXXX OK Completed\r\n");
|
|
|
|
// Wait for the response
|
|
QVERIFY(socketDevice.waitForRead());
|
|
char c;
|
|
QVERIFY(socketDevice.read(&c, sizeof(c)) == -1);
|
|
QCOMPARE(socketDevice.error(), QAbstractSocket::RemoteHostClosedError);
|
|
QCOMPARE(socketDevice.state(), QAbstractSocket::UnconnectedState);
|
|
}
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
void tst_QHttpSocketEngine::ensureEofTriggersNotification()
|
|
{
|
|
QList<QByteArray> serverData;
|
|
// Set the handshake and server response data
|
|
serverData << "HTTP/1.0 200 Connection established\r\n\r\n" << "0";
|
|
MiniHttpServer server(serverData);
|
|
|
|
QTcpSocket socket;
|
|
connect(&socket, SIGNAL(connected()), SLOT(exitLoopSlot()));
|
|
socket.setProxy(QNetworkProxy(QNetworkProxy::HttpProxy, server.serverAddress().toString(),
|
|
server.serverPort()));
|
|
socket.connectToHost("0.1.2.3", 12345);
|
|
|
|
QTestEventLoop::instance().enterLoop(5);
|
|
if (QTestEventLoop::instance().timeout())
|
|
QFAIL("Connect timed out");
|
|
|
|
QCOMPARE(socket.state(), QTcpSocket::ConnectedState);
|
|
// Disable read notification on server response
|
|
socket.setReadBufferSize(1);
|
|
socket.putChar(0);
|
|
|
|
// Wait for the response
|
|
connect(&socket, SIGNAL(readyRead()), SLOT(exitLoopSlot()));
|
|
QTestEventLoop::instance().enterLoop(5);
|
|
if (QTestEventLoop::instance().timeout())
|
|
QFAIL("Read timed out");
|
|
|
|
QCOMPARE(socket.state(), QTcpSocket::ConnectedState);
|
|
QCOMPARE(socket.bytesAvailable(), 1);
|
|
// Trigger a read notification
|
|
socket.readAll();
|
|
// Check for pending EOF at input
|
|
QCOMPARE(socket.bytesAvailable(), 0);
|
|
QCOMPARE(socket.state(), QTcpSocket::ConnectedState);
|
|
|
|
// Try to read EOF
|
|
connect(&socket, SIGNAL(disconnected()), SLOT(exitLoopSlot()));
|
|
QTestEventLoop::instance().enterLoop(5);
|
|
if (QTestEventLoop::instance().timeout())
|
|
QFAIL("Disconnect timed out");
|
|
|
|
// Check that it's closed
|
|
QCOMPARE(socket.state(), QTcpSocket::UnconnectedState);
|
|
}
|
|
|
|
QTEST_MAIN(tst_QHttpSocketEngine)
|
|
#include "tst_qhttpsocketengine.moc"
|