21#include "stringfilters.h"
25#include <QtCore/QRegularExpression>
26#include <QtCore/QVariant>
27#include <QtCore/QJsonArray>
28#include <QtCore/QJsonObject>
29#include <QtCore/QJsonDocument>
33 bool autoescape)
const
39 .replace(QLatin1Char(
'\\'), QStringLiteral(
"\\\\"))
41 .replace(QLatin1Char(
'\"'), QStringLiteral(
"\\\""))
43 .replace(QLatin1Char(
'\''), QStringLiteral(
"\\\'"));
49 bool autoescape)
const
54 if (safeString.get().isEmpty())
57 return QVariant(safeString.get().at(0).toUpper()
59 safeString.get().right(safeString.get().size() - 1)));
62EscapeJsFilter::EscapeJsFilter() {}
64static QList<std::pair<QString, QString>> getJsEscapes()
66 QList<std::pair<QString, QString>> jsEscapes;
67 jsEscapes << std::pair<QString, QString>(QChar::fromLatin1(
'\\'),
68 QStringLiteral(
"\\u005C"))
69 << std::pair<QString, QString>(QChar::fromLatin1(
'\''),
70 QStringLiteral(
"\\u0027"))
71 << std::pair<QString, QString>(QChar::fromLatin1(
'\"'),
72 QStringLiteral(
"\\u0022"))
73 << std::pair<QString, QString>(QChar::fromLatin1(
'>'),
74 QStringLiteral(
"\\u003E"))
75 << std::pair<QString, QString>(QChar::fromLatin1(
'<'),
76 QStringLiteral(
"\\u003C"))
77 << std::pair<QString, QString>(QChar::fromLatin1(
'&'),
78 QStringLiteral(
"\\u0026"))
79 << std::pair<QString, QString>(QChar::fromLatin1(
'='),
80 QStringLiteral(
"\\u003D"))
81 << std::pair<QString, QString>(QChar::fromLatin1(
'-'),
82 QStringLiteral(
"\\u002D"))
83 << std::pair<QString, QString>(QChar::fromLatin1(
';'),
84 QStringLiteral(
"\\u003B"))
85 << std::pair<QString, QString>(QChar(0x2028), QStringLiteral(
"\\u2028"))
86 << std::pair<QString, QString>(QChar(0x2029),
87 QStringLiteral(
"\\u2029"));
89 for (
auto i = 0; i < 32; ++i) {
90 jsEscapes << std::pair<QString, QString>(
92 QStringLiteral(
"\\u00")
93 + QStringLiteral(
"%1").arg(i, 2, 16, QLatin1Char(
'0')).toUpper());
100 bool autoescape)
const
106 static const auto jsEscapes = getJsEscapes();
108 for (
auto &
escape : jsEscapes) {
109 retString = retString.replace(
escape.first,
escape.second);
116 bool autoescape)
const
122 const QRegularExpression fixAmpersandsRegexp(
123 QStringLiteral(
"&(?!(\\w+|#\\d+);)"));
125 safeString.get().replace(fixAmpersandsRegexp, QStringLiteral(
"&"));
131 bool autoescape)
const
137 auto inputSafe = retString.isSafe();
139 retString.get().remove(argString);
141 if (inputSafe && argString.get() != QChar::fromLatin1(
';'))
148 bool autoescape)
const
157 bool autoescape)
const
161 auto lines = safeString.get().split(QLatin1Char(
'\n'));
162 auto width = QString::number(lines.size()).size();
164 const auto shouldEscape = (autoescape && !safeString.isSafe());
165 for (
auto i = 0; i < lines.size(); ++i) {
167 = QStringLiteral(
"%1. %2")
169 .arg(shouldEscape ?
QString(
escape(lines.at(i))) : lines.at(i));
172 return SafeString(lines.join(QChar::fromLatin1(
'\n')),
true);
176 bool autoescape)
const
185 bool autoescape)
const
191 else if (input.userType() == qMetaTypeId<QVariantList>()) {
192 a = toString(input.value<QVariantList>());
200 bool autoescape)
const
207 auto it = str.begin();
208 const auto end = str.end();
211 for (; it != end; ++it) {
216 toUpper = it->isSpace();
224 bool autoescape)
const
230 auto numWords = s.get().toInt(&ok);
233 return input.toString();
237 auto words = inputString.split(QLatin1Char(
' '), Qt::SkipEmptyParts);
239 if (words.size() > numWords) {
240 words = words.mid(0, numWords);
241 if (!words.at(words.size() - 1).endsWith(QStringLiteral(
"..."))) {
242 words << QStringLiteral(
"...");
245 return words.join(QChar::fromLatin1(
' '));
249 bool autoescape)
const
258 bool autoescape)
const
262 return QString::number(
267 bool autoescape)
const
275 bool autoescape)
const
283 bool autoescape)
const
287 const auto valueWidth = value.size();
289 const auto totalPadding = width - valueWidth;
290 const auto rightPadding = totalPadding >> 1;
292 return value.leftJustified(valueWidth + rightPadding).rightJustified(width);
296 bool autoescape)
const
305 bool autoescape)
const
314 bool autoescape)
const
319 = QStringLiteral(
"(%1)").arg(tags.join(QChar::fromLatin1(
'|')));
320 const QRegularExpression startTag(
321 QStringLiteral(
"<%1(/?>|(\\s+[^>]*>))").arg(tagRe));
322 const QRegularExpression endTag(QStringLiteral(
"</%1>").arg(tagRe));
325 const auto safeInput = value.isSafe();
326 value.get().remove(startTag);
327 value.get().remove(endTag);
335 bool autoescape)
const
339 static QRegularExpression tagRe(QStringLiteral(
"<[^>]*>"),
340 QRegularExpression::InvertedGreedinessOption);
349 bool autoescape)
const
353 auto width = argument.value<
int>();
354 auto partList = _input.split(QLatin1Char(
' '), Qt::SkipEmptyParts);
355 if (partList.isEmpty())
357 auto output = partList.takeFirst();
358 auto pos = output.size() - output.lastIndexOf(QLatin1Char(
'\n')) - 1;
359 Q_FOREACH (
const QString &part, partList) {
361 if (part.contains(QLatin1Char(
'\n'))) {
362 lines = part.split(QLatin1Char(
'\n'));
366 pos += lines.first().size() + 1;
368 output.append(QLatin1Char(
'\n'));
369 pos += lines.last().size();
371 output.append(QLatin1Char(
' '));
372 if (lines.size() > 1)
373 pos += lines.last().size();
382 bool autoescape)
const
386 switch (input.userType()) {
388 case QMetaType::UInt:
389 case QMetaType::LongLong:
390 case QMetaType::ULongLong:
391 case QMetaType::Double:
392 inputDouble = input.toDouble();
399 if (argument.isValid())
404 return QString::number(inputDouble,
'f', precision);
409 bool autoescape)
const
414 if (input.userType() == qMetaTypeId<QVariantList>()) {
415 const auto inputList = input.value<QVariantList>();
416 for (
const auto &item : inputList) {
425 bool autoescape)
const
429 static const QRegularExpression re(QStringLiteral(
"\n{2,}"));
432 Q_FOREACH (
const QString &bit, inputString.get().split(re)) {
433 auto _bit =
SafeString(bit, inputString.isSafe());
436 _bit.get().replace(QLatin1Char(
'\n'), QStringLiteral(
"<br />"));
437 output.append(QStringLiteral(
"<p>%1</p>").arg(_bit));
439 return markSafe(output.join(QStringLiteral(
"\n\n")));
444 bool autoescape)
const
452 inputString.get().replace(QLatin1Char(
'\n'), QStringLiteral(
"<br />")));
458 output.reserve(input.size());
460 auto it = input.constBegin();
461 const auto end = input.constEnd();
462 static const QChar asciiEndPoint(128);
463 for (; it != end; ++it)
464 if (*it < asciiEndPoint)
472 bool autoescape)
const
477 inputString = inputString.normalized(QString::NormalizationForm_KD);
478 inputString = nofailStringToAscii(inputString);
479 inputString = inputString.trimmed()
481 .remove(QRegularExpression(QStringLiteral(
"[^\\w\\s-]")))
482 .replace(QRegularExpression(QStringLiteral(
"[-\\s]+")), QStringLiteral(
"-"));
488 bool autoescape)
const
494 bool numberConvert =
true;
497 if (input.canConvert<qreal>()) {
498 size = input.toReal(&numberConvert);
499 if (!numberConvert) {
500 qWarning(
"%s",
"Failed to convert input file size into floating point value.");
504 if (!numberConvert) {
505 qWarning(
"%s",
"Failed to convert input file size into floating point value.");
511 qreal multiplier = 1.0f;
513 if (!arg.get().isEmpty()) {
514 const auto argList = arg.get().split(QLatin1Char(
','), Qt::SkipEmptyParts);
515 const auto numArgs = argList.size();
517 unitSystem = argList.at(0).toInt(&numberConvert);
518 if (!numberConvert) {
519 qWarning(
"%s",
"Failed to convert filse size format unit system into integer. Falling back to default 10.");
525 precision = argList.at(1).toInt(&numberConvert);
526 if (!numberConvert) {
527 qWarning(
"%s",
"Failed to convert file size format decimal precision into integer. Falling back to default 2.");
533 multiplier = argList.at(2).toDouble(&numberConvert);
534 if (!numberConvert) {
535 qWarning(
"%s",
"Failed to convert file size format multiplier into double value. Falling back to default 1.0");
538 if (multiplier == 0.0f) {
539 qWarning(
"%s",
"It makes no sense to multiply the file size by zero. Using default value 1.0.");
546 const double sizeMult = size * multiplier;
548 if (unitSystem == 10) {
549 if ((sizeMult > -1000) && (sizeMult < 1000)) {
552 }
else if (unitSystem == 2) {
553 if ((sizeMult > - 1024) && (sizeMult < 1024)) {
558 const std::pair<qreal,QString> sizePair =
calcFileSize(size, unitSystem, multiplier);
560 const QString retString = QString::number(sizePair.first,
'f', precision) + QLatin1Char(
' ') + sizePair.second;
562 ret.setValue(retString);
573 if(retString.length() < count)
return retString;
574 retString.truncate(count);
575 retString.append(QStringLiteral(
"..."));
582 const int len = input.length();
583 esc.reserve(
static_cast<int>(len * 1.2));
584 for (
int i = 0; i < len; ++i) {
585 const QChar ch = input.at(i);
586 if (ch == QLatin1Char(
'>')) {
587 esc += QStringLiteral(
"\\\\u003E");
588 }
else if (ch == QLatin1Char(
'<')) {
589 esc += QStringLiteral(
"\\\\u003C");
590 }
else if (ch == QLatin1Char(
'&')) {
591 esc += QStringLiteral(
"\\\\u0026");
603 if (input.isNull() || !input.isValid()) {
610 if (input.canConvert<QJsonDocument>()) {
611 json = input.toJsonDocument();
612 }
else if (input.canConvert<QJsonObject>()) {
613 json.setObject(input.toJsonObject());
614 }
else if (input.canConvert<QJsonArray>()) {
615 json.setArray(input.toJsonArray());
617 qWarning(
"%s",
"Can not convert input data into QJsonObject or QJSonArray.");
621 QString jsonString = QString::fromUtf8(json.toJson(QJsonDocument::Compact));
622 jsonString = escapeJson(jsonString);
624 const QString scriptString = u
"<script id=\"" + arg + u
"\" type=\"application/json\">" + jsonString + u
"</script>";
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
SafeString escape(const QString &input) const
SafeString conditionalEscape(const SafeString &input) const
A QString wrapper class for containing whether a string is safe or needs to be escaped.
const NestedString & get() const
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument=QVariant(), bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument=QVariant(), bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
QVariant doFilter(const QVariant &input, const QVariant &argument={}, bool autoescape={}) const override
std::pair< qreal, QString > calcFileSize(qreal size, int unitSystem=10, qreal multiplier=1.0)
bool isSafeString(const QVariant &input)
Cutelee::SafeString getSafeString(const QVariant &input)
Cutelee::SafeString markSafe(const Cutelee::SafeString &input)
Cutelee::SafeString markForEscaping(const Cutelee::SafeString &input)
Utility functions used throughout Cutelee.