mirror of
https://github.com/QtExcel/QXlsx.git
synced 2025-01-16 04:42:53 +08:00
299 lines
6.4 KiB
C++
299 lines
6.4 KiB
C++
#ifndef RECURSE_REQUEST_HPP
|
|
#define RECURSE_REQUEST_HPP
|
|
|
|
#include <QtGlobal>
|
|
#include <QTcpSocket>
|
|
#include <QHash>
|
|
#include <QUrl>
|
|
#include <QUrlQuery>
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
#include <QRegularExpression>
|
|
#endif
|
|
|
|
class Request
|
|
{
|
|
|
|
public:
|
|
//!
|
|
//! \brief data
|
|
//! client request buffer data
|
|
//!
|
|
QString data;
|
|
|
|
//!
|
|
//! \brief socket
|
|
//! underlying client socket
|
|
//!
|
|
QTcpSocket *socket;
|
|
|
|
//!
|
|
//! \brief body_parsed
|
|
//! Data to be filled by body parsing middleware
|
|
//!
|
|
QHash<QString, QVariant> body_parsed;
|
|
|
|
//!
|
|
//! \brief body
|
|
//!
|
|
QString body;
|
|
|
|
//!
|
|
//! \brief method
|
|
//! HTTP method, eg: GET
|
|
//!
|
|
QString method;
|
|
|
|
//!
|
|
//! \brief protocol
|
|
//! Request protocol, eg: HTTP
|
|
//!
|
|
QString protocol;
|
|
|
|
//!
|
|
//! \brief secure
|
|
//! Shorthand for protocol == "HTTPS" to check if a requet was issued via TLS
|
|
//!
|
|
bool secure = protocol == "HTTPS";
|
|
|
|
//!
|
|
//! \brief url
|
|
//! HTTP request url, eg: /helloworld
|
|
//!
|
|
QUrl url;
|
|
|
|
//!
|
|
//! \brief query
|
|
//! query strings
|
|
//!
|
|
QUrlQuery query;
|
|
|
|
//!
|
|
//! \brief params
|
|
//!r
|
|
//! request parameters that can be filled by router middlewares
|
|
//! it's easier to provide container here (which doesn't have to be used)
|
|
QHash<QString, QString> params;
|
|
|
|
//!
|
|
//! \brief length
|
|
//! HTTP request Content-Length
|
|
//!
|
|
qint64 length = 0;
|
|
|
|
//!
|
|
//! \brief ip
|
|
//! Client ip address
|
|
//!
|
|
QHostAddress ip;
|
|
|
|
//!
|
|
//! \brief hostname
|
|
//! HTTP hostname from "Host" HTTP header
|
|
//!
|
|
QString hostname;
|
|
|
|
//!
|
|
//! \brief getHeader
|
|
//! return header value, keys are saved in lowercase
|
|
//! \param key QString
|
|
//! \return QString header value
|
|
//!
|
|
QString getHeader(const QString &key)
|
|
{
|
|
return m_headers[key];
|
|
}
|
|
|
|
//!
|
|
//! \brief getRawHeader
|
|
//! return original header name as sent by client
|
|
//! \param key QString case-sensitive key of the header
|
|
//! \return QString header value as sent by client
|
|
//!
|
|
QHash<QString, QString> getRawHeaders()
|
|
{
|
|
return m_headers;
|
|
}
|
|
|
|
//!
|
|
//! \brief getCookie
|
|
//! return cookie value with lowercase name
|
|
//! \param key case-insensitive cookie name
|
|
//! \return
|
|
//!
|
|
QString getCookie(const QString &key)
|
|
{
|
|
return m_cookies[key.toLower()];
|
|
}
|
|
|
|
//!
|
|
//! \brief getRawCookie
|
|
//! return cookie name as sent by client
|
|
//! \param key case-sensitive cookie name
|
|
//! \return
|
|
//!
|
|
QString getRawCookie(const QString &key)
|
|
{
|
|
return m_cookies[key];
|
|
}
|
|
|
|
//!
|
|
//! \brief getParam
|
|
//! return params value
|
|
//! \param key of the param, eg: name
|
|
//! \return value of the param, eg: johnny
|
|
//!
|
|
QString getParam(const QString &key)
|
|
{
|
|
return params.value(key);
|
|
}
|
|
|
|
//!
|
|
//! \brief parse
|
|
//! parse data from request
|
|
//!
|
|
//! \param QString request
|
|
//! \return true on success, false otherwise, considered bad request
|
|
//!
|
|
bool parse(QString request);
|
|
|
|
private:
|
|
//!
|
|
//! \brief header
|
|
//! HTTP request headers, eg: header["content-type"] = "text/plain"
|
|
//!
|
|
QHash<QString, QString> m_headers;
|
|
|
|
|
|
//!
|
|
//! \brief cookies
|
|
//! HTTP cookies in key/value form
|
|
//!
|
|
QHash<QString, QString> m_cookies;
|
|
|
|
//!
|
|
//! \brief httpRx
|
|
//! match HTTP request line
|
|
//!
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
QRegularExpression httpRx = QRegularExpression("^(?=[A-Z]).* \\/.* HTTP\\/[0-9]\\.[0-9]\\r\\n");
|
|
#else
|
|
QRegExp httpRx = QRegExp("^(?=[A-Z]).* \\/.* HTTP\\/[0-9]\\.[0-9]\\r\\n");
|
|
#endif
|
|
|
|
|
|
};
|
|
|
|
inline bool Request::parse(QString request)
|
|
{
|
|
// buffer all data
|
|
this->data += request;
|
|
|
|
// Save client ip address
|
|
this->ip = this->socket->peerAddress();
|
|
|
|
// if no header is present, just append all data to request.body
|
|
if ( ! this->data.contains( httpRx ) )
|
|
{
|
|
this->body.append(this->data);
|
|
return true;
|
|
}
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
auto data_list = this->data.split("\r\n");
|
|
#else
|
|
auto data_list = this->data.splitRef("\r\n");
|
|
#endif
|
|
|
|
bool is_body = false;
|
|
|
|
for (int i = 0; i < data_list.size(); ++i)
|
|
{
|
|
if (is_body)
|
|
{
|
|
this->body.append(data_list.at(i));
|
|
this->length += this->body.size();
|
|
continue;
|
|
}
|
|
|
|
auto entity_item = data_list.at(i).split(":");
|
|
|
|
if (entity_item.length() < 2 && entity_item.at(0).size() < 1 && !is_body)
|
|
{
|
|
is_body = true;
|
|
continue;
|
|
}
|
|
else if (i == 0 && entity_item.length() < 2)
|
|
{
|
|
auto first_line = entity_item.at(0).split(" ");
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
this->method = first_line.at(0);
|
|
#else
|
|
this->method = first_line.at(0).toString();
|
|
#endif
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
this->url = first_line.at(1);
|
|
#else
|
|
this->url = first_line.at(1).toString();
|
|
#endif
|
|
|
|
this->query.setQuery(this->url.query());
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
this->protocol = first_line.at(2);
|
|
#else
|
|
this->protocol = first_line.at(2).toString();
|
|
#endif
|
|
|
|
continue;
|
|
}
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
m_headers[entity_item.at(0).toLower()] = entity_item.at(1);
|
|
#else
|
|
m_headers[entity_item.at(0).toString().toLower()] = entity_item.at(1).toString();
|
|
#endif
|
|
|
|
}
|
|
|
|
if (m_headers.contains("host"))
|
|
this->hostname = m_headers["host"];
|
|
|
|
// extract cookies
|
|
// eg: USER_TOKEN=Yes;test=val
|
|
if (m_headers.contains("cookie"))
|
|
{
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
for (const auto &cookie : m_headers["cookie"].split(";"))
|
|
#else
|
|
for (const auto &cookie : m_headers["cookie"].splitRef(";"))
|
|
#endif
|
|
{
|
|
int split = cookie.indexOf("=");
|
|
if (split == -1)
|
|
continue;
|
|
|
|
auto key = cookie.left(split);
|
|
if (!key.size())
|
|
continue;
|
|
|
|
auto value = cookie.mid(split + 1);
|
|
|
|
#if QT_VERSION_MAJOR >= 6 // Qt6 or higher version
|
|
m_cookies[key.toLower()] = value;
|
|
#else
|
|
m_cookies[key.toString().toLower()] = value.toString();
|
|
#endif
|
|
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
#endif
|