Masterctrl/Lib/QtHTTPServer/include/QtHttpServer/qhttpserverfutureresponse.h
2021-05-26 04:15:57 -04:00

163 lines
4.8 KiB
C++

/****************************************************************************
**
** Copyright (C) 2020 Mikhail Svetkin <mikhail.svetkin@gmail.com>
** 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$
**
****************************************************************************/
#ifndef QHTTPSERVERFUTURERESPONSE_H
#define QHTTPSERVERFUTURERESPONSE_H
#include <QtHttpServer/qhttpserverresponse.h>
#include <QtCore/qbytearray.h>
#include <QtCore/qfuture.h>
#include <QtConcurrent>
#include <mutex>
QT_BEGIN_NAMESPACE
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
template <>
class QFutureInterface<QHttpServerResponse> : public QFutureInterfaceBase
{
public:
QFutureInterface(State initialState = NoState)
: QFutureInterfaceBase(initialState)
{
refT();
}
QFutureInterface(const QFutureInterface &other)
: QFutureInterfaceBase(other)
{
refT();
}
~QFutureInterface()
{
if (!derefT())
resultStoreBase().template clear<QHttpServerResponse>();
}
static QFutureInterface canceledResult()
{ return QFutureInterface(State(Started | Finished | Canceled)); }
QFutureInterface &operator=(const QFutureInterface &other)
{
other.refT();
if (!derefT())
resultStoreBase().template clear<QHttpServerResponse>();
QFutureInterfaceBase::operator=(other);
return *this;
}
inline QFuture<QHttpServerResponse> future()
{
return QFuture<QHttpServerResponse>(this);
}
void reportAndMoveResult(QHttpServerResponse &&result, int index = -1)
{
#if QT_VERSION < QT_VERSION_CHECK(5, 14, 0)
std::lock_guard<QMutex> locker{*mutex()};
#else
std::lock_guard<QMutex> locker{mutex(0)};
#endif
if (queryState(Canceled) || queryState(Finished))
return;
QtPrivate::ResultStoreBase &store = resultStoreBase();
const int oldResultCount = store.count();
const int insertIndex = store.addResult(
index, static_cast<void *>(new QHttpServerResponse(std::move_if_noexcept(result))));
if (!store.filterMode() || oldResultCount < store.count()) // Let's make sure it's not in pending results.
reportResultsReady(insertIndex, store.count());
}
void reportFinished()
{
QFutureInterfaceBase::reportFinished();
}
QHttpServerResponse takeResult();
};
#endif // QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
namespace QtConcurrent {
template <>
class RunFunctionTask<QHttpServerResponse> : public RunFunctionTaskBase<QHttpServerResponse>
{
public:
void run() override
{
if (this->isCanceled()) {
this->reportFinished();
return;
}
#ifndef QT_NO_EXCEPTIONS
try {
#endif
this->runFunctor();
#ifndef QT_NO_EXCEPTIONS
} catch (QException &e) {
QFutureInterface<QHttpServerResponse>::reportException(e);
} catch (...) {
QFutureInterface<QHttpServerResponse>::reportException(QUnhandledException());
}
#endif
this->reportAndMoveResult(std::move_if_noexcept(result));
this->reportFinished();
}
QHttpServerResponse result{QHttpServerResponse::StatusCode::NotFound};
};
}
class QHttpServerFutureResponsePrivate;
class Q_HTTPSERVER_EXPORT QHttpServerFutureResponse : public QHttpServerResponse
{
Q_DECLARE_PRIVATE(QHttpServerFutureResponse)
public:
using QHttpServerResponse::QHttpServerResponse;
QHttpServerFutureResponse(const QFuture<QHttpServerResponse> &futureResponse);
virtual void write(QHttpServerResponder &&responder) const override;
protected:
QHttpServerFutureResponse(QHttpServerFutureResponsePrivate *d);
};
QT_END_NAMESPACE
#endif // QHTTPSERVERFUTURERESPONSE_H