2018-12-31 22:04:05 +03:00
|
|
|
//
|
2023-01-01 00:28:08 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/telegram/Photo.h"
|
|
|
|
|
2022-06-02 17:52:12 +03:00
|
|
|
#include "td/telegram/Dimensions.h"
|
2018-12-27 22:24:44 +03:00
|
|
|
#include "td/telegram/files/FileEncryptionKey.h"
|
2019-01-20 00:54:29 +03:00
|
|
|
#include "td/telegram/files/FileLocation.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/telegram/files/FileManager.h"
|
2022-04-10 01:15:49 +03:00
|
|
|
#include "td/telegram/files/FileType.h"
|
2018-06-26 02:43:11 +03:00
|
|
|
#include "td/telegram/net/DcId.h"
|
2022-04-10 01:15:49 +03:00
|
|
|
#include "td/telegram/PhotoFormat.h"
|
|
|
|
#include "td/telegram/PhotoSizeSource.h"
|
2023-01-20 17:31:33 +03:00
|
|
|
#include "td/telegram/Td.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2021-01-01 15:59:53 +03:00
|
|
|
#include "td/utils/algorithm.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/utils/common.h"
|
|
|
|
#include "td/utils/format.h"
|
|
|
|
#include "td/utils/logging.h"
|
2023-01-18 22:25:25 +03:00
|
|
|
#include "td/utils/overloaded.h"
|
2021-05-17 15:21:11 +03:00
|
|
|
#include "td/utils/SliceBuilder.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2022-04-16 06:01:05 +03:00
|
|
|
#include <algorithm>
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
2022-12-22 22:38:30 +03:00
|
|
|
int64 get_profile_photo_id(const tl_object_ptr<telegram_api::UserProfilePhoto> &profile_photo_ptr) {
|
|
|
|
if (profile_photo_ptr != nullptr && profile_photo_ptr->get_id() == telegram_api::userProfilePhoto::ID) {
|
|
|
|
return static_cast<const telegram_api::userProfilePhoto *>(profile_photo_ptr.get())->photo_id_;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-08 12:26:35 +03:00
|
|
|
ProfilePhoto get_profile_photo(FileManager *file_manager, UserId user_id, int64 user_access_hash,
|
2018-12-31 22:04:05 +03:00
|
|
|
tl_object_ptr<telegram_api::UserProfilePhoto> &&profile_photo_ptr) {
|
|
|
|
ProfilePhoto result;
|
|
|
|
int32 profile_photo_id =
|
|
|
|
profile_photo_ptr == nullptr ? telegram_api::userProfilePhotoEmpty::ID : profile_photo_ptr->get_id();
|
|
|
|
switch (profile_photo_id) {
|
|
|
|
case telegram_api::userProfilePhotoEmpty::ID:
|
|
|
|
break;
|
|
|
|
case telegram_api::userProfilePhoto::ID: {
|
|
|
|
auto profile_photo = move_tl_object_as<telegram_api::userProfilePhoto>(profile_photo_ptr);
|
2022-12-22 22:41:11 +03:00
|
|
|
if (profile_photo->photo_id_ == 0 || profile_photo->photo_id_ == -2) {
|
|
|
|
LOG(ERROR) << "Receive a profile photo without identifier " << to_string(profile_photo);
|
|
|
|
break;
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2019-06-08 12:26:35 +03:00
|
|
|
auto dc_id = DcId::create(profile_photo->dc_id_);
|
2021-11-01 21:53:23 +03:00
|
|
|
result.has_animation = profile_photo->has_video_;
|
2022-12-09 15:52:21 +03:00
|
|
|
result.is_personal = profile_photo->personal_;
|
2018-12-31 22:04:05 +03:00
|
|
|
result.id = profile_photo->photo_id_;
|
2021-04-09 17:01:58 +03:00
|
|
|
result.minithumbnail = profile_photo->stripped_thumb_.as_slice().str();
|
2022-04-10 01:15:49 +03:00
|
|
|
result.small_file_id = register_photo_size(
|
2021-10-27 16:52:22 +03:00
|
|
|
file_manager, PhotoSizeSource::dialog_photo(DialogId(user_id), user_access_hash, false), result.id,
|
|
|
|
0 /*access_hash*/, "" /*file_reference*/, DialogId(), 0 /*file_size*/, dc_id, PhotoFormat::Jpeg);
|
2022-04-10 01:15:49 +03:00
|
|
|
result.big_file_id = register_photo_size(
|
2021-10-27 16:52:22 +03:00
|
|
|
file_manager, PhotoSizeSource::dialog_photo(DialogId(user_id), user_access_hash, true), result.id,
|
|
|
|
0 /*access_hash*/, "" /*file_reference*/, DialogId(), 0 /*file_size*/, dc_id, PhotoFormat::Jpeg);
|
2018-12-31 22:04:05 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
tl_object_ptr<td_api::profilePhoto> get_profile_photo_object(FileManager *file_manager,
|
2020-07-08 22:59:31 +03:00
|
|
|
const ProfilePhoto &profile_photo) {
|
|
|
|
if (!profile_photo.small_file_id.is_valid()) {
|
2018-12-31 22:04:05 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
2020-07-08 22:59:31 +03:00
|
|
|
return td_api::make_object<td_api::profilePhoto>(
|
|
|
|
profile_photo.id, file_manager->get_file_object(profile_photo.small_file_id),
|
2021-04-09 17:01:58 +03:00
|
|
|
file_manager->get_file_object(profile_photo.big_file_id), get_minithumbnail_object(profile_photo.minithumbnail),
|
2022-12-09 15:52:21 +03:00
|
|
|
profile_photo.has_animation, profile_photo.is_personal);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2022-04-29 10:50:10 +03:00
|
|
|
bool need_update_profile_photo(const ProfilePhoto &from, const ProfilePhoto &to) {
|
|
|
|
return from.id != to.id || need_update_dialog_photo(from, to);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &operator<<(StringBuilder &string_builder, const ProfilePhoto &profile_photo) {
|
2021-03-15 03:32:14 +03:00
|
|
|
return string_builder << "<ID = " << profile_photo.id << ", small_file_id = " << profile_photo.small_file_id
|
2020-06-24 11:45:36 +03:00
|
|
|
<< ", big_file_id = " << profile_photo.big_file_id
|
2022-12-09 15:52:21 +03:00
|
|
|
<< ", has_animation = " << profile_photo.has_animation
|
|
|
|
<< ", is_personal = " << profile_photo.is_personal << '>';
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2019-06-08 12:26:35 +03:00
|
|
|
DialogPhoto get_dialog_photo(FileManager *file_manager, DialogId dialog_id, int64 dialog_access_hash,
|
|
|
|
tl_object_ptr<telegram_api::ChatPhoto> &&chat_photo_ptr) {
|
2018-12-31 22:04:05 +03:00
|
|
|
int32 chat_photo_id = chat_photo_ptr == nullptr ? telegram_api::chatPhotoEmpty::ID : chat_photo_ptr->get_id();
|
|
|
|
|
|
|
|
DialogPhoto result;
|
|
|
|
switch (chat_photo_id) {
|
|
|
|
case telegram_api::chatPhotoEmpty::ID:
|
|
|
|
break;
|
|
|
|
case telegram_api::chatPhoto::ID: {
|
|
|
|
auto chat_photo = move_tl_object_as<telegram_api::chatPhoto>(chat_photo_ptr);
|
|
|
|
|
2019-06-08 12:26:35 +03:00
|
|
|
auto dc_id = DcId::create(chat_photo->dc_id_);
|
2021-11-01 21:53:23 +03:00
|
|
|
result.has_animation = chat_photo->has_video_;
|
2022-12-09 15:52:21 +03:00
|
|
|
result.is_personal = false;
|
2021-04-09 17:01:58 +03:00
|
|
|
result.minithumbnail = chat_photo->stripped_thumb_.as_slice().str();
|
2021-10-27 16:52:22 +03:00
|
|
|
result.small_file_id =
|
2022-04-10 01:15:49 +03:00
|
|
|
register_photo_size(file_manager, PhotoSizeSource::dialog_photo(dialog_id, dialog_access_hash, false),
|
|
|
|
chat_photo->photo_id_, 0, "", DialogId(), 0, dc_id, PhotoFormat::Jpeg);
|
2021-10-27 16:52:22 +03:00
|
|
|
result.big_file_id =
|
2022-04-10 01:15:49 +03:00
|
|
|
register_photo_size(file_manager, PhotoSizeSource::dialog_photo(dialog_id, dialog_access_hash, true),
|
|
|
|
chat_photo->photo_id_, 0, "", DialogId(), 0, dc_id, PhotoFormat::Jpeg);
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-07-07 13:00:56 +03:00
|
|
|
tl_object_ptr<td_api::chatPhotoInfo> get_chat_photo_info_object(FileManager *file_manager,
|
|
|
|
const DialogPhoto *dialog_photo) {
|
2018-12-31 22:04:05 +03:00
|
|
|
if (dialog_photo == nullptr || !dialog_photo->small_file_id.is_valid()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2020-07-07 13:00:56 +03:00
|
|
|
return td_api::make_object<td_api::chatPhotoInfo>(file_manager->get_file_object(dialog_photo->small_file_id),
|
2020-07-08 22:59:31 +03:00
|
|
|
file_manager->get_file_object(dialog_photo->big_file_id),
|
2021-04-09 17:01:58 +03:00
|
|
|
get_minithumbnail_object(dialog_photo->minithumbnail),
|
2022-12-09 15:52:21 +03:00
|
|
|
dialog_photo->has_animation, dialog_photo->is_personal);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2019-01-20 06:34:47 +03:00
|
|
|
vector<FileId> dialog_photo_get_file_ids(const DialogPhoto &dialog_photo) {
|
|
|
|
vector<FileId> result;
|
|
|
|
if (dialog_photo.small_file_id.is_valid()) {
|
|
|
|
result.push_back(dialog_photo.small_file_id);
|
|
|
|
}
|
|
|
|
if (dialog_photo.big_file_id.is_valid()) {
|
|
|
|
result.push_back(dialog_photo.big_file_id);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-12-09 15:52:21 +03:00
|
|
|
DialogPhoto as_fake_dialog_photo(const Photo &photo, DialogId dialog_id, bool is_personal) {
|
2019-06-08 12:26:35 +03:00
|
|
|
DialogPhoto result;
|
2020-06-23 21:04:26 +03:00
|
|
|
if (!photo.is_empty()) {
|
2019-06-08 12:26:35 +03:00
|
|
|
for (auto &size : photo.photos) {
|
|
|
|
if (size.type == 'a') {
|
|
|
|
result.small_file_id = size.file_id;
|
|
|
|
} else if (size.type == 'c') {
|
|
|
|
result.big_file_id = size.file_id;
|
|
|
|
}
|
|
|
|
}
|
2021-04-16 17:43:51 +03:00
|
|
|
result.minithumbnail = photo.minithumbnail;
|
2020-07-08 22:59:31 +03:00
|
|
|
result.has_animation = !photo.animations.empty();
|
2022-12-09 15:52:21 +03:00
|
|
|
result.is_personal = is_personal;
|
2019-06-08 12:26:35 +03:00
|
|
|
if (!result.small_file_id.is_valid() || !result.big_file_id.is_valid()) {
|
2021-05-31 20:06:08 +03:00
|
|
|
LOG(ERROR) << "Failed to convert " << photo << " to chat photo of " << dialog_id;
|
2019-06-08 12:26:35 +03:00
|
|
|
return DialogPhoto();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-12-09 15:52:21 +03:00
|
|
|
DialogPhoto as_dialog_photo(FileManager *file_manager, DialogId dialog_id, int64 dialog_access_hash, const Photo &photo,
|
|
|
|
bool is_personal) {
|
2022-04-28 20:33:24 +03:00
|
|
|
DialogPhoto result;
|
2022-12-09 15:52:21 +03:00
|
|
|
static_cast<DialogPhoto &>(result) = as_fake_dialog_photo(photo, dialog_id, is_personal);
|
2020-07-20 17:46:54 +03:00
|
|
|
if (!result.small_file_id.is_valid()) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto reregister_photo = [&](bool is_big, FileId file_id) {
|
|
|
|
auto file_view = file_manager->get_file_view(file_id);
|
|
|
|
CHECK(file_view.has_remote_location());
|
|
|
|
auto remote = file_view.remote_location();
|
|
|
|
CHECK(remote.is_photo());
|
|
|
|
CHECK(!remote.is_web());
|
2022-04-28 20:33:24 +03:00
|
|
|
remote.set_source(PhotoSizeSource::dialog_photo(dialog_id, dialog_access_hash, is_big));
|
2022-04-28 22:29:10 +03:00
|
|
|
return file_manager->register_remote(std::move(remote), FileLocationSource::FromServer, DialogId(), 0, 0,
|
|
|
|
file_view.remote_name());
|
2020-07-20 17:46:54 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
result.small_file_id = reregister_photo(false, result.small_file_id);
|
|
|
|
result.big_file_id = reregister_photo(true, result.big_file_id);
|
|
|
|
|
2020-07-20 17:05:49 +03:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-12-09 15:52:21 +03:00
|
|
|
ProfilePhoto as_profile_photo(FileManager *file_manager, UserId user_id, int64 user_access_hash, const Photo &photo,
|
|
|
|
bool is_personal) {
|
2022-04-28 20:33:24 +03:00
|
|
|
ProfilePhoto result;
|
2022-12-09 15:52:21 +03:00
|
|
|
static_cast<DialogPhoto &>(result) =
|
|
|
|
as_dialog_photo(file_manager, DialogId(user_id), user_access_hash, photo, is_personal);
|
2022-04-28 20:33:24 +03:00
|
|
|
if (result.small_file_id.is_valid()) {
|
|
|
|
result.id = photo.id.get();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-04-28 21:02:44 +03:00
|
|
|
bool is_same_dialog_photo(FileManager *file_manager, DialogId dialog_id, const Photo &photo,
|
2022-12-09 15:52:21 +03:00
|
|
|
const DialogPhoto &dialog_photo, bool is_personal) {
|
2022-04-28 21:02:44 +03:00
|
|
|
auto get_unique_file_id = [file_manager](FileId file_id) {
|
|
|
|
return file_manager->get_file_view(file_id).get_unique_file_id();
|
|
|
|
};
|
2022-12-09 15:52:21 +03:00
|
|
|
auto fake_photo = as_fake_dialog_photo(photo, dialog_id, is_personal);
|
2022-04-28 21:02:44 +03:00
|
|
|
return get_unique_file_id(fake_photo.small_file_id) == get_unique_file_id(dialog_photo.small_file_id) &&
|
|
|
|
get_unique_file_id(fake_photo.big_file_id) == get_unique_file_id(dialog_photo.big_file_id);
|
|
|
|
}
|
|
|
|
|
2022-04-29 10:50:10 +03:00
|
|
|
bool need_update_dialog_photo(const DialogPhoto &from, const DialogPhoto &to) {
|
|
|
|
return from.small_file_id != to.small_file_id || from.big_file_id != to.big_file_id ||
|
2022-12-09 15:52:21 +03:00
|
|
|
from.has_animation != to.has_animation || from.is_personal != to.is_personal;
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &operator<<(StringBuilder &string_builder, const DialogPhoto &dialog_photo) {
|
|
|
|
return string_builder << "<small_file_id = " << dialog_photo.small_file_id
|
2020-06-24 11:45:36 +03:00
|
|
|
<< ", big_file_id = " << dialog_photo.big_file_id
|
2022-12-09 15:52:21 +03:00
|
|
|
<< ", has_animation = " << dialog_photo.has_animation
|
|
|
|
<< ", is_personal = " << dialog_photo.is_personal << '>';
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2020-05-31 22:22:15 +03:00
|
|
|
static tl_object_ptr<td_api::photoSize> get_photo_size_object(FileManager *file_manager, const PhotoSize *photo_size) {
|
2020-05-30 01:48:56 +03:00
|
|
|
if (photo_size == nullptr || !photo_size->file_id.is_valid()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2020-05-31 22:22:15 +03:00
|
|
|
return td_api::make_object<td_api::photoSize>(
|
|
|
|
photo_size->type ? std::string(1, static_cast<char>(photo_size->type))
|
|
|
|
: std::string(), // TODO replace string type with integer type
|
2020-08-25 15:09:24 +03:00
|
|
|
file_manager->get_file_object(photo_size->file_id), photo_size->dimensions.width, photo_size->dimensions.height,
|
|
|
|
vector<int32>(photo_size->progressive_sizes));
|
2020-05-30 01:48:56 +03:00
|
|
|
}
|
|
|
|
|
2020-05-31 22:22:15 +03:00
|
|
|
static vector<td_api::object_ptr<td_api::photoSize>> get_photo_sizes_object(FileManager *file_manager,
|
|
|
|
const vector<PhotoSize> &photo_sizes) {
|
2019-01-17 03:27:43 +03:00
|
|
|
auto sizes = transform(photo_sizes, [file_manager](const PhotoSize &photo_size) {
|
|
|
|
return get_photo_size_object(file_manager, &photo_size);
|
|
|
|
});
|
2020-10-12 10:17:02 +03:00
|
|
|
std::stable_sort(sizes.begin(), sizes.end(), [](const auto &lhs, const auto &rhs) {
|
2018-03-09 16:56:42 +03:00
|
|
|
if (lhs->photo_->expected_size_ != rhs->photo_->expected_size_) {
|
|
|
|
return lhs->photo_->expected_size_ < rhs->photo_->expected_size_;
|
|
|
|
}
|
2018-04-03 10:35:04 +03:00
|
|
|
return static_cast<uint32>(lhs->width_) * static_cast<uint32>(lhs->height_) <
|
|
|
|
static_cast<uint32>(rhs->width_) * static_cast<uint32>(rhs->height_);
|
2018-03-09 16:56:42 +03:00
|
|
|
});
|
2020-10-12 10:31:57 +03:00
|
|
|
td::remove_if(sizes, [](const auto &size) {
|
|
|
|
return !size->photo_->local_->can_be_downloaded_ && !size->photo_->local_->is_downloading_completed_;
|
|
|
|
});
|
2019-01-17 03:27:43 +03:00
|
|
|
return sizes;
|
2018-03-09 16:56:42 +03:00
|
|
|
}
|
|
|
|
|
2020-07-07 11:41:01 +03:00
|
|
|
static tl_object_ptr<td_api::animatedChatPhoto> get_animated_chat_photo_object(FileManager *file_manager,
|
|
|
|
const AnimationSize *animation_size) {
|
|
|
|
if (animation_size == nullptr || !animation_size->file_id.is_valid()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return td_api::make_object<td_api::animatedChatPhoto>(animation_size->dimensions.width,
|
|
|
|
file_manager->get_file_object(animation_size->file_id),
|
|
|
|
animation_size->main_frame_timestamp);
|
|
|
|
}
|
|
|
|
|
2021-08-01 06:17:51 +03:00
|
|
|
Photo get_encrypted_file_photo(FileManager *file_manager, unique_ptr<EncryptedFile> &&file,
|
2019-06-17 02:05:56 +03:00
|
|
|
tl_object_ptr<secret_api::decryptedMessageMediaPhoto> &&photo,
|
|
|
|
DialogId owner_dialog_id) {
|
2018-12-31 22:04:05 +03:00
|
|
|
FileId file_id = file_manager->register_remote(
|
2019-06-08 12:26:35 +03:00
|
|
|
FullRemoteFileLocation(FileType::Encrypted, file->id_, file->access_hash_, DcId::create(file->dc_id_), string()),
|
2018-03-08 23:12:31 +03:00
|
|
|
FileLocationSource::FromServer, owner_dialog_id, photo->size_, 0,
|
|
|
|
PSTRING() << static_cast<uint64>(file->id_) << ".jpg");
|
2018-12-31 22:04:05 +03:00
|
|
|
file_manager->set_encryption_key(file_id, FileEncryptionKey{photo->key_.as_slice(), photo->iv_.as_slice()});
|
|
|
|
|
|
|
|
Photo res;
|
2020-06-23 20:50:16 +03:00
|
|
|
res.id = 0;
|
2018-12-31 22:04:05 +03:00
|
|
|
res.date = 0;
|
|
|
|
|
|
|
|
if (!photo->thumb_.empty()) {
|
2019-03-01 22:51:33 +03:00
|
|
|
res.photos.push_back(get_secret_thumbnail_photo_size(file_manager, std::move(photo->thumb_), owner_dialog_id,
|
|
|
|
photo->thumb_w_, photo->thumb_h_));
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
PhotoSize s;
|
|
|
|
s.type = 'i';
|
2022-05-26 17:13:31 +03:00
|
|
|
s.dimensions = get_dimensions(photo->w_, photo->h_, nullptr);
|
2018-12-31 22:04:05 +03:00
|
|
|
s.size = photo->size_;
|
|
|
|
s.file_id = file_id;
|
|
|
|
res.photos.push_back(s);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2023-01-20 17:31:33 +03:00
|
|
|
Photo get_photo(Td *td, tl_object_ptr<telegram_api::Photo> &&photo, DialogId owner_dialog_id) {
|
2019-06-08 12:26:35 +03:00
|
|
|
if (photo == nullptr || photo->get_id() == telegram_api::photoEmpty::ID) {
|
2020-06-23 20:50:16 +03:00
|
|
|
return Photo();
|
2019-06-08 12:26:35 +03:00
|
|
|
}
|
|
|
|
CHECK(photo->get_id() == telegram_api::photo::ID);
|
2023-01-20 17:31:33 +03:00
|
|
|
return get_photo(td, move_tl_object_as<telegram_api::photo>(photo), owner_dialog_id);
|
2019-06-08 12:26:35 +03:00
|
|
|
}
|
|
|
|
|
2023-01-20 17:31:33 +03:00
|
|
|
Photo get_photo(Td *td, tl_object_ptr<telegram_api::photo> &&photo, DialogId owner_dialog_id) {
|
2020-06-23 19:40:37 +03:00
|
|
|
CHECK(photo != nullptr);
|
2018-12-31 22:04:05 +03:00
|
|
|
Photo res;
|
|
|
|
|
|
|
|
res.id = photo->id_;
|
|
|
|
res.date = photo->date_;
|
2021-11-01 21:53:23 +03:00
|
|
|
res.has_stickers = photo->has_stickers_;
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2020-06-23 21:04:26 +03:00
|
|
|
if (res.is_empty()) {
|
2021-03-15 03:32:14 +03:00
|
|
|
LOG(ERROR) << "Receive photo with identifier " << res.id.get();
|
2019-06-17 02:41:57 +03:00
|
|
|
res.id = -3;
|
|
|
|
}
|
|
|
|
|
2020-06-24 14:13:33 +03:00
|
|
|
DcId dc_id = DcId::create(photo->dc_id_);
|
2018-12-31 22:04:05 +03:00
|
|
|
for (auto &size_ptr : photo->sizes_) {
|
2023-01-20 17:31:33 +03:00
|
|
|
auto photo_size = get_photo_size(td->file_manager_.get(), PhotoSizeSource::thumbnail(FileType::Photo, 0),
|
|
|
|
photo->id_, photo->access_hash_, photo->file_reference_.as_slice().str(), dc_id,
|
2021-10-27 16:52:22 +03:00
|
|
|
owner_dialog_id, std::move(size_ptr), PhotoFormat::Jpeg);
|
2019-03-01 22:51:33 +03:00
|
|
|
if (photo_size.get_offset() == 0) {
|
2019-07-20 02:33:49 +03:00
|
|
|
PhotoSize &size = photo_size.get<0>();
|
2022-05-30 04:01:55 +03:00
|
|
|
if (size.type == 0 || size.type == 't' || size.type == 'i' || size.type == 'p' || size.type == 'u' ||
|
|
|
|
size.type == 'v') {
|
2019-07-20 02:33:49 +03:00
|
|
|
LOG(ERROR) << "Skip unallowed photo size " << size;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
res.photos.push_back(std::move(size));
|
2019-03-01 22:51:33 +03:00
|
|
|
} else {
|
|
|
|
res.minithumbnail = std::move(photo_size.get<1>());
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2020-06-24 14:13:33 +03:00
|
|
|
for (auto &size_ptr : photo->video_sizes_) {
|
2023-01-20 17:31:33 +03:00
|
|
|
auto animation =
|
2023-01-23 11:51:00 +03:00
|
|
|
process_video_size(td, PhotoSizeSource::thumbnail(FileType::Photo, 0), photo->id_, photo->access_hash_,
|
2023-01-20 17:31:33 +03:00
|
|
|
photo->file_reference_.as_slice().str(), dc_id, owner_dialog_id, std::move(size_ptr));
|
2023-01-18 22:25:25 +03:00
|
|
|
if (animation.empty()) {
|
|
|
|
continue;
|
2020-07-07 11:41:01 +03:00
|
|
|
}
|
2023-01-18 22:25:25 +03:00
|
|
|
animation.visit(overloaded(
|
2023-01-23 00:57:40 +03:00
|
|
|
[&](AnimationSize &&animation_size) {
|
2023-01-18 22:25:25 +03:00
|
|
|
if (animation_size.type != 0 && animation_size.dimensions.width == animation_size.dimensions.height) {
|
|
|
|
res.animations.push_back(std::move(animation_size));
|
|
|
|
}
|
|
|
|
},
|
2023-01-23 00:57:40 +03:00
|
|
|
[&](unique_ptr<StickerPhotoSize> &&sticker_photo_size) {
|
|
|
|
res.sticker_photo_size = std::move(sticker_photo_size);
|
|
|
|
}));
|
2020-06-24 14:13:33 +03:00
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-09-27 22:44:40 +03:00
|
|
|
Photo get_web_document_photo(FileManager *file_manager, tl_object_ptr<telegram_api::WebDocument> web_document,
|
|
|
|
DialogId owner_dialog_id) {
|
|
|
|
PhotoSize s = get_web_document_photo_size(file_manager, FileType::Photo, owner_dialog_id, std::move(web_document));
|
|
|
|
Photo photo;
|
2020-06-23 20:50:16 +03:00
|
|
|
if (s.file_id.is_valid() && s.type != 'v' && s.type != 'g') {
|
2018-09-27 22:44:40 +03:00
|
|
|
photo.id = 0;
|
|
|
|
photo.photos.push_back(s);
|
|
|
|
}
|
|
|
|
return photo;
|
|
|
|
}
|
|
|
|
|
2020-07-06 18:45:43 +03:00
|
|
|
tl_object_ptr<td_api::photo> get_photo_object(FileManager *file_manager, const Photo &photo) {
|
|
|
|
if (photo.is_empty()) {
|
2018-12-31 22:04:05 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2020-07-06 18:45:43 +03:00
|
|
|
return td_api::make_object<td_api::photo>(photo.has_stickers, get_minithumbnail_object(photo.minithumbnail),
|
|
|
|
get_photo_sizes_object(file_manager, photo.photos));
|
2018-09-12 23:59:10 +03:00
|
|
|
}
|
|
|
|
|
2020-07-07 13:48:56 +03:00
|
|
|
tl_object_ptr<td_api::chatPhoto> get_chat_photo_object(FileManager *file_manager, const Photo &photo) {
|
2020-07-06 18:45:43 +03:00
|
|
|
if (photo.is_empty()) {
|
2020-07-06 15:26:29 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2022-05-30 04:01:55 +03:00
|
|
|
const AnimationSize *small_animation = nullptr;
|
|
|
|
const AnimationSize *big_animation = nullptr;
|
|
|
|
for (auto &animation : photo.animations) {
|
|
|
|
if (animation.type == 'p') {
|
|
|
|
small_animation = &animation;
|
|
|
|
} else if (animation.type == 'u') {
|
|
|
|
big_animation = &animation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (big_animation == nullptr && small_animation != nullptr) {
|
|
|
|
LOG(ERROR) << "Have small animation without big animation in " << photo;
|
|
|
|
small_animation = nullptr;
|
|
|
|
}
|
2023-01-23 11:56:09 +03:00
|
|
|
auto chat_photo_sticker =
|
|
|
|
photo.sticker_photo_size == nullptr ? nullptr : photo.sticker_photo_size->get_chat_photo_sticker_object();
|
2020-06-30 12:57:54 +03:00
|
|
|
return td_api::make_object<td_api::chatPhoto>(
|
|
|
|
photo.id.get(), photo.date, get_minithumbnail_object(photo.minithumbnail),
|
2022-05-30 04:01:55 +03:00
|
|
|
get_photo_sizes_object(file_manager, photo.photos), get_animated_chat_photo_object(file_manager, big_animation),
|
2023-01-23 11:56:09 +03:00
|
|
|
get_animated_chat_photo_object(file_manager, small_animation), std::move(chat_photo_sticker));
|
2020-07-06 15:26:29 +03:00
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
void photo_delete_thumbnail(Photo &photo) {
|
|
|
|
for (size_t i = 0; i < photo.photos.size(); i++) {
|
|
|
|
if (photo.photos[i].type == 't') {
|
|
|
|
photo.photos.erase(photo.photos.begin() + i);
|
2019-03-16 23:13:18 +03:00
|
|
|
return;
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-03 15:44:33 +03:00
|
|
|
bool photo_has_input_media(FileManager *file_manager, const Photo &photo, bool is_secret, bool is_bot) {
|
2018-12-31 22:04:05 +03:00
|
|
|
if (photo.photos.empty() || photo.photos.back().type != 'i') {
|
|
|
|
LOG(ERROR) << "Wrong photo: " << photo;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
auto file_id = photo.photos.back().file_id;
|
|
|
|
auto file_view = file_manager->get_file_view(file_id);
|
|
|
|
if (is_secret) {
|
2018-03-27 16:11:15 +03:00
|
|
|
if (!file_view.is_encrypted_secret() || !file_view.has_remote_location()) {
|
2018-01-31 12:18:40 +03:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &size : photo.photos) {
|
|
|
|
if (size.type == 't' && size.file_id.is_valid()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2018-12-31 22:04:05 +03:00
|
|
|
} else {
|
|
|
|
if (file_view.is_encrypted()) {
|
|
|
|
return false;
|
|
|
|
}
|
2020-05-03 15:44:33 +03:00
|
|
|
if (is_bot && file_view.has_remote_location()) {
|
|
|
|
return true;
|
|
|
|
}
|
2019-02-10 02:39:58 +03:00
|
|
|
return /* file_view.has_remote_location() || */ file_view.has_url();
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tl_object_ptr<telegram_api::InputMedia> photo_get_input_media(FileManager *file_manager, const Photo &photo,
|
|
|
|
tl_object_ptr<telegram_api::InputFile> input_file,
|
2022-12-20 18:06:21 +03:00
|
|
|
int32 ttl, bool has_spoiler) {
|
2018-12-31 22:04:05 +03:00
|
|
|
if (!photo.photos.empty()) {
|
|
|
|
auto file_id = photo.photos.back().file_id;
|
|
|
|
auto file_view = file_manager->get_file_view(file_id);
|
|
|
|
if (file_view.is_encrypted()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2019-11-17 22:41:28 +03:00
|
|
|
if (file_view.has_remote_location() && !file_view.main_remote_location().is_web() && input_file == nullptr) {
|
2018-12-31 22:04:05 +03:00
|
|
|
int32 flags = 0;
|
|
|
|
if (ttl != 0) {
|
|
|
|
flags |= telegram_api::inputMediaPhoto::TTL_SECONDS_MASK;
|
|
|
|
}
|
2022-12-23 18:27:38 +03:00
|
|
|
if (has_spoiler) {
|
|
|
|
flags |= telegram_api::inputMediaPhoto::SPOILER_MASK;
|
|
|
|
}
|
2022-12-08 20:58:37 +03:00
|
|
|
return make_tl_object<telegram_api::inputMediaPhoto>(flags, false /*ignored*/,
|
|
|
|
file_view.main_remote_location().as_input_photo(), ttl);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
if (file_view.has_url()) {
|
|
|
|
int32 flags = 0;
|
|
|
|
if (ttl != 0) {
|
|
|
|
flags |= telegram_api::inputMediaPhotoExternal::TTL_SECONDS_MASK;
|
|
|
|
}
|
2022-12-23 18:27:38 +03:00
|
|
|
if (has_spoiler) {
|
|
|
|
flags |= telegram_api::inputMediaPhotoExternal::SPOILER_MASK;
|
|
|
|
}
|
2022-12-28 21:59:41 +03:00
|
|
|
LOG(INFO) << "Create inputMediaPhotoExternal with a URL " << file_view.url() << " and self-destruct time " << ttl;
|
2022-12-08 20:58:37 +03:00
|
|
|
return make_tl_object<telegram_api::inputMediaPhotoExternal>(flags, false /*ignored*/, file_view.url(), ttl);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
2019-02-12 04:50:30 +03:00
|
|
|
if (input_file == nullptr) {
|
|
|
|
CHECK(!file_view.has_remote_location());
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
if (input_file != nullptr) {
|
|
|
|
int32 flags = 0;
|
|
|
|
vector<tl_object_ptr<telegram_api::InputDocument>> added_stickers;
|
|
|
|
if (photo.has_stickers) {
|
|
|
|
flags |= telegram_api::inputMediaUploadedPhoto::STICKERS_MASK;
|
|
|
|
added_stickers = file_manager->get_input_documents(photo.sticker_file_ids);
|
|
|
|
}
|
|
|
|
if (ttl != 0) {
|
|
|
|
flags |= telegram_api::inputMediaUploadedPhoto::TTL_SECONDS_MASK;
|
|
|
|
}
|
2022-12-15 22:41:55 +03:00
|
|
|
|
|
|
|
CHECK(!photo.photos.empty());
|
2022-12-20 18:06:21 +03:00
|
|
|
if (has_spoiler) {
|
2022-12-15 22:41:55 +03:00
|
|
|
flags |= telegram_api::inputMediaUploadedPhoto::SPOILER_MASK;
|
2022-12-08 20:58:37 +03:00
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2022-12-08 20:58:37 +03:00
|
|
|
return make_tl_object<telegram_api::inputMediaUploadedPhoto>(flags, false /*ignored*/, std::move(input_file),
|
2018-12-31 22:04:05 +03:00
|
|
|
std::move(added_stickers), ttl);
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
SecretInputMedia photo_get_secret_input_media(FileManager *file_manager, const Photo &photo,
|
|
|
|
tl_object_ptr<telegram_api::InputEncryptedFile> input_file,
|
|
|
|
const string &caption, BufferSlice thumbnail) {
|
|
|
|
FileId file_id;
|
|
|
|
int32 width = 0;
|
|
|
|
int32 height = 0;
|
|
|
|
|
|
|
|
FileId thumbnail_file_id;
|
|
|
|
int32 thumbnail_width = 0;
|
|
|
|
int32 thumbnail_height = 0;
|
|
|
|
for (const auto &size : photo.photos) {
|
|
|
|
if (size.type == 'i') {
|
|
|
|
file_id = size.file_id;
|
|
|
|
width = size.dimensions.width;
|
|
|
|
height = size.dimensions.height;
|
|
|
|
}
|
|
|
|
if (size.type == 't') {
|
|
|
|
thumbnail_file_id = size.file_id;
|
|
|
|
thumbnail_width = size.dimensions.width;
|
|
|
|
thumbnail_height = size.dimensions.height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (file_id.empty()) {
|
|
|
|
LOG(ERROR) << "NO SIZE";
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
auto file_view = file_manager->get_file_view(file_id);
|
|
|
|
auto &encryption_key = file_view.encryption_key();
|
2018-06-08 22:23:49 +03:00
|
|
|
if (!file_view.is_encrypted_secret() || encryption_key.empty()) {
|
2018-12-31 22:04:05 +03:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
if (file_view.has_remote_location()) {
|
2019-11-17 22:41:28 +03:00
|
|
|
LOG(INFO) << "Photo has remote location";
|
|
|
|
input_file = file_view.main_remote_location().as_input_encrypted_file();
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
if (input_file == nullptr) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
if (thumbnail_file_id.is_valid() && thumbnail.empty()) {
|
|
|
|
return {};
|
|
|
|
}
|
2022-05-11 17:17:20 +03:00
|
|
|
auto size = file_view.size();
|
|
|
|
if (size < 0 || size >= 1000000000) {
|
|
|
|
size = 0;
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2022-05-30 04:01:55 +03:00
|
|
|
return SecretInputMedia{
|
|
|
|
std::move(input_file),
|
|
|
|
make_tl_object<secret_api::decryptedMessageMediaPhoto>(
|
|
|
|
std::move(thumbnail), thumbnail_width, thumbnail_height, width, height, static_cast<int32>(size),
|
|
|
|
BufferSlice(encryption_key.key_slice()), BufferSlice(encryption_key.iv_slice()), caption)};
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2019-01-06 02:15:07 +03:00
|
|
|
vector<FileId> photo_get_file_ids(const Photo &photo) {
|
2020-06-24 14:13:33 +03:00
|
|
|
auto result = transform(photo.photos, [](auto &size) { return size.file_id; });
|
2020-07-07 11:41:01 +03:00
|
|
|
if (!photo.animations.empty()) {
|
2020-06-28 17:44:56 +03:00
|
|
|
// photo file IDs must be first
|
2020-07-07 11:41:01 +03:00
|
|
|
append(result, transform(photo.animations, [](auto &size) { return size.file_id; }));
|
2020-06-24 14:13:33 +03:00
|
|
|
}
|
|
|
|
return result;
|
2019-01-06 02:15:07 +03:00
|
|
|
}
|
|
|
|
|
2022-09-22 18:25:20 +03:00
|
|
|
FileId get_photo_upload_file_id(const Photo &photo) {
|
|
|
|
for (auto &size : photo.photos) {
|
|
|
|
if (size.type == 'i') {
|
|
|
|
return size.file_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return FileId();
|
|
|
|
}
|
|
|
|
|
|
|
|
FileId get_photo_any_file_id(const Photo &photo) {
|
|
|
|
const auto &sizes = photo.photos;
|
|
|
|
if (!sizes.empty()) {
|
|
|
|
return sizes.back().file_id;
|
|
|
|
}
|
|
|
|
return FileId();
|
|
|
|
}
|
|
|
|
|
|
|
|
FileId get_photo_thumbnail_file_id(const Photo &photo) {
|
|
|
|
for (auto &size : photo.photos) {
|
|
|
|
if (size.type == 't') {
|
|
|
|
return size.file_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return FileId();
|
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
bool operator==(const Photo &lhs, const Photo &rhs) {
|
2023-01-18 22:25:25 +03:00
|
|
|
return lhs.id.get() == rhs.id.get() && lhs.photos == rhs.photos && lhs.animations == rhs.animations &&
|
2023-01-20 17:31:33 +03:00
|
|
|
lhs.sticker_photo_size == rhs.sticker_photo_size;
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const Photo &lhs, const Photo &rhs) {
|
|
|
|
return !(lhs == rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &operator<<(StringBuilder &string_builder, const Photo &photo) {
|
2021-03-15 03:32:14 +03:00
|
|
|
string_builder << "[ID = " << photo.id.get() << ", photos = " << format::as_array(photo.photos);
|
2020-07-07 11:41:01 +03:00
|
|
|
if (!photo.animations.empty()) {
|
|
|
|
string_builder << ", animations = " << format::as_array(photo.animations);
|
2020-06-24 14:13:33 +03:00
|
|
|
}
|
2023-01-23 00:57:40 +03:00
|
|
|
if (photo.sticker_photo_size != nullptr) {
|
|
|
|
string_builder << ", sticker = " << *photo.sticker_photo_size;
|
2023-01-18 22:25:25 +03:00
|
|
|
}
|
2021-03-15 03:32:14 +03:00
|
|
|
return string_builder << ']';
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2019-01-22 19:08:41 +03:00
|
|
|
tl_object_ptr<telegram_api::userProfilePhoto> convert_photo_to_profile_photo(
|
2022-12-09 15:52:21 +03:00
|
|
|
const tl_object_ptr<telegram_api::photo> &photo, bool is_personal) {
|
2019-06-08 12:26:35 +03:00
|
|
|
if (photo == nullptr) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-04-16 17:43:51 +03:00
|
|
|
bool have_photo_small = false;
|
|
|
|
bool have_photo_big = false;
|
2019-01-22 19:08:41 +03:00
|
|
|
for (auto &size_ptr : photo->sizes_) {
|
|
|
|
switch (size_ptr->get_id()) {
|
|
|
|
case telegram_api::photoSizeEmpty::ID:
|
|
|
|
break;
|
|
|
|
case telegram_api::photoSize::ID: {
|
|
|
|
auto size = static_cast<const telegram_api::photoSize *>(size_ptr.get());
|
|
|
|
if (size->type_ == "a") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_small = true;
|
2019-01-22 19:08:41 +03:00
|
|
|
} else if (size->type_ == "c") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_big = true;
|
2019-01-22 19:08:41 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case telegram_api::photoCachedSize::ID: {
|
|
|
|
auto size = static_cast<const telegram_api::photoCachedSize *>(size_ptr.get());
|
|
|
|
if (size->type_ == "a") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_small = true;
|
2019-01-22 19:08:41 +03:00
|
|
|
} else if (size->type_ == "c") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_big = true;
|
2019-01-22 19:08:41 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-03-01 22:51:33 +03:00
|
|
|
case telegram_api::photoStrippedSize::ID:
|
|
|
|
break;
|
2020-08-25 15:09:24 +03:00
|
|
|
case telegram_api::photoSizeProgressive::ID: {
|
|
|
|
auto size = static_cast<const telegram_api::photoSizeProgressive *>(size_ptr.get());
|
|
|
|
if (size->type_ == "a") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_small = true;
|
2020-08-25 15:09:24 +03:00
|
|
|
} else if (size->type_ == "c") {
|
2021-04-16 17:43:51 +03:00
|
|
|
have_photo_big = true;
|
2020-08-25 15:09:24 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-01-22 19:08:41 +03:00
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-04-16 17:43:51 +03:00
|
|
|
if (!have_photo_small || !have_photo_big) {
|
2019-01-22 19:08:41 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
2021-11-01 21:53:23 +03:00
|
|
|
bool has_video = !photo->video_sizes_.empty();
|
2022-12-09 15:52:21 +03:00
|
|
|
return make_tl_object<telegram_api::userProfilePhoto>(0, has_video, is_personal, photo->id_, BufferSlice(),
|
|
|
|
photo->dc_id_);
|
2019-01-22 19:08:41 +03:00
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
} // namespace td
|