9 #include "multipartformdataparser.h" 13 #include <QJsonDocument> 15 #include <QJsonObject> 20 d_ptr(new RequestPrivate)
22 d_ptr->engineRequest = engineRequest;
28 qDeleteAll(d_ptr->uploads);
36 return d->engineRequest->remoteAddress;
44 quint32 data = d->engineRequest->remoteAddress.toIPv4Address(&ok);
48 return d->engineRequest->remoteAddress.toString();
58 if (!d->remoteHostname.isEmpty()) {
59 ret = d->remoteHostname;
64 if (ptr.
error() != QHostInfo::NoError) {
65 qCDebug(CUTELYST_REQUEST) <<
"DNS lookup for the client hostname failed" << d->engineRequest->remoteAddress;
70 ret = d->remoteHostname;
77 return d->engineRequest->remotePort;
85 if (!(d->parserStatus & RequestPrivate::UrlParsed)) {
87 if (d->engineRequest->serverAddress.isEmpty()) {
93 uri.
setScheme(d->engineRequest->isSecure ? QStringLiteral(
"https") : QStringLiteral(
"http"));
98 if (!d->engineRequest->query.isEmpty()) {
103 d->parserStatus |= RequestPrivate::UrlParsed;
112 if (!(d->parserStatus & RequestPrivate::BaseParsed)) {
113 base = d->engineRequest->isSecure ? QStringLiteral(
"https://") : QStringLiteral(
"http://");
116 if (d->engineRequest->serverAddress.isEmpty()) {
126 d->parserStatus |= RequestPrivate::BaseParsed;
134 return d->engineRequest->path;
176 return d->engineRequest->isSecure;
188 if (!(d->parserStatus & RequestPrivate::BodyParsed)) {
211 return RequestPrivate::paramsMultiMapToVariantMap(
bodyParameters());
217 if (!(d->parserStatus & RequestPrivate::BodyParsed)) {
229 while (it != query.
constEnd() && it.
key() == key) {
239 if (!(d->parserStatus & RequestPrivate::QueryParsed)) {
242 return d->queryKeywords;
253 if (!(d->parserStatus & RequestPrivate::QueryParsed)) {
256 return d->queryParam;
265 while (it != query.
constEnd() && it.
key() == key) {
275 if (!(d->parserStatus & RequestPrivate::CookiesParsed)) {
279 return d->cookies.value(name);
287 if (!(d->parserStatus & RequestPrivate::CookiesParsed)) {
291 auto it = d->cookies.constFind(name);
292 while (it != d->cookies.constEnd() && it.key() == name) {
302 if (!(d->parserStatus & RequestPrivate::CookiesParsed)) {
311 return d->engineRequest->headers;
317 return d->engineRequest->method;
323 return d->engineRequest->method.compare(u
"POST") == 0;
329 return d->engineRequest->method.compare(u
"GET") == 0;
335 return d->engineRequest->method.compare(u
"HEAD") == 0;
341 return d->engineRequest->method.compare(u
"PUT") == 0;
347 return d->engineRequest->method.compare(u
"PATCH") == 0;
353 return d->engineRequest->method.compare(u
"DELETE") == 0;
359 return d->engineRequest->protocol;
365 return d->engineRequest->headers.header(QStringLiteral(
"X_REQUESTED_WITH")).compare(u
"XMLHttpRequest") == 0;
371 return d->engineRequest->remoteUser;
377 if (!(d->parserStatus & RequestPrivate::BodyParsed)) {
386 if (!(d->parserStatus & RequestPrivate::BodyParsed)) {
389 return d->uploadsMap;
396 const auto range = map.equal_range(name);
397 for (
auto i = range.first; i != range.second; ++i) {
412 ret.
replace(it.key(), it.value());
440 void RequestPrivate::parseUrlQuery()
const 443 if (engineRequest->query.size()) {
445 if (engineRequest->query.indexOf(
'=') < 0) {
447 queryKeywords = Utils::decodePercentEncoding(&aux);
449 if (parserStatus & RequestPrivate::UrlParsed) {
450 queryParam = Utils::decodePercentEncoding(engineRequest->query.data(), engineRequest->query.size());
454 queryParam = Utils::decodePercentEncoding(aux.
data(), aux.
size());
458 parserStatus |= RequestPrivate::QueryParsed;
461 void RequestPrivate::parseBody()
const 464 parserStatus |= RequestPrivate::BodyParsed;
468 bool sequencial = body->isSequential();
469 qint64 posOrig = body->pos();
470 if (sequencial && posOrig) {
471 qCWarning(CUTELYST_REQUEST) <<
"Can not parse sequential post body out of beginning";
472 parserStatus |= RequestPrivate::BodyParsed;
476 const QString contentTypeKey = QStringLiteral(
"CONTENT_TYPE");
477 const QString contentType = engineRequest->headers.header(contentTypeKey);
478 if (contentType.
startsWith(u
"application/x-www-form-urlencoded", Qt::CaseInsensitive)) {
486 bodyParam = Utils::decodePercentEncoding(line.
data(), line.
size());
488 }
else if (contentType.
startsWith(u
"multipart/form-data", Qt::CaseInsensitive)) {
494 for (
Upload *upload : ups) {
495 if (upload->filename().isEmpty() && upload->headers().header(contentTypeKey).isEmpty()) {
499 uploadsMap.insert(upload->name(), upload);
503 }
else if (contentType.
startsWith(u
"application/json", Qt::CaseInsensitive)) {
515 parserStatus |= RequestPrivate::BodyParsed;
518 static inline bool isSlit(
QChar c)
523 int findNextSplit(
const QString &text,
int from,
int length)
525 while (from < length) {
526 if (isSlit(text.
at(from))) {
534 static inline bool isLWS(
QChar c)
539 static int nextNonWhitespace(
const QString &text,
int from,
int length)
545 while (from < length) {
546 if (isLWS(text.
at(from)))
556 static std::pair<QString, QString> nextField(
const QString &text,
int &position)
558 std::pair<QString, QString> ret;
563 const int length = text.
length();
564 position = nextNonWhitespace(text, position, length);
566 int semiColonPosition = findNextSplit(text, position, length);
567 if (semiColonPosition < 0)
568 semiColonPosition = length;
571 if (equalsPosition < 0 || equalsPosition > semiColonPosition) {
575 ret.first = text.
mid(position, equalsPosition - position).
trimmed();
576 int secondLength = semiColonPosition - equalsPosition - 1;
577 if (secondLength > 0) {
578 ret.second = text.
mid(equalsPosition + 1, secondLength).
trimmed();
581 position = semiColonPosition;
585 void RequestPrivate::parseCookies()
const 587 const QString cookieString = engineRequest->headers.header(QStringLiteral(
"COOKIE"));
589 const int length = cookieString.
length();
590 while (position < length) {
591 const auto field = nextField(cookieString, position);
592 if (field.first.isEmpty()) {
598 if (field.second.isEmpty()) {
602 cookies.insert(field.first, field.second);
606 parserStatus |= RequestPrivate::CookiesParsed;
609 QVariantMap RequestPrivate::paramsMultiMapToVariantMap(
const ParamsMultiMap ¶ms)
615 ret.insert(ret.constBegin(), end.key(), end.value());
620 #include "moc_request.cpp" QJsonDocument fromJson(const QByteArray &json, QJsonParseError *error)
QString url(QUrl::FormattingOptions options) const const
int indexOf(QChar ch, int from, Qt::CaseSensitivity cs) const const
Request(EngineRequest *engineRequest)
QString & append(QChar ch)
QString method() const noexcept
typename QMap< Key, T >::const_iterator constFind(const Key &key, const T &value) const const
QJsonDocument toJsonDocument() const const
QJsonArray array() const const
typename QMap< Key, T >::iterator replace(const Key &key, const T &value)
QVariant bodyData() const
QMap< Key, T > & unite(const QMap< Key, T > &other)
bool isDelete() const noexcept
QJsonObject object() const const
QMap::const_iterator constBegin() const const
QStringList arguments() const noexcept
QJsonArray bodyJsonArray() const
QIODevice * body
The QIODevice containing the body (if any) of the request.
QString toString() const const
void setMatch(const QString &match)
QString match() const noexcept
QString protocol() const noexcept
Cutelyst Upload handles file upload request
void setPath(const QString &path, QUrl::ParsingMode mode)
ParamsMultiMap queryParams() const
ParamsMultiMap bodyParameters() const
void addQueryItem(const QString &key, const QString &value)
QVector< Upload * > uploads() const
QHostAddress address() const noexcept
void setAuthority(const QString &authority, QUrl::ParsingMode mode)
QString fromUtf8(const char *str, int size)
QString addressString() const
const Key & key() const const
QMultiMap< QString, Upload * > uploadsMap() const
Headers headers() const noexcept
bool isGet() const noexcept
QString trimmed() const const
QMap::const_iterator constEnd() const const
bool startsWith(const QString &s, Qt::CaseSensitivity cs) const const
ParamsMultiMap cookies() const
void setScheme(const QString &scheme)
The Cutelyst namespace holds all public Cutelyst API.
QJsonObject bodyJsonObject() const
void setCaptures(const QStringList &captures)
void setArguments(const QStringList &arguments)
QJsonDocument bodyJsonDocument() const
QVariant fromValue(const T &value)
QHostInfo fromName(const QString &name)
ParamsMultiMap queryParameters() const
bool isPost() const noexcept
ParamsMultiMap mangleParams(const ParamsMultiMap &args, bool append=false) const
bool xhr() const noexcept
QStringList captures() const noexcept
QString mid(int position, int n) const const
const QChar at(int position) const const
void setQuery(const QString &query, QUrl::ParsingMode mode)
QString queryKeywords() const
void push_back(const T &value)
QString fromLatin1(const char *str, int size)
QString cookie(const QString &name) const
bool isPut() const noexcept
void prepend(const T &value)
QVariantMap bodyParametersVariant() const
void setHost(const QString &host, QUrl::ParsingMode mode)
bool secure() const noexcept
bool isPatch() const noexcept
QUrl uriWith(const ParamsMultiMap &args, bool append=false) const
QList::const_iterator constEnd() const const
QList::const_iterator constBegin() const const
QHostInfo::HostInfoError error() const const
QVariantMap queryParametersVariant() const
QString remoteUser() const noexcept
bool isHead() const noexcept
QString path() const noexcept
Engine * engine() const noexcept
QString hostName() const const