Removed QHttpHeader class and cleaned out QHttpSocket tests.
This commit is contained in:
@@ -1,61 +0,0 @@
|
||||
/**
|
||||
* 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.
|
||||
*/
|
||||
|
||||
#ifndef QHTTPENGINE_QHTTPHEADER_H
|
||||
#define QHTTPENGINE_QHTTPHEADER_H
|
||||
|
||||
#include "config.h"
|
||||
|
||||
/**
|
||||
* @brief Single HTTP header
|
||||
*
|
||||
* This class encapsulates a single HTTP header, including its name and value.
|
||||
*/
|
||||
class QHTTPENGINE_EXPORT QHttpHeader
|
||||
{
|
||||
public:
|
||||
|
||||
/**
|
||||
* @brief Create a header from the specified values
|
||||
*/
|
||||
QHttpHeader(const QByteArray &name, const QByteArray &value)
|
||||
: name(name),
|
||||
value(value)
|
||||
{}
|
||||
|
||||
/**
|
||||
* @brief Comparison operator
|
||||
*
|
||||
* Note that comparison between header names is case-insensitive.
|
||||
*/
|
||||
bool operator==(const QHttpHeader &rhs) const
|
||||
{
|
||||
return name.toLower() == rhs.name.toLower() && value == rhs.value;
|
||||
}
|
||||
|
||||
QByteArray name;
|
||||
QByteArray value;
|
||||
};
|
||||
|
||||
#endif // QHTTPENGINE_QHTTPHEADER_H
|
||||
@@ -22,22 +22,10 @@
|
||||
* IN THE SOFTWARE.
|
||||
*/
|
||||
|
||||
#include <QBuffer>
|
||||
#include <QList>
|
||||
#include <QPair>
|
||||
#include <QSignalSpy>
|
||||
#include <QObject>
|
||||
#include <QTest>
|
||||
|
||||
#include "common/qiodevicecounter.h"
|
||||
#include "common/qsocketpair.h"
|
||||
#include "core/qhttpsocket.h"
|
||||
#include "util/qhttpparser.h"
|
||||
#include "util/qiodevicecopier.h"
|
||||
|
||||
typedef QPair<QByteArray, QByteArray> Header;
|
||||
typedef QList<Header> HeaderList;
|
||||
|
||||
const QByteArray TestData = "test";
|
||||
|
||||
class TestQHttpSocket : public QObject
|
||||
{
|
||||
@@ -45,214 +33,8 @@ class TestQHttpSocket : public QObject
|
||||
|
||||
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 = QHttpParser::split(bufferData, "\r\n\r\n", 1);
|
||||
QList<QByteArray> lines = QHttpParser::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"
|
||||
|
||||
Reference in New Issue
Block a user