Remove all JSON marshalling code from QObjectHandler.
This commit is contained in:
@@ -1973,7 +1973,7 @@ INCLUDE_FILE_PATTERNS =
|
|||||||
# recursively expanded use the := operator instead of the = operator.
|
# recursively expanded use the := operator instead of the = operator.
|
||||||
# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
|
# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
|
||||||
|
|
||||||
PREDEFINED =
|
PREDEFINED = DOXYGEN
|
||||||
|
|
||||||
# If the MACRO_EXPANSION and EXPAND_ONLY_PREDEF tags are set to YES then this
|
# If the MACRO_EXPANSION and EXPAND_ONLY_PREDEF tags are set to YES then this
|
||||||
# tag can be used to specify a list of macro names that should be expanded. The
|
# tag can be used to specify a list of macro names that should be expanded. The
|
||||||
|
|||||||
@@ -39,10 +39,7 @@ class QHTTPENGINE_EXPORT QObjectHandlerPrivate;
|
|||||||
* This handler enables incoming requests to be processed by slots in a
|
* This handler enables incoming requests to be processed by slots in a
|
||||||
* QObject-derived class or functor. Methods are registered by providing a
|
* QObject-derived class or functor. Methods are registered by providing a
|
||||||
* name and slot to invoke. The slot may take a pointer to the QHttpSocket for
|
* name and slot to invoke. The slot may take a pointer to the QHttpSocket for
|
||||||
* the request as an argument. For requests that include a body, the content
|
* the request as an argument.
|
||||||
* is parsed as a JSON object and provided as a second parameter to the slot.
|
|
||||||
* The slot is expected to return a QVariantMap containing the response, which
|
|
||||||
* will be encoded as a JSON-document.
|
|
||||||
*
|
*
|
||||||
* To use this class, simply create an instance and call the appropriate
|
* To use this class, simply create an instance and call the appropriate
|
||||||
* registerMethod() overload. For example:
|
* registerMethod() overload. For example:
|
||||||
@@ -52,7 +49,7 @@ class QHTTPENGINE_EXPORT QObjectHandlerPrivate;
|
|||||||
* {
|
* {
|
||||||
* Q_OBJECT
|
* Q_OBJECT
|
||||||
* public slots:
|
* public slots:
|
||||||
* QVariantMap something(QHttpSocket *socket);
|
* void something(QHttpSocket *socket);
|
||||||
* };
|
* };
|
||||||
*
|
*
|
||||||
* QObjectHandler handler;
|
* QObjectHandler handler;
|
||||||
@@ -69,7 +66,7 @@ class QHTTPENGINE_EXPORT QObjectHandlerPrivate;
|
|||||||
* @code
|
* @code
|
||||||
* QObjectHandler handler;
|
* QObjectHandler handler;
|
||||||
* handler.registerMethod("something", [](QHttpSocket *socket) {
|
* handler.registerMethod("something", [](QHttpSocket *socket) {
|
||||||
* return QVariantMap();
|
* // do something
|
||||||
* });
|
* });
|
||||||
* @endcode
|
* @endcode
|
||||||
*/
|
*/
|
||||||
@@ -89,62 +86,64 @@ public:
|
|||||||
*
|
*
|
||||||
* This overload uses the traditional connection syntax with macros.
|
* This overload uses the traditional connection syntax with macros.
|
||||||
*/
|
*/
|
||||||
void registerMethod(const QString &name, QObject *receiver, const char *method, int acceptedStatusCodes = QHttpSocket::GET);
|
void registerMethod(const QString &name, QObject *receiver, const char *method);
|
||||||
|
|
||||||
|
#ifdef DOXYGEN
|
||||||
/**
|
/**
|
||||||
* @brief Register a method
|
* @brief Register a method
|
||||||
*
|
*
|
||||||
* This overload uses the new connection syntax with member pointers.
|
* This overload uses the new connection syntax with member pointers.
|
||||||
*/
|
*/
|
||||||
template <typename Func1>
|
void registerMethod(const QString &name, QObject *receiver, PointerToMemberFunction method);
|
||||||
inline void registerMethod(const QString &name,
|
|
||||||
typename QtPrivate::FunctionPointer<Func1>::Object *receiver,
|
|
||||||
Func1 slot, int acceptedStatusCodes = QHttpSocket::GET) {
|
|
||||||
|
|
||||||
typedef QtPrivate::FunctionPointer<Func1> SlotType;
|
|
||||||
|
|
||||||
// Ensure the slot doesn't have too many parameters
|
|
||||||
Q_STATIC_ASSERT_X(int(SlotType::ArgumentCount) <= 2,
|
|
||||||
"The slot must have no more than two arguments.");
|
|
||||||
|
|
||||||
// Ensure the parameters are of the correct type
|
|
||||||
Q_STATIC_ASSERT_X((QtPrivate::CheckCompatibleArguments<QtPrivate::List<QHttpSocket*, QVariantMap>, typename SlotType::Arguments>::value),
|
|
||||||
"The slot parameters do not match");
|
|
||||||
|
|
||||||
// Ensure the return value is correct
|
|
||||||
Q_STATIC_ASSERT_X((QtPrivate::AreArgumentsCompatible<typename SlotType::ReturnType, QVariantMap>::value),
|
|
||||||
"Return type of the slot is not compatible with the return type of the signal.");
|
|
||||||
|
|
||||||
// Invoke the implementation
|
|
||||||
registerMethodImpl(name, receiver,
|
|
||||||
new QtPrivate::QSlotObject<Func1, typename SlotType::Arguments, void>(slot),
|
|
||||||
acceptedStatusCodes);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Register a method
|
* @brief Register a method
|
||||||
*
|
*
|
||||||
* This overload uses the new functor syntax (without context).
|
* This overload uses the new functor syntax (without context).
|
||||||
*/
|
*/
|
||||||
template <typename Func1>
|
void registerMethod(const QString &name, Functor functor);
|
||||||
inline void registerMethod(const QString &name, Func1 slot, int acceptedStatusCodes = QHttpSocket::GET) {
|
|
||||||
registerMethod(name, Q_NULLPTR, slot, acceptedStatusCodes);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Register a method
|
* @brief Register a method
|
||||||
*
|
*
|
||||||
* This overload uses the new functor syntax (with context).
|
* This overload uses the new functor syntax (with context).
|
||||||
*/
|
*/
|
||||||
|
void registerMethod(const QString &name, QObject *receiver, Functor functor);
|
||||||
|
#else
|
||||||
|
template <typename Func1>
|
||||||
|
inline void registerMethod(const QString &name,
|
||||||
|
typename QtPrivate::FunctionPointer<Func1>::Object *receiver,
|
||||||
|
Func1 slot) {
|
||||||
|
|
||||||
|
typedef QtPrivate::FunctionPointer<Func1> SlotType;
|
||||||
|
|
||||||
|
// Ensure the slot doesn't have too many arguments
|
||||||
|
Q_STATIC_ASSERT_X(int(SlotType::ArgumentCount) == 1,
|
||||||
|
"The slot must have exactly one argument.");
|
||||||
|
|
||||||
|
// Ensure the argument is of the correct type
|
||||||
|
Q_STATIC_ASSERT_X((QtPrivate::AreArgumentsCompatible<QHttpSocket*, typename QtPrivate::List_Select<typename SlotType::Arguments, 0>::Value>::value),
|
||||||
|
"The slot parameters do not match");
|
||||||
|
|
||||||
|
// Invoke the implementation
|
||||||
|
registerMethodImpl(name, receiver, new QtPrivate::QSlotObject<Func1, typename SlotType::Arguments, void>(slot));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Func1>
|
||||||
|
inline void registerMethod(const QString &name, Func1 slot) {
|
||||||
|
registerMethod(name, Q_NULLPTR, slot);
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Func1>
|
template <typename Func1>
|
||||||
inline typename QtPrivate::QEnableIf<!QtPrivate::FunctionPointer<Func1>::IsPointerToMemberFunction &&
|
inline typename QtPrivate::QEnableIf<!QtPrivate::FunctionPointer<Func1>::IsPointerToMemberFunction &&
|
||||||
!QtPrivate::is_same<const char*, Func1>::value, void>::Type
|
!QtPrivate::is_same<const char*, Func1>::value, void>::Type
|
||||||
registerMethod(const QString &name, QObject *context, Func1 slot, int acceptedStatusCodes = QHttpSocket::GET) {
|
registerMethod(const QString &name, QObject *context, Func1 slot) {
|
||||||
|
|
||||||
// There is an easier way to do this but then the header wouldn't
|
// There is an easier way to do this but then the header wouldn't
|
||||||
// compile on non-C++11 compilers
|
// compile on non-C++11 compilers
|
||||||
return registerMethod_functor(name, context, slot, &Func1::operator(), acceptedStatusCodes);
|
return registerMethod_functor(name, context, slot, &Func1::operator());
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
@@ -156,29 +155,23 @@ protected:
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
template <typename Func1, typename Func1Operator>
|
template <typename Func1, typename Func1Operator>
|
||||||
inline void registerMethod_functor(const QString &name, QObject *context, Func1 slot, Func1Operator, int acceptedStatusCodes) {
|
inline void registerMethod_functor(const QString &name, QObject *context, Func1 slot, Func1Operator) {
|
||||||
|
|
||||||
typedef QtPrivate::FunctionPointer<Func1Operator> SlotType;
|
typedef QtPrivate::FunctionPointer<Func1Operator> SlotType;
|
||||||
|
|
||||||
// Ensure the slot doesn't have too many parameters
|
// Ensure the slot doesn't have too many arguments
|
||||||
Q_STATIC_ASSERT_X(int(SlotType::ArgumentCount) <= 2,
|
Q_STATIC_ASSERT_X(int(SlotType::ArgumentCount) == 1,
|
||||||
"The slot must have no more than two arguments.");
|
"The slot must have exactly one argument.");
|
||||||
|
|
||||||
// Ensure the parameters are of the correct type
|
// Ensure the argument is of the correct type
|
||||||
Q_STATIC_ASSERT_X((QtPrivate::CheckCompatibleArguments<QtPrivate::List<QHttpSocket*, QVariantMap>, typename SlotType::Arguments>::value),
|
Q_STATIC_ASSERT_X((QtPrivate::AreArgumentsCompatible<QHttpSocket*, typename QtPrivate::List_Select<typename SlotType::Arguments, 0>::Value>::value),
|
||||||
"The slot parameters do not match");
|
"The slot parameters do not match");
|
||||||
|
|
||||||
// Ensure the return value is correct
|
|
||||||
Q_STATIC_ASSERT_X((QtPrivate::AreArgumentsCompatible<typename SlotType::ReturnType, QVariantMap>::value),
|
|
||||||
"Return type of the slot is not compatible with the return type of the signal.");
|
|
||||||
|
|
||||||
registerMethodImpl(name, context,
|
registerMethodImpl(name, context,
|
||||||
new QtPrivate::QFunctorSlotObject<Func1, SlotType::ArgumentCount,
|
new QtPrivate::QFunctorSlotObject<Func1, 1, typename SlotType::Arguments, void>(slot));
|
||||||
typename QtPrivate::List_Left<QtPrivate::List<QHttpSocket*, QVariantMap>, SlotType::ArgumentCount>::Value, void>(slot),
|
|
||||||
acceptedStatusCodes);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void registerMethodImpl(const QString &name, QObject *receiver, QtPrivate::QSlotObjectBase *slotObj, int acceptedStatusCodes);
|
void registerMethodImpl(const QString &name, QObject *receiver, QtPrivate::QSlotObjectBase *slotObj);
|
||||||
|
|
||||||
QObjectHandlerPrivate *const d;
|
QObjectHandlerPrivate *const d;
|
||||||
friend class QObjectHandlerPrivate;
|
friend class QObjectHandlerPrivate;
|
||||||
|
|||||||
@@ -21,9 +21,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <QGenericArgument>
|
#include <QGenericArgument>
|
||||||
#include <QJsonParseError>
|
|
||||||
#include <QJsonDocument>
|
|
||||||
#include <QJsonObject>
|
|
||||||
#include <QMetaMethod>
|
#include <QMetaMethod>
|
||||||
|
|
||||||
#include <QHttpEngine/QObjectHandler>
|
#include <QHttpEngine/QObjectHandler>
|
||||||
@@ -36,75 +33,6 @@ QObjectHandlerPrivate::QObjectHandlerPrivate(QObjectHandler *handler)
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void QObjectHandlerPrivate::invokeSlot(QHttpSocket *socket, const QString &path)
|
|
||||||
{
|
|
||||||
Method m = map.value(path);
|
|
||||||
QVariantMap parameters;
|
|
||||||
|
|
||||||
// If data was supplied, decode it as JSON
|
|
||||||
if (socket->bytesAvailable()) {
|
|
||||||
QJsonParseError error;
|
|
||||||
QJsonDocument document = QJsonDocument::fromJson(socket->readAll(), &error);
|
|
||||||
|
|
||||||
// Ensure that the document is valid
|
|
||||||
if (error.error != QJsonParseError::NoError) {
|
|
||||||
socket->writeError(QHttpSocket::BadRequest);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
parameters = document.object().toVariantMap();
|
|
||||||
}
|
|
||||||
|
|
||||||
QVariantMap retVal;
|
|
||||||
|
|
||||||
// Invoke the slot
|
|
||||||
if (m.oldSlot) {
|
|
||||||
|
|
||||||
// Obtain the slot index
|
|
||||||
int index = m.receiver->metaObject()->indexOfSlot(m.slot.method + 1);
|
|
||||||
if (index == -1) {
|
|
||||||
socket->writeError(QHttpSocket::InternalServerError);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
QMetaMethod method = m.receiver->metaObject()->method(index);
|
|
||||||
|
|
||||||
// Ensure the parameters are correct
|
|
||||||
QList<QByteArray> params = method.parameterTypes();
|
|
||||||
if (params.count() > 0 && params.at(0) != "QHttpSocket*" ||
|
|
||||||
params.count() > 1 && params.at(1) != "QVariantMap" ||
|
|
||||||
params.count() > 2 ||
|
|
||||||
method.returnType() != QMetaType::QVariantMap) {
|
|
||||||
socket->writeError(QHttpSocket::InternalServerError);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Invoke the method
|
|
||||||
if (!m.receiver->metaObject()->method(index).invoke(
|
|
||||||
m.receiver,
|
|
||||||
Q_RETURN_ARG(QVariantMap, retVal),
|
|
||||||
Q_ARG(QHttpSocket*, socket),
|
|
||||||
Q_ARG(QVariantMap, parameters))) {
|
|
||||||
socket->writeError(QHttpSocket::InternalServerError);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
void *args[3] = {
|
|
||||||
&retVal,
|
|
||||||
&socket,
|
|
||||||
¶meters
|
|
||||||
};
|
|
||||||
m.slot.slotObj->call(m.receiver, args);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert the return value to JSON and write it to the socket
|
|
||||||
QByteArray data = QJsonDocument(QJsonObject::fromVariantMap(retVal)).toJson();
|
|
||||||
socket->setHeader("Content-Length", QByteArray::number(data.length()));
|
|
||||||
socket->setHeader("Content-Type", "application/json");
|
|
||||||
socket->write(data);
|
|
||||||
socket->close();
|
|
||||||
}
|
|
||||||
|
|
||||||
QObjectHandler::QObjectHandler(QObject *parent)
|
QObjectHandler::QObjectHandler(QObject *parent)
|
||||||
: QHttpHandler(parent),
|
: QHttpHandler(parent),
|
||||||
d(new QObjectHandlerPrivate(this))
|
d(new QObjectHandlerPrivate(this))
|
||||||
@@ -119,31 +47,48 @@ void QObjectHandler::process(QHttpSocket *socket, const QString &path)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure the method is accepted
|
|
||||||
QObjectHandlerPrivate::Method m = d->map.value(path);
|
QObjectHandlerPrivate::Method m = d->map.value(path);
|
||||||
if (!(m.acceptedMethods & socket->method())) {
|
|
||||||
// TODO: accept header
|
|
||||||
socket->writeError(QHttpSocket::MethodNotAllowed);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the slot has finished receiving all of the data, jump directly to
|
// Invoke the slot
|
||||||
// invokeSlot(), otherwise, wait until we have the rest of it
|
if (m.oldSlot) {
|
||||||
if (socket->bytesAvailable() >= socket->contentLength()) {
|
|
||||||
d->invokeSlot(socket, path);
|
// Obtain the slot index
|
||||||
|
int index = m.receiver->metaObject()->indexOfSlot(m.slot.method + 1);
|
||||||
|
if (index == -1) {
|
||||||
|
socket->writeError(QHttpSocket::InternalServerError);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QMetaMethod method = m.receiver->metaObject()->method(index);
|
||||||
|
|
||||||
|
// Ensure the parameter is correct
|
||||||
|
QList<QByteArray> params = method.parameterTypes();
|
||||||
|
if (params.count() != 1 || params.at(0) != "QHttpSocket*") {
|
||||||
|
socket->writeError(QHttpSocket::InternalServerError);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Invoke the method
|
||||||
|
if (!m.receiver->metaObject()->method(index).invoke(
|
||||||
|
m.receiver, Q_ARG(QHttpSocket*, socket))) {
|
||||||
|
socket->writeError(QHttpSocket::InternalServerError);
|
||||||
|
return;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
connect(socket, &QHttpSocket::readChannelFinished, [this, socket, path]() {
|
void *args[] = {
|
||||||
d->invokeSlot(socket, path);
|
Q_NULLPTR,
|
||||||
});
|
&socket
|
||||||
|
};
|
||||||
|
m.slot.slotObj->call(m.receiver, args);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QObjectHandler::registerMethod(const QString &name, QObject *receiver, const char *method, int acceptedStatusCodes)
|
void QObjectHandler::registerMethod(const QString &name, QObject *receiver, const char *method)
|
||||||
{
|
{
|
||||||
d->map.insert(name, QObjectHandlerPrivate::Method(receiver, method, acceptedStatusCodes));
|
d->map.insert(name, QObjectHandlerPrivate::Method(receiver, method));
|
||||||
}
|
}
|
||||||
|
|
||||||
void QObjectHandler::registerMethodImpl(const QString &name, QObject *receiver, QtPrivate::QSlotObjectBase *slotObj, int acceptedStatusCodes)
|
void QObjectHandler::registerMethodImpl(const QString &name, QObject *receiver, QtPrivate::QSlotObjectBase *slotObj)
|
||||||
{
|
{
|
||||||
d->map.insert(name, QObjectHandlerPrivate::Method(receiver, slotObj, acceptedStatusCodes));
|
d->map.insert(name, QObjectHandlerPrivate::Method(receiver, slotObj));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,18 +37,16 @@ public:
|
|||||||
|
|
||||||
explicit QObjectHandlerPrivate(QObjectHandler *handler);
|
explicit QObjectHandlerPrivate(QObjectHandler *handler);
|
||||||
|
|
||||||
void invokeSlot(QHttpSocket *socket, const QString &path);
|
|
||||||
|
|
||||||
// In order to invoke the slot, a "pointer" to it needs to be stored in a
|
// In order to invoke the slot, a "pointer" to it needs to be stored in a
|
||||||
// map that lets us look up information by method name
|
// map that lets us look up information by method name
|
||||||
|
|
||||||
class Method {
|
class Method {
|
||||||
public:
|
public:
|
||||||
Method() {}
|
Method() {}
|
||||||
Method(QObject *receiver, const char *method, int acceptedMethods)
|
Method(QObject *receiver, const char *method)
|
||||||
: receiver(receiver), oldSlot(true), slot(method), acceptedMethods(acceptedMethods) {}
|
: receiver(receiver), oldSlot(true), slot(method) {}
|
||||||
Method(QObject *receiver, QtPrivate::QSlotObjectBase *slotObj, int acceptedMethods)
|
Method(QObject *receiver, QtPrivate::QSlotObjectBase *slotObj)
|
||||||
: receiver(receiver), oldSlot(false), slot(slotObj), acceptedMethods(acceptedMethods) {}
|
: receiver(receiver), oldSlot(false), slot(slotObj) {}
|
||||||
|
|
||||||
QObject *receiver;
|
QObject *receiver;
|
||||||
bool oldSlot;
|
bool oldSlot;
|
||||||
@@ -59,7 +57,6 @@ public:
|
|||||||
const char *method;
|
const char *method;
|
||||||
QtPrivate::QSlotObjectBase *slotObj;
|
QtPrivate::QSlotObjectBase *slotObj;
|
||||||
} slot;
|
} slot;
|
||||||
int acceptedMethods;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
QMap<QString, Method> map;
|
QMap<QString, Method> map;
|
||||||
|
|||||||
@@ -20,11 +20,8 @@
|
|||||||
* IN THE SOFTWARE.
|
* IN THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <QJsonDocument>
|
|
||||||
#include <QJsonObject>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QTest>
|
#include <QTest>
|
||||||
#include <QVariantMap>
|
|
||||||
|
|
||||||
#include <QHttpEngine/QHttpSocket>
|
#include <QHttpEngine/QHttpSocket>
|
||||||
#include <QHttpEngine/QObjectHandler>
|
#include <QHttpEngine/QObjectHandler>
|
||||||
@@ -38,12 +35,11 @@ class DummyAPI : public QObject
|
|||||||
|
|
||||||
public Q_SLOTS:
|
public Q_SLOTS:
|
||||||
|
|
||||||
int invalidReturnValue() { return 0; }
|
void wrongArgumentCount() {}
|
||||||
QVariantMap invalidArguments(int) { return QVariantMap(); }
|
void wrongArgumentType(int) {}
|
||||||
QVariantMap noParameters() { return QVariantMap(); }
|
void valid(QHttpSocket *socket) {
|
||||||
QVariantMap oneParameter(QHttpSocket *) { return QVariantMap(); }
|
socket->writeError(QHttpSocket::OK);
|
||||||
QVariantMap twoParameters(QHttpSocket *, QVariantMap) { return QVariantMap(); }
|
}
|
||||||
QVariantMap echoPost(QHttpSocket *, QVariantMap d) { return d; }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class TestQObjectHandler : public QObject
|
class TestQObjectHandler : public QObject
|
||||||
@@ -59,75 +55,31 @@ private Q_SLOTS:
|
|||||||
|
|
||||||
void TestQObjectHandler::testOldConnection_data()
|
void TestQObjectHandler::testOldConnection_data()
|
||||||
{
|
{
|
||||||
QTest::addColumn<bool>("registerPost");
|
|
||||||
QTest::addColumn<bool>("requestPost");
|
|
||||||
QTest::addColumn<QByteArray>("slot");
|
QTest::addColumn<QByteArray>("slot");
|
||||||
QTest::addColumn<int>("statusCode");
|
QTest::addColumn<int>("statusCode");
|
||||||
QTest::addColumn<QVariantMap>("data");
|
|
||||||
|
|
||||||
QTest::newRow("invalid return")
|
QTest::newRow("wrong argument count")
|
||||||
<< false
|
<< QByteArray(SLOT(wrongArgumentCount()))
|
||||||
<< false
|
<< static_cast<int>(QHttpSocket::InternalServerError);
|
||||||
<< QByteArray(SLOT(invalidReturnValue()))
|
|
||||||
<< static_cast<int>(QHttpSocket::InternalServerError)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("invalid arguments")
|
QTest::newRow("wrong argument type")
|
||||||
<< false
|
<< QByteArray(SLOT(wrongArgumentType(int)))
|
||||||
<< false
|
<< static_cast<int>(QHttpSocket::InternalServerError);
|
||||||
<< QByteArray(SLOT(invalidArguments(int)))
|
|
||||||
<< static_cast<int>(QHttpSocket::InternalServerError)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("no parameters")
|
QTest::newRow("valid")
|
||||||
<< false
|
<< QByteArray(SLOT(valid(QHttpSocket*)))
|
||||||
<< false
|
<< static_cast<int>(QHttpSocket::OK);
|
||||||
<< QByteArray(SLOT(noParameters()))
|
|
||||||
<< static_cast<int>(QHttpSocket::OK)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("one parameter")
|
|
||||||
<< false
|
|
||||||
<< false
|
|
||||||
<< QByteArray(SLOT(oneParameter(QHttpSocket*)))
|
|
||||||
<< static_cast<int>(QHttpSocket::OK)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("two parameters")
|
|
||||||
<< false
|
|
||||||
<< false
|
|
||||||
<< QByteArray(SLOT(twoParameters(QHttpSocket*,QVariantMap)))
|
|
||||||
<< static_cast<int>(QHttpSocket::OK)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("invalid method")
|
|
||||||
<< true
|
|
||||||
<< false
|
|
||||||
<< QByteArray(SLOT(echoPost(QHttpSocket*,QVariantMap)))
|
|
||||||
<< static_cast<int>(QHttpSocket::MethodNotAllowed)
|
|
||||||
<< QVariantMap();
|
|
||||||
|
|
||||||
QTest::newRow("post data")
|
|
||||||
<< true
|
|
||||||
<< true
|
|
||||||
<< QByteArray(SLOT(echoPost(QHttpSocket*,QVariantMap)))
|
|
||||||
<< static_cast<int>(QHttpSocket::OK)
|
|
||||||
<< QVariantMap{{"a", "a"}, {"b", 1}};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestQObjectHandler::testOldConnection()
|
void TestQObjectHandler::testOldConnection()
|
||||||
{
|
{
|
||||||
QFETCH(bool, registerPost);
|
|
||||||
QFETCH(bool, requestPost);
|
|
||||||
QFETCH(QByteArray, slot);
|
QFETCH(QByteArray, slot);
|
||||||
QFETCH(int, statusCode);
|
QFETCH(int, statusCode);
|
||||||
QFETCH(QVariantMap, data);
|
|
||||||
|
|
||||||
QObjectHandler handler;
|
QObjectHandler handler;
|
||||||
DummyAPI api;
|
DummyAPI api;
|
||||||
|
|
||||||
handler.registerMethod("test", &api, slot.constData(),
|
handler.registerMethod("test", &api, slot.constData());
|
||||||
registerPost ? QHttpSocket::POST : QHttpSocket::GET);
|
|
||||||
|
|
||||||
QSocketPair pair;
|
QSocketPair pair;
|
||||||
QTRY_VERIFY(pair.isConnected());
|
QTRY_VERIFY(pair.isConnected());
|
||||||
@@ -135,26 +87,11 @@ void TestQObjectHandler::testOldConnection()
|
|||||||
QSimpleHttpClient client(pair.client());
|
QSimpleHttpClient client(pair.client());
|
||||||
QHttpSocket socket(pair.server(), &pair);
|
QHttpSocket socket(pair.server(), &pair);
|
||||||
|
|
||||||
if (requestPost) {
|
client.sendHeaders("GET", "test");
|
||||||
QByteArray buff = QJsonDocument(QJsonObject::fromVariantMap(data)).toJson();
|
|
||||||
client.sendHeaders("POST", "test", QHttpSocket::QHttpHeaderMap{
|
|
||||||
{"Content-Length", QByteArray::number(buff.length())},
|
|
||||||
});
|
|
||||||
client.sendData(buff);
|
|
||||||
} else {
|
|
||||||
client.sendHeaders("GET", "test");
|
|
||||||
}
|
|
||||||
|
|
||||||
QTRY_VERIFY(socket.isHeadersParsed());
|
QTRY_VERIFY(socket.isHeadersParsed());
|
||||||
|
|
||||||
handler.route(&socket, socket.path());
|
handler.route(&socket, socket.path());
|
||||||
QTRY_COMPARE(client.statusCode(), statusCode);
|
QTRY_COMPARE(client.statusCode(), statusCode);
|
||||||
|
|
||||||
if (requestPost) {
|
|
||||||
QVERIFY(client.headers().contains("Content-Length"));
|
|
||||||
QTRY_COMPARE(client.data().length(), client.headers().value("Content-Length").toInt());
|
|
||||||
QCOMPARE(QJsonDocument::fromJson(client.data()).object(), QJsonObject::fromVariantMap(data));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestQObjectHandler::testNewConnection()
|
void TestQObjectHandler::testNewConnection()
|
||||||
@@ -163,17 +100,17 @@ void TestQObjectHandler::testNewConnection()
|
|||||||
DummyAPI api;
|
DummyAPI api;
|
||||||
|
|
||||||
// Connect to object slot
|
// Connect to object slot
|
||||||
handler.registerMethod("0", &api, &DummyAPI::noParameters);
|
handler.registerMethod("0", &api, &DummyAPI::valid);
|
||||||
handler.registerMethod("1", &api, &DummyAPI::oneParameter);
|
|
||||||
handler.registerMethod("2", &api, &DummyAPI::twoParameters);
|
|
||||||
|
|
||||||
// Connect to functor
|
// Connect to functor
|
||||||
handler.registerMethod("3", []() { return QVariantMap(); });
|
handler.registerMethod("1", [](QHttpSocket *socket) {
|
||||||
handler.registerMethod("4", &api, []() { return QVariantMap(); });
|
socket->writeError(QHttpSocket::OK);
|
||||||
handler.registerMethod("5", &api, [](QHttpSocket*) { return QVariantMap(); });
|
});
|
||||||
handler.registerMethod("6", &api, [](QHttpSocket*, QVariantMap d) { return d; });
|
handler.registerMethod("2", &api, [](QHttpSocket *socket) {
|
||||||
|
socket->writeError(QHttpSocket::OK);
|
||||||
|
});
|
||||||
|
|
||||||
for (int i = 0; i < 7; ++i) {
|
for (int i = 0; i < 3; ++i) {
|
||||||
QSocketPair pair;
|
QSocketPair pair;
|
||||||
QTRY_VERIFY(pair.isConnected());
|
QTRY_VERIFY(pair.isConnected());
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user