Fix crash in case of incorrect Text entities.

This commit is contained in:
John Preston 2018-01-18 14:04:25 +03:00
parent 7814ee0f7a
commit f88cbf3d4b
4 changed files with 65 additions and 15 deletions

View File

@ -493,6 +493,31 @@ void SetAnnotation(const std::string &key, const QString &value) {
}
}
void SetAnnotationHex(const std::string &key, const QString &value) {
if (value.isEmpty()) {
return SetAnnotation(key, value);
}
const auto utf = value.toUtf8();
auto buffer = std::string();
buffer.reserve(4 * utf.size());
const auto hexDigit = [](std::uint8_t value) {
if (value >= 10) {
return 'A' + (value - 10);
}
return '0' + value;
};
const auto appendHex = [&](std::uint8_t value) {
buffer.push_back('\\');
buffer.push_back('x');
buffer.push_back(hexDigit(value / 16));
buffer.push_back(hexDigit(value % 16));
};
for (const auto ch : utf) {
appendHex(ch);
}
ProcessAnnotations[key] = std::move(buffer);
}
void SetAnnotationRef(const std::string &key, const QString *valuePtr) {
if (valuePtr) {
ProcessAnnotationRefs[key] = valuePtr;

View File

@ -34,6 +34,7 @@ Status Restart(); // can be only CantOpen or Started
void Finish();
void SetAnnotation(const std::string &key, const QString &value);
void SetAnnotationHex(const std::string &key, const QString &value);
inline void ClearAnnotation(const std::string &key) {
SetAnnotation(key, QString());
}

View File

@ -496,7 +496,9 @@ HistoryMessage::HistoryMessage(
initMedia(msg.has_media() ? (&msg.vmedia) : nullptr);
auto text = TextUtilities::Clean(qs(msg.vmessage));
auto entities = msg.has_entities() ? TextUtilities::EntitiesFromMTP(msg.ventities.v) : EntitiesInText();
auto entities = msg.has_entities()
? TextUtilities::EntitiesFromMTP(msg.ventities.v)
: EntitiesInText();
setText({ text, entities });
}

View File

@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include <private/qharfbuzz_p.h>
#include "core/click_handler_types.h"
#include "core/crash_reports.h"
#include "ui/text/text_block.h"
#include "lang/lang_keys.h"
#include "platform/platform_specific.h"
@ -293,19 +294,15 @@ public:
++waitingEntity;
if (links.size() >= 0x7FFF) {
while (waitingEntity != entitiesEnd && (
waitingEntity->type() == EntityInTextUrl ||
waitingEntity->type() == EntityInTextCustomUrl ||
waitingEntity->type() == EntityInTextEmail ||
waitingEntity->type() == EntityInTextHashtag ||
waitingEntity->type() == EntityInTextMention ||
waitingEntity->type() == EntityInTextMentionName ||
waitingEntity->type() == EntityInTextBotCommand ||
waitingEntity->length() <= 0)) {
while (waitingEntity != entitiesEnd
&& (isLinkEntity(*waitingEntity)
|| isInvalidEntity(*waitingEntity))) {
++waitingEntity;
}
} else {
while (waitingEntity != entitiesEnd && waitingEntity->length() <= 0) ++waitingEntity;
while (waitingEntity != entitiesEnd && isInvalidEntity(*waitingEntity)) {
++waitingEntity;
}
}
return true;
}
@ -479,9 +476,13 @@ public:
for (int l = len - emojiLookback - 1; l > 0; --l) {
_t->_text.push_back(*++ptr);
}
if (e->hasPostfix() && _t->_text.at(_t->_text.size() - 1).unicode() != Ui::Emoji::kPostfix) {
_t->_text.push_back(QChar(Ui::Emoji::kPostfix));
++len;
if (e->hasPostfix()) {
Assert(!_t->_text.isEmpty());
const auto last = _t->_text[_t->_text.size() - 1];
if (last.unicode() != Ui::Emoji::kPostfix) {
_t->_text.push_back(QChar(Ui::Emoji::kPostfix));
++len;
}
}
createBlock(-len);
@ -530,6 +531,25 @@ public:
parse(options);
}
bool isInvalidEntity(const EntityInText &entity) const {
const auto length = entity.length();
return (start + entity.offset() + length > end) || (length <= 0);
}
bool isLinkEntity(const EntityInText &entity) const {
const auto type = entity.type();
const auto urls = {
EntityInTextUrl,
EntityInTextCustomUrl,
EntityInTextEmail,
EntityInTextHashtag,
EntityInTextMention,
EntityInTextMentionName,
EntityInTextBotCommand
};
return ranges::find(urls, type) != std::end(urls);
}
void parse(const TextParseOptions &options) {
if (options.maxw > 0 && options.maxh > 0) {
stopAfterWidth = ((options.maxh / _t->_st->font->height) + 1) * options.maxw;
@ -540,7 +560,9 @@ public:
entitiesEnd = source.entities.cend();
waitingEntity = source.entities.cbegin();
while (waitingEntity != entitiesEnd && waitingEntity->length() <= 0) ++waitingEntity;
while (waitingEntity != entitiesEnd && isInvalidEntity(*waitingEntity)) {
++waitingEntity;
}
int firstMonospaceOffset = EntityInText::firstMonospaceOffset(source.entities, end - start);
ptr = start;