1
0
mirror of https://github.com/QtExcel/QXlsx.git synced 2025-01-16 04:42:53 +08:00
QXlsx/WebServer/request.hpp
Andreas Deininger 073bec284d Fix typos
2024-02-29 20:38:51 -03:00

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 request 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