Files
qhttpengine/tests/TestQHttpSocket.cpp
2015-05-20 22:08:55 -07:00

259 lines
7.9 KiB
C++

/**
* The MIT License (MIT)
*
* Copyright (c) 2015 Nathan Osman
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
#include <QBuffer>
#include <QList>
#include <QPair>
#include <QSignalSpy>
#include <QTest>
#include "common/qiodevicecounter.h"
#include "common/qsocketpair.h"
#include "qhttpengine.h"
#include "qhttpsocket.h"
#include "qiodevicecopier.h"
typedef QPair<QByteArray, QByteArray> Header;
typedef QList<Header> HeaderList;
const QByteArray TestData = "test";
class TestQHttpSocket : public QObject
{
Q_OBJECT
private Q_SLOTS:
void initTestCase();
void testRequest_data();
void testRequest();
void testResponse_data();
void testResponse();
};
void TestQHttpSocket::initTestCase()
{
qRegisterMetaType<QHttpSocket::Error>("Error");
}
void TestQHttpSocket::testRequest_data()
{
QTest::addColumn<QByteArray>("request");
QTest::addColumn<QHttpSocket::Error>("error");
QTest::addColumn<QByteArray>("method");
QTest::addColumn<QByteArray>("path");
QTest::addColumn<HeaderList>("headers");
QTest::addColumn<QByteArray>("data");
QTest::newRow("simple GET")
<< QByteArray("GET /test HTTP/1.0\r\n\r\n")
<< QHttpSocket::NoError
<< QByteArray("GET")
<< QByteArray("/test")
<< HeaderList()
<< QByteArray();
QTest::newRow("request header")
<< QByteArray("GET / HTTP/1.0\r\nX-Test1: test\r\nX-Test2: test\r\n\r\n")
<< QHttpSocket::NoError
<< QByteArray("GET")
<< QByteArray("/")
<< (HeaderList() << Header("X-Test1", "test") << Header("X-Test2", "test"))
<< QByteArray();
QTest::newRow("data")
<< QByteArray("POST / HTTP/1.0\r\n\r\n")
<< QHttpSocket::NoError
<< QByteArray("POST")
<< QByteArray("/")
<< HeaderList()
<< TestData;
QTest::newRow("malformed request line")
<< QByteArray("test\r\n\r\n")
<< QHttpSocket::MalformedRequestLine;
QTest::newRow("malformed request header")
<< QByteArray("GET / HTTP/1.0\r\ntest\r\n\r\n")
<< QHttpSocket::MalformedRequestHeader;
QTest::newRow("invalid HTTP version")
<< QByteArray("GET / HTTP/1.2\r\n\r\n")
<< QHttpSocket::InvalidHttpVersion;
}
void TestQHttpSocket::testRequest()
{
QFETCH(QByteArray, request);
QFETCH(QHttpSocket::Error, error);
QSocketPair pair;
QTRY_VERIFY(pair.isConnected());
QHttpSocket socket(pair.server());
pair.client()->write(request);
if(error == QHttpSocket::NoError) {
QFETCH(QByteArray, method);
QFETCH(QByteArray, path);
QFETCH(HeaderList, headers);
QFETCH(QByteArray, data);
// Ensure that the headersParsedChanged() signal is emitted
QSignalSpy hPChangedSpy(&socket, SIGNAL(headersParsedChanged(bool)));
QTRY_COMPARE(hPChangedSpy.count(), 1);
// Compare the parsed properties to their proper values
QCOMPARE(socket.method(), method);
QCOMPARE(socket.path(), path);
QCOMPARE(socket.headers().count(), headers.count());
foreach(Header header, headers) {
QCOMPARE(socket.header(header.first), header.second);
}
if(!data.isNull()) {
// Watch for readyRead() signals as the data is written to the client
QSignalSpy readyReadSpy(&socket, SIGNAL(readyRead()));
pair.client()->write(data);
// Ensure that the signal was emitted and the data is correct
QTRY_COMPARE(socket.bytesAvailable(), data.length());
QCOMPARE(socket.readAll(), data);
QVERIFY(readyReadSpy.count() > 0);
}
} else {
// Ensure that the errorChanged() signal is emitted
QSignalSpy errorSpy(&socket, SIGNAL(errorChanged(Error)));
QTRY_COMPARE(errorSpy.count(), 1);
// Ensure the error value is correct
QCOMPARE(errorSpy.at(0).at(0).value<QHttpSocket::Error>(), error);
QCOMPARE(socket.error(), error);
}
}
void TestQHttpSocket::testResponse_data()
{
QTest::addColumn<QByteArray>("statusLine");
QTest::addColumn<QByteArray>("statusCode");
QTest::addColumn<HeaderList>("headers");
QTest::addColumn<QByteArray>("data");
QTest::newRow("default values")
<< QByteArray("HTTP/1.0 200 OK")
<< QByteArray()
<< HeaderList()
<< QByteArray();
QTest::newRow("status code")
<< QByteArray("HTTP/1.0 404 FILE NOT FOUND")
<< QByteArray("404 FILE NOT FOUND")
<< HeaderList()
<< QByteArray();
QTest::newRow("response header")
<< QByteArray("HTTP/1.0 200 OK")
<< QByteArray()
<< (HeaderList() << Header("X-Test1", "test") << Header("X-Test2", "test"))
<< QByteArray();
QTest::newRow("data")
<< QByteArray("HTTP/1.0 200 OK")
<< QByteArray()
<< HeaderList()
<< TestData;
}
void TestQHttpSocket::testResponse()
{
QFETCH(QByteArray, statusLine);
QFETCH(QByteArray, statusCode);
QFETCH(HeaderList, headers);
QFETCH(QByteArray, data);
QSocketPair pair;
QTRY_VERIFY(pair.isConnected());
QHttpSocket socket(pair.server());
if(!statusCode.isNull()) {
socket.setStatusCode(statusCode);
}
foreach(Header header, headers) {
socket.setHeader(header.first, header.second);
}
// Create a buffer for storing data received by the HTTP client
QByteArray bufferData;
QBuffer buffer(&bufferData);
buffer.open(QIODevice::WriteOnly);
// Copy data from the client into the buffer as it becomes available
QIODeviceCopier copier(pair.client(), &buffer);
copier.start();
// Write the headers and wait until they are received by the client
socket.writeHeaders();
QTRY_VERIFY(bufferData.indexOf("\r\n\r\n") != -1);
if(!data.isNull()) {
// Store the length of the headers
qint64 headerLength = bufferData.length();
// Write the data to the socket, counting the amount written
QIODeviceCounter counter(&socket);
socket.write(data);
// Verify that the expected amount was written and read
QTRY_COMPARE(counter.bytesWritten(), data.length());
QTRY_COMPARE(bufferData.length(), headerLength + data.length());
}
// Split the response into fragments and compare the expected values
QList<QByteArray> parts = QHttpEngine::split(bufferData, "\r\n\r\n", 1);
QList<QByteArray> lines = QHttpEngine::split(parts.at(0), "\r\n");
QCOMPARE(lines.takeFirst(), statusLine);
QCOMPARE(lines.count(), headers.count());
foreach(Header header, headers) {
QVERIFY(lines.contains(header.first + ": " + header.second));
}
if(!data.isNull()) {
QCOMPARE(parts.at(1), data);
}
}
QTEST_MAIN(TestQHttpSocket)
#include "TestQHttpSocket.moc"