/* This file is part of Telegram Desktop, the official desktop application for the Telegram messaging service. For license and copyright information please follow this link: https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "storage/serialize_document.h" #include "storage/serialize_common.h" #include "storage/serialize_peer.h" #include "data/data_session.h" #include "data/stickers/data_stickers.h" #include "ui/image/image.h" #include "main/main_session.h" namespace Serialize { namespace { constexpr auto kVersionTag = int32(0x7FFFFFFF); constexpr auto kVersion = 5; enum StickerSetType { StickerSetTypeEmpty = 0, StickerSetTypeID = 1, StickerSetTypeShortName = 2, StickerSetTypeEmoji = 3, StickerSetTypeMasks = 4, }; } // namespace void Document::writeToStream(QDataStream &stream, DocumentData *document) { stream << quint64(document->id) << quint64(document->_access) << qint32(document->date) << document->_fileReference << qint32(kVersionTag) << qint32(kVersion) << document->filename() << document->mimeString() << qint32(document->_dc) // FileSize: Right now any file size fits 32 bit. << qint32(uint32(document->size)) << qint32(document->dimensions.width()) << qint32(document->dimensions.height()) << qint32(document->type); if (const auto sticker = document->sticker()) { stream << sticker->alt; if (sticker->setType == Data::StickersType::Emoji) { stream << qint32(StickerSetTypeEmoji); } else if (sticker->setType == Data::StickersType::Masks) { stream << qint32(StickerSetTypeMasks); } else if (sticker->set.id) { stream << qint32(StickerSetTypeID); } else { stream << qint32(StickerSetTypeEmpty); } } stream << qint32(document->getDuration()); if (document->type == StickerDocument) { const auto premium = document->isPremiumSticker() || document->isPremiumEmoji(); stream << qint32(premium ? 1 : 0); stream << qint32(document->emojiUsesTextColor() ? 1 : 0); } writeImageLocation(stream, document->thumbnailLocation()); stream << qint32(document->thumbnailByteSize()); writeImageLocation(stream, document->videoThumbnailLocation()); stream << qint32(document->videoThumbnailByteSize()) << qint32(document->inlineThumbnailIsPath() ? 1 : 0) << document->inlineThumbnailBytes(); } DocumentData *Document::readFromStreamHelper( not_null session, int streamAppVersion, QDataStream &stream, const StickerSetInfo *info) { quint64 id, access; QString name, mime; qint32 date, dc, size, width, height, type, versionTag, version = 0; QByteArray fileReference; stream >> id >> access >> date; if (streamAppVersion >= 9061) { if (streamAppVersion >= 1003013) { stream >> fileReference; } stream >> versionTag; if (versionTag == kVersionTag) { stream >> version; } } else { versionTag = 0; version = 0; } stream >> name >> mime >> dc >> size // FileSize: Right now any file size fits 32 bit. >> width >> height >> type; QVector attributes; if (!name.isEmpty()) { attributes.push_back(MTP_documentAttributeFilename(MTP_string(name))); } qint32 duration = -1; qint32 isPremiumSticker = 0; qint32 useTextColor = 0; if (type == StickerDocument) { QString alt; qint32 typeOfSet; stream >> alt >> typeOfSet; if (version >= 3) { stream >> duration; if (version >= 4) { stream >> isPremiumSticker; if (version >= 5) { stream >> useTextColor; } } } if (typeOfSet == StickerSetTypeEmpty) { attributes.push_back(MTP_documentAttributeSticker(MTP_flags(0), MTP_string(alt), MTP_inputStickerSetEmpty(), MTPMaskCoords())); } else if (info) { if (info->setId == Data::Stickers::DefaultSetId || info->setId == Data::Stickers::CloudRecentSetId || info->setId == Data::Stickers::CloudRecentAttachedSetId || info->setId == Data::Stickers::FavedSetId || info->setId == Data::Stickers::CustomSetId) { typeOfSet = StickerSetTypeEmpty; } switch (typeOfSet) { case StickerSetTypeID: { attributes.push_back(MTP_documentAttributeSticker(MTP_flags(0), MTP_string(alt), MTP_inputStickerSetID(MTP_long(info->setId), MTP_long(info->accessHash)), MTPMaskCoords())); } break; case StickerSetTypeMasks: { attributes.push_back(MTP_documentAttributeSticker(MTP_flags(MTPDdocumentAttributeSticker::Flag::f_mask), MTP_string(alt), MTP_inputStickerSetID(MTP_long(info->setId), MTP_long(info->accessHash)), MTPMaskCoords())); } break; case StickerSetTypeEmoji: { if (version < 5) { // We didn't store useTextColor yet, can't use. stream.setStatus(QDataStream::ReadCorruptData); return nullptr; } using Flag = MTPDdocumentAttributeCustomEmoji::Flag; attributes.push_back(MTP_documentAttributeCustomEmoji( MTP_flags((isPremiumSticker ? Flag(0) : Flag::f_free) | (useTextColor ? Flag::f_text_color : Flag(0))), MTP_string(alt), MTP_inputStickerSetID( MTP_long(info->setId), MTP_long(info->accessHash)))); } break; case StickerSetTypeEmpty: default: { attributes.push_back(MTP_documentAttributeSticker(MTP_flags(0), MTP_string(alt), MTP_inputStickerSetEmpty(), MTPMaskCoords())); } break; } } } else { stream >> duration; if (type == AnimatedDocument) { attributes.push_back(MTP_documentAttributeAnimated()); } } std::optional videoThumb; qint32 thumbnailByteSize = 0, videoThumbnailByteSize = 0; qint32 inlineThumbnailIsPath = 0; QByteArray inlineThumbnailBytes; const auto thumb = readImageLocation(streamAppVersion, stream); if (version >= 1) { stream >> thumbnailByteSize; videoThumb = readImageLocation(streamAppVersion, stream); stream >> videoThumbnailByteSize; if (version >= 2) { stream >> inlineThumbnailIsPath >> inlineThumbnailBytes; } } else { videoThumb = ImageLocation(); } if (width > 0 && height > 0) { if (duration >= 0) { auto flags = MTPDdocumentAttributeVideo::Flags(0); if (type == RoundVideoDocument) { flags |= MTPDdocumentAttributeVideo::Flag::f_round_message; } attributes.push_back(MTP_documentAttributeVideo(MTP_flags(flags), MTP_int(duration), MTP_int(width), MTP_int(height))); } else { attributes.push_back(MTP_documentAttributeImageSize(MTP_int(width), MTP_int(height))); } } if ((stream.status() != QDataStream::Ok) || (!dc && !access) || !thumb || !videoThumb) { stream.setStatus(QDataStream::ReadCorruptData); return nullptr; } const auto storage = std::get_if( &thumb->file().data); if (thumb->valid() && (!storage || !storage->isDocumentThumbnail())) { stream.setStatus(QDataStream::ReadCorruptData); // We can't convert legacy thumbnail location to modern, because // size letter ('s' or 'm') is lost, it was not saved in legacy. return nullptr; } return session->data().document( id, access, fileReference, date, attributes, mime, InlineImageLocation{ inlineThumbnailBytes, (inlineThumbnailIsPath == 1), }, ImageWithLocation{ .location = *thumb, .bytesCount = thumbnailByteSize }, ImageWithLocation{ .location = *videoThumb, .bytesCount = videoThumbnailByteSize }, (isPremiumSticker == 1), dc, int64(uint32(size))); } DocumentData *Document::readStickerFromStream( not_null session, int streamAppVersion, QDataStream &stream, const StickerSetInfo &info) { return readFromStreamHelper(session, streamAppVersion, stream, &info); } DocumentData *Document::readFromStream( not_null session, int streamAppVersion, QDataStream &stream) { return readFromStreamHelper(session, streamAppVersion, stream, nullptr); } int Document::sizeInStream(DocumentData *document) { int result = 0; // id + access + date result += sizeof(quint64) + sizeof(quint64) + sizeof(qint32); // file_reference + version tag + version result += bytearraySize(document->_fileReference) + sizeof(qint32) * 2; // + namelen + name + mimelen + mime + dc + size result += stringSize(document->filename()) + stringSize(document->mimeString()) + sizeof(qint32) + sizeof(qint32); // + width + height result += sizeof(qint32) + sizeof(qint32); // + type result += sizeof(qint32); if (auto sticker = document->sticker()) { // type == StickerDocument // + altlen + alt + type-of-set result += stringSize(sticker->alt) + sizeof(qint32); } else { // + duration result += sizeof(qint32); } // + thumb loc result += Serialize::imageLocationSize(document->thumbnailLocation()); result += sizeof(qint32); // thumbnail_byte_size result += Serialize::imageLocationSize(document->videoThumbnailLocation()); result += sizeof(qint32); // video_thumbnail_byte_size result += sizeof(qint32) + Serialize::bytearraySize(document->inlineThumbnailBytes()); return result; } } // namespace Serialize