/**************************************************************************** ** ** Copyright (C) 2019 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtHttpServer module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:GPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 or (at your option) any later version ** approved by the KDE Free Qt Foundation. The licenses are as published by ** the Free Software Foundation and appearing in the file LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include #include #include Q_DECLARE_METATYPE(QNetworkAccessManager::Operation); QT_BEGIN_NAMESPACE struct HttpServer : QAbstractHttpServer { QHttpServerRouter router; HttpServer() : QAbstractHttpServer() { connect(this, &QAbstractHttpServer::missingHandler, [] (const QHttpServerRequest &request, QTcpSocket *socket) { makeResponder(request, socket).write(QHttpServerResponder::StatusCode::NotFound); }); } template void route(const char *path, const QHttpServerRequest::Methods methods, ViewHandler &&viewHandler) { auto rule = new QHttpServerRouterRule( path, methods, [this, &viewHandler] (const QRegularExpressionMatch &match, const QHttpServerRequest &request, QTcpSocket *socket) { auto boundViewHandler = router.bindCaptured( std::forward(viewHandler), match); boundViewHandler(makeResponder(request, socket)); }); router.addRule(rule); } template void route(const char *path, ViewHandler &&viewHandler) { route(path, QHttpServerRequest::Method::All, std::forward(viewHandler)); } bool handleRequest(const QHttpServerRequest &request, QTcpSocket *socket) override { return router.handleRequest(request, socket); } }; class tst_QHttpServerRouter : public QObject { Q_OBJECT private slots: void initTestCase(); void routerRule_data(); void routerRule(); void viewHandlerNoArg(); void viewHandlerOneArg(); void viewHandlerTwoArgs(); void viewHandlerResponder(); void viewHandlerRequest(); void viewHandlerLastTwoSpecials(); private: HttpServer httpserver; QString urlBase; }; void tst_QHttpServerRouter::initTestCase() { httpserver.route("/page/", [] (const quint64 &page, QHttpServerResponder &&responder) { responder.write(QString("page: %1").arg(page).toUtf8(), "text/plain"); }); httpserver.route("/post-only", QHttpServerRequest::Method::Post, [] (QHttpServerResponder &&responder) { responder.write(QString("post-test").toUtf8(), "text/plain"); }); httpserver.route("/get-only", QHttpServerRequest::Method::Get, [] (QHttpServerResponder &&responder) { responder.write(QString("get-test").toUtf8(), "text/plain"); }); urlBase = QStringLiteral("http://localhost:%1%2").arg(httpserver.listen()); } void tst_QHttpServerRouter::routerRule_data() { QTest::addColumn("url"); QTest::addColumn("code"); QTest::addColumn("type"); QTest::addColumn("body"); QTest::addColumn("replyType"); QTest::addRow("/page/1") << "/page/1" << 200 << "text/plain" << "page: 1" << QNetworkAccessManager::GetOperation; QTest::addRow("/page/-1") << "/page/-1" << 404 << "application/x-empty" << "" << QNetworkAccessManager::GetOperation; QTest::addRow("/post-only [GET]") << "/post-only" << 404 << "application/x-empty" << "" << QNetworkAccessManager::GetOperation; QTest::addRow("/post-only [DELETE]") << "/post-only" << 404 << "application/x-empty" << "" << QNetworkAccessManager::DeleteOperation; QTest::addRow("/post-only [POST]") << "/post-only" << 200 << "text/plain" << "post-test" << QNetworkAccessManager::PostOperation; QTest::addRow("/get-only [GET]") << "/get-only" << 200 << "text/plain" << "get-test" << QNetworkAccessManager::GetOperation; QTest::addRow("/get-only [POST]") << "/get-only" << 404 << "application/x-empty" << "" << QNetworkAccessManager::PostOperation; QTest::addRow("/get-only [DELETE]") << "/get-only" << 404 << "application/x-empty" << "" << QNetworkAccessManager::DeleteOperation; } void tst_QHttpServerRouter::routerRule() { QFETCH(QString, url); QFETCH(int, code); QFETCH(QString, type); QFETCH(QString, body); QFETCH(QNetworkAccessManager::Operation, replyType); QNetworkAccessManager networkAccessManager; QNetworkReply *reply; QNetworkRequest request(QUrl(urlBase.arg(url))); switch (replyType) { case QNetworkAccessManager::GetOperation: reply = networkAccessManager.get(request); break; case QNetworkAccessManager::PostOperation: request.setHeader(QNetworkRequest::ContentTypeHeader, type); reply = networkAccessManager.post(request, QByteArray("post body")); break; case QNetworkAccessManager::DeleteOperation: reply = networkAccessManager.deleteResource(request); break; default: QFAIL("The replyType does not supported"); } QTRY_VERIFY(reply->isFinished()); QCOMPARE(reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt(), code); QCOMPARE(reply->header(QNetworkRequest::ContentTypeHeader), type); QCOMPARE(reply->readAll(), body); } void tst_QHttpServerRouter::viewHandlerNoArg() { auto viewNonArg = [] () { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 0, "viewNonArg: Args::Count == 0"); static_assert(Args::CapturableCount == 0, "viewNonArg: Args::CapturableCount == 0"); static_assert(Args::PlaceholdersCount == 0, "viewNonArg: Args::PlaceholdersCount == 0"); static_assert(Args::Last::IsRequest::Value == 0, "viewNonArg: Args::Last::IsRequest::Value == 0"); static_assert(Args::Last::IsRequest::Valid == 0, "viewNonArg: Args::Last::IsRequest::Valid == 0"); static_assert(Args::Last::IsResponder::Value == 0, "viewNonArg: Args::Last::IsResponder::Value == 0"); static_assert(Args::Last::IsResponder::Valid == 0, "viewNonArg: Args::Last::IsResponder::Valid == 0"); static_assert(Args::Last::IsSpecial::Value == 0, "viewNonArg: Args::Last::IsSpecial::Value == 0"); static_assert(Args::Last::IsSpecial::Valid == 0, "viewNonArg: Args::Last::IsSpecial::Valid == 0"); static_assert(Args::Last::IsSimple::Value == 0, "viewNonArg: Args::Last::IsSimple::Value == 0"); static_assert(Args::Last::IsSimple::Valid == 0, "viewNonArg: Args::Last::IsSimple::Valid == 0"); static_assert(Args::Last::Valid == 0, "viewOneArg: Args::Last::Valid"); static_assert(Args::Last::StaticAssert, "viewOneArg: Args::Last::StaticAssert"); static_assert(Args::Last::isType(), "viewNonArg: Args::Last::isType()"); using Arg1 = typename Args::template Arg<1>; static_assert(Arg1::IsRequest::Value == 0, "viewNonArg: Args::Arg<1>::isRequest::Value == 0"); static_assert(Arg1::IsRequest::Valid == 0, "viewNonArg: Args::Arg<1>::IsRequest::Valid == 0"); static_assert(Arg1::IsResponder::Value == 0, "viewNonArg: Args::Arg<1>::IsResponder::Value == 0"); static_assert(Arg1::IsResponder::Valid == 0, "viewNonArg: Args::Arg<1>::IsResponder::Valid == 0"); static_assert(Arg1::IsSpecial::Value == 0, "viewNonArg: Args::Arg<1>::IsSpecial::Value == 0"); static_assert(Arg1::IsSpecial::Valid == 0, "viewNonArg: Args::Arg<1>::IsSpecial::Valid == 0"); static_assert(Arg1::Valid == 0, "viewOneArg: Args::Arg<1>::Valid"); static_assert(Arg1::StaticAssert, "viewOneArg: Args::Arg<1>::Valid::StaticAssert"); static_assert(Arg1::isType(), "viewNonArg: Args::Arg<1>::isType()"); static_assert(Args::Valid, "viewNonArg: Args::Valid"); static_assert(Args::StaticAssert, "viewNonArg: Args::StaticAssert"); } void tst_QHttpServerRouter::viewHandlerOneArg() { auto view = [] (const quint64 &) { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 1, "viewOneArg: Args::Count == 1"); static_assert(Args::CapturableCount == 1, "viewOneArg: Args::CapturableCount == 1"); static_assert(Args::PlaceholdersCount == 0, "viewOneArg: Args::PlaceholdersCount == 0"); static_assert(Args::Last::IsRequest::Value == 0, "viewOneArg: Args::Last::IsRequest::Value == 0"); static_assert(Args::Last::IsRequest::Valid == 0, "viewOneArg: Args::Last::IsRequest::Valid == 0"); static_assert(Args::Last::IsResponder::Value == 0, "viewOneArg: Args::Last::IsResponder::Value == 0"); static_assert(Args::Last::IsResponder::Valid == 0, "viewOneArg: Args::Last::IsResponder::Valid == 0"); static_assert(Args::Last::IsSpecial::Value == 0, "viewOneArg: Args::Last::IsSpecial::Value == 0"); static_assert(Args::Last::IsSpecial::Valid == 0, "viewOneArg: Args::Last::IsSpecial::Valid == 0"); static_assert(Args::Last::IsSimple::Value, "viewOneArg: Args::Last::IsSimple::Value"); static_assert(Args::Last::IsSimple::Valid, "viewOneArg: Args::Last::IsSimple::Valid"); static_assert(Args::Last::Valid, "viewOneArg: Args::Last::Valid"); static_assert(Args::Last::StaticAssert, "viewOneArg: Args::Last::StaticAssert"); static_assert(Args::Last::isType(), "viewNonArg: Args::Last::isType()"); static_assert(Args::Valid, "viewOneArg: Args::Valid"); static_assert(Args::StaticAssert, "viewOneArg: Args::StaticAssert"); } void tst_QHttpServerRouter::viewHandlerTwoArgs() { auto view = [] (const quint64 &, const QHttpServerResponder &) { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 2, "viewTwoArgs: Args::Count == 2"); static_assert(Args::CapturableCount == 1, "viewTwoArgs: Args::CapturableCount == 1"); static_assert(Args::PlaceholdersCount == 1, "viewTwoArgs: Args::PlaceholdersCount == 0"); using Arg0 = typename Args::template Arg<0>; static_assert(Arg0::IsRequest::Value == 0, "viewTwoArgs: Args::Arg0::IsRequest::Value == 0"); static_assert(Arg0::IsRequest::Valid == 0, "viewTwoArgs: Args::Arg0::IsRequest::Valid == 0"); static_assert(Arg0::IsResponder::Value == 0, "viewTwoArgs: Args::Arg0::IsResponder::Value == 0"); static_assert(Arg0::IsResponder::Valid == 0, "viewTwoArgs: Args::Arg0::IsResponder::Valid == 0"); static_assert(Arg0::IsSpecial::Value == 0, "viewTwoArgs: Args::Arg0::IsSpecial::Value == 0"); static_assert(Arg0::IsSpecial::Valid == 0, "viewTwoArgs: Args::Arg0::IsSpecial::Valid == 0"); static_assert(Arg0::IsSimple::Value, "viewTwoArgs: Args::Arg0::IsSimple::Value"); static_assert(Arg0::IsSimple::Valid, "viewTwoArgs: Args::Arg0::IsSimple::Valid"); static_assert(Arg0::Valid, "viewTwoArgs: Args::Arg0::Valid"); static_assert(Arg0::StaticAssert, "viewTwoArgs: Args::Arg0::StaticAssert"); static_assert(Arg0::isType(), "viewNonArg: Args::Arg0::isType()"); using Arg1 = typename Args::template Arg<1>; static_assert(Arg1::IsRequest::Value == 0, "viewTwoArgs: Args::Arg1::IsRequest::Value == 0"); static_assert(Arg1::IsRequest::Valid == 0, "viewTwoArgs: Args::Arg1::IsRequest::Valid == 0"); static_assert(Arg1::IsResponder::Value, "viewTwoArgs: Args::Arg1::IsResponder::Value"); static_assert(Arg1::IsResponder::Valid == 0, "viewTwoArgs: Args::Arg1::IsResponder::Valid == 0"); static_assert(Arg1::IsSpecial::Value == 1, "viewTwoArgs: Args::Arg1::IsSpecial::Value"); static_assert(Arg1::IsSpecial::Valid == 0, "viewTwoArgs: Args::Arg1::IsSpecial::Valid == 0"); static_assert(Arg1::IsSimple::Value == 0, "viewTwoArgs: Args::Arg1::IsSimple::Value == 0"); static_assert(Arg1::IsSimple::Valid == 0, "viewTwoArgs: Args::Arg1::IsSimple::Valid == 0"); static_assert(Arg1::Valid == 0, "viewTwoArgs: Args::Arg1::Valid"); // StaticAssert is disabled in tests static_assert(Arg1::StaticAssert, "viewOneArg: Args::Arg1::StaticAssert"); static_assert(Arg1::isType(), "viewTwoArgs: Args::Arg1::isType>()"); static_assert(Args::Valid == 0, "viewTwoArgs: Args::Valid == 0"); // StaticAssert is disabled in tests static_assert(Args::StaticAssert, "viewTwoArgs: Args::StaticAssert"); } void tst_QHttpServerRouter::viewHandlerResponder() { auto view = [] (QHttpServerResponder &&) { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 1, "viewResponder: Args::Count == 1"); static_assert(Args::CapturableCount == 0, "viewResponder: Args::CapturableCount == 0"); static_assert(Args::PlaceholdersCount == 1, "viewResponder: Args::PlaceholdersCount == 1"); static_assert(Args::Last::IsRequest::Value == 0, "viewResponder: Args::Last::IsRequest::Value == 0"); static_assert(Args::Last::IsRequest::Valid == 0, "viewResponder: Args::Last::IsRequest::Valid == 0"); static_assert(Args::Last::IsResponder::Value, "viewResponder: Args::Last::IsResponder::Value"); static_assert(Args::Last::IsResponder::Valid, "viewResponder: Args::Last::IsResponder::Valid"); static_assert(Args::Last::IsSpecial::Value, "viewResponder: Args::Last::IsSpecial::Value"); static_assert(Args::Last::IsSpecial::Valid, "viewResponder: Args::Last::IsSpecial::Valid"); static_assert(Args::Last::IsSimple::Value == 0, "viewResponder: Args::Last::IsSimple::Value == 0"); static_assert(Args::Last::IsSimple::Valid == 0, "viewResponder: Args::Last::IsSimple::Valid == 0"); static_assert(Args::Last::Valid, "viewResponder: Args::Last::Valid"); static_assert(Args::Last::StaticAssert, "viewResponder: Args::Last::StaticAssert"); static_assert(Args::Last::isType(), "viewNonArg: Args::Last::isType()"); static_assert(Args::Valid, "viewResponder: Args::Valid"); static_assert(Args::StaticAssert, "viewResponder: Args::StaticAssert"); } void tst_QHttpServerRouter::viewHandlerRequest() { auto view = [] (const QHttpServerRequest &) { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 1, "viewResponder: Args::Count == 1"); static_assert(Args::CapturableCount == 0, "viewResponder: Args::CapturableCount == 0"); static_assert(Args::PlaceholdersCount == 1, "viewResponder: Args::PlaceholdersCount == 1"); static_assert(Args::Last::IsRequest::Value, "viewResponder: Args::Last::IsRequest::Value"); static_assert(Args::Last::IsRequest::Valid, "viewResponder: Args::Last::IsRequest::Valid"); static_assert(Args::Last::IsResponder::Value == 0, "viewResponder: Args::Last::IsResponder::Value == 0"); static_assert(Args::Last::IsResponder::Valid == 0, "viewResponder: Args::Last::IsResponder::Valid == 0"); static_assert(Args::Last::IsSpecial::Value, "viewResponder: Args::Last::IsSpecial::Value"); static_assert(Args::Last::IsSpecial::Valid, "viewResponder: Args::Last::IsSpecial::Valid"); static_assert(Args::Last::IsSimple::Value == 0, "viewResponder: Args::Last::IsSimple::Value == 0"); static_assert(Args::Last::IsSimple::Valid == 0, "viewResponder: Args::Last::IsSimple::Valid == 0"); static_assert(Args::Last::Valid, "viewResponder: Args::Last::Valid"); static_assert(Args::Last::StaticAssert, "viewResponder: Args::Last::StaticAssert"); static_assert(Args::Last::isType(), "viewNonArg: Args::Last::isType()"); static_assert(Args::Valid, "viewResponder: Args::Valid"); static_assert(Args::StaticAssert, "viewResponder: Args::StaticAssert"); } void tst_QHttpServerRouter::viewHandlerLastTwoSpecials() { auto view = [] (const QHttpServerRequest &, QHttpServerResponder &&) { }; using ViewTraits = QHttpServerRouterViewTraits; using Args = typename ViewTraits::Arguments; static_assert(Args::Count == 2, "viewTwoSpecialArgs: Args::Count == 2"); static_assert(Args::CapturableCount == 0, "viewTwoSpecialArgs: Args::CapturableCount == 1"); static_assert(Args::PlaceholdersCount == 2, "viewTwoSpecialArgs: Args::PlaceholdersCount == 0"); using Arg0 = typename Args::template Arg<0>; static_assert(Arg0::IsRequest::Value, "viewTwoSpecialArgs: Args::Arg0::IsRequest::Value"); static_assert(Arg0::IsRequest::Valid, "viewTwoSpecialArgs: Args::Arg0::IsRequest::Valid"); static_assert(Arg0::IsResponder::Value == 0, "viewTwoSpecialArgs: Args::Arg0::IsResponder::Value == 0"); static_assert(Arg0::IsResponder::Valid == 0, "viewTwoSpecialArgs: Args::Arg0::IsResponder::Valid == 0"); static_assert(Arg0::IsSpecial::Value, "viewTwoSpecialArgs: Args::Arg0::IsSpecial::Value"); static_assert(Arg0::IsSpecial::Valid, "viewTwoSpecialArgs: Args::Arg0::IsSpecial::Valid"); static_assert(Arg0::IsSimple::Value == 0, "viewTwoSpecialArgs: Args::Arg0::IsSimple::Value == 0"); static_assert(Arg0::IsSimple::Valid == 0, "viewTwoSpecialArgs: Args::Arg0::IsSimple::Valid == 0"); static_assert(Arg0::Valid, "viewTwoSpecialArgs: Args::Arg0::Valid"); // StaticAssert is disabled in tests static_assert(Arg0::StaticAssert, "viewTwoSpecialArgs: Args::Arg0::StaticAssert"); static_assert(Arg0::isType(), "viewNonArg: Args::Arg0::isType()"); using Arg1 = typename Args::template Arg<1>; static_assert(Arg1::IsRequest::Value == 0, "viewTwoSpecialArgs: Args::Arg1::IsRequest::Value == 0"); static_assert(Arg1::IsRequest::Valid == 0, "viewTwoSpecialArgs: Args::Arg1::IsRequest::Valid == 0"); static_assert(Arg1::IsResponder::Value, "viewTwoSpecialArgs: Args::Arg1::IsResponder::Value"); static_assert(Arg1::IsResponder::Valid, "viewTwoSpecialArgs: Args::Arg1::IsResponder::Valid"); static_assert(Arg1::IsSpecial::Value, "viewTwoSpecialArgs: Args::Arg1::IsSpecial::Value"); static_assert(Arg1::IsSpecial::Valid, "viewTwoSpecialArgs: Args::Arg1::IsSpecial::Valid"); static_assert(Arg1::IsSimple::Value == 0, "viewTwoSpecialArgs: Args::Arg1::IsSimple::Value == 0"); static_assert(Arg1::IsSimple::Valid == 0, "viewTwoSpecialArgs: Args::Arg1::IsSimple::Valid == 0"); static_assert(Arg1::Valid, "viewTwoSpecialArgs: Args::Arg1::Valid"); static_assert(Arg1::StaticAssert, "viewTwoSpecialArgs: Args::Arg1::StaticAssert"); static_assert(Arg1::isType(), "viewTwoSpecialArgs: Args::Arg1::isType()"); static_assert(Args::Valid, "viewTwoSpecialArgs: Args::Valid"); // StaticAssert is disabled in tests static_assert(Args::StaticAssert, "viewTwoSpecialArgs: Args::StaticAssert"); } QT_END_NAMESPACE QTEST_MAIN(tst_QHttpServerRouter) #include "tst_qhttpserverrouter.moc"