// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 // // 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/Td.h" #include "td/telegram/net/ConnectionCreator.h" #include "td/telegram/net/DcId.h" #include "td/telegram/net/MtprotoHeader.h" #include "td/telegram/net/NetQuery.h" #include "td/telegram/net/NetQueryDelayer.h" #include "td/telegram/net/NetQueryDispatcher.h" #include "td/telegram/net/NetStatsManager.h" #include "td/telegram/net/NetType.h" #include "td/telegram/net/TempAuthKeyWatchdog.h" #include "td/telegram/AccessRights.h" #include "td/telegram/AnimationsManager.h" #include "td/telegram/AudiosManager.h" #include "td/telegram/AuthManager.h" #include "td/telegram/CallbackQueriesManager.h" #include "td/telegram/CallId.h" #include "td/telegram/CallManager.h" #include "td/telegram/ChannelId.h" #include "td/telegram/ChatId.h" #include "td/telegram/ConfigManager.h" #include "td/telegram/ConfigShared.h" #include "td/telegram/ContactsManager.h" #include "td/telegram/DeviceTokenManager.h" #include "td/telegram/DialogId.h" #include "td/telegram/DialogParticipant.h" #include "td/telegram/DocumentsManager.h" #include "td/telegram/files/FileGcParameters.h" #include "td/telegram/files/FileId.h" #include "td/telegram/files/FileManager.h" #include "td/telegram/Global.h" #include "td/telegram/HashtagHints.h" #include "td/telegram/InlineQueriesManager.h" #include "td/telegram/LanguagePackManager.h" #include "td/telegram/MessageEntity.h" #include "td/telegram/MessageId.h" #include "td/telegram/MessagesManager.h" #include "td/telegram/misc.h" #include "td/telegram/PasswordManager.h" #include "td/telegram/Photo.h" #include "td/telegram/PrivacyManager.h" #include "td/telegram/SecretChatId.h" #include "td/telegram/SecretChatsManager.h" #include "td/telegram/SecureManager.h" #include "td/telegram/SecureValue.h" #include "td/telegram/StateManager.h" #include "td/telegram/StickersManager.h" #include "td/telegram/StorageManager.h" #include "td/telegram/TdDb.h" #include "td/telegram/TopDialogManager.h" #include "td/telegram/UpdatesManager.h" #include "td/telegram/VideoNotesManager.h" #include "td/telegram/VideosManager.h" #include "td/telegram/VoiceNotesManager.h" #include "td/telegram/WebPageId.h" #include "td/telegram/WebPagesManager.h" #include "td/telegram/td_api.hpp" #include "td/telegram/telegram_api.h" #include "td/telegram/telegram_api.hpp" #include "td/actor/actor.h" #include "td/actor/PromiseFuture.h" #include "td/db/binlog/BinlogEvent.h" #include "td/mtproto/utils.h" // for create_storer, fetch_result, etc, TODO #include "td/utils/buffer.h" #include "td/utils/filesystem.h" #include "td/utils/format.h" #include "td/utils/MimeType.h" #include "td/utils/misc.h" #include "td/utils/PathView.h" #include "td/utils/port/path.h" #include "td/utils/Random.h" #include "td/utils/Slice.h" #include "td/utils/Status.h" #include "td/utils/Timer.h" #include "td/utils/tl_parsers.h" #include "td/utils/utf8.h" #include #include #include namespace td { namespace { DbKey as_db_key(string key) { // Database will still be effectively not encrypted, but // 1. sqlite db will be protected from corruption, because that's how sqlcipher works // 2. security through obscurity // 3. no need for reencryption of sqlite db if (key.empty()) { return DbKey::raw_key("cucumber"); } return DbKey::raw_key(std::move(key)); } } // namespace void Td::ResultHandler::set_td(Td *new_td) { CHECK(td == nullptr); td = new_td; } void Td::ResultHandler::on_result(NetQueryPtr query) { CHECK(query->is_ready()); if (query->is_ok()) { on_result(query->id(), std::move(query->ok())); } else { on_error(query->id(), std::move(query->error())); } query->clear(); } void Td::ResultHandler::send_query(NetQueryPtr query) { td->add_handler(query->id(), shared_from_this()); td->send(std::move(query)); } class GetNearestDcQuery : public Td::ResultHandler { Promise promise_; public: explicit GetNearestDcQuery(Promise &&promise) : promise_(std::move(promise)) { } void send() { send_query(G()->net_query_creator().create(create_storer(telegram_api::help_getNearestDc()), DcId::main(), NetQuery::Type::Common, NetQuery::AuthFlag::Off)); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } promise_.set_value(std::move(result_ptr.ok()->country_)); } void on_error(uint64 id, Status status) override { LOG(ERROR) << "GetNearestDc returned " << status; promise_.set_error(std::move(status)); } }; class GetWallpapersQuery : public Td::ResultHandler { Promise> promise_; public: explicit GetWallpapersQuery(Promise> &&promise) : promise_(std::move(promise)) { } void send() { send_query(G()->net_query_creator().create(create_storer(telegram_api::account_getWallPapers()))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } auto wallpapers = result_ptr.move_as_ok(); auto results = make_tl_object(); results->wallpapers_.reserve(wallpapers.size()); for (auto &wallpaper_ptr : wallpapers) { CHECK(wallpaper_ptr != nullptr); switch (wallpaper_ptr->get_id()) { case telegram_api::wallPaper::ID: { auto wallpaper = move_tl_object_as(wallpaper_ptr); vector> sizes; sizes.reserve(wallpaper->sizes_.size()); for (auto &size_ptr : wallpaper->sizes_) { auto photo_size = get_photo_size(td->file_manager_.get(), FileType::Wallpaper, 0, 0, DialogId(), std::move(size_ptr), false); sizes.push_back(get_photo_size_object(td->file_manager_.get(), &photo_size)); } sort_photo_sizes(sizes); results->wallpapers_.push_back( make_tl_object(wallpaper->id_, std::move(sizes), wallpaper->color_)); break; } case telegram_api::wallPaperSolid::ID: { auto wallpaper = move_tl_object_as(wallpaper_ptr); results->wallpapers_.push_back(make_tl_object( wallpaper->id_, vector>(), wallpaper->bg_color_)); break; } default: UNREACHABLE(); } } promise_.set_value(std::move(results)); } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; class GetRecentMeUrlsQuery : public Td::ResultHandler { Promise> promise_; public: explicit GetRecentMeUrlsQuery(Promise> &&promise) : promise_(std::move(promise)) { } void send(const string &referrer) { send_query(G()->net_query_creator().create(create_storer(telegram_api::help_getRecentMeUrls(referrer)))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } auto urls_full = result_ptr.move_as_ok(); td->contacts_manager_->on_get_users(std::move(urls_full->users_)); td->contacts_manager_->on_get_chats(std::move(urls_full->chats_)); auto urls = std::move(urls_full->urls_); auto results = make_tl_object(); results->urls_.reserve(urls.size()); for (auto &url_ptr : urls) { CHECK(url_ptr != nullptr); tl_object_ptr result = make_tl_object(); switch (url_ptr->get_id()) { case telegram_api::recentMeUrlUser::ID: { auto url = move_tl_object_as(url_ptr); result->url_ = std::move(url->url_); UserId user_id(url->user_id_); if (!user_id.is_valid()) { LOG(ERROR) << "Receive invalid " << user_id; result = nullptr; break; } result->type_ = make_tl_object( td->contacts_manager_->get_user_id_object(user_id, "tMeUrlTypeUser")); break; } case telegram_api::recentMeUrlChat::ID: { auto url = move_tl_object_as(url_ptr); result->url_ = std::move(url->url_); ChannelId channel_id(url->chat_id_); if (!channel_id.is_valid()) { LOG(ERROR) << "Receive invalid " << channel_id; result = nullptr; break; } result->type_ = make_tl_object( td->contacts_manager_->get_supergroup_id_object(channel_id, "tMeUrlTypeSupergroup")); break; } case telegram_api::recentMeUrlChatInvite::ID: { auto url = move_tl_object_as(url_ptr); result->url_ = std::move(url->url_); td->contacts_manager_->on_get_dialog_invite_link_info(result->url_, std::move(url->chat_invite_)); result->type_ = make_tl_object( td->contacts_manager_->get_chat_invite_link_info_object(result->url_)); break; } case telegram_api::recentMeUrlStickerSet::ID: { auto url = move_tl_object_as(url_ptr); result->url_ = std::move(url->url_); auto sticker_set_id = td->stickers_manager_->on_get_sticker_set_covered(std::move(url->set_), false); if (sticker_set_id == 0) { LOG(ERROR) << "Receive invalid sticker set"; result = nullptr; break; } result->type_ = make_tl_object(sticker_set_id); break; } case telegram_api::recentMeUrlUnknown::ID: // skip result = nullptr; break; default: UNREACHABLE(); } if (result != nullptr) { results->urls_.push_back(std::move(result)); } } promise_.set_value(std::move(results)); } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; class SendCustomRequestQuery : public Td::ResultHandler { Promise promise_; public: explicit SendCustomRequestQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(const string &method, const string ¶meters) { send_query(G()->net_query_creator().create(create_storer( telegram_api::bots_sendCustomRequest(method, make_tl_object(parameters))))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } promise_.set_value(std::move(result_ptr.ok()->data_)); } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; class AnswerCustomQueryQuery : public Td::ResultHandler { Promise promise_; public: explicit AnswerCustomQueryQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(int64 custom_query_id, const string &data) { send_query(G()->net_query_creator().create(create_storer( telegram_api::bots_answerWebhookJSONQuery(custom_query_id, make_tl_object(data))))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } bool result = result_ptr.ok(); if (!result) { LOG(INFO) << "Sending answer to a custom query has failed"; } promise_.set_value(Unit()); } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; class SetBotUpdatesStatusQuery : public Td::ResultHandler { public: void send(int32 pending_update_count, const string &error_message) { send_query(G()->net_query_creator().create( create_storer(telegram_api::help_setBotUpdatesStatus(pending_update_count, error_message)))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } bool result = result_ptr.ok(); LOG_IF(WARNING, !result) << "Set bot updates status has failed"; } void on_error(uint64 id, Status status) override { LOG(WARNING) << "Receive error for SetBotUpdatesStatus: " << status; status.ignore(); } }; class UpdateStatusQuery : public Td::ResultHandler { bool is_offline_; public: NetQueryRef send(bool is_offline) { is_offline_ = is_offline; auto net_query = G()->net_query_creator().create(create_storer(telegram_api::account_updateStatus(is_offline))); auto result = net_query.get_weak(); send_query(std::move(net_query)); return result; } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } bool result = result_ptr.ok(); LOG(INFO) << "UpdateStatus returned " << result; td->on_update_status_success(!is_offline_); } void on_error(uint64 id, Status status) override { if (status.code() != NetQuery::Cancelled) { LOG(ERROR) << "Receive error for UpdateStatusQuery: " << status; } status.ignore(); } }; class GetInviteTextQuery : public Td::ResultHandler { Promise promise_; public: explicit GetInviteTextQuery(Promise &&promise) : promise_(std::move(promise)) { } void send() { send_query(G()->net_query_creator().create(create_storer(telegram_api::help_getInviteText()))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } promise_.set_value(std::move(result_ptr.ok()->message_)); } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; class GetDeepLinkInfoQuery : public Td::ResultHandler { Promise> promise_; public: explicit GetDeepLinkInfoQuery(Promise> &&promise) : promise_(std::move(promise)) { } void send(Slice link) { Slice link_scheme("tg:"); if (begins_with(link, link_scheme)) { link.remove_prefix(link_scheme.size()); if (begins_with(link, "//")) { link.remove_prefix(2); } } size_t pos = 0; while (pos < link.size() && link[pos] != '/' && link[pos] != '?' && link[pos] != '#') { pos++; } link.truncate(pos); send_query(G()->net_query_creator().create(create_storer(telegram_api::help_getDeepLinkInfo(link.str())), DcId::main(), NetQuery::Type::Common, NetQuery::AuthFlag::Off)); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, result_ptr.move_as_error()); } auto result = result_ptr.move_as_ok(); switch (result->get_id()) { case telegram_api::help_deepLinkInfoEmpty::ID: return promise_.set_value(nullptr); case telegram_api::help_deepLinkInfo::ID: { auto info = telegram_api::move_object_as(result); bool need_update = (info->flags_ & telegram_api::help_deepLinkInfo::UPDATE_APP_MASK) != 0; auto entities = get_message_entities(nullptr, std::move(info->entities_), "GetDeepLinkInfoQuery"); auto status = fix_formatted_text(info->message_, entities, true, true, true, true); if (status.is_error()) { LOG(ERROR) << "Receive error " << status << " while parsing deep link info " << info->message_; if (!clean_input_string(info->message_)) { info->message_.clear(); } entities.clear(); } FormattedText text{std::move(info->message_), std::move(entities)}; return promise_.set_value( td_api::make_object(get_formatted_text_object(text), need_update)); } default: UNREACHABLE(); } } void on_error(uint64 id, Status status) override { promise_.set_error(std::move(status)); } }; template class RequestActor : public Actor { public: RequestActor(ActorShared td_id, uint64 request_id) : td_id_(std::move(td_id)), td(td_id_.get().get_actor_unsafe()), request_id_(request_id) { } void loop() override { PromiseActor promise; FutureActor future; init_promise_future(&promise, &future); do_run(PromiseCreator::from_promise_actor(std::move(promise))); if (future.is_ready()) { if (future.is_error()) { do_send_error(future.move_as_error()); } else { do_set_result(future.move_as_ok()); do_send_result(); } stop(); } else { if (--tries_left_ == 0) { future.close(); do_send_error(Status::Error(400, "Requested data is unaccessible")); return stop(); } future.set_event(EventCreator::raw(actor_id(), nullptr)); future_ = std::move(future); } } void raw_event(const Event::Raw &event) override { if (future_.is_error()) { auto error = future_.move_as_error(); if (error == Status::Error::Hangup>()) { // dropping query due to lost authorization or lost promise // td may be already closed, so we should check is auth_manager_ is empty bool is_authorized = td->auth_manager_ && td->auth_manager_->is_authorized(); if (is_authorized) { LOG(ERROR) << "Promise was lost"; do_send_error(Status::Error(500, "Query can't be answered due to bug in the TDLib")); } else { do_send_error(Status::Error(401, "Unauthorized")); } return stop(); } do_send_error(std::move(error)); stop(); } else { do_set_result(future_.move_as_ok()); loop(); } } void on_start_migrate(int32 /*sched_id*/) override { UNREACHABLE(); } void on_finish_migrate() override { UNREACHABLE(); } int get_tries() const { return tries_left_; } void set_tries(int32 tries) { tries_left_ = tries; } protected: ActorShared td_id_; Td *td; void send_result(tl_object_ptr &&result) { send_closure(td_id_, &Td::send_result, request_id_, std::move(result)); } void send_error(Status &&status) { LOG(INFO) << "Receive error for query: " << status; send_closure(td_id_, &Td::send_error, request_id_, std::move(status)); } private: virtual void do_run(Promise &&promise) = 0; virtual void do_send_result() { send_result(make_tl_object()); } virtual void do_send_error(Status &&status) { send_error(std::move(status)); } virtual void do_set_result(T &&result) { CHECK((std::is_same::value)); // all other results should be implicitly handled by overriding this method } void hangup() override { do_send_error(Status::Error(500, "Request aborted")); stop(); } friend class RequestOnceActor; uint64 request_id_; int tries_left_ = 2; FutureActor future_; }; class RequestOnceActor : public RequestActor<> { public: RequestOnceActor(ActorShared td_id, uint64 request_id) : RequestActor(std::move(td_id), request_id) { } void loop() override { if (get_tries() < 2) { do_send_result(); stop(); return; } RequestActor::loop(); } }; /*** Td ***/ /** Td queries **/ class TestQuery : public Td::ResultHandler { public: explicit TestQuery(uint64 request_id) : request_id_(request_id) { } void send() { send_query(G()->net_query_creator().create(create_storer(telegram_api::help_getConfig()))); } void on_result(uint64 id, BufferSlice packet) override { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(id, Status::Error(500, "Fetch failed")); } LOG(DEBUG) << "TestOK: " << to_string(result_ptr.ok()); send_closure(G()->td(), &Td::send_result, request_id_, make_tl_object()); } void on_error(uint64 id, Status status) override { status.ignore(); LOG(ERROR) << "Test query failed: " << status; } private: uint64 request_id_; }; class GetAccountTtlRequest : public RequestActor { int32 account_ttl_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(account_ttl_)); return; } td->contacts_manager_->get_account_ttl(std::move(promise)); } void do_set_result(int32 &&result) override { account_ttl_ = result; } void do_send_result() override { send_result(make_tl_object(account_ttl_)); } public: GetAccountTtlRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetActiveSessionsRequest : public RequestActor> { tl_object_ptr sessions_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(sessions_)); return; } td->contacts_manager_->get_active_sessions(std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { sessions_ = std::move(result); } void do_send_result() override { CHECK(sessions_ != nullptr); send_result(std::move(sessions_)); } public: GetActiveSessionsRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetConnectedWebsitesRequest : public RequestActor> { tl_object_ptr connected_websites_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(connected_websites_)); return; } td->contacts_manager_->get_connected_websites(std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { connected_websites_ = std::move(result); } void do_send_result() override { CHECK(connected_websites_ != nullptr); send_result(std::move(connected_websites_)); } public: GetConnectedWebsitesRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetMeRequest : public RequestActor<> { UserId user_id_; void do_run(Promise &&promise) override { user_id_ = td->contacts_manager_->get_me(std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_user_object(user_id_)); } public: GetMeRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetUserRequest : public RequestActor<> { UserId user_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_user(user_id_, get_tries(), std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_user_object(user_id_)); } public: GetUserRequest(ActorShared td, uint64 request_id, int32 user_id) : RequestActor(std::move(td), request_id), user_id_(user_id) { set_tries(3); } }; class GetUserFullInfoRequest : public RequestActor<> { UserId user_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_user_full(user_id_, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_user_full_info_object(user_id_)); } public: GetUserFullInfoRequest(ActorShared td, uint64 request_id, int32 user_id) : RequestActor(std::move(td), request_id), user_id_(user_id) { } }; class GetGroupRequest : public RequestActor<> { ChatId chat_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_chat(chat_id_, get_tries(), std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_basic_group_object(chat_id_)); } public: GetGroupRequest(ActorShared td, uint64 request_id, int32 chat_id) : RequestActor(std::move(td), request_id), chat_id_(chat_id) { set_tries(3); } }; class GetGroupFullInfoRequest : public RequestActor<> { ChatId chat_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_chat_full(chat_id_, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_basic_group_full_info_object(chat_id_)); } public: GetGroupFullInfoRequest(ActorShared td, uint64 request_id, int32 chat_id) : RequestActor(std::move(td), request_id), chat_id_(chat_id) { } }; class GetSupergroupRequest : public RequestActor<> { ChannelId channel_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_channel(channel_id_, get_tries(), std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_supergroup_object(channel_id_)); } public: GetSupergroupRequest(ActorShared td, uint64 request_id, int32 channel_id) : RequestActor(std::move(td), request_id), channel_id_(channel_id) { set_tries(3); } }; class GetSupergroupFullInfoRequest : public RequestActor<> { ChannelId channel_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_channel_full(channel_id_, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_channel_full_info_object(channel_id_)); } public: GetSupergroupFullInfoRequest(ActorShared td, uint64 request_id, int32 channel_id) : RequestActor(std::move(td), request_id), channel_id_(channel_id) { } }; class GetSecretChatRequest : public RequestActor<> { SecretChatId secret_chat_id_; void do_run(Promise &&promise) override { td->contacts_manager_->get_secret_chat(secret_chat_id_, get_tries() < 2, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_secret_chat_object(secret_chat_id_)); } public: GetSecretChatRequest(ActorShared td, uint64 request_id, int32 secret_chat_id) : RequestActor(std::move(td), request_id), secret_chat_id_(secret_chat_id) { } }; class GetChatRequest : public RequestActor<> { DialogId dialog_id_; bool dialog_found_ = false; void do_run(Promise &&promise) override { dialog_found_ = td->messages_manager_->load_dialog(dialog_id_, get_tries(), std::move(promise)); } void do_send_result() override { if (!dialog_found_) { send_error(Status::Error(400, "Chat is not accessible")); } else { send_result(td->messages_manager_->get_chat_object(dialog_id_)); } } public: GetChatRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id) { set_tries(3); } }; class GetChatsRequest : public RequestActor<> { DialogDate offset_; int32 limit_; vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->messages_manager_->get_dialogs(offset_, limit_, get_tries() < 2, std::move(promise)); } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: GetChatsRequest(ActorShared td, uint64 request_id, int64 offset_order, int64 offset_dialog_id, int32 limit) : RequestActor(std::move(td), request_id), offset_(offset_order, DialogId(offset_dialog_id)), limit_(limit) { // 1 for database + 1 for server request + 1 for server request at the end + 1 for return + 1 just in case set_tries(5); } }; class SearchPublicChatRequest : public RequestActor<> { string username_; DialogId dialog_id_; void do_run(Promise &&promise) override { dialog_id_ = td->messages_manager_->search_public_dialog(username_, get_tries() < 3, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_chat_object(dialog_id_)); } public: SearchPublicChatRequest(ActorShared td, uint64 request_id, string username) : RequestActor(std::move(td), request_id), username_(std::move(username)) { set_tries(3); } }; class SearchPublicChatsRequest : public RequestActor<> { string query_; vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->messages_manager_->search_public_dialogs(query_, std::move(promise)); } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: SearchPublicChatsRequest(ActorShared td, uint64 request_id, string query) : RequestActor(std::move(td), request_id), query_(std::move(query)) { } }; class SearchChatsRequest : public RequestActor<> { string query_; int32 limit_; vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->messages_manager_->search_dialogs(query_, limit_, std::move(promise)).second; } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: SearchChatsRequest(ActorShared td, uint64 request_id, string query, int32 limit) : RequestActor(std::move(td), request_id), query_(std::move(query)), limit_(limit) { } }; class SearchChatsOnServerRequest : public RequestActor<> { string query_; int32 limit_; vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->messages_manager_->search_dialogs_on_server(query_, limit_, std::move(promise)); } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: SearchChatsOnServerRequest(ActorShared td, uint64 request_id, string query, int32 limit) : RequestActor(std::move(td), request_id), query_(std::move(query)), limit_(limit) { } }; class GetGroupsInCommonRequest : public RequestActor<> { UserId user_id_; DialogId offset_dialog_id_; int32 limit_; vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->messages_manager_->get_common_dialogs(user_id_, offset_dialog_id_, limit_, get_tries() < 2, std::move(promise)); } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: GetGroupsInCommonRequest(ActorShared td, uint64 request_id, int32 user_id, int64 offset_dialog_id, int32 limit) : RequestActor(std::move(td), request_id), user_id_(user_id), offset_dialog_id_(offset_dialog_id), limit_(limit) { } }; class CheckChatUsernameRequest : public RequestActor { DialogId dialog_id_; string username_; CheckDialogUsernameResult result_ = CheckDialogUsernameResult::Ok; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(result_)); return; } td->contacts_manager_->check_dialog_username(dialog_id_, username_, std::move(promise)); } void do_set_result(CheckDialogUsernameResult &&result) override { result_ = std::move(result); } void do_send_result() override { send_result(ContactsManager::get_check_chat_username_result_object(result_)); } public: CheckChatUsernameRequest(ActorShared td, uint64 request_id, int64 dialog_id, string username) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id), username_(std::move(username)) { } }; class GetCreatedPublicChatsRequest : public RequestActor<> { vector dialog_ids_; void do_run(Promise &&promise) override { dialog_ids_ = td->contacts_manager_->get_created_public_dialogs(std::move(promise)); } void do_send_result() override { send_result(MessagesManager::get_chats_object(dialog_ids_)); } public: GetCreatedPublicChatsRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetMessageRequest : public RequestOnceActor { FullMessageId full_message_id_; void do_run(Promise &&promise) override { td->messages_manager_->get_message(full_message_id_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: GetMessageRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id) : RequestOnceActor(std::move(td), request_id), full_message_id_(DialogId(dialog_id), MessageId(message_id)) { } }; class GetRepliedMessageRequest : public RequestOnceActor { DialogId dialog_id_; MessageId message_id_; MessageId replied_message_id_; void do_run(Promise &&promise) override { replied_message_id_ = td->messages_manager_->get_replied_message(dialog_id_, message_id_, get_tries() < 3, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object({dialog_id_, replied_message_id_})); } public: GetRepliedMessageRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id) : RequestOnceActor(std::move(td), request_id), dialog_id_(dialog_id), message_id_(message_id) { set_tries(3); } }; class GetChatPinnedMessageRequest : public RequestActor { DialogId dialog_id_; MessageId pinned_message_id_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(pinned_message_id_)); return; } td->messages_manager_->get_dialog_pinned_message(dialog_id_, std::move(promise)); } void do_set_result(MessageId &&result) override { pinned_message_id_ = result; } void do_send_result() override { send_result(td->messages_manager_->get_message_object({dialog_id_, pinned_message_id_})); } public: GetChatPinnedMessageRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id) { } }; class GetMessagesRequest : public RequestOnceActor { DialogId dialog_id_; vector message_ids_; void do_run(Promise &&promise) override { td->messages_manager_->get_messages(dialog_id_, message_ids_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(-1, dialog_id_, message_ids_)); } public: GetMessagesRequest(ActorShared td, uint64 request_id, int64 dialog_id, const vector &message_ids) : RequestOnceActor(std::move(td), request_id) , dialog_id_(dialog_id) , message_ids_(MessagesManager::get_message_ids(message_ids)) { } }; class GetPublicMessageLinkRequest : public RequestActor<> { FullMessageId full_message_id_; bool for_group_; string link_; string html_; void do_run(Promise &&promise) override { std::tie(link_, html_) = td->messages_manager_->get_public_message_link(full_message_id_, for_group_, std::move(promise)); } void do_send_result() override { send_result(make_tl_object(link_, html_)); } public: GetPublicMessageLinkRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, bool for_group) : RequestActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , for_group_(for_group) { } }; class EditMessageTextRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr reply_markup_; tl_object_ptr input_message_content_; void do_run(Promise &&promise) override { td->messages_manager_->edit_message_text(full_message_id_, std::move(reply_markup_), std::move(input_message_content_), std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: EditMessageTextRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr reply_markup, tl_object_ptr input_message_content) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , reply_markup_(std::move(reply_markup)) , input_message_content_(std::move(input_message_content)) { } }; class EditMessageLiveLocationRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr reply_markup_; tl_object_ptr location_; void do_run(Promise &&promise) override { td->messages_manager_->edit_message_live_location(full_message_id_, std::move(reply_markup_), std::move(location_), std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: EditMessageLiveLocationRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr reply_markup, tl_object_ptr location) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , reply_markup_(std::move(reply_markup)) , location_(std::move(location)) { } }; class EditMessageMediaRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr reply_markup_; tl_object_ptr input_message_content_; void do_run(Promise &&promise) override { td->messages_manager_->edit_message_media(full_message_id_, std::move(reply_markup_), std::move(input_message_content_), std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: EditMessageMediaRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr reply_markup, tl_object_ptr input_message_content) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , reply_markup_(std::move(reply_markup)) , input_message_content_(std::move(input_message_content)) { } }; class EditMessageCaptionRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr reply_markup_; tl_object_ptr caption_; void do_run(Promise &&promise) override { td->messages_manager_->edit_message_caption(full_message_id_, std::move(reply_markup_), std::move(caption_), std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: EditMessageCaptionRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr reply_markup, tl_object_ptr caption) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , reply_markup_(std::move(reply_markup)) , caption_(std::move(caption)) { } }; class EditMessageReplyMarkupRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr reply_markup_; void do_run(Promise &&promise) override { td->messages_manager_->edit_message_reply_markup(full_message_id_, std::move(reply_markup_), std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: EditMessageReplyMarkupRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr reply_markup) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , reply_markup_(std::move(reply_markup)) { } }; class SetGameScoreRequest : public RequestOnceActor { FullMessageId full_message_id_; bool edit_message_; UserId user_id_; int32 score_; bool force_; void do_run(Promise &&promise) override { td->messages_manager_->set_game_score(full_message_id_, edit_message_, user_id_, score_, force_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_message_object(full_message_id_)); } public: SetGameScoreRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, bool edit_message, int32 user_id, int32 score, bool force) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , edit_message_(edit_message) , user_id_(user_id) , score_(score) , force_(force) { } }; class GetGameHighScoresRequest : public RequestOnceActor { FullMessageId full_message_id_; UserId user_id_; int64 random_id_; void do_run(Promise &&promise) override { random_id_ = td->messages_manager_->get_game_high_scores(full_message_id_, user_id_, std::move(promise)); } void do_send_result() override { CHECK(random_id_ != 0); send_result(td->messages_manager_->get_game_high_scores_object(random_id_)); } public: GetGameHighScoresRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, int32 user_id) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , user_id_(user_id) , random_id_(0) { } }; class GetInlineGameHighScoresRequest : public RequestOnceActor { string inline_message_id_; UserId user_id_; int64 random_id_; void do_run(Promise &&promise) override { random_id_ = td->messages_manager_->get_inline_game_high_scores(inline_message_id_, user_id_, std::move(promise)); } void do_send_result() override { CHECK(random_id_ != 0); send_result(td->messages_manager_->get_game_high_scores_object(random_id_)); } public: GetInlineGameHighScoresRequest(ActorShared td, uint64 request_id, string inline_message_id, int32 user_id) : RequestOnceActor(std::move(td), request_id) , inline_message_id_(std::move(inline_message_id)) , user_id_(user_id) , random_id_(0) { } }; class GetChatHistoryRequest : public RequestActor<> { DialogId dialog_id_; MessageId from_message_id_; int32 offset_; int32 limit_; bool only_local_; tl_object_ptr messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->get_dialog_history(dialog_id_, from_message_id_, offset_, limit_, get_tries() - 1, only_local_, std::move(promise)); } void do_send_result() override { send_result(std::move(messages_)); } public: GetChatHistoryRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 from_message_id, int32 offset, int32 limit, bool only_local) : RequestActor(std::move(td), request_id) , dialog_id_(dialog_id) , from_message_id_(from_message_id) , offset_(offset) , limit_(limit) , only_local_(only_local) { if (!only_local_) { set_tries(4); } } }; class SearchChatMessagesRequest : public RequestActor<> { DialogId dialog_id_; string query_; UserId sender_user_id_; MessageId from_message_id_; int32 offset_; int32 limit_; tl_object_ptr filter_; int64 random_id_; std::pair> messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->search_dialog_messages(dialog_id_, query_, sender_user_id_, from_message_id_, offset_, limit_, filter_, random_id_, get_tries() == 3, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(messages_.first, dialog_id_, messages_.second)); } void do_send_error(Status &&status) override { if (status.message() == "SEARCH_QUERY_EMPTY") { messages_.first = 0; messages_.second.clear(); return do_send_result(); } send_error(std::move(status)); } public: SearchChatMessagesRequest(ActorShared td, uint64 request_id, int64 dialog_id, string query, int32 user_id, int64 from_message_id, int32 offset, int32 limit, tl_object_ptr filter) : RequestActor(std::move(td), request_id) , dialog_id_(dialog_id) , query_(std::move(query)) , sender_user_id_(user_id) , from_message_id_(from_message_id) , offset_(offset) , limit_(limit) , filter_(std::move(filter)) , random_id_(0) { set_tries(3); } }; class OfflineSearchMessagesRequest : public RequestActor<> { DialogId dialog_id_; string query_; int64 from_search_id_; int32 limit_; tl_object_ptr filter_; int64 random_id_; std::pair> messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->offline_search_messages(dialog_id_, query_, from_search_id_, limit_, filter_, random_id_, std::move(promise)); } void do_send_result() override { vector> result; result.reserve(messages_.second.size()); for (auto full_message_id : messages_.second) { result.push_back(td->messages_manager_->get_message_object(full_message_id)); } send_result(make_tl_object(std::move(result), messages_.first)); } public: OfflineSearchMessagesRequest(ActorShared td, uint64 request_id, int64 dialog_id, string query, int64 from_search_id, int32 limit, tl_object_ptr filter) : RequestActor(std::move(td), request_id) , dialog_id_(dialog_id) , query_(std::move(query)) , from_search_id_(from_search_id) , limit_(limit) , filter_(std::move(filter)) , random_id_(0) { } }; class SearchMessagesRequest : public RequestActor<> { string query_; int32 offset_date_; DialogId offset_dialog_id_; MessageId offset_message_id_; int32 limit_; int64 random_id_; std::pair> messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->search_messages(query_, offset_date_, offset_dialog_id_, offset_message_id_, limit_, random_id_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(messages_.first, messages_.second)); } void do_send_error(Status &&status) override { if (status.message() == "SEARCH_QUERY_EMPTY") { messages_.first = 0; messages_.second.clear(); return do_send_result(); } send_error(std::move(status)); } public: SearchMessagesRequest(ActorShared td, uint64 request_id, string query, int32 offset_date, int64 offset_dialog_id, int64 offset_message_id, int32 limit) : RequestActor(std::move(td), request_id) , query_(std::move(query)) , offset_date_(offset_date) , offset_dialog_id_(offset_dialog_id) , offset_message_id_(offset_message_id) , limit_(limit) , random_id_(0) { } }; class SearchCallMessagesRequest : public RequestActor<> { MessageId from_message_id_; int32 limit_; bool only_missed_; int64 random_id_; std::pair> messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->search_call_messages(from_message_id_, limit_, only_missed_, random_id_, get_tries() == 3, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(messages_.first, messages_.second)); } public: SearchCallMessagesRequest(ActorShared td, uint64 request_id, int64 from_message_id, int32 limit, bool only_missed) : RequestActor(std::move(td), request_id) , from_message_id_(from_message_id) , limit_(limit) , only_missed_(only_missed) , random_id_(0) { set_tries(3); } }; class SearchChatRecentLocationMessagesRequest : public RequestActor<> { DialogId dialog_id_; int32 limit_; int64 random_id_; std::pair> messages_; void do_run(Promise &&promise) override { messages_ = td->messages_manager_->search_dialog_recent_location_messages(dialog_id_, limit_, random_id_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(messages_.first, dialog_id_, messages_.second)); } public: SearchChatRecentLocationMessagesRequest(ActorShared td, uint64 request_id, int64 dialog_id, int32 limit) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id), limit_(limit), random_id_(0) { } }; class GetActiveLiveLocationMessagesRequest : public RequestActor<> { vector full_message_ids_; void do_run(Promise &&promise) override { full_message_ids_ = td->messages_manager_->get_active_live_location_messages(std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_messages_object(-1, full_message_ids_)); } public: GetActiveLiveLocationMessagesRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetChatMessageByDateRequest : public RequestOnceActor { DialogId dialog_id_; int32 date_; int64 random_id_; void do_run(Promise &&promise) override { random_id_ = td->messages_manager_->get_dialog_message_by_date(dialog_id_, date_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_dialog_message_by_date_object(random_id_)); } public: GetChatMessageByDateRequest(ActorShared td, uint64 request_id, int64 dialog_id, int32 date) : RequestOnceActor(std::move(td), request_id), dialog_id_(dialog_id), date_(date), random_id_(0) { } }; class GetChatMessageCountRequest : public RequestActor<> { DialogId dialog_id_; tl_object_ptr filter_; bool return_local_; int64 random_id_; int32 result_ = 0; void do_run(Promise &&promise) override { result_ = td->messages_manager_->get_dialog_message_count(dialog_id_, filter_, return_local_, random_id_, std::move(promise)); } void do_send_result() override { send_result(td_api::make_object(result_)); } public: GetChatMessageCountRequest(ActorShared td, uint64 request_id, int64 dialog_id, tl_object_ptr filter, bool return_local) : RequestActor(std::move(td), request_id) , dialog_id_(dialog_id) , filter_(std::move(filter)) , return_local_(return_local) , random_id_(0) { } }; class GetWebPagePreviewRequest : public RequestOnceActor { td_api::object_ptr text_; int64 request_id_ = 0; void do_run(Promise &&promise) override { request_id_ = td->web_pages_manager_->get_web_page_preview(std::move(text_), std::move(promise)); } void do_send_result() override { send_result(td->web_pages_manager_->get_web_page_preview_result(request_id_)); } public: GetWebPagePreviewRequest(ActorShared td, uint64 request_id, td_api::object_ptr text) : RequestOnceActor(std::move(td), request_id), text_(std::move(text)) { } }; class GetWebPageInstantViewRequest : public RequestActor<> { string url_; bool force_full_; WebPageId web_page_id_; void do_run(Promise &&promise) override { web_page_id_ = td->web_pages_manager_->get_web_page_instant_view(url_, force_full_, get_tries() < 3, std::move(promise)); } void do_send_result() override { send_result(td->web_pages_manager_->get_web_page_instant_view_object(web_page_id_)); } public: GetWebPageInstantViewRequest(ActorShared td, uint64 request_id, string url, bool force_full) : RequestActor(std::move(td), request_id), url_(std::move(url)), force_full_(force_full) { set_tries(3); } }; class CreateChatRequest : public RequestActor<> { DialogId dialog_id_; bool force_; void do_run(Promise &&promise) override { td->messages_manager_->create_dialog(dialog_id_, force_, std::move(promise)); } void do_send_result() override { send_result(td->messages_manager_->get_chat_object(dialog_id_)); } public: CreateChatRequest(ActorShared td, uint64 request_id, DialogId dialog_id, bool force) : RequestActor<>(std::move(td), request_id), dialog_id_(dialog_id), force_(force) { } }; class CreateNewGroupChatRequest : public RequestActor<> { vector user_ids_; string title_; int64 random_id_; DialogId dialog_id_; void do_run(Promise &&promise) override { dialog_id_ = td->messages_manager_->create_new_group_chat(user_ids_, title_, random_id_, std::move(promise)); } void do_send_result() override { CHECK(dialog_id_.is_valid()); send_result(td->messages_manager_->get_chat_object(dialog_id_)); } public: CreateNewGroupChatRequest(ActorShared td, uint64 request_id, vector user_ids, string title) : RequestActor(std::move(td), request_id), title_(std::move(title)), random_id_(0) { for (auto user_id : user_ids) { user_ids_.emplace_back(user_id); } } }; class CreateNewSecretChatRequest : public RequestActor { UserId user_id_; SecretChatId secret_chat_id_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(secret_chat_id_)); return; } td->messages_manager_->create_new_secret_chat(user_id_, std::move(promise)); } void do_set_result(SecretChatId &&result) override { secret_chat_id_ = result; LOG(INFO) << "New " << secret_chat_id_ << " created"; } void do_send_result() override { CHECK(secret_chat_id_.is_valid()); // SecretChatActor will send this update by himself. // But since the update may still be on its way, we will update essential fields here. td->contacts_manager_->on_update_secret_chat( secret_chat_id_, 0 /* no access_hash */, user_id_, SecretChatState::Unknown, true /* it is outbound chat */, -1 /* unknown ttl */, 0 /* unknown creation date */, "" /* no key_hash */, 0); DialogId dialog_id(secret_chat_id_); td->messages_manager_->force_create_dialog(dialog_id, "create new secret chat"); send_result(td->messages_manager_->get_chat_object(dialog_id)); } public: CreateNewSecretChatRequest(ActorShared td, uint64 request_id, int32 user_id) : RequestActor(std::move(td), request_id), user_id_(user_id) { } }; class CreateNewSupergroupChatRequest : public RequestActor<> { string title_; bool is_megagroup_; string description_; int64 random_id_; DialogId dialog_id_; void do_run(Promise &&promise) override { dialog_id_ = td->messages_manager_->create_new_channel_chat(title_, is_megagroup_, description_, random_id_, std::move(promise)); } void do_send_result() override { CHECK(dialog_id_.is_valid()); send_result(td->messages_manager_->get_chat_object(dialog_id_)); } public: CreateNewSupergroupChatRequest(ActorShared td, uint64 request_id, string title, bool is_megagroup, string description) : RequestActor(std::move(td), request_id) , title_(std::move(title)) , is_megagroup_(is_megagroup) , description_(std::move(description)) , random_id_(0) { } }; class UpgradeGroupChatToSupergroupChatRequest : public RequestActor<> { string title_; DialogId dialog_id_; DialogId result_dialog_id_; void do_run(Promise &&promise) override { result_dialog_id_ = td->messages_manager_->migrate_dialog_to_megagroup(dialog_id_, std::move(promise)); } void do_send_result() override { CHECK(result_dialog_id_.is_valid()); send_result(td->messages_manager_->get_chat_object(result_dialog_id_)); } public: UpgradeGroupChatToSupergroupChatRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id) { } }; class GetChatMemberRequest : public RequestActor<> { DialogId dialog_id_; UserId user_id_; int64 random_id_; DialogParticipant dialog_participant_; void do_run(Promise &&promise) override { dialog_participant_ = td->messages_manager_->get_dialog_participant(dialog_id_, user_id_, random_id_, get_tries() < 3, std::move(promise)); } void do_send_result() override { if (!td->contacts_manager_->have_user(user_id_)) { return send_error(Status::Error(3, "User not found")); } send_result(td->contacts_manager_->get_chat_member_object(dialog_participant_)); } void do_send_error(Status &&status) override { send_error(std::move(status)); } public: GetChatMemberRequest(ActorShared td, uint64 request_id, int64 dialog_id, int32 user_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id), user_id_(user_id), random_id_(0) { set_tries(3); } }; class SearchChatMembersRequest : public RequestActor<> { DialogId dialog_id_; string query_; int32 limit_; DialogParticipantsFilter filter_; int64 random_id_ = 0; std::pair> participants_; void do_run(Promise &&promise) override { participants_ = td->messages_manager_->search_dialog_participants(dialog_id_, query_, limit_, filter_, random_id_, get_tries() < 3, std::move(promise)); } void do_send_result() override { // TODO create function get_chat_members_object vector> result; result.reserve(participants_.second.size()); for (auto participant : participants_.second) { result.push_back(td->contacts_manager_->get_chat_member_object(participant)); } send_result(make_tl_object(participants_.first, std::move(result))); } public: SearchChatMembersRequest(ActorShared td, uint64 request_id, int64 dialog_id, string &&query, int32 limit, DialogParticipantsFilter filter) : RequestActor(std::move(td), request_id) , dialog_id_(dialog_id) , query_(std::move(query)) , limit_(limit) , filter_(filter) { set_tries(3); } }; class GetChatAdministratorsRequest : public RequestActor<> { DialogId dialog_id_; vector user_ids_; void do_run(Promise &&promise) override { user_ids_ = td->messages_manager_->get_dialog_administrators(dialog_id_, get_tries(), std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_users_object(-1, user_ids_)); } public: GetChatAdministratorsRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id) { set_tries(3); } }; class GenerateChatInviteLinkRequest : public RequestOnceActor { DialogId dialog_id_; void do_run(Promise &&promise) override { td->messages_manager_->export_dialog_invite_link(dialog_id_, std::move(promise)); } void do_send_result() override { send_result(make_tl_object(td->messages_manager_->get_dialog_invite_link(dialog_id_))); } public: GenerateChatInviteLinkRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestOnceActor(std::move(td), request_id), dialog_id_(dialog_id) { } }; class CheckChatInviteLinkRequest : public RequestActor<> { string invite_link_; void do_run(Promise &&promise) override { td->contacts_manager_->check_dialog_invite_link(invite_link_, std::move(promise)); } void do_send_result() override { auto result = td->contacts_manager_->get_chat_invite_link_info_object(invite_link_); CHECK(result != nullptr); send_result(std::move(result)); } public: CheckChatInviteLinkRequest(ActorShared td, uint64 request_id, string invite_link) : RequestActor(std::move(td), request_id), invite_link_(std::move(invite_link)) { } }; class JoinChatByInviteLinkRequest : public RequestActor { string invite_link_; DialogId dialog_id_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(dialog_id_)); return; } td->contacts_manager_->import_dialog_invite_link(invite_link_, std::move(promise)); } void do_set_result(DialogId &&result) override { dialog_id_ = result; } void do_send_result() override { CHECK(dialog_id_.is_valid()); td->messages_manager_->force_create_dialog(dialog_id_, "join chat by invite link"); send_result(td->messages_manager_->get_chat_object(dialog_id_)); } public: JoinChatByInviteLinkRequest(ActorShared td, uint64 request_id, string invite_link) : RequestActor(std::move(td), request_id), invite_link_(std::move(invite_link)) { } }; class GetChatEventLogRequest : public RequestOnceActor { DialogId dialog_id_; string query_; int64 from_event_id_; int32 limit_; tl_object_ptr filters_; vector user_ids_; int64 random_id_ = 0; void do_run(Promise &&promise) override { random_id_ = td->messages_manager_->get_dialog_event_log(dialog_id_, query_, from_event_id_, limit_, filters_, user_ids_, std::move(promise)); } void do_send_result() override { CHECK(random_id_ != 0); send_result(td->messages_manager_->get_chat_events_object(random_id_)); } public: GetChatEventLogRequest(ActorShared td, uint64 request_id, int64 dialog_id, string &&query, int64 from_event_id, int32 limit, tl_object_ptr &&filters, vector user_ids) : RequestOnceActor(std::move(td), request_id) , dialog_id_(dialog_id) , query_(std::move(query)) , from_event_id_(from_event_id) , limit_(limit) , filters_(std::move(filters)) { for (auto user_id : user_ids) { user_ids_.emplace_back(user_id); } } }; class GetBlockedUsersRequest : public RequestOnceActor { int32 offset_; int32 limit_; int64 random_id_; void do_run(Promise &&promise) override { random_id_ = td->contacts_manager_->get_blocked_users(offset_, limit_, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_blocked_users_object(random_id_)); } public: GetBlockedUsersRequest(ActorShared td, uint64 request_id, int32 offset, int32 limit) : RequestOnceActor(std::move(td), request_id), offset_(offset), limit_(limit), random_id_(0) { } }; class ImportContactsRequest : public RequestActor<> { vector> contacts_; int64 random_id_; std::pair, vector> imported_contacts_; void do_run(Promise &&promise) override { imported_contacts_ = td->contacts_manager_->import_contacts(contacts_, random_id_, std::move(promise)); } void do_send_result() override { CHECK(imported_contacts_.first.size() == contacts_.size()); CHECK(imported_contacts_.second.size() == contacts_.size()); send_result(make_tl_object(transform(imported_contacts_.first, [this](UserId user_id) { return td->contacts_manager_->get_user_id_object( user_id, "ImportContactsRequest"); }), std::move(imported_contacts_.second))); } public: ImportContactsRequest(ActorShared td, uint64 request_id, vector> &&contacts) : RequestActor(std::move(td), request_id), contacts_(std::move(contacts)), random_id_(0) { set_tries(3); // load_contacts + import_contacts } }; class SearchContactsRequest : public RequestActor<> { string query_; int32 limit_; std::pair> user_ids_; void do_run(Promise &&promise) override { user_ids_ = td->contacts_manager_->search_contacts(query_, limit_, std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_users_object(user_ids_.first, user_ids_.second)); } public: SearchContactsRequest(ActorShared td, uint64 request_id, string query, int32 limit) : RequestActor(std::move(td), request_id), query_(std::move(query)), limit_(limit) { } }; class RemoveContactsRequest : public RequestActor<> { vector user_ids_; void do_run(Promise &&promise) override { td->contacts_manager_->remove_contacts(user_ids_, std::move(promise)); } public: RemoveContactsRequest(ActorShared td, uint64 request_id, vector &&user_ids) : RequestActor(std::move(td), request_id) { for (auto user_id : user_ids) { user_ids_.emplace_back(user_id); } set_tries(3); // load_contacts + delete_contacts } }; class GetImportedContactCountRequest : public RequestActor<> { int32 imported_contact_count_ = 0; void do_run(Promise &&promise) override { imported_contact_count_ = td->contacts_manager_->get_imported_contact_count(std::move(promise)); } void do_send_result() override { send_result(td_api::make_object(imported_contact_count_)); } public: GetImportedContactCountRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class ChangeImportedContactsRequest : public RequestActor<> { vector> contacts_; size_t contacts_size_; int64 random_id_; std::pair, vector> imported_contacts_; void do_run(Promise &&promise) override { imported_contacts_ = td->contacts_manager_->change_imported_contacts(std::move(contacts_), random_id_, std::move(promise)); } void do_send_result() override { CHECK(imported_contacts_.first.size() == contacts_size_); CHECK(imported_contacts_.second.size() == contacts_size_); send_result(make_tl_object(transform(imported_contacts_.first, [this](UserId user_id) { return td->contacts_manager_->get_user_id_object( user_id, "ChangeImportedContactsRequest"); }), std::move(imported_contacts_.second))); } public: ChangeImportedContactsRequest(ActorShared td, uint64 request_id, vector> &&contacts) : RequestActor(std::move(td), request_id) , contacts_(std::move(contacts)) , contacts_size_(contacts_.size()) , random_id_(0) { set_tries(4); // load_contacts + load_local_contacts + (import_contacts + delete_contacts) } }; class GetRecentInlineBotsRequest : public RequestActor<> { vector user_ids_; void do_run(Promise &&promise) override { user_ids_ = td->inline_queries_manager_->get_recent_inline_bots(std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_users_object(-1, user_ids_)); } public: GetRecentInlineBotsRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetSupergroupMembersRequest : public RequestActor<> { ChannelId channel_id_; tl_object_ptr filter_; int32 offset_; int32 limit_; int64 random_id_ = 0; std::pair> participants_; void do_run(Promise &&promise) override { participants_ = td->contacts_manager_->get_channel_participants(channel_id_, filter_, string(), offset_, limit_, -1, random_id_, get_tries() < 3, std::move(promise)); } void do_send_result() override { // TODO create function get_chat_members_object vector> result; result.reserve(participants_.second.size()); for (auto participant : participants_.second) { result.push_back(td->contacts_manager_->get_chat_member_object(participant)); } send_result(make_tl_object(participants_.first, std::move(result))); } public: GetSupergroupMembersRequest(ActorShared td, uint64 request_id, int32 channel_id, tl_object_ptr &&filter, int32 offset, int32 limit) : RequestActor(std::move(td), request_id) , channel_id_(channel_id) , filter_(std::move(filter)) , offset_(offset) , limit_(limit) { set_tries(3); } }; class GetUserProfilePhotosRequest : public RequestActor<> { UserId user_id_; int32 offset_; int32 limit_; std::pair> photos; void do_run(Promise &&promise) override { photos = td->contacts_manager_->get_user_profile_photos(user_id_, offset_, limit_, std::move(promise)); } void do_send_result() override { // TODO create function get_user_profile_photos_object vector> result; result.reserve(photos.second.size()); for (auto photo : photos.second) { result.push_back(get_photo_object(td->file_manager_.get(), photo)); } send_result(make_tl_object(photos.first, std::move(result))); } public: GetUserProfilePhotosRequest(ActorShared td, uint64 request_id, int32 user_id, int32 offset, int32 limit) : RequestActor(std::move(td), request_id), user_id_(user_id), offset_(offset), limit_(limit) { } }; class GetScopeNotificationSettingsRequest : public RequestActor<> { NotificationSettingsScope scope_; const ScopeNotificationSettings *notification_settings_ = nullptr; void do_run(Promise &&promise) override { notification_settings_ = td->messages_manager_->get_scope_notification_settings(scope_, std::move(promise)); } void do_send_result() override { CHECK(notification_settings_ != nullptr); send_result(td->messages_manager_->get_scope_notification_settings_object(notification_settings_)); } public: GetScopeNotificationSettingsRequest(ActorShared td, uint64 request_id, NotificationSettingsScope scope) : RequestActor(std::move(td), request_id), scope_(scope) { } }; class GetChatReportSpamStateRequest : public RequestActor<> { DialogId dialog_id_; bool can_report_spam_ = false; void do_run(Promise &&promise) override { can_report_spam_ = td->messages_manager_->get_dialog_report_spam_state(dialog_id_, std::move(promise)); } void do_send_result() override { send_result(make_tl_object(can_report_spam_)); } public: GetChatReportSpamStateRequest(ActorShared td, uint64 request_id, int64 dialog_id) : RequestActor(std::move(td), request_id), dialog_id_(dialog_id) { } }; class GetStickersRequest : public RequestActor<> { string emoji_; int32 limit_; vector sticker_ids_; void do_run(Promise &&promise) override { sticker_ids_ = td->stickers_manager_->get_stickers(emoji_, limit_, get_tries() < 2, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_stickers_object(sticker_ids_)); } public: GetStickersRequest(ActorShared td, uint64 request_id, string &&emoji, int32 limit) : RequestActor(std::move(td), request_id), emoji_(std::move(emoji)), limit_(limit) { set_tries(5); } }; class SearchStickersRequest : public RequestActor<> { string emoji_; int32 limit_; vector sticker_ids_; void do_run(Promise &&promise) override { sticker_ids_ = td->stickers_manager_->search_stickers(emoji_, limit_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_stickers_object(sticker_ids_)); } public: SearchStickersRequest(ActorShared td, uint64 request_id, string &&emoji, int32 limit) : RequestActor(std::move(td), request_id), emoji_(std::move(emoji)), limit_(limit) { } }; class GetInstalledStickerSetsRequest : public RequestActor<> { bool is_masks_; vector sticker_set_ids_; void do_run(Promise &&promise) override { sticker_set_ids_ = td->stickers_manager_->get_installed_sticker_sets(is_masks_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(-1, sticker_set_ids_, 1)); } public: GetInstalledStickerSetsRequest(ActorShared td, uint64 request_id, bool is_masks) : RequestActor(std::move(td), request_id), is_masks_(is_masks) { } }; class GetArchivedStickerSetsRequest : public RequestActor<> { bool is_masks_; int64 offset_sticker_set_id_; int32 limit_; int32 total_count_; vector sticker_set_ids_; void do_run(Promise &&promise) override { std::tie(total_count_, sticker_set_ids_) = td->stickers_manager_->get_archived_sticker_sets( is_masks_, offset_sticker_set_id_, limit_, get_tries() < 2, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(total_count_, sticker_set_ids_, 1)); } public: GetArchivedStickerSetsRequest(ActorShared td, uint64 request_id, bool is_masks, int64 offset_sticker_set_id, int32 limit) : RequestActor(std::move(td), request_id) , is_masks_(is_masks) , offset_sticker_set_id_(offset_sticker_set_id) , limit_(limit) { } }; class GetTrendingStickerSetsRequest : public RequestActor<> { vector sticker_set_ids_; void do_run(Promise &&promise) override { sticker_set_ids_ = td->stickers_manager_->get_featured_sticker_sets(std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(-1, sticker_set_ids_, 5)); } public: GetTrendingStickerSetsRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetAttachedStickerSetsRequest : public RequestActor<> { FileId file_id_; vector sticker_set_ids_; void do_run(Promise &&promise) override { sticker_set_ids_ = td->stickers_manager_->get_attached_sticker_sets(file_id_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(-1, sticker_set_ids_, 5)); } public: GetAttachedStickerSetsRequest(ActorShared td, uint64 request_id, int32 file_id) : RequestActor(std::move(td), request_id), file_id_(file_id, 0) { } }; class GetStickerSetRequest : public RequestActor<> { int64 set_id_; int64 sticker_set_id_; void do_run(Promise &&promise) override { sticker_set_id_ = td->stickers_manager_->get_sticker_set(set_id_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_set_object(sticker_set_id_)); } public: GetStickerSetRequest(ActorShared td, uint64 request_id, int64 set_id) : RequestActor(std::move(td), request_id), set_id_(set_id) { set_tries(3); } }; class SearchStickerSetRequest : public RequestActor<> { string name_; int64 sticker_set_id_; void do_run(Promise &&promise) override { sticker_set_id_ = td->stickers_manager_->search_sticker_set(name_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_set_object(sticker_set_id_)); } public: SearchStickerSetRequest(ActorShared td, uint64 request_id, string &&name) : RequestActor(std::move(td), request_id), name_(std::move(name)) { set_tries(3); } }; class SearchInstalledStickerSetsRequest : public RequestActor<> { bool is_masks_; string query_; int32 limit_; std::pair> sticker_set_ids_; void do_run(Promise &&promise) override { sticker_set_ids_ = td->stickers_manager_->search_installed_sticker_sets(is_masks_, query_, limit_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(sticker_set_ids_.first, sticker_set_ids_.second, 5)); } public: SearchInstalledStickerSetsRequest(ActorShared td, uint64 request_id, bool is_masks, string &&query, int32 limit) : RequestActor(std::move(td), request_id), is_masks_(is_masks), query_(std::move(query)), limit_(limit) { } }; class SearchStickerSetsRequest : public RequestActor<> { string query_; vector sticker_set_ids_; void do_run(Promise &&promise) override { sticker_set_ids_ = td->stickers_manager_->search_sticker_sets(query_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_sticker_sets_object(-1, sticker_set_ids_, 5)); } public: SearchStickerSetsRequest(ActorShared td, uint64 request_id, string &&query) : RequestActor(std::move(td), request_id), query_(std::move(query)) { } }; class ChangeStickerSetRequest : public RequestOnceActor { int64 set_id_; bool is_installed_; bool is_archived_; void do_run(Promise &&promise) override { td->stickers_manager_->change_sticker_set(set_id_, is_installed_, is_archived_, std::move(promise)); } public: ChangeStickerSetRequest(ActorShared td, uint64 request_id, int64 set_id, bool is_installed, bool is_archived) : RequestOnceActor(std::move(td), request_id) , set_id_(set_id) , is_installed_(is_installed) , is_archived_(is_archived) { set_tries(4); } }; class UploadStickerFileRequest : public RequestOnceActor { UserId user_id_; tl_object_ptr sticker_; FileId file_id; void do_run(Promise &&promise) override { file_id = td->stickers_manager_->upload_sticker_file(user_id_, sticker_, std::move(promise)); } void do_send_result() override { send_result(td->file_manager_->get_file_object(file_id)); } public: UploadStickerFileRequest(ActorShared td, uint64 request_id, int32 user_id, tl_object_ptr &&sticker) : RequestOnceActor(std::move(td), request_id), user_id_(user_id), sticker_(std::move(sticker)) { } }; class CreateNewStickerSetRequest : public RequestOnceActor { UserId user_id_; string title_; string name_; bool is_masks_; vector> stickers_; void do_run(Promise &&promise) override { td->stickers_manager_->create_new_sticker_set(user_id_, title_, name_, is_masks_, std::move(stickers_), std::move(promise)); } void do_send_result() override { auto set_id = td->stickers_manager_->search_sticker_set(name_, Auto()); if (set_id == 0) { return send_error(Status::Error(500, "Created sticker set not found")); } send_result(td->stickers_manager_->get_sticker_set_object(set_id)); } public: CreateNewStickerSetRequest(ActorShared td, uint64 request_id, int32 user_id, string &&title, string &&name, bool is_masks, vector> &&stickers) : RequestOnceActor(std::move(td), request_id) , user_id_(user_id) , title_(std::move(title)) , name_(std::move(name)) , is_masks_(is_masks) , stickers_(std::move(stickers)) { } }; class AddStickerToSetRequest : public RequestOnceActor { UserId user_id_; string name_; tl_object_ptr sticker_; void do_run(Promise &&promise) override { td->stickers_manager_->add_sticker_to_set(user_id_, name_, std::move(sticker_), std::move(promise)); } void do_send_result() override { auto set_id = td->stickers_manager_->search_sticker_set(name_, Auto()); if (set_id == 0) { return send_error(Status::Error(500, "Sticker set not found")); } send_result(td->stickers_manager_->get_sticker_set_object(set_id)); } public: AddStickerToSetRequest(ActorShared td, uint64 request_id, int32 user_id, string &&name, tl_object_ptr &&sticker) : RequestOnceActor(std::move(td), request_id) , user_id_(user_id) , name_(std::move(name)) , sticker_(std::move(sticker)) { } }; class GetRecentStickersRequest : public RequestActor<> { bool is_attached_; vector sticker_ids_; void do_run(Promise &&promise) override { sticker_ids_ = td->stickers_manager_->get_recent_stickers(is_attached_, std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_stickers_object(sticker_ids_)); } public: GetRecentStickersRequest(ActorShared td, uint64 request_id, bool is_attached) : RequestActor(std::move(td), request_id), is_attached_(is_attached) { } }; class AddRecentStickerRequest : public RequestActor<> { bool is_attached_; tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->stickers_manager_->add_recent_sticker(is_attached_, input_file_, std::move(promise)); } public: AddRecentStickerRequest(ActorShared td, uint64 request_id, bool is_attached, tl_object_ptr &&input_file) : RequestActor(std::move(td), request_id), is_attached_(is_attached), input_file_(std::move(input_file)) { set_tries(3); } }; class RemoveRecentStickerRequest : public RequestActor<> { bool is_attached_; tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->stickers_manager_->remove_recent_sticker(is_attached_, input_file_, std::move(promise)); } public: RemoveRecentStickerRequest(ActorShared td, uint64 request_id, bool is_attached, tl_object_ptr &&input_file) : RequestActor(std::move(td), request_id), is_attached_(is_attached), input_file_(std::move(input_file)) { set_tries(3); } }; class ClearRecentStickersRequest : public RequestActor<> { bool is_attached_; void do_run(Promise &&promise) override { td->stickers_manager_->clear_recent_stickers(is_attached_, std::move(promise)); } public: ClearRecentStickersRequest(ActorShared td, uint64 request_id, bool is_attached) : RequestActor(std::move(td), request_id), is_attached_(is_attached) { set_tries(3); } }; class GetFavoriteStickersRequest : public RequestActor<> { vector sticker_ids_; void do_run(Promise &&promise) override { sticker_ids_ = td->stickers_manager_->get_favorite_stickers(std::move(promise)); } void do_send_result() override { send_result(td->stickers_manager_->get_stickers_object(sticker_ids_)); } public: GetFavoriteStickersRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class AddFavoriteStickerRequest : public RequestOnceActor { tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->stickers_manager_->add_favorite_sticker(input_file_, std::move(promise)); } public: AddFavoriteStickerRequest(ActorShared td, uint64 request_id, tl_object_ptr &&input_file) : RequestOnceActor(std::move(td), request_id), input_file_(std::move(input_file)) { set_tries(3); } }; class RemoveFavoriteStickerRequest : public RequestOnceActor { tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->stickers_manager_->remove_favorite_sticker(input_file_, std::move(promise)); } public: RemoveFavoriteStickerRequest(ActorShared td, uint64 request_id, tl_object_ptr &&input_file) : RequestOnceActor(std::move(td), request_id), input_file_(std::move(input_file)) { set_tries(3); } }; class GetStickerEmojisRequest : public RequestActor<> { tl_object_ptr input_file_; vector emojis_; void do_run(Promise &&promise) override { emojis_ = td->stickers_manager_->get_sticker_emojis(input_file_, std::move(promise)); } void do_send_result() override { send_result(make_tl_object(std::move(emojis_))); } public: GetStickerEmojisRequest(ActorShared td, uint64 request_id, tl_object_ptr &&input_file) : RequestActor(std::move(td), request_id), input_file_(std::move(input_file)) { set_tries(3); } }; class GetSavedAnimationsRequest : public RequestActor<> { vector animation_ids_; void do_run(Promise &&promise) override { animation_ids_ = td->animations_manager_->get_saved_animations(std::move(promise)); } void do_send_result() override { send_result(make_tl_object(transform(std::move(animation_ids_), [td = td](FileId animation_id) { return td->animations_manager_->get_animation_object(animation_id, "GetSavedAnimationsRequest"); }))); } public: GetSavedAnimationsRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class AddSavedAnimationRequest : public RequestOnceActor { tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->animations_manager_->add_saved_animation(input_file_, std::move(promise)); } public: AddSavedAnimationRequest(ActorShared td, uint64 request_id, tl_object_ptr &&input_file) : RequestOnceActor(std::move(td), request_id), input_file_(std::move(input_file)) { set_tries(3); } }; class RemoveSavedAnimationRequest : public RequestOnceActor { tl_object_ptr input_file_; void do_run(Promise &&promise) override { td->animations_manager_->remove_saved_animation(input_file_, std::move(promise)); } public: RemoveSavedAnimationRequest(ActorShared td, uint64 request_id, tl_object_ptr &&input_file) : RequestOnceActor(std::move(td), request_id), input_file_(std::move(input_file)) { set_tries(3); } }; class GetInlineQueryResultsRequest : public RequestOnceActor { UserId bot_user_id_; DialogId dialog_id_; Location user_location_; string query_; string offset_; uint64 query_hash_; void do_run(Promise &&promise) override { query_hash_ = td->inline_queries_manager_->send_inline_query(bot_user_id_, dialog_id_, user_location_, query_, offset_, std::move(promise)); } void do_send_result() override { send_result(td->inline_queries_manager_->get_inline_query_results_object(query_hash_)); } public: GetInlineQueryResultsRequest(ActorShared td, uint64 request_id, int32 bot_user_id, int64 dialog_id, const tl_object_ptr &user_location, string query, string offset) : RequestOnceActor(std::move(td), request_id) , bot_user_id_(bot_user_id) , dialog_id_(dialog_id) , user_location_(user_location) , query_(std::move(query)) , offset_(std::move(offset)) , query_hash_(0) { } }; class GetCallbackQueryAnswerRequest : public RequestOnceActor { FullMessageId full_message_id_; tl_object_ptr payload_; int64 result_id_; void do_run(Promise &&promise) override { result_id_ = td->callback_queries_manager_->send_callback_query(full_message_id_, payload_, std::move(promise)); } void do_send_result() override { send_result(td->callback_queries_manager_->get_callback_query_answer_object(result_id_)); } void do_send_error(Status &&status) override { if (status.code() == 502 && td->messages_manager_->is_message_edited_recently(full_message_id_, 31)) { return send_result(make_tl_object()); } send_error(std::move(status)); } public: GetCallbackQueryAnswerRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr payload) : RequestOnceActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , payload_(std::move(payload)) , result_id_(0) { } }; class GetPaymentFormRequest : public RequestActor> { FullMessageId full_message_id_; tl_object_ptr payment_form_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(payment_form_)); return; } td->messages_manager_->get_payment_form(full_message_id_, std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { payment_form_ = std::move(result); } void do_send_result() override { CHECK(payment_form_ != nullptr); send_result(std::move(payment_form_)); } public: GetPaymentFormRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id) : RequestActor(std::move(td), request_id), full_message_id_(DialogId(dialog_id), MessageId(message_id)) { } }; class ValidateOrderInfoRequest : public RequestActor> { FullMessageId full_message_id_; tl_object_ptr order_info_; bool allow_save_; tl_object_ptr validated_order_info_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(validated_order_info_)); return; } td->messages_manager_->validate_order_info(full_message_id_, std::move(order_info_), allow_save_, std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { validated_order_info_ = std::move(result); } void do_send_result() override { CHECK(validated_order_info_ != nullptr); send_result(std::move(validated_order_info_)); } public: ValidateOrderInfoRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, tl_object_ptr order_info, bool allow_save) : RequestActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , order_info_(std::move(order_info)) , allow_save_(allow_save) { } }; class SendPaymentFormRequest : public RequestActor> { FullMessageId full_message_id_; string order_info_id_; string shipping_option_id_; tl_object_ptr credentials_; tl_object_ptr payment_result_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(payment_result_)); return; } td->messages_manager_->send_payment_form(full_message_id_, order_info_id_, shipping_option_id_, credentials_, std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { payment_result_ = std::move(result); } void do_send_result() override { CHECK(payment_result_ != nullptr); send_result(std::move(payment_result_)); } public: SendPaymentFormRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id, string order_info_id, string shipping_option_id, tl_object_ptr credentials) : RequestActor(std::move(td), request_id) , full_message_id_(DialogId(dialog_id), MessageId(message_id)) , order_info_id_(std::move(order_info_id)) , shipping_option_id_(std::move(shipping_option_id)) , credentials_(std::move(credentials)) { } }; class GetPaymentReceiptRequest : public RequestActor> { FullMessageId full_message_id_; tl_object_ptr payment_receipt_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(payment_receipt_)); return; } td->messages_manager_->get_payment_receipt(full_message_id_, std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { payment_receipt_ = std::move(result); } void do_send_result() override { CHECK(payment_receipt_ != nullptr); send_result(std::move(payment_receipt_)); } public: GetPaymentReceiptRequest(ActorShared td, uint64 request_id, int64 dialog_id, int64 message_id) : RequestActor(std::move(td), request_id), full_message_id_(DialogId(dialog_id), MessageId(message_id)) { } }; class GetSavedOrderInfoRequest : public RequestActor> { tl_object_ptr order_info_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(order_info_)); return; } get_saved_order_info(std::move(promise)); } void do_set_result(tl_object_ptr &&result) override { order_info_ = std::move(result); } void do_send_result() override { send_result(std::move(order_info_)); } public: GetSavedOrderInfoRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetSupportUserRequest : public RequestActor<> { UserId user_id_; void do_run(Promise &&promise) override { user_id_ = td->contacts_manager_->get_support_user(std::move(promise)); } void do_send_result() override { send_result(td->contacts_manager_->get_user_object(user_id_)); } public: GetSupportUserRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetWallpapersRequest : public RequestActor> { tl_object_ptr wallpapers_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(wallpapers_)); return; } td->create_handler(std::move(promise))->send(); } void do_set_result(tl_object_ptr &&result) override { wallpapers_ = std::move(result); } void do_send_result() override { CHECK(wallpapers_ != nullptr); send_result(std::move(wallpapers_)); } public: GetWallpapersRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetRecentlyVisitedTMeUrlsRequest : public RequestActor> { string referrer_; tl_object_ptr urls_; void do_run(Promise> &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(urls_)); return; } td->create_handler(std::move(promise))->send(referrer_); } void do_set_result(tl_object_ptr &&result) override { urls_ = std::move(result); } void do_send_result() override { CHECK(urls_ != nullptr); send_result(std::move(urls_)); } public: GetRecentlyVisitedTMeUrlsRequest(ActorShared td, uint64 request_id, string referrer) : RequestActor(std::move(td), request_id), referrer_(std::move(referrer)) { } }; class SendCustomRequestRequest : public RequestActor { string method_; string parameters_; string request_result_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(request_result_)); return; } td->create_handler(std::move(promise))->send(method_, parameters_); } void do_set_result(string &&result) override { request_result_ = std::move(result); } void do_send_result() override { send_result(make_tl_object(request_result_)); } public: SendCustomRequestRequest(ActorShared td, uint64 request_id, string &&method, string &¶meters) : RequestActor(std::move(td), request_id), method_(method), parameters_(parameters) { } }; class GetCountryCodeRequest : public RequestActor { string country_code_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(country_code_)); return; } td->create_handler(std::move(promise))->send(); } void do_set_result(string &&result) override { country_code_ = std::move(result); } void do_send_result() override { send_result(make_tl_object(country_code_)); } public: GetCountryCodeRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; class GetInviteTextRequest : public RequestActor { string text_; void do_run(Promise &&promise) override { if (get_tries() < 2) { promise.set_value(std::move(text_)); return; } td->create_handler(std::move(promise))->send(); } void do_set_result(string &&result) override { text_ = std::move(result); } void do_send_result() override { send_result(make_tl_object(text_)); } public: GetInviteTextRequest(ActorShared td, uint64 request_id) : RequestActor(std::move(td), request_id) { } }; /** Td **/ Td::Td(std::unique_ptr callback) : callback_(std::move(callback)) { } void Td::on_alarm_timeout_callback(void *td_ptr, int64 alarm_id) { auto td = static_cast(td_ptr); auto td_id = td->actor_id(td); send_closure_later(td_id, &Td::on_alarm_timeout, alarm_id); } void Td::on_alarm_timeout(int64 alarm_id) { if (alarm_id == ONLINE_ALARM_ID) { on_online_updated(false, true); return; } if (alarm_id == PING_SERVER_ALARM_ID) { if (!close_flag_ && updates_manager_ != nullptr) { updates_manager_->ping_server(); alarm_timeout_.set_timeout_in(PING_SERVER_ALARM_ID, PING_SERVER_TIMEOUT + Random::fast(0, PING_SERVER_TIMEOUT / 5)); } return; } if (alarm_id == TERMS_OF_SERVICE_ALARM_ID) { if (!close_flag_) { get_terms_of_service( this, PromiseCreator::lambda([actor_id = actor_id(this)](Result> result) { send_closure(actor_id, &Td::on_get_terms_of_service, std::move(result), false); })); } return; } auto it = pending_alarms_.find(alarm_id); CHECK(it != pending_alarms_.end()); uint64 request_id = it->second; pending_alarms_.erase(alarm_id); send_result(request_id, make_tl_object()); } void Td::on_online_updated(bool force, bool send_update) { if (close_flag_ >= 2 || auth_manager_->is_bot() || !auth_manager_->is_authorized()) { return; } if (force || is_online_) { contacts_manager_->set_my_online_status(is_online_, send_update, true); if (!update_status_query_.empty()) { LOG(INFO) << "Cancel previous update status query"; cancel_query(update_status_query_); } update_status_query_ = create_handler()->send(!is_online_); } if (is_online_) { alarm_timeout_.set_timeout_in(ONLINE_ALARM_ID, ONLINE_TIMEOUT); } else { alarm_timeout_.cancel_timeout(ONLINE_ALARM_ID); } } void Td::on_update_status_success(bool is_online) { if (is_online == is_online_) { if (!update_status_query_.empty()) { update_status_query_ = NetQueryRef(); } contacts_manager_->set_my_online_status(is_online_, true, false); } } void Td::on_get_terms_of_service(Result> result, bool dummy) { int32 expires_in = 0; if (result.is_error()) { expires_in = Random::fast(10, 60); } else { auto terms = std::move(result.ok().second); if (terms.get_id().empty()) { expires_in = min(max(result.ok().first, G()->unix_time() + 60) - G()->unix_time(), 86400); } else { send_update( make_tl_object(terms.get_id().str(), terms.get_terms_of_service_object())); } } if (expires_in > 0) { schedule_get_terms_of_service(expires_in); } } void Td::schedule_get_terms_of_service(int32 expires_in) { if (!close_flag_ && !auth_manager_->is_bot()) { alarm_timeout_.set_timeout_in(TERMS_OF_SERVICE_ALARM_ID, expires_in); } } void Td::on_channel_unban_timeout(int64 channel_id_long) { if (close_flag_ >= 2) { return; } contacts_manager_->on_channel_unban_timeout(ChannelId(narrow_cast(channel_id_long))); } bool Td::is_online() const { return is_online_; } bool Td::is_authentication_request(int32 id) { switch (id) { case td_api::setTdlibParameters::ID: case td_api::checkDatabaseEncryptionKey::ID: case td_api::setDatabaseEncryptionKey::ID: case td_api::getAuthorizationState::ID: case td_api::setAuthenticationPhoneNumber::ID: case td_api::resendAuthenticationCode::ID: case td_api::checkAuthenticationCode::ID: case td_api::checkAuthenticationPassword::ID: case td_api::requestAuthenticationPasswordRecovery::ID: case td_api::recoverAuthenticationPassword::ID: case td_api::deleteAccount::ID: case td_api::logOut::ID: case td_api::close::ID: case td_api::destroy::ID: case td_api::checkAuthenticationBotToken::ID: return true; default: return false; } } bool Td::is_synchronous_request(int32 id) { switch (id) { case td_api::getTextEntities::ID: case td_api::parseTextEntities::ID: case td_api::getFileMimeType::ID: case td_api::getFileExtension::ID: case td_api::cleanFileName::ID: case td_api::getLanguagePackString::ID: return true; default: return false; } } bool Td::is_preinitialization_request(int32 id) { switch (id) { case td_api::setAlarm::ID: case td_api::testUseUpdate::ID: case td_api::testUseError::ID: case td_api::testCallEmpty::ID: case td_api::testSquareInt::ID: case td_api::testCallString::ID: case td_api::testCallBytes::ID: case td_api::testCallVectorInt::ID: case td_api::testCallVectorIntObject::ID: case td_api::testCallVectorString::ID: case td_api::testCallVectorStringObject::ID: return true; default: return false; } } bool Td::is_preauthentication_request(int32 id) { switch (id) { case td_api::processDcUpdate::ID: case td_api::getLanguagePackInfo::ID: case td_api::getLanguagePackStrings::ID: case td_api::setCustomLanguage::ID: case td_api::deleteLanguage::ID: case td_api::getOption::ID: case td_api::setOption::ID: case td_api::setNetworkType::ID: case td_api::getNetworkStatistics::ID: case td_api::addNetworkStatistics::ID: case td_api::resetNetworkStatistics::ID: case td_api::getCountryCode::ID: case td_api::getDeepLinkInfo::ID: case td_api::addProxy::ID: case td_api::editProxy::ID: case td_api::enableProxy::ID: case td_api::disableProxy::ID: case td_api::removeProxy::ID: case td_api::getProxies::ID: case td_api::pingProxy::ID: case td_api::testNetwork::ID: return true; default: return false; } } void Td::request(uint64 id, tl_object_ptr function) { request_set_.insert(id); if (id == 0) { LOG(ERROR) << "Receive request with id == 0"; return send_error_raw(id, 400, "Wrong request id == 0"); } if (function == nullptr) { LOG(ERROR) << "Receive empty request"; return send_error_raw(id, 400, "Request is empty"); } VLOG(td_requests) << "Receive request " << id << ": " << to_string(function); int32 function_id = function->get_id(); switch (state_) { case State::WaitParameters: { switch (function_id) { case td_api::getAuthorizationState::ID: return send_closure(actor_id(this), &Td::send_result, id, td_api::make_object()); case td_api::setTdlibParameters::ID: return answer_ok_query( id, set_parameters(std::move(move_tl_object_as(function)->parameters_))); default: if (is_synchronous_request(function_id) || is_preinitialization_request(function_id)) { break; } if (is_preauthentication_request(function_id)) { pending_preauthentication_requests_.emplace_back(id, std::move(function)); return; } return send_error_raw(id, 401, "Initialization parameters are needed: call setTdlibParameters first"); } break; } case State::Decrypt: { string encryption_key; switch (function_id) { case td_api::getAuthorizationState::ID: return send_closure( actor_id(this), &Td::send_result, id, td_api::make_object(encryption_info_.is_encrypted)); case td_api::checkDatabaseEncryptionKey::ID: { auto check_key = move_tl_object_as(function); encryption_key = std::move(check_key->encryption_key_); break; } case td_api::setDatabaseEncryptionKey::ID: { auto set_key = move_tl_object_as(function); encryption_key = std::move(set_key->new_encryption_key_); break; } case td_api::close::ID: return close(); case td_api::destroy::ID: return destroy(); default: if (is_synchronous_request(function_id) || is_preinitialization_request(function_id)) { break; } if (is_preauthentication_request(function_id)) { pending_preauthentication_requests_.emplace_back(id, std::move(function)); return; } return send_error_raw(id, 401, "Database encryption key is needed: call checkDatabaseEncryptionKey first"); } return answer_ok_query(id, init(as_db_key(encryption_key))); } case State::Close: { if (function_id == td_api::getAuthorizationState::ID) { if (close_flag_ == 5) { return send_closure(actor_id(this), &Td::send_result, id, td_api::make_object()); } else { return send_closure(actor_id(this), &Td::send_result, id, td_api::make_object()); } } if (is_synchronous_request(function_id)) { break; } return send_error_raw(id, 401, "Unauthorized"); } case State::Run: break; } if ((auth_manager_ == nullptr || !auth_manager_->is_authorized()) && !is_preauthentication_request(function_id) && !is_preinitialization_request(function_id) && !is_synchronous_request(function_id) && !is_authentication_request(function_id)) { return send_error_raw(id, 401, "Unauthorized"); } downcast_call(*function, [this, id](auto &request) { this->on_request(id, request); }); } td_api::object_ptr Td::static_request(td_api::object_ptr function) { VLOG(td_requests) << "Receive static request: " << to_string(function); td_api::object_ptr response; downcast_call(*function, [&response](auto &request) { response = Td::do_static_request(request); }); VLOG(td_requests) << "Sending result for static request: " << to_string(response); return response; } void Td::add_handler(uint64 id, std::shared_ptr handler) { result_handlers_.emplace_back(id, handler); } std::shared_ptr Td::extract_handler(uint64 id) { std::shared_ptr result; for (size_t i = 0; i < result_handlers_.size(); i++) { if (result_handlers_[i].first == id) { result = std::move(result_handlers_[i].second); result_handlers_.erase(result_handlers_.begin() + i); break; } } return result; } void Td::invalidate_handler(ResultHandler *handler) { for (size_t i = 0; i < result_handlers_.size(); i++) { if (result_handlers_[i].second.get() == handler) { result_handlers_.erase(result_handlers_.begin() + i); i--; } } } void Td::send(NetQueryPtr &&query) { VLOG(net_query) << "Send " << query << " to dispatcher"; query->debug("Td: send to NetQueryDispatcher"); query->set_callback(actor_shared(this, 1)); G()->net_query_dispatcher().dispatch(std::move(query)); } void Td::update_qts(int32 qts) { if (close_flag_ > 1) { return; } updates_manager_->set_qts(qts); } void Td::force_get_difference() { if (close_flag_) { return; } updates_manager_->get_difference("force_get_difference"); } void Td::on_result(NetQueryPtr query) { query->debug("Td: received from DcManager"); VLOG(net_query) << "on_result " << query; if (close_flag_ > 1) { return; } if (query->id() == 0) { if (query->is_error()) { query->clear(); updates_manager_->schedule_get_difference("error in update"); LOG(ERROR) << "Error in update"; return; } auto ok = query->move_as_ok(); TlBufferParser parser(&ok); auto ptr = telegram_api::Updates::fetch(parser); if (parser.get_error()) { LOG(ERROR) << "Failed to fetch update: " << parser.get_error() << format::as_hex_dump<4>(ok.as_slice()); updates_manager_->schedule_get_difference("failed to fetch update"); } else { updates_manager_->on_get_updates(std::move(ptr)); if (auth_manager_->is_bot()) { alarm_timeout_.set_timeout_in(PING_SERVER_ALARM_ID, PING_SERVER_TIMEOUT + Random::fast(0, PING_SERVER_TIMEOUT / 5)); } } return; } auto handler = extract_handler(query->id()); if (handler == nullptr) { query->clear(); LOG_IF(WARNING, !query->is_ok() || query->ok_tl_constructor() != telegram_api::upload_file::ID) << tag("NetQuery", query) << " is ignored: no handlers found"; return; } handler->on_result(std::move(query)); } bool Td::is_internal_config_option(Slice name) { switch (name[0]) { case 'a': return name == "auth"; case 'c': return name == "call_ring_timeout_ms" || name == "call_receive_timeout_ms" || name == "channels_read_media_period"; case 'd': return name == "dc_txt_domain_name"; case 'e': return name == "edit_time_limit" || name == "expect_blocking"; case 'l': return name == "language_pack_version"; case 'm': return name == "my_phone_number"; case 'r': return name == "revoke_pm_inbox" || name == "revoke_time_limit" || name == "revoke_pm_time_limit" || name == "rating_e_decay" || name == "recent_stickers_limit"; case 's': return name == "saved_animations_limit"; case 'w': return name == "webfile_dc_id"; default: return false; } } void Td::on_config_option_updated(const string &name) { if (close_flag_) { return; } if (name == "auth") { return on_authorization_lost(); } else if (name == "saved_animations_limit") { return animations_manager_->on_update_saved_animations_limit(G()->shared_config().get_option_integer(name)); } else if (name == "recent_stickers_limit") { return stickers_manager_->on_update_recent_stickers_limit(G()->shared_config().get_option_integer(name)); } else if (name == "favorite_stickers_limit") { stickers_manager_->on_update_favorite_stickers_limit(G()->shared_config().get_option_integer(name)); } else if (name == "my_id") { G()->set_my_id(G()->shared_config().get_option_integer(name)); } else if (name == "session_count") { G()->net_query_dispatcher().update_session_count(); } else if (name == "use_pfs") { G()->net_query_dispatcher().update_use_pfs(); } else if (name == "use_storage_optimizer") { send_closure(storage_manager_, &StorageManager::update_use_storage_optimizer); } else if (name == "rating_e_decay") { return send_closure(top_dialog_manager_, &TopDialogManager::update_rating_e_decay); } else if (name == "disable_top_chats") { send_closure(top_dialog_manager_, &TopDialogManager::update_is_enabled, !G()->shared_config().get_option_boolean(name)); } else if (name == "is_emulator") { if (G()->mtproto_header().set_is_emulator(G()->shared_config().get_option_boolean(name))) { G()->net_query_dispatcher().update_mtproto_header(); } } else if (name == "language_pack") { send_closure(language_pack_manager_, &LanguagePackManager::on_language_pack_changed); if (G()->mtproto_header().set_language_pack(G()->shared_config().get_option_string(name))) { G()->net_query_dispatcher().update_mtproto_header(); } } else if (name == "language_code") { send_closure(language_pack_manager_, &LanguagePackManager::on_language_code_changed); if (G()->mtproto_header().set_language_code(G()->shared_config().get_option_string(name))) { G()->net_query_dispatcher().update_mtproto_header(); } } else if (name == "language_pack_version") { send_closure(language_pack_manager_, &LanguagePackManager::on_language_pack_version_changed, -1); return; } else if (is_internal_config_option(name)) { return; } // send_closure was already used in the callback send_update(make_tl_object(name, G()->shared_config().get_option_value(name))); } tl_object_ptr Td::get_connection_state_object(StateManager::State state) { switch (state) { case StateManager::State::Empty: UNREACHABLE(); return nullptr; case StateManager::State::WaitingForNetwork: return make_tl_object(); case StateManager::State::ConnectingToProxy: return make_tl_object(); case StateManager::State::Connecting: return make_tl_object(); case StateManager::State::Updating: return make_tl_object(); case StateManager::State::Ready: return make_tl_object(); default: UNREACHABLE(); return nullptr; } } void Td::on_connection_state_changed(StateManager::State new_state) { if (new_state == connection_state_) { LOG(ERROR) << "State manager sends update about unchanged state " << static_cast(new_state); return; } connection_state_ = new_state; send_closure(actor_id(this), &Td::send_update, make_tl_object(get_connection_state_object(connection_state_))); } void Td::on_authorization_lost() { LOG(WARNING) << "on_authorization_lost"; destroy(); } static td_api::object_ptr make_error(int32 code, CSlice error) { return td_api::make_object(code, error.str()); } void Td::start_up() { always_wait_for_mailbox(); uint64 check_endianness = 0x0706050403020100; auto check_endianness_raw = reinterpret_cast(&check_endianness); for (unsigned char c = 0; c < 8; c++) { auto symbol = check_endianness_raw[static_cast(c)]; LOG_IF(FATAL, symbol != c) << "TDLib requires little-endian platform"; } alarm_timeout_.set_callback(on_alarm_timeout_callback); alarm_timeout_.set_callback_data(static_cast(this)); CHECK(state_ == State::WaitParameters); send_update(td_api::make_object( td_api::make_object())); } void Td::tear_down() { CHECK(close_flag_ == 5); } void Td::hangup_shared() { auto token = get_link_token(); auto type = Container::type_from_id(token); if (type == RequestActorIdType) { request_actors_.erase(get_link_token()); dec_request_actor_refcnt(); } else if (type == ActorIdType) { dec_actor_refcnt(); } else { LOG(FATAL) << "Unknown hangup_shared of type " << type; } } void Td::hangup() { close(); dec_stop_cnt(); } ActorShared Td::create_reference() { inc_actor_refcnt(); return actor_shared(this, ActorIdType); } void Td::inc_actor_refcnt() { actor_refcnt_++; } void Td::dec_actor_refcnt() { actor_refcnt_--; if (actor_refcnt_ == 0) { if (close_flag_ == 2) { create_reference(); close_flag_ = 3; } else if (close_flag_ == 3) { LOG(WARNING) << "ON_ACTORS_CLOSED"; Timer timer; animations_manager_.reset(); LOG(DEBUG) << "AnimationsManager was cleared " << timer; audios_manager_.reset(); LOG(DEBUG) << "AudiosManager was cleared " << timer; auth_manager_.reset(); LOG(DEBUG) << "AuthManager was cleared " << timer; contacts_manager_.reset(); LOG(DEBUG) << "ContactsManager was cleared " << timer; documents_manager_.reset(); LOG(DEBUG) << "DocumentsManager was cleared " << timer; file_manager_.reset(); LOG(DEBUG) << "FileManager was cleared " << timer; inline_queries_manager_.reset(); LOG(DEBUG) << "InlineQueriesManager was cleared " << timer; messages_manager_.reset(); LOG(DEBUG) << "MessagesManager was cleared " << timer; stickers_manager_.reset(); LOG(DEBUG) << "StickersManager was cleared " << timer; updates_manager_.reset(); LOG(DEBUG) << "UpdatesManager was cleared " << timer; video_notes_manager_.reset(); LOG(DEBUG) << "VideoNotesManager was cleared " << timer; videos_manager_.reset(); LOG(DEBUG) << "VideosManager was cleared " << timer; voice_notes_manager_.reset(); LOG(DEBUG) << "VoiceNotesManager was cleared " << timer; web_pages_manager_.reset(); LOG(DEBUG) << "WebPagesManager was cleared " << timer; Promise<> promise = PromiseCreator::lambda([actor_id = create_reference()](Unit) mutable { actor_id.reset(); }); if (destroy_flag_) { G()->close_and_destroy_all(std::move(promise)); } else { G()->close_all(std::move(promise)); } // NetQueryDispatcher will be closed automatically close_flag_ = 4; } else if (close_flag_ == 4) { on_closed(); } else { UNREACHABLE(); } } } void Td::on_closed() { LOG(WARNING) << "ON_CLOSED"; close_flag_ = 5; send_update( td_api::make_object(td_api::make_object())); callback_->on_closed(); dec_stop_cnt(); } void Td::dec_stop_cnt() { stop_cnt_--; if (stop_cnt_ == 0) { stop(); } } void Td::inc_request_actor_refcnt() { request_actor_refcnt_++; } void Td::dec_request_actor_refcnt() { request_actor_refcnt_--; if (request_actor_refcnt_ == 0) { LOG(WARNING) << "no request actors"; clear(); dec_actor_refcnt(); // remove guard } } void Td::clear_handlers() { result_handlers_.clear(); } void Td::clear_requests() { while (!pending_alarms_.empty()) { auto it = pending_alarms_.begin(); auto alarm_id = it->first; pending_alarms_.erase(it); alarm_timeout_.cancel_timeout(alarm_id); } while (!request_set_.empty()) { uint64 id = *request_set_.begin(); if (destroy_flag_) { send_error_impl(id, make_error(401, "Unauthorized")); } else { send_error_impl(id, make_error(500, "Request aborted")); } } } void Td::clear() { if (close_flag_ >= 2) { return; } LOG(INFO) << "Clear Td"; close_flag_ = 2; Timer timer; if (destroy_flag_) { for (auto &option : G()->shared_config().get_options()) { if (!is_internal_config_option(option.first)) { send_update(make_tl_object(option.first, make_tl_object())); } } } LOG(DEBUG) << "Options was cleared " << timer; G()->net_query_creator().stop_check(); clear_handlers(); LOG(DEBUG) << "Handlers was cleared " << timer; G()->net_query_dispatcher().stop(); LOG(DEBUG) << "NetQueryDispatcher was stopped " << timer; state_manager_.reset(); LOG(DEBUG) << "StateManager was cleared " << timer; clear_requests(); if (is_online_) { is_online_ = false; alarm_timeout_.cancel_timeout(ONLINE_ALARM_ID); } alarm_timeout_.cancel_timeout(PING_SERVER_ALARM_ID); alarm_timeout_.cancel_timeout(TERMS_OF_SERVICE_ALARM_ID); LOG(DEBUG) << "Requests was answered " << timer; // close all pure actors call_manager_.reset(); LOG(DEBUG) << "CallManager was cleared " << timer; change_phone_number_manager_.reset(); LOG(DEBUG) << "ChangePhoneNumberManager was cleared " << timer; config_manager_.reset(); LOG(DEBUG) << "ConfigManager was cleared " << timer; confirm_phone_number_manager_.reset(); LOG(DEBUG) << "ConfirmPhoneNumberManager was cleared " << timer; device_token_manager_.reset(); LOG(DEBUG) << "DeviceTokenManager was cleared " << timer; hashtag_hints_.reset(); LOG(DEBUG) << "HashtagHints was cleared " << timer; language_pack_manager_.reset(); LOG(DEBUG) << "LanguagePackManager was cleared " << timer; net_stats_manager_.reset(); LOG(DEBUG) << "NetStatsManager was cleared " << timer; password_manager_.reset(); LOG(DEBUG) << "PasswordManager was cleared " << timer; privacy_manager_.reset(); LOG(DEBUG) << "PrivacyManager was cleared " << timer; secure_manager_.reset(); LOG(DEBUG) << "SecureManager was cleared " << timer; secret_chats_manager_.reset(); LOG(DEBUG) << "SecretChatsManager was cleared " << timer; storage_manager_.reset(); LOG(DEBUG) << "StorageManager was cleared " << timer; top_dialog_manager_.reset(); LOG(DEBUG) << "TopDialogManager was cleared " << timer; verify_phone_number_manager_.reset(); LOG(DEBUG) << "VerifyPhoneNumberManager was cleared " << timer; G()->set_connection_creator(ActorOwn()); LOG(DEBUG) << "ConnectionCreator was cleared " << timer; // clear actors which are unique pointers animations_manager_actor_.reset(); LOG(DEBUG) << "AnimationsManager actor was cleared " << timer; auth_manager_actor_.reset(); LOG(DEBUG) << "AuthManager actor was cleared " << timer; contacts_manager_actor_.reset(); LOG(DEBUG) << "ContactsManager actor was cleared " << timer; file_manager_actor_.reset(); LOG(DEBUG) << "FileManager actor was cleared " << timer; inline_queries_manager_actor_.reset(); LOG(DEBUG) << "InlineQueriesManager actor was cleared " << timer; messages_manager_actor_.reset(); // TODO: Stop silent LOG(DEBUG) << "MessagesManager actor was cleared " << timer; stickers_manager_actor_.reset(); LOG(DEBUG) << "StickersManager actor was cleared " << timer; updates_manager_actor_.reset(); LOG(DEBUG) << "UpdatesManager actor was cleared " << timer; web_pages_manager_actor_.reset(); LOG(DEBUG) << "WebPagesManager actor was cleared " << timer; } void Td::close() { close_impl(false); } void Td::destroy() { close_impl(true); } void Td::close_impl(bool destroy_flag) { destroy_flag_ |= destroy_flag; if (close_flag_) { return; } if (state_ == State::WaitParameters) { clear_requests(); return on_closed(); } if (state_ == State::Decrypt) { clear_requests(); if (destroy_flag) { TdDb::destroy(parameters_); } state_ = State::Close; close_flag_ = 4; return dec_actor_refcnt(); } state_ = State::Close; close_flag_ = 1; G()->set_close_flag(); send_closure(auth_manager_actor_, &AuthManager::on_closing); LOG(WARNING) << "Close " << tag("destroy", destroy_flag); // wait till all request_actors will stop. request_actors_.clear(); G()->td_db()->flush_all(); send_closure_later(actor_id(this), &Td::dec_request_actor_refcnt); // remove guard } class Td::DownloadFileCallback : public FileManager::DownloadCallback { public: void on_progress(FileId file_id) override { } void on_download_ok(FileId file_id) override { } void on_download_error(FileId file_id, Status error) override { } }; class Td::UploadFileCallback : public FileManager::UploadCallback { public: void on_progress(FileId file_id) override { } void on_upload_ok(FileId file_id, tl_object_ptr input_file) override { // cancel file upload of the file to allow next upload with the same file to succeed send_closure(G()->file_manager(), &FileManager::upload, file_id, nullptr, 0, 0); } void on_upload_encrypted_ok(FileId file_id, tl_object_ptr input_file) override { // cancel file upload of the file to allow next upload with the same file to succeed send_closure(G()->file_manager(), &FileManager::upload, file_id, nullptr, 0, 0); } void on_upload_secure_ok(FileId file_id, tl_object_ptr input_file) override { // cancel file upload of the file to allow next upload with the same file to succeed send_closure(G()->file_manager(), &FileManager::upload, file_id, nullptr, 0, 0); } void on_upload_error(FileId file_id, Status error) override { } }; int VERBOSITY_NAME(td_init) = VERBOSITY_NAME(DEBUG) + 3; template void Td::complete_pending_preauthentication_requests(const T &func) { for (auto &request : pending_preauthentication_requests_) { if (request.second != nullptr && func(request.second->get_id())) { downcast_call(*request.second, [this, id = request.first](auto &request) { this->on_request(id, request); }); request.second = nullptr; } } } Status Td::init(DbKey key) { auto current_scheduler_id = Scheduler::instance()->sched_id(); auto scheduler_count = Scheduler::instance()->sched_count(); VLOG(td_init) << "Begin to init database"; TdDb::Events events; TRY_RESULT(td_db, TdDb::open(min(current_scheduler_id + 1, scheduler_count - 1), parameters_, std::move(key), events)); LOG(INFO) << "Successfully inited database in " << tag("database_directory", parameters_.database_directory) << " and " << tag("files_directory", parameters_.files_directory); G()->init(parameters_, actor_id(this), std::move(td_db)).ensure(); // Init all managers and actors VLOG(td_init) << "Create StateManager"; class StateManagerCallback : public StateManager::Callback { public: explicit StateManagerCallback(ActorShared td) : td_(std::move(td)) { } bool on_state(StateManager::State state) override { send_closure(td_, &Td::on_connection_state_changed, state); return td_.is_alive(); } private: ActorShared td_; }; state_manager_ = create_actor("State manager"); send_closure(state_manager_, &StateManager::add_callback, make_unique(create_reference())); G()->set_state_manager(state_manager_.get()); connection_state_ = StateManager::State::Empty; VLOG(td_init) << "Create ConnectionCreator"; { auto connection_creator = create_actor("ConnectionCreator", create_reference()); auto net_stats_manager = create_actor("NetStatsManager", create_reference()); // How else could I let two actor know about each other, without quite complex async logic? auto net_stats_manager_ptr = net_stats_manager->get_actor_unsafe(); net_stats_manager_ptr->init(); connection_creator->get_actor_unsafe()->set_net_stats_callback(net_stats_manager_ptr->get_common_stats_callback(), net_stats_manager_ptr->get_media_stats_callback()); G()->set_net_stats_file_callbacks(net_stats_manager_ptr->get_file_stats_callbacks()); G()->set_connection_creator(std::move(connection_creator)); net_stats_manager_ = std::move(net_stats_manager); complete_pending_preauthentication_requests([](int32 id) { switch (id) { case td_api::processDcUpdate::ID: case td_api::setNetworkType::ID: case td_api::getNetworkStatistics::ID: case td_api::addNetworkStatistics::ID: case td_api::resetNetworkStatistics::ID: case td_api::addProxy::ID: case td_api::editProxy::ID: case td_api::enableProxy::ID: case td_api::disableProxy::ID: case td_api::removeProxy::ID: case td_api::getProxies::ID: case td_api::getProxyLink::ID: case td_api::pingProxy::ID: return true; default: return false; } }); } VLOG(td_init) << "Create TempAuthKeyWatchdog"; auto temp_auth_key_watchdog = create_actor("TempAuthKeyWatchdog"); G()->set_temp_auth_key_watchdog(std::move(temp_auth_key_watchdog)); VLOG(td_init) << "Create ConfigManager and ConfigShared"; class ConfigSharedCallback : public ConfigShared::Callback { public: void on_option_updated(const string &name, const string &value) override { send_closure(G()->td(), &Td::on_config_option_updated, name); } }; send_closure( actor_id(this), &Td::send_update, make_tl_object("version", make_tl_object(TDLIB_VERSION))); G()->set_shared_config( std::make_unique(G()->td_db()->get_config_pmc(), std::make_unique())); config_manager_ = create_actor("ConfigManager", create_reference()); G()->set_config_manager(config_manager_.get()); complete_pending_preauthentication_requests([](int32 id) { switch (id) { case td_api::getOption::ID: case td_api::setOption::ID: return true; default: return false; } }); options_.language_pack = G()->shared_config().get_option_string("language_pack"); options_.language_code = G()->shared_config().get_option_string("language_code"); options_.is_emulator = G()->shared_config().get_option_boolean("is_emulator"); // options_.proxy = Proxy(); G()->set_mtproto_header(std::make_unique(options_)); if (!G()->shared_config().have_option("message_text_length_max")) { G()->shared_config().set_option_integer("message_text_length_max", 4096); } if (!G()->shared_config().have_option("message_caption_length_max")) { G()->shared_config().set_option_integer("message_caption_length_max", 200); } VLOG(td_init) << "Create NetQueryDispatcher"; auto net_query_dispatcher = std::make_unique([&] { return create_reference(); }); G()->set_net_query_dispatcher(std::move(net_query_dispatcher)); VLOG(td_init) << "Create AuthManager"; auth_manager_ = std::make_unique(parameters_.api_id, parameters_.api_hash, create_reference()); auth_manager_actor_ = register_actor("AuthManager", auth_manager_.get()); VLOG(td_init) << "Create FileManager"; download_file_callback_ = std::make_shared(); upload_file_callback_ = std::make_shared(); class FileManagerContext : public FileManager::Context { public: explicit FileManagerContext(Td *td) : td_(td) { } void on_new_file(int64 size) final { send_closure(G()->storage_manager(), &StorageManager::on_new_file, size); } void on_file_updated(FileId file_id) final { send_closure(G()->td(), &Td::send_update, make_tl_object(td_->file_manager_->get_file_object(file_id))); } ActorShared<> create_reference() final { return td_->create_reference(); } private: Td *td_; }; file_manager_ = std::make_unique(std::make_unique(this)); file_manager_actor_ = register_actor("FileManager", file_manager_.get()); file_manager_->init_actor(); G()->set_file_manager(file_manager_actor_.get()); VLOG(td_init) << "Create Managers"; audios_manager_ = make_unique(this); callback_queries_manager_ = make_unique(this); documents_manager_ = make_unique(this); video_notes_manager_ = make_unique(this); videos_manager_ = make_unique(this); voice_notes_manager_ = make_unique(this); animations_manager_ = std::make_unique(this, create_reference()); animations_manager_actor_ = register_actor("AnimationsManager", animations_manager_.get()); G()->set_animations_manager(animations_manager_actor_.get()); contacts_manager_ = std::make_unique(this, create_reference()); contacts_manager_actor_ = register_actor("ContactsManager", contacts_manager_.get()); G()->set_contacts_manager(contacts_manager_actor_.get()); inline_queries_manager_ = std::make_unique(this, create_reference()); inline_queries_manager_actor_ = register_actor("InlineQueriesManager", inline_queries_manager_.get()); messages_manager_ = std::make_unique(this, create_reference()); messages_manager_actor_ = register_actor("MessagesManager", messages_manager_.get()); G()->set_messages_manager(messages_manager_actor_.get()); stickers_manager_ = std::make_unique(this, create_reference()); stickers_manager_actor_ = register_actor("StickersManager", stickers_manager_.get()); G()->set_stickers_manager(stickers_manager_actor_.get()); updates_manager_ = std::make_unique(this, create_reference()); updates_manager_actor_ = register_actor("UpdatesManager", updates_manager_.get()); G()->set_updates_manager(updates_manager_actor_.get()); web_pages_manager_ = std::make_unique(this, create_reference()); web_pages_manager_actor_ = register_actor("WebPagesManager", web_pages_manager_.get()); G()->set_web_pages_manager(web_pages_manager_actor_.get()); call_manager_ = create_actor("CallManager", create_reference()); G()->set_call_manager(call_manager_.get()); change_phone_number_manager_ = create_actor( "ChangePhoneNumberManager", PhoneNumberManager::Type::ChangePhone, create_reference()); confirm_phone_number_manager_ = create_actor( "ConfirmPhoneNumberManager", PhoneNumberManager::Type::ConfirmPhone, create_reference()); device_token_manager_ = create_actor("DeviceTokenManager", create_reference()); hashtag_hints_ = create_actor("HashtagHints", "text", create_reference()); language_pack_manager_ = create_actor("LanguagePackManager", create_reference()); G()->set_language_pack_manager(language_pack_manager_.get()); password_manager_ = create_actor("PasswordManager", create_reference()); G()->set_password_manager(password_manager_.get()); privacy_manager_ = create_actor("PrivacyManager", create_reference()); secret_chats_manager_ = create_actor("SecretChatsManager", create_reference()); G()->set_secret_chats_manager(secret_chats_manager_.get()); secure_manager_ = create_actor("SecureManager", create_reference()); storage_manager_ = create_actor("StorageManager", create_reference(), min(current_scheduler_id + 2, scheduler_count - 1)); G()->set_storage_manager(storage_manager_.get()); top_dialog_manager_ = create_actor("TopDialogManager", create_reference()); G()->set_top_dialog_manager(top_dialog_manager_.get()); verify_phone_number_manager_ = create_actor( "VerifyPhoneNumberManager", PhoneNumberManager::Type::VerifyPhone, create_reference()); VLOG(td_init) << "Send binlog events"; for (auto &event : events.user_events) { contacts_manager_->on_binlog_user_event(std::move(event)); } for (auto &event : events.channel_events) { contacts_manager_->on_binlog_channel_event(std::move(event)); } // chats may contain links to channels, so should be inited after for (auto &event : events.chat_events) { contacts_manager_->on_binlog_chat_event(std::move(event)); } for (auto &event : events.secret_chat_events) { contacts_manager_->on_binlog_secret_chat_event(std::move(event)); } for (auto &event : events.web_page_events) { web_pages_manager_->on_binlog_web_page_event(std::move(event)); } // Send binlog events to managers // // 1. Actors must receive all binlog events before other queries. // // -- All actors have one "entry point". So there is only one way to send query to them. So all queries are ordered // for each Actor. // // // 2. An actor must not make some decisions before all binlog events are processed. // For example, SecretChatActor must not send RequestKey, before it receives logevent with RequestKey and understands // that RequestKey was already sent. // // -- G()->wait_binlog_replay_finish(Promise<>); // // 3. During replay of binlog some queries may be sent to other actors. They shouldn't process such events before all // their binlog events are processed. So actor may receive some old queries. It must be in it's actual state in // orded to handle them properly. // // -- Use send_closure_later, so actors don't even start process binlog events, before all binlog events are sent for (auto &event : events.to_secret_chats_manager) { send_closure_later(secret_chats_manager_, &SecretChatsManager::replay_binlog_event, std::move(event)); } send_closure_later(messages_manager_actor_, &MessagesManager::on_binlog_events, std::move(events.to_messages_manager)); // NB: be very careful. This notification may be received before all binlog events are. G()->on_binlog_replay_finish(); send_closure(secret_chats_manager_, &SecretChatsManager::binlog_replay_finish); VLOG(td_init) << "Ping datacenter"; if (!auth_manager_->is_authorized()) { create_handler(Promise())->send(); } else { updates_manager_->get_difference("init"); schedule_get_terms_of_service(0); } complete_pending_preauthentication_requests([](int32 id) { return true; }); state_ = State::Run; return Status::OK(); } void Td::send_update(tl_object_ptr &&object) { auto object_id = object->get_id(); if (close_flag_ >= 5 && object_id != td_api::updateAuthorizationState::ID) { // just in case return; } switch (object_id) { case td_api::updateFavoriteStickers::ID: case td_api::updateInstalledStickerSets::ID: case td_api::updateRecentStickers::ID: case td_api::updateSavedAnimations::ID: case td_api::updateUserStatus::ID: VLOG(td_requests) << "Sending update: " << oneline(to_string(object)); break; case td_api::updateTrendingStickerSets::ID: VLOG(td_requests) << "Sending update: updateTrendingStickerSets { ... }"; break; case td_api::updateChatReadInbox::ID * 0: LOG(ERROR) << "Sending update: " << oneline(to_string(object)); break; default: VLOG(td_requests) << "Sending update: " << to_string(object); } callback_->on_result(0, std::move(object)); } void Td::send_result(uint64 id, tl_object_ptr object) { if (id == 0) { LOG(ERROR) << "Sending " << to_string(object) << " through send_result"; return; } auto it = request_set_.find(id); if (it != request_set_.end()) { request_set_.erase(it); VLOG(td_requests) << "Sending result for request " << id << ": " << to_string(object); if (object == nullptr) { object = make_tl_object(404, "Not Found"); } callback_->on_result(id, std::move(object)); } } void Td::send_error_impl(uint64 id, tl_object_ptr error) { CHECK(id != 0); CHECK(callback_ != nullptr); CHECK(error != nullptr); auto it = request_set_.find(id); if (it != request_set_.end()) { request_set_.erase(it); VLOG(td_requests) << "Sending error for request " << id << ": " << oneline(to_string(error)); callback_->on_error(id, std::move(error)); } } void Td::send_error(uint64 id, Status error) { send_error_impl(id, make_tl_object(error.code(), error.message().str())); error.ignore(); } void Td::send_error_raw(uint64 id, int32 code, CSlice error) { send_closure(actor_id(this), &Td::send_error_impl, id, make_error(code, error)); } void Td::answer_ok_query(uint64 id, Status status) { if (status.is_error()) { send_closure(actor_id(this), &Td::send_error, id, std::move(status)); } else { send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } } template Promise Td::create_request_promise(uint64 id) { return PromiseCreator::lambda([id = id, actor_id = actor_id(this)](Result r_state) { if (r_state.is_error()) { send_closure(actor_id, &Td::send_error, id, r_state.move_as_error()); } else { send_closure(actor_id, &Td::send_result, id, r_state.move_as_ok()); } }); } Promise Td::create_ok_request_promise(uint64 id) { return PromiseCreator::lambda([id = id, actor_id = actor_id(this)](Result result) mutable { if (result.is_error()) { send_closure(actor_id, &Td::send_error, id, result.move_as_error()); } else { send_closure(actor_id, &Td::send_result, id, td_api::make_object()); } }); } #define CLEAN_INPUT_STRING(field_name) \ if (!clean_input_string(field_name)) { \ return send_error_raw(id, 400, "Strings must be encoded in UTF-8"); \ } #define CHECK_IS_BOT() \ if (!auth_manager_->is_bot()) { \ return send_error_raw(id, 400, "Only bots can use the method"); \ } #define CHECK_IS_USER() \ if (auth_manager_->is_bot()) { \ return send_error_raw(id, 400, "The method is not available for bots"); \ } #define CREATE_NO_ARGS_REQUEST(name) \ auto slot_id = request_actors_.create(ActorOwn<>(), RequestActorIdType); \ inc_request_actor_refcnt(); \ *request_actors_.get(slot_id) = create_actor(#name, actor_shared(this, slot_id), id); #define CREATE_REQUEST(name, ...) \ auto slot_id = request_actors_.create(ActorOwn<>(), RequestActorIdType); \ inc_request_actor_refcnt(); \ *request_actors_.get(slot_id) = create_actor(#name, actor_shared(this, slot_id), id, __VA_ARGS__); #define CREATE_REQUEST_PROMISE() auto promise = create_request_promise::ReturnType>(id) #define CREATE_OK_REQUEST_PROMISE() \ static_assert(std::is_same::ReturnType, td_api::object_ptr>::value, ""); \ auto promise = create_ok_request_promise(id) Status Td::fix_parameters(TdParameters ¶meters) { if (parameters.database_directory.empty()) { VLOG(td_init) << "Fix database_directory"; parameters.database_directory = "."; } if (parameters.files_directory.empty()) { VLOG(td_init) << "Fix files_directory"; parameters.files_directory = parameters.database_directory; } if (parameters.use_message_db) { VLOG(td_init) << "Fix use_chat_info_db"; parameters.use_chat_info_db = true; } if (parameters.use_chat_info_db) { VLOG(td_init) << "Fix use_file_db"; parameters.use_file_db = true; } if (parameters.api_id <= 0) { VLOG(td_init) << "Invalid api_id"; return Status::Error(400, "Valid api_id must be provided. Can be obtained at https://my.telegram.org"); } if (parameters.api_hash.empty()) { VLOG(td_init) << "Invalid api_hash"; return Status::Error(400, "Valid api_hash must be provided. Can be obtained at https://my.telegram.org"); } auto prepare_dir = [](string dir) -> Result { CHECK(!dir.empty()); if (dir.back() != TD_DIR_SLASH) { dir += TD_DIR_SLASH; } TRY_STATUS(mkpath(dir, 0750)); TRY_RESULT(real_dir, realpath(dir, true)); if (dir.back() != TD_DIR_SLASH) { dir += TD_DIR_SLASH; } return real_dir; }; auto r_database_directory = prepare_dir(parameters.database_directory); if (r_database_directory.is_error()) { VLOG(td_init) << "Invalid database_directory"; return Status::Error(400, PSLICE() << "Can't init database in the directory \"" << parameters.database_directory << "\": " << r_database_directory.error()); } parameters.database_directory = r_database_directory.move_as_ok(); auto r_files_directory = prepare_dir(parameters.files_directory); if (r_files_directory.is_error()) { VLOG(td_init) << "Invalid files_directory"; return Status::Error(400, PSLICE() << "Can't init files directory \"" << parameters.files_directory << "\": " << r_files_directory.error()); } parameters.files_directory = r_files_directory.move_as_ok(); return Status::OK(); } Status Td::set_parameters(td_api::object_ptr parameters) { VLOG(td_init) << "Begin to set TDLib parameters"; if (parameters == nullptr) { VLOG(td_init) << "Empty parameters"; return Status::Error(400, "Parameters aren't specified"); } if (!clean_input_string(parameters->api_hash_) && !clean_input_string(parameters->system_language_code_) && !clean_input_string(parameters->device_model_) && !clean_input_string(parameters->system_version_) && !clean_input_string(parameters->application_version_)) { VLOG(td_init) << "Wrong string encoding"; return Status::Error(400, "Strings must be encoded in UTF-8"); } parameters_.use_test_dc = parameters->use_test_dc_; parameters_.database_directory = parameters->database_directory_; parameters_.files_directory = parameters->files_directory_; parameters_.api_id = parameters->api_id_; parameters_.api_hash = parameters->api_hash_; parameters_.use_file_db = parameters->use_file_database_; parameters_.enable_storage_optimizer = parameters->enable_storage_optimizer_; parameters_.ignore_file_names = parameters->ignore_file_names_; parameters_.use_secret_chats = parameters->use_secret_chats_; parameters_.use_chat_info_db = parameters->use_chat_info_database_; parameters_.use_message_db = parameters->use_message_database_; VLOG(td_init) << "Fix parameters..."; TRY_STATUS(fix_parameters(parameters_)); VLOG(td_init) << "Check binlog encryption..."; TRY_RESULT(encryption_info, TdDb::check_encryption(parameters_)); encryption_info_ = std::move(encryption_info); VLOG(td_init) << "Create Global"; set_context(std::make_shared()); inc_request_actor_refcnt(); // guard inc_actor_refcnt(); // guard VLOG(td_init) << "Create MtprotoHeader::Options"; options_.api_id = parameters->api_id_; options_.system_language_code = trim(parameters->system_language_code_); options_.device_model = trim(parameters->device_model_); options_.system_version = trim(parameters->system_version_); options_.application_version = trim(parameters->application_version_); if (options_.system_language_code.empty()) { return Status::Error(400, "System language code must be non-empty"); } if (options_.device_model.empty()) { return Status::Error(400, "Device model must be non-empty"); } if (options_.system_version.empty()) { return Status::Error(400, "System version must be non-empty"); } if (options_.application_version.empty()) { return Status::Error(400, "Application version must be non-empty"); } if (options_.api_id != 21724) { options_.application_version += ", TDLib "; options_.application_version += TDLIB_VERSION; } options_.language_pack = ""; options_.language_code = ""; options_.is_emulator = false; options_.proxy = Proxy(); state_ = State::Decrypt; VLOG(td_init) << "Send authorizationStateWaitEncryptionKey"; send_closure(actor_id(this), &Td::send_update, td_api::make_object( td_api::make_object(encryption_info_.is_encrypted))); VLOG(td_init) << "Finish set parameters"; return Status::OK(); } void Td::on_request(uint64 id, const td_api::setTdlibParameters &request) { send_error_raw(id, 400, "Unexpected setTdlibParameters"); } void Td::on_request(uint64 id, const td_api::checkDatabaseEncryptionKey &request) { send_error_raw(id, 400, "Unexpected checkDatabaseEncryptionKey"); } void Td::on_request(uint64 id, td_api::setDatabaseEncryptionKey &request) { CREATE_OK_REQUEST_PROMISE(); G()->td_db()->get_binlog()->change_key(as_db_key(std::move(request.new_encryption_key_)), std::move(promise)); } void Td::on_request(uint64 id, const td_api::getAuthorizationState &request) { send_closure(auth_manager_actor_, &AuthManager::get_state, id); } void Td::on_request(uint64 id, td_api::setAuthenticationPhoneNumber &request) { CLEAN_INPUT_STRING(request.phone_number_); send_closure(auth_manager_actor_, &AuthManager::set_phone_number, id, std::move(request.phone_number_), request.allow_flash_call_, request.is_current_phone_number_); } void Td::on_request(uint64 id, const td_api::resendAuthenticationCode &request) { send_closure(auth_manager_actor_, &AuthManager::resend_authentication_code, id); } void Td::on_request(uint64 id, td_api::checkAuthenticationCode &request) { CLEAN_INPUT_STRING(request.code_); CLEAN_INPUT_STRING(request.first_name_); CLEAN_INPUT_STRING(request.last_name_); send_closure(auth_manager_actor_, &AuthManager::check_code, id, std::move(request.code_), std::move(request.first_name_), std::move(request.last_name_)); } void Td::on_request(uint64 id, td_api::checkAuthenticationPassword &request) { CLEAN_INPUT_STRING(request.password_); send_closure(auth_manager_actor_, &AuthManager::check_password, id, std::move(request.password_)); } void Td::on_request(uint64 id, const td_api::requestAuthenticationPasswordRecovery &request) { send_closure(auth_manager_actor_, &AuthManager::request_password_recovery, id); } void Td::on_request(uint64 id, td_api::recoverAuthenticationPassword &request) { CLEAN_INPUT_STRING(request.recovery_code_); send_closure(auth_manager_actor_, &AuthManager::recover_password, id, std::move(request.recovery_code_)); } void Td::on_request(uint64 id, const td_api::logOut &request) { // will call Td::destroy later send_closure(auth_manager_actor_, &AuthManager::logout, id); } void Td::on_request(uint64 id, const td_api::close &request) { // send response before actually closing send_closure(actor_id(this), &Td::send_result, id, td_api::make_object()); close(); } void Td::on_request(uint64 id, const td_api::destroy &request) { // send response before actually destroying send_closure(actor_id(this), &Td::send_result, id, td_api::make_object()); destroy(); } void Td::on_request(uint64 id, td_api::checkAuthenticationBotToken &request) { CLEAN_INPUT_STRING(request.token_); send_closure(auth_manager_actor_, &AuthManager::check_bot_token, id, std::move(request.token_)); } void Td::on_request(uint64 id, td_api::getPasswordState &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::get_state, std::move(promise)); } void Td::on_request(uint64 id, td_api::setPassword &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.old_password_); CLEAN_INPUT_STRING(request.new_password_); CLEAN_INPUT_STRING(request.new_hint_); CLEAN_INPUT_STRING(request.new_recovery_email_address_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::set_password, std::move(request.old_password_), std::move(request.new_password_), std::move(request.new_hint_), request.set_recovery_email_address_, std::move(request.new_recovery_email_address_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setRecoveryEmailAddress &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CLEAN_INPUT_STRING(request.new_recovery_email_address_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::set_recovery_email_address, std::move(request.password_), std::move(request.new_recovery_email_address_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getRecoveryEmailAddress &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::get_recovery_email_address, std::move(request.password_), std::move(promise)); } void Td::on_request(uint64 id, td_api::requestPasswordRecovery &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::request_password_recovery, std::move(promise)); } void Td::on_request(uint64 id, td_api::recoverPassword &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.recovery_code_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::recover_password, std::move(request.recovery_code_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getTemporaryPasswordState &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::get_temp_password_state, std::move(promise)); } void Td::on_request(uint64 id, td_api::createTemporaryPassword &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::create_temp_password, std::move(request.password_), request.valid_for_, std::move(promise)); } void Td::on_request(uint64 id, td_api::processDcUpdate &request) { CLEAN_INPUT_STRING(request.dc_); CLEAN_INPUT_STRING(request.addr_); CREATE_OK_REQUEST_PROMISE(); auto dc_id_raw = to_integer(request.dc_); if (!DcId::is_valid(dc_id_raw)) { return promise.set_error(Status::Error("Invalid dc id")); } send_closure(G()->connection_creator(), &ConnectionCreator::on_dc_update, DcId::internal(dc_id_raw), request.addr_, std::move(promise)); } void Td::on_request(uint64 id, td_api::registerDevice &request) { CHECK_IS_USER(); if (request.device_token_ == nullptr) { return send_error_raw(id, 400, "Device token should not be empty"); } CREATE_REQUEST_PROMISE(); send_closure(device_token_manager_, &DeviceTokenManager::register_device, std::move(request.device_token_), std::move(request.other_user_ids_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getUserPrivacySettingRules &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(privacy_manager_, &PrivacyManager::get_privacy, std::move(request.setting_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setUserPrivacySettingRules &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(privacy_manager_, &PrivacyManager::set_privacy, std::move(request.setting_), std::move(request.rules_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::getAccountTtl &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetAccountTtlRequest); } void Td::on_request(uint64 id, const td_api::setAccountTtl &request) { CHECK_IS_USER(); if (request.ttl_ == nullptr) { return send_error_raw(id, 400, "New account TTL should not be empty"); } CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_account_ttl(request.ttl_->days_, std::move(promise)); } void Td::on_request(uint64 id, td_api::deleteAccount &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.reason_); send_closure(auth_manager_actor_, &AuthManager::delete_account, id, request.reason_); } void Td::on_request(uint64 id, td_api::changePhoneNumber &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.phone_number_); send_closure(change_phone_number_manager_, &PhoneNumberManager::set_phone_number, id, std::move(request.phone_number_), request.allow_flash_call_, request.is_current_phone_number_); } void Td::on_request(uint64 id, td_api::checkChangePhoneNumberCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.code_); send_closure(change_phone_number_manager_, &PhoneNumberManager::check_code, id, std::move(request.code_)); } void Td::on_request(uint64 id, td_api::resendChangePhoneNumberCode &request) { CHECK_IS_USER(); send_closure(change_phone_number_manager_, &PhoneNumberManager::resend_authentication_code, id); } void Td::on_request(uint64 id, const td_api::getActiveSessions &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetActiveSessionsRequest); } void Td::on_request(uint64 id, const td_api::terminateSession &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->terminate_session(request.session_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::terminateAllOtherSessions &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->terminate_all_other_sessions(std::move(promise)); } void Td::on_request(uint64 id, const td_api::getConnectedWebsites &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetConnectedWebsitesRequest); } void Td::on_request(uint64 id, const td_api::disconnectWebsite &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->disconnect_website(request.website_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::disconnectAllWebsites &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->disconnect_all_websites(std::move(promise)); } void Td::on_request(uint64 id, const td_api::getMe &request) { CREATE_NO_ARGS_REQUEST(GetMeRequest); } void Td::on_request(uint64 id, const td_api::getUser &request) { CREATE_REQUEST(GetUserRequest, request.user_id_); } void Td::on_request(uint64 id, const td_api::getUserFullInfo &request) { CREATE_REQUEST(GetUserFullInfoRequest, request.user_id_); } void Td::on_request(uint64 id, const td_api::getBasicGroup &request) { CREATE_REQUEST(GetGroupRequest, request.basic_group_id_); } void Td::on_request(uint64 id, const td_api::getBasicGroupFullInfo &request) { CREATE_REQUEST(GetGroupFullInfoRequest, request.basic_group_id_); } void Td::on_request(uint64 id, const td_api::getSupergroup &request) { CREATE_REQUEST(GetSupergroupRequest, request.supergroup_id_); } void Td::on_request(uint64 id, const td_api::getSupergroupFullInfo &request) { CREATE_REQUEST(GetSupergroupFullInfoRequest, request.supergroup_id_); } void Td::on_request(uint64 id, const td_api::getSecretChat &request) { CREATE_REQUEST(GetSecretChatRequest, request.secret_chat_id_); } void Td::on_request(uint64 id, const td_api::getChat &request) { CREATE_REQUEST(GetChatRequest, request.chat_id_); } void Td::on_request(uint64 id, const td_api::getMessage &request) { CREATE_REQUEST(GetMessageRequest, request.chat_id_, request.message_id_); } void Td::on_request(uint64 id, const td_api::getRepliedMessage &request) { CREATE_REQUEST(GetRepliedMessageRequest, request.chat_id_, request.message_id_); } void Td::on_request(uint64 id, const td_api::getChatPinnedMessage &request) { CREATE_REQUEST(GetChatPinnedMessageRequest, request.chat_id_); } void Td::on_request(uint64 id, const td_api::getMessages &request) { CREATE_REQUEST(GetMessagesRequest, request.chat_id_, request.message_ids_); } void Td::on_request(uint64 id, const td_api::getPublicMessageLink &request) { CREATE_REQUEST(GetPublicMessageLinkRequest, request.chat_id_, request.message_id_, request.for_album_); } void Td::on_request(uint64 id, const td_api::getFile &request) { send_closure(actor_id(this), &Td::send_result, id, file_manager_->get_file_object(FileId(request.file_id_, 0))); } void Td::on_request(uint64 id, td_api::getRemoteFile &request) { CLEAN_INPUT_STRING(request.remote_file_id_); auto r_file_id = file_manager_->from_persistent_id( request.remote_file_id_, request.file_type_ == nullptr ? FileType::Temp : from_td_api(*request.file_type_)); if (r_file_id.is_error()) { send_closure(actor_id(this), &Td::send_error, id, r_file_id.move_as_error()); } else { send_closure(actor_id(this), &Td::send_result, id, file_manager_->get_file_object(r_file_id.ok())); } } void Td::on_request(uint64 id, td_api::getStorageStatistics &request) { CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(result.ok().as_td_api()); } }); send_closure(storage_manager_, &StorageManager::get_storage_stats, request.chat_limit_, std::move(query_promise)); } void Td::on_request(uint64 id, td_api::getStorageStatisticsFast &request) { CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(result.ok().as_td_api()); } }); send_closure(storage_manager_, &StorageManager::get_storage_stats_fast, std::move(query_promise)); } void Td::on_request(uint64 id, td_api::optimizeStorage &request) { std::vector file_types; for (auto &file_type : request.file_types_) { if (file_type == nullptr) { return send_error_raw(id, 400, "File type should not be empty"); } file_types.push_back(from_td_api(*file_type)); } std::vector owner_dialog_ids; for (auto chat_id : request.chat_ids_) { DialogId dialog_id(chat_id); if (!dialog_id.is_valid() && dialog_id != DialogId()) { return send_error_raw(id, 400, "Wrong chat id"); } owner_dialog_ids.push_back(dialog_id); } std::vector exclude_owner_dialog_ids; for (auto chat_id : request.exclude_chat_ids_) { DialogId dialog_id(chat_id); if (!dialog_id.is_valid() && dialog_id != DialogId()) { return send_error_raw(id, 400, "Wrong chat id"); } exclude_owner_dialog_ids.push_back(dialog_id); } FileGcParameters parameters(request.size_, request.ttl_, request.count_, request.immunity_delay_, std::move(file_types), std::move(owner_dialog_ids), std::move(exclude_owner_dialog_ids), request.chat_limit_); CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(result.ok().as_td_api()); } }); send_closure(storage_manager_, &StorageManager::run_gc, std::move(parameters), std::move(query_promise)); } void Td::on_request(uint64 id, td_api::getNetworkStatistics &request) { CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(result.ok().as_td_api()); } }); send_closure(net_stats_manager_, &NetStatsManager::get_network_stats, request.only_current_, std::move(query_promise)); } void Td::on_request(uint64 id, td_api::resetNetworkStatistics &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(net_stats_manager_, &NetStatsManager::reset_network_stats); promise.set_value(Unit()); } void Td::on_request(uint64 id, td_api::addNetworkStatistics &request) { if (request.entry_ == nullptr) { return send_error_raw(id, 400, "Network statistics entry should not be empty"); } NetworkStatsEntry entry; switch (request.entry_->get_id()) { case td_api::networkStatisticsEntryFile::ID: { auto file_entry = move_tl_object_as(request.entry_); entry.is_call = false; if (file_entry->file_type_ != nullptr) { entry.file_type = from_td_api(*file_entry->file_type_); } entry.net_type = from_td_api(file_entry->network_type_); entry.rx = file_entry->received_bytes_; entry.tx = file_entry->sent_bytes_; break; } case td_api::networkStatisticsEntryCall::ID: { auto call_entry = move_tl_object_as(request.entry_); entry.is_call = true; entry.net_type = from_td_api(call_entry->network_type_); entry.rx = call_entry->received_bytes_; entry.tx = call_entry->sent_bytes_; entry.duration = call_entry->duration_; break; } default: UNREACHABLE(); } if (entry.net_type == NetType::None) { return send_error_raw(id, 400, "Network statistics entry can't be increased for NetworkTypeNone"); } if (entry.rx > (1ll << 40) || entry.rx < 0) { return send_error_raw(id, 400, "Wrong received bytes value"); } if (entry.tx > (1ll << 40) || entry.tx < 0) { return send_error_raw(id, 400, "Wrong sent bytes value"); } if (entry.count > (1 << 30) || entry.count < 0) { return send_error_raw(id, 400, "Wrong count value"); } if (entry.duration > (1 << 30) || entry.duration < 0) { return send_error_raw(id, 400, "Wrong duration value"); } send_closure(net_stats_manager_, &NetStatsManager::add_network_stats, entry); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::setNetworkType &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(state_manager_, &StateManager::on_network, from_td_api(request.type_)); promise.set_value(Unit()); } void Td::on_request(uint64 id, td_api::getTopChats &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); if (request.category_ == nullptr) { return promise.set_error(Status::Error(400, "Top chat category should not be empty")); } if (request.limit_ <= 0) { return promise.set_error(Status::Error(400, "Limit must be positive")); } auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result> result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(MessagesManager::get_chats_object(result.ok())); } }); send_closure(top_dialog_manager_, &TopDialogManager::get_top_dialogs, top_dialog_category_from_td_api(*request.category_), request.limit_, std::move(query_promise)); } void Td::on_request(uint64 id, const td_api::removeTopChat &request) { CHECK_IS_USER(); if (request.category_ == nullptr) { return send_error_raw(id, 400, "Top chat category should not be empty"); } send_closure(top_dialog_manager_, &TopDialogManager::remove_dialog, top_dialog_category_from_td_api(*request.category_), DialogId(request.chat_id_), messages_manager_->get_input_peer(DialogId(request.chat_id_), AccessRights::Read)); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, const td_api::getChats &request) { CHECK_IS_USER(); CREATE_REQUEST(GetChatsRequest, request.offset_order_, request.offset_chat_id_, request.limit_); } void Td::on_request(uint64 id, td_api::searchPublicChat &request) { CLEAN_INPUT_STRING(request.username_); CREATE_REQUEST(SearchPublicChatRequest, request.username_); } void Td::on_request(uint64 id, td_api::searchPublicChats &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchPublicChatsRequest, request.query_); } void Td::on_request(uint64 id, td_api::searchChats &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchChatsRequest, request.query_, request.limit_); } void Td::on_request(uint64 id, td_api::searchChatsOnServer &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchChatsOnServerRequest, request.query_, request.limit_); } void Td::on_request(uint64 id, const td_api::getGroupsInCommon &request) { CHECK_IS_USER(); CREATE_REQUEST(GetGroupsInCommonRequest, request.user_id_, request.offset_chat_id_, request.limit_); } void Td::on_request(uint64 id, td_api::checkChatUsername &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.username_); CREATE_REQUEST(CheckChatUsernameRequest, request.chat_id_, std::move(request.username_)); } void Td::on_request(uint64 id, const td_api::getCreatedPublicChats &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetCreatedPublicChatsRequest); } void Td::on_request(uint64 id, const td_api::addRecentlyFoundChat &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->add_recently_found_dialog(DialogId(request.chat_id_))); } void Td::on_request(uint64 id, const td_api::removeRecentlyFoundChat &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->remove_recently_found_dialog(DialogId(request.chat_id_))); } void Td::on_request(uint64 id, const td_api::clearRecentlyFoundChats &request) { CHECK_IS_USER(); messages_manager_->clear_recently_found_dialogs(); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, const td_api::openChat &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->open_dialog(DialogId(request.chat_id_))); } void Td::on_request(uint64 id, const td_api::closeChat &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->close_dialog(DialogId(request.chat_id_))); } void Td::on_request(uint64 id, const td_api::viewMessages &request) { CHECK_IS_USER(); answer_ok_query( id, messages_manager_->view_messages( DialogId(request.chat_id_), MessagesManager::get_message_ids(request.message_ids_), request.force_read_)); } void Td::on_request(uint64 id, const td_api::openMessageContent &request) { CHECK_IS_USER(); answer_ok_query( id, messages_manager_->open_message_content({DialogId(request.chat_id_), MessageId(request.message_id_)})); } void Td::on_request(uint64 id, const td_api::getChatHistory &request) { CHECK_IS_USER(); CREATE_REQUEST(GetChatHistoryRequest, request.chat_id_, request.from_message_id_, request.offset_, request.limit_, request.only_local_); } void Td::on_request(uint64 id, const td_api::deleteChatHistory &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->delete_dialog_history(DialogId(request.chat_id_), request.remove_from_chat_list_, std::move(promise)); } void Td::on_request(uint64 id, td_api::searchChatMessages &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchChatMessagesRequest, request.chat_id_, std::move(request.query_), request.sender_user_id_, request.from_message_id_, request.offset_, request.limit_, std::move(request.filter_)); } void Td::on_request(uint64 id, td_api::searchSecretMessages &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(OfflineSearchMessagesRequest, request.chat_id_, std::move(request.query_), request.from_search_id_, request.limit_, std::move(request.filter_)); } void Td::on_request(uint64 id, td_api::searchMessages &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchMessagesRequest, std::move(request.query_), request.offset_date_, request.offset_chat_id_, request.offset_message_id_, request.limit_); } void Td::on_request(uint64 id, td_api::searchCallMessages &request) { CHECK_IS_USER(); CREATE_REQUEST(SearchCallMessagesRequest, request.from_message_id_, request.limit_, request.only_missed_); } void Td::on_request(uint64 id, const td_api::searchChatRecentLocationMessages &request) { CHECK_IS_USER(); CREATE_REQUEST(SearchChatRecentLocationMessagesRequest, request.chat_id_, request.limit_); } void Td::on_request(uint64 id, const td_api::getActiveLiveLocationMessages &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetActiveLiveLocationMessagesRequest); } void Td::on_request(uint64 id, const td_api::getChatMessageByDate &request) { CREATE_REQUEST(GetChatMessageByDateRequest, request.chat_id_, request.date_); } void Td::on_request(uint64 id, td_api::getChatMessageCount &request) { CHECK_IS_USER(); CREATE_REQUEST(GetChatMessageCountRequest, request.chat_id_, std::move(request.filter_), request.return_local_); } void Td::on_request(uint64 id, const td_api::deleteMessages &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->delete_messages(DialogId(request.chat_id_), MessagesManager::get_message_ids(request.message_ids_), request.revoke_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::deleteChatMessagesFromUser &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->delete_dialog_messages_from_user(DialogId(request.chat_id_), UserId(request.user_id_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::readAllChatMentions &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->read_all_dialog_mentions(DialogId(request.chat_id_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendMessage &request) { DialogId dialog_id(request.chat_id_); auto r_new_message_id = messages_manager_->send_message( dialog_id, MessageId(request.reply_to_message_id_), request.disable_notification_, request.from_background_, std::move(request.reply_markup_), std::move(request.input_message_content_)); if (r_new_message_id.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_new_message_id.move_as_error()); } CHECK(r_new_message_id.ok().is_valid()); send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_message_object({dialog_id, r_new_message_id.ok()})); } void Td::on_request(uint64 id, td_api::sendMessageAlbum &request) { DialogId dialog_id(request.chat_id_); auto r_message_ids = messages_manager_->send_message_group(dialog_id, MessageId(request.reply_to_message_id_), request.disable_notification_, request.from_background_, std::move(request.input_message_contents_)); if (r_message_ids.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_message_ids.move_as_error()); } send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_messages_object(-1, dialog_id, r_message_ids.ok())); } void Td::on_request(uint64 id, td_api::sendBotStartMessage &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.parameter_); DialogId dialog_id(request.chat_id_); auto r_new_message_id = messages_manager_->send_bot_start_message(UserId(request.bot_user_id_), dialog_id, request.parameter_); if (r_new_message_id.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_new_message_id.move_as_error()); } CHECK(r_new_message_id.ok().is_valid()); send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_message_object({dialog_id, r_new_message_id.ok()})); } void Td::on_request(uint64 id, td_api::sendInlineQueryResultMessage &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.result_id_); DialogId dialog_id(request.chat_id_); auto r_new_message_id = messages_manager_->send_inline_query_result_message( dialog_id, MessageId(request.reply_to_message_id_), request.disable_notification_, request.from_background_, request.query_id_, request.result_id_); if (r_new_message_id.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_new_message_id.move_as_error()); } CHECK(r_new_message_id.ok().is_valid()); send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_message_object({dialog_id, r_new_message_id.ok()})); } void Td::on_request(uint64 id, const td_api::sendChatSetTtlMessage &request) { DialogId dialog_id(request.chat_id_); auto r_new_message_id = messages_manager_->send_dialog_set_ttl_message(dialog_id, request.ttl_); if (r_new_message_id.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_new_message_id.move_as_error()); } CHECK(r_new_message_id.ok().is_valid()); send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_message_object({dialog_id, r_new_message_id.ok()})); } void Td::on_request(uint64 id, td_api::addLocalMessage &request) { CHECK_IS_USER(); DialogId dialog_id(request.chat_id_); auto r_new_message_id = messages_manager_->add_local_message( dialog_id, UserId(request.sender_user_id_), MessageId(request.reply_to_message_id_), request.disable_notification_, std::move(request.input_message_content_)); if (r_new_message_id.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_new_message_id.move_as_error()); } CHECK(r_new_message_id.ok().is_valid()); send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_message_object({dialog_id, r_new_message_id.ok()})); } void Td::on_request(uint64 id, td_api::editMessageText &request) { CREATE_REQUEST(EditMessageTextRequest, request.chat_id_, request.message_id_, std::move(request.reply_markup_), std::move(request.input_message_content_)); } void Td::on_request(uint64 id, td_api::editMessageLiveLocation &request) { CREATE_REQUEST(EditMessageLiveLocationRequest, request.chat_id_, request.message_id_, std::move(request.reply_markup_), std::move(request.location_)); } void Td::on_request(uint64 id, td_api::editMessageMedia &request) { CREATE_REQUEST(EditMessageMediaRequest, request.chat_id_, request.message_id_, std::move(request.reply_markup_), std::move(request.input_message_content_)); } void Td::on_request(uint64 id, td_api::editMessageCaption &request) { CREATE_REQUEST(EditMessageCaptionRequest, request.chat_id_, request.message_id_, std::move(request.reply_markup_), std::move(request.caption_)); } void Td::on_request(uint64 id, td_api::editMessageReplyMarkup &request) { CHECK_IS_BOT(); CREATE_REQUEST(EditMessageReplyMarkupRequest, request.chat_id_, request.message_id_, std::move(request.reply_markup_)); } void Td::on_request(uint64 id, td_api::editInlineMessageText &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->edit_inline_message_text(std::move(request.inline_message_id_), std::move(request.reply_markup_), std::move(request.input_message_content_), std::move(promise)); } void Td::on_request(uint64 id, td_api::editInlineMessageLiveLocation &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->edit_inline_message_live_location(std::move(request.inline_message_id_), std::move(request.reply_markup_), std::move(request.location_), std::move(promise)); } void Td::on_request(uint64 id, td_api::editInlineMessageMedia &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->edit_inline_message_media(std::move(request.inline_message_id_), std::move(request.reply_markup_), std::move(request.input_message_content_), std::move(promise)); } void Td::on_request(uint64 id, td_api::editInlineMessageCaption &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->edit_inline_message_caption(std::move(request.inline_message_id_), std::move(request.reply_markup_), std::move(request.caption_), std::move(promise)); } void Td::on_request(uint64 id, td_api::editInlineMessageReplyMarkup &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->edit_inline_message_reply_markup(std::move(request.inline_message_id_), std::move(request.reply_markup_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setGameScore &request) { CHECK_IS_BOT(); CREATE_REQUEST(SetGameScoreRequest, request.chat_id_, request.message_id_, request.edit_message_, request.user_id_, request.score_, request.force_); } void Td::on_request(uint64 id, td_api::setInlineGameScore &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_inline_game_score(std::move(request.inline_message_id_), request.edit_message_, UserId(request.user_id_), request.score_, request.force_, std::move(promise)); } void Td::on_request(uint64 id, td_api::getGameHighScores &request) { CHECK_IS_BOT(); CREATE_REQUEST(GetGameHighScoresRequest, request.chat_id_, request.message_id_, request.user_id_); } void Td::on_request(uint64 id, td_api::getInlineGameHighScores &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.inline_message_id_); CREATE_REQUEST(GetInlineGameHighScoresRequest, std::move(request.inline_message_id_), request.user_id_); } void Td::on_request(uint64 id, const td_api::deleteChatReplyMarkup &request) { CHECK_IS_USER(); answer_ok_query( id, messages_manager_->delete_dialog_reply_markup(DialogId(request.chat_id_), MessageId(request.message_id_))); } void Td::on_request(uint64 id, td_api::sendChatAction &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->send_dialog_action(DialogId(request.chat_id_), std::move(request.action_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendChatScreenshotTakenNotification &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->send_screenshot_taken_notification_message(DialogId(request.chat_id_))); } void Td::on_request(uint64 id, const td_api::forwardMessages &request) { DialogId dialog_id(request.chat_id_); auto r_message_ids = messages_manager_->forward_messages( dialog_id, DialogId(request.from_chat_id_), MessagesManager::get_message_ids(request.message_ids_), request.disable_notification_, request.from_background_, false, request.as_album_); if (r_message_ids.is_error()) { return send_closure(actor_id(this), &Td::send_error, id, r_message_ids.move_as_error()); } send_closure(actor_id(this), &Td::send_result, id, messages_manager_->get_messages_object(-1, dialog_id, r_message_ids.ok())); } void Td::on_request(uint64 id, td_api::getWebPagePreview &request) { CHECK_IS_USER(); CREATE_REQUEST(GetWebPagePreviewRequest, std::move(request.text_)); } void Td::on_request(uint64 id, td_api::getWebPageInstantView &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.url_); CREATE_REQUEST(GetWebPageInstantViewRequest, std::move(request.url_), request.force_full_); } void Td::on_request(uint64 id, const td_api::createPrivateChat &request) { CREATE_REQUEST(CreateChatRequest, DialogId(UserId(request.user_id_)), request.force_); } void Td::on_request(uint64 id, const td_api::createBasicGroupChat &request) { CREATE_REQUEST(CreateChatRequest, DialogId(ChatId(request.basic_group_id_)), request.force_); } void Td::on_request(uint64 id, const td_api::createSupergroupChat &request) { CREATE_REQUEST(CreateChatRequest, DialogId(ChannelId(request.supergroup_id_)), request.force_); } void Td::on_request(uint64 id, td_api::createSecretChat &request) { CREATE_REQUEST(CreateChatRequest, DialogId(SecretChatId(request.secret_chat_id_)), true); } void Td::on_request(uint64 id, td_api::createNewBasicGroupChat &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.title_); CREATE_REQUEST(CreateNewGroupChatRequest, request.user_ids_, std::move(request.title_)); } void Td::on_request(uint64 id, td_api::createNewSupergroupChat &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.title_); CLEAN_INPUT_STRING(request.description_); CREATE_REQUEST(CreateNewSupergroupChatRequest, std::move(request.title_), !request.is_channel_, std::move(request.description_)); } void Td::on_request(uint64 id, td_api::createNewSecretChat &request) { CREATE_REQUEST(CreateNewSecretChatRequest, request.user_id_); } void Td::on_request(uint64 id, td_api::createCall &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(result.ok().as_td_api()); } }); if (!request.protocol_) { return query_promise.set_error(Status::Error(5, "Call protocol must not be empty")); } UserId user_id(request.user_id_); auto input_user = contacts_manager_->get_input_user(user_id); if (input_user == nullptr) { return query_promise.set_error(Status::Error(6, "User not found")); } if (!G()->shared_config().get_option_boolean("calls_enabled")) { return query_promise.set_error(Status::Error(7, "Calls are not enabled for the current user")); } send_closure(G()->call_manager(), &CallManager::create_call, user_id, std::move(input_user), CallProtocol::from_td_api(*request.protocol_), std::move(query_promise)); } void Td::on_request(uint64 id, td_api::discardCall &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); send_closure(G()->call_manager(), &CallManager::discard_call, CallId(request.call_id_), request.is_disconnected_, request.duration_, request.connection_id_, std::move(promise)); } void Td::on_request(uint64 id, td_api::acceptCall &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); if (!request.protocol_) { return promise.set_error(Status::Error(5, "Call protocol must not be empty")); } send_closure(G()->call_manager(), &CallManager::accept_call, CallId(request.call_id_), CallProtocol::from_td_api(*request.protocol_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendCallRating &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.comment_); CREATE_OK_REQUEST_PROMISE(); send_closure(G()->call_manager(), &CallManager::rate_call, CallId(request.call_id_), request.rating_, std::move(request.comment_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendCallDebugInformation &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.debug_information_); CREATE_OK_REQUEST_PROMISE(); send_closure(G()->call_manager(), &CallManager::send_call_debug_information, CallId(request.call_id_), std::move(request.debug_information_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::upgradeBasicGroupChatToSupergroupChat &request) { CHECK_IS_USER(); CREATE_REQUEST(UpgradeGroupChatToSupergroupChatRequest, request.chat_id_); } void Td::on_request(uint64 id, td_api::setChatTitle &request) { CLEAN_INPUT_STRING(request.title_); CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_dialog_title(DialogId(request.chat_id_), request.title_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::setChatPhoto &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_dialog_photo(DialogId(request.chat_id_), request.photo_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setChatDraftMessage &request) { CHECK_IS_USER(); answer_ok_query( id, messages_manager_->set_dialog_draft_message(DialogId(request.chat_id_), std::move(request.draft_message_))); } void Td::on_request(uint64 id, const td_api::toggleChatIsPinned &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->toggle_dialog_is_pinned(DialogId(request.chat_id_), request.is_pinned_)); } void Td::on_request(uint64 id, const td_api::toggleChatIsMarkedAsUnread &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->toggle_dialog_is_marked_as_unread(DialogId(request.chat_id_), request.is_marked_as_unread_)); } void Td::on_request(uint64 id, const td_api::toggleChatDefaultDisableNotification &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->toggle_dialog_silent_send_message(DialogId(request.chat_id_), request.default_disable_notification_)); } void Td::on_request(uint64 id, const td_api::setPinnedChats &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->set_pinned_dialogs( transform(request.chat_ids_, [](int64 chat_id) { return DialogId(chat_id); }))); } void Td::on_request(uint64 id, td_api::setChatClientData &request) { answer_ok_query( id, messages_manager_->set_dialog_client_data(DialogId(request.chat_id_), std::move(request.client_data_))); } void Td::on_request(uint64 id, const td_api::joinChat &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_dialog_participant_status(DialogId(request.chat_id_), contacts_manager_->get_my_id("joinChat"), td_api::make_object(), std::move(promise)); } void Td::on_request(uint64 id, const td_api::leaveChat &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_dialog_participant_status( DialogId(request.chat_id_), contacts_manager_->get_my_id("leaveChat"), td_api::make_object(), std::move(promise)); } void Td::on_request(uint64 id, const td_api::addChatMember &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->add_dialog_participant(DialogId(request.chat_id_), UserId(request.user_id_), request.forward_limit_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::addChatMembers &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); vector user_ids; for (auto &user_id : request.user_ids_) { user_ids.emplace_back(user_id); } messages_manager_->add_dialog_participants(DialogId(request.chat_id_), user_ids, std::move(promise)); } void Td::on_request(uint64 id, td_api::setChatMemberStatus &request) { CREATE_OK_REQUEST_PROMISE(); messages_manager_->set_dialog_participant_status(DialogId(request.chat_id_), UserId(request.user_id_), request.status_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getChatMember &request) { CREATE_REQUEST(GetChatMemberRequest, request.chat_id_, request.user_id_); } void Td::on_request(uint64 id, td_api::searchChatMembers &request) { CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchChatMembersRequest, request.chat_id_, std::move(request.query_), request.limit_, get_dialog_participants_filter(request.filter_)); } void Td::on_request(uint64 id, td_api::getChatAdministrators &request) { CREATE_REQUEST(GetChatAdministratorsRequest, request.chat_id_); } void Td::on_request(uint64 id, const td_api::generateChatInviteLink &request) { CREATE_REQUEST(GenerateChatInviteLinkRequest, request.chat_id_); } void Td::on_request(uint64 id, td_api::checkChatInviteLink &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.invite_link_); CREATE_REQUEST(CheckChatInviteLinkRequest, request.invite_link_); } void Td::on_request(uint64 id, td_api::joinChatByInviteLink &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.invite_link_); CREATE_REQUEST(JoinChatByInviteLinkRequest, request.invite_link_); } void Td::on_request(uint64 id, td_api::getChatEventLog &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(GetChatEventLogRequest, request.chat_id_, std::move(request.query_), request.from_event_id_, request.limit_, std::move(request.filters_), std::move(request.user_ids_)); } void Td::on_request(uint64 id, const td_api::clearAllDraftMessages &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->clear_all_draft_messages(request.exclude_secret_chats_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::downloadFile &request) { auto priority = request.priority_; if (!(1 <= priority && priority <= 32)) { return send_error_raw(id, 5, "Download priority must be in [1;32] range"); } file_manager_->download(FileId(request.file_id_, 0), download_file_callback_, priority); auto file = file_manager_->get_file_object(FileId(request.file_id_, 0), false); if (file->id_ == 0) { return send_error_raw(id, 400, "Invalid file id"); } send_closure(actor_id(this), &Td::send_result, id, std::move(file)); } void Td::on_request(uint64 id, const td_api::cancelDownloadFile &request) { file_manager_->download(FileId(request.file_id_, 0), nullptr, request.only_if_pending_ ? -1 : 0); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::uploadFile &request) { auto priority = request.priority_; if (!(1 <= priority && priority <= 32)) { return send_error_raw(id, 5, "Upload priority must be in [1;32] range"); } auto file_type = request.file_type_ == nullptr ? FileType::Temp : from_td_api(*request.file_type_); bool is_secret = file_type == FileType::Encrypted || file_type == FileType::EncryptedThumbnail; bool is_secure = file_type == FileType::Secure; auto r_file_id = file_manager_->get_input_file_id(file_type, request.file_, DialogId(), false, is_secret, !is_secure && !is_secret, is_secure); if (r_file_id.is_error()) { return send_error_raw(id, 400, r_file_id.error().message()); } auto file_id = r_file_id.ok(); auto upload_file_id = file_manager_->dup_file_id(file_id); file_manager_->upload(upload_file_id, upload_file_callback_, priority, 0); send_closure(actor_id(this), &Td::send_result, id, file_manager_->get_file_object(upload_file_id, false)); } void Td::on_request(uint64 id, const td_api::cancelUploadFile &request) { file_manager_->upload(FileId(request.file_id_, 0), nullptr, 0, 0); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, const td_api::setFileGenerationProgress &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(file_manager_actor_, &FileManager::external_file_generate_progress, request.generation_id_, request.expected_size_, request.local_prefix_size_, std::move(promise)); } void Td::on_request(uint64 id, td_api::finishFileGeneration &request) { Status status; if (request.error_ != nullptr) { CLEAN_INPUT_STRING(request.error_->message_); status = Status::Error(request.error_->code_, request.error_->message_); } CREATE_OK_REQUEST_PROMISE(); send_closure(file_manager_actor_, &FileManager::external_file_generate_finish, request.generation_id_, std::move(status), std::move(promise)); } void Td::on_request(uint64 id, const td_api::deleteFile &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(file_manager_actor_, &FileManager::delete_file, FileId(request.file_id_, 0), std::move(promise), "td_api::deleteFile"); } void Td::on_request(uint64 id, const td_api::blockUser &request) { CHECK_IS_USER(); answer_ok_query(id, contacts_manager_->block_user(UserId(request.user_id_))); } void Td::on_request(uint64 id, const td_api::unblockUser &request) { CHECK_IS_USER(); answer_ok_query(id, contacts_manager_->unblock_user(UserId(request.user_id_))); } void Td::on_request(uint64 id, const td_api::getBlockedUsers &request) { CHECK_IS_USER(); CREATE_REQUEST(GetBlockedUsersRequest, request.offset_, request.limit_); } void Td::on_request(uint64 id, td_api::importContacts &request) { CHECK_IS_USER(); for (auto &contact : request.contacts_) { if (contact == nullptr) { return send_error_raw(id, 5, "Contact must not be empty"); } CLEAN_INPUT_STRING(contact->phone_number_); CLEAN_INPUT_STRING(contact->first_name_); CLEAN_INPUT_STRING(contact->last_name_); } CREATE_REQUEST(ImportContactsRequest, std::move(request.contacts_)); } void Td::on_request(uint64 id, const td_api::getContacts &request) { CHECK_IS_USER(); CREATE_REQUEST(SearchContactsRequest, string(), 1000000); } void Td::on_request(uint64 id, td_api::searchContacts &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchContactsRequest, request.query_, request.limit_); } void Td::on_request(uint64 id, td_api::removeContacts &request) { CHECK_IS_USER(); CREATE_REQUEST(RemoveContactsRequest, std::move(request.user_ids_)); } void Td::on_request(uint64 id, const td_api::getImportedContactCount &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetImportedContactCountRequest); } void Td::on_request(uint64 id, td_api::changeImportedContacts &request) { CHECK_IS_USER(); for (auto &contact : request.contacts_) { if (contact == nullptr) { return send_error_raw(id, 5, "Contact must not be empty"); } CLEAN_INPUT_STRING(contact->phone_number_); CLEAN_INPUT_STRING(contact->first_name_); CLEAN_INPUT_STRING(contact->last_name_); } CREATE_REQUEST(ChangeImportedContactsRequest, std::move(request.contacts_)); } void Td::on_request(uint64 id, const td_api::clearImportedContacts &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->clear_imported_contacts(std::move(promise)); } void Td::on_request(uint64 id, const td_api::getRecentInlineBots &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetRecentInlineBotsRequest); } void Td::on_request(uint64 id, td_api::setName &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.first_name_); CLEAN_INPUT_STRING(request.last_name_); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_name(request.first_name_, request.last_name_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setBio &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.bio_); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_bio(request.bio_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setUsername &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.username_); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_username(request.username_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setProfilePhoto &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_profile_photo(request.photo_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::deleteProfilePhoto &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->delete_profile_photo(request.profile_photo_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getUserProfilePhotos &request) { CREATE_REQUEST(GetUserProfilePhotosRequest, request.user_id_, request.offset_, request.limit_); } void Td::on_request(uint64 id, const td_api::toggleBasicGroupAdministrators &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->toggle_chat_administrators(ChatId(request.basic_group_id_), request.everyone_is_administrator_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setSupergroupUsername &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.username_); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_channel_username(ChannelId(request.supergroup_id_), request.username_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::setSupergroupStickerSet &request) { CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_channel_sticker_set(ChannelId(request.supergroup_id_), request.sticker_set_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::toggleSupergroupInvites &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->toggle_channel_invites(ChannelId(request.supergroup_id_), request.anyone_can_invite_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::toggleSupergroupSignMessages &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->toggle_channel_sign_messages(ChannelId(request.supergroup_id_), request.sign_messages_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::toggleSupergroupIsAllHistoryAvailable &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->toggle_channel_is_all_history_available(ChannelId(request.supergroup_id_), request.is_all_history_available_, std::move(promise)); } void Td::on_request(uint64 id, td_api::setSupergroupDescription &request) { CLEAN_INPUT_STRING(request.description_); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->set_channel_description(ChannelId(request.supergroup_id_), request.description_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::pinSupergroupMessage &request) { CREATE_OK_REQUEST_PROMISE(); contacts_manager_->pin_channel_message(ChannelId(request.supergroup_id_), MessageId(request.message_id_), request.disable_notification_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::unpinSupergroupMessage &request) { CREATE_OK_REQUEST_PROMISE(); contacts_manager_->unpin_channel_message(ChannelId(request.supergroup_id_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::reportSupergroupSpam &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->report_channel_spam(ChannelId(request.supergroup_id_), UserId(request.user_id_), MessagesManager::get_message_ids(request.message_ids_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getSupergroupMembers &request) { CREATE_REQUEST(GetSupergroupMembersRequest, request.supergroup_id_, std::move(request.filter_), request.offset_, request.limit_); } void Td::on_request(uint64 id, const td_api::deleteSupergroup &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); contacts_manager_->delete_channel(ChannelId(request.supergroup_id_), std::move(promise)); } void Td::on_request(uint64 id, td_api::closeSecretChat &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(secret_chats_manager_, &SecretChatsManager::cancel_chat, SecretChatId(request.secret_chat_id_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getStickers &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.emoji_); CREATE_REQUEST(GetStickersRequest, std::move(request.emoji_), request.limit_); } void Td::on_request(uint64 id, td_api::searchStickers &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.emoji_); CREATE_REQUEST(SearchStickersRequest, std::move(request.emoji_), request.limit_); } void Td::on_request(uint64 id, const td_api::getInstalledStickerSets &request) { CHECK_IS_USER(); CREATE_REQUEST(GetInstalledStickerSetsRequest, request.is_masks_); } void Td::on_request(uint64 id, const td_api::getArchivedStickerSets &request) { CHECK_IS_USER(); CREATE_REQUEST(GetArchivedStickerSetsRequest, request.is_masks_, request.offset_sticker_set_id_, request.limit_); } void Td::on_request(uint64 id, const td_api::getTrendingStickerSets &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetTrendingStickerSetsRequest); } void Td::on_request(uint64 id, const td_api::getAttachedStickerSets &request) { CHECK_IS_USER(); CREATE_REQUEST(GetAttachedStickerSetsRequest, request.file_id_); } void Td::on_request(uint64 id, const td_api::getStickerSet &request) { CREATE_REQUEST(GetStickerSetRequest, request.set_id_); } void Td::on_request(uint64 id, td_api::searchStickerSet &request) { CLEAN_INPUT_STRING(request.name_); CREATE_REQUEST(SearchStickerSetRequest, std::move(request.name_)); } void Td::on_request(uint64 id, td_api::searchInstalledStickerSets &request) { CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchInstalledStickerSetsRequest, request.is_masks_, std::move(request.query_), request.limit_); } void Td::on_request(uint64 id, td_api::searchStickerSets &request) { CLEAN_INPUT_STRING(request.query_); CREATE_REQUEST(SearchStickerSetsRequest, std::move(request.query_)); } void Td::on_request(uint64 id, const td_api::changeStickerSet &request) { CHECK_IS_USER(); CREATE_REQUEST(ChangeStickerSetRequest, request.set_id_, request.is_installed_, request.is_archived_); } void Td::on_request(uint64 id, const td_api::viewTrendingStickerSets &request) { CHECK_IS_USER(); stickers_manager_->view_featured_sticker_sets(request.sticker_set_ids_); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::reorderInstalledStickerSets &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); stickers_manager_->reorder_installed_sticker_sets(request.is_masks_, request.sticker_set_ids_, std::move(promise)); } void Td::on_request(uint64 id, td_api::uploadStickerFile &request) { CHECK_IS_BOT(); CREATE_REQUEST(UploadStickerFileRequest, request.user_id_, std::move(request.png_sticker_)); } void Td::on_request(uint64 id, td_api::createNewStickerSet &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.title_); CLEAN_INPUT_STRING(request.name_); CREATE_REQUEST(CreateNewStickerSetRequest, request.user_id_, std::move(request.title_), std::move(request.name_), request.is_masks_, std::move(request.stickers_)); } void Td::on_request(uint64 id, td_api::addStickerToSet &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.name_); CREATE_REQUEST(AddStickerToSetRequest, request.user_id_, std::move(request.name_), std::move(request.sticker_)); } void Td::on_request(uint64 id, td_api::setStickerPositionInSet &request) { CHECK_IS_BOT(); CREATE_OK_REQUEST_PROMISE(); stickers_manager_->set_sticker_position_in_set(request.sticker_, request.position_, std::move(promise)); } void Td::on_request(uint64 id, td_api::removeStickerFromSet &request) { CHECK_IS_BOT(); CREATE_OK_REQUEST_PROMISE(); stickers_manager_->remove_sticker_from_set(request.sticker_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getRecentStickers &request) { CHECK_IS_USER(); CREATE_REQUEST(GetRecentStickersRequest, request.is_attached_); } void Td::on_request(uint64 id, td_api::addRecentSticker &request) { CHECK_IS_USER(); CREATE_REQUEST(AddRecentStickerRequest, request.is_attached_, std::move(request.sticker_)); } void Td::on_request(uint64 id, td_api::removeRecentSticker &request) { CHECK_IS_USER(); CREATE_REQUEST(RemoveRecentStickerRequest, request.is_attached_, std::move(request.sticker_)); } void Td::on_request(uint64 id, td_api::clearRecentStickers &request) { CHECK_IS_USER(); CREATE_REQUEST(ClearRecentStickersRequest, request.is_attached_); } void Td::on_request(uint64 id, const td_api::getFavoriteStickers &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetFavoriteStickersRequest); } void Td::on_request(uint64 id, td_api::addFavoriteSticker &request) { CHECK_IS_USER(); CREATE_REQUEST(AddFavoriteStickerRequest, std::move(request.sticker_)); } void Td::on_request(uint64 id, td_api::removeFavoriteSticker &request) { CHECK_IS_USER(); CREATE_REQUEST(RemoveFavoriteStickerRequest, std::move(request.sticker_)); } void Td::on_request(uint64 id, td_api::getStickerEmojis &request) { CHECK_IS_USER(); CREATE_REQUEST(GetStickerEmojisRequest, std::move(request.sticker_)); } void Td::on_request(uint64 id, const td_api::getSavedAnimations &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetSavedAnimationsRequest); } void Td::on_request(uint64 id, td_api::addSavedAnimation &request) { CHECK_IS_USER(); CREATE_REQUEST(AddSavedAnimationRequest, std::move(request.animation_)); } void Td::on_request(uint64 id, td_api::removeSavedAnimation &request) { CHECK_IS_USER(); CREATE_REQUEST(RemoveSavedAnimationRequest, std::move(request.animation_)); } void Td::on_request(uint64 id, const td_api::getScopeNotificationSettings &request) { CHECK_IS_USER(); if (request.scope_ == nullptr) { return send_error_raw(id, 400, "Scope must not be empty"); } CREATE_REQUEST(GetScopeNotificationSettingsRequest, MessagesManager::get_notification_settings_scope(request.scope_)); } void Td::on_request(uint64 id, const td_api::getChatReportSpamState &request) { CHECK_IS_USER(); CREATE_REQUEST(GetChatReportSpamStateRequest, request.chat_id_); } void Td::on_request(uint64 id, const td_api::changeChatReportSpamState &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->change_dialog_report_spam_state(DialogId(request.chat_id_), request.is_spam_chat_, std::move(promise)); } void Td::on_request(uint64 id, td_api::reportChat &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); messages_manager_->report_dialog(DialogId(request.chat_id_), request.reason_, MessagesManager::get_message_ids(request.message_ids_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setChatNotificationSettings &request) { CHECK_IS_USER(); answer_ok_query(id, messages_manager_->set_dialog_notification_settings(DialogId(request.chat_id_), std::move(request.notification_settings_))); } void Td::on_request(uint64 id, td_api::setScopeNotificationSettings &request) { CHECK_IS_USER(); if (request.scope_ == nullptr) { return send_error_raw(id, 400, "Scope must not be empty"); } answer_ok_query(id, messages_manager_->set_scope_notification_settings( MessagesManager::get_notification_settings_scope(request.scope_), std::move(request.notification_settings_))); } void Td::on_request(uint64 id, const td_api::resetAllNotificationSettings &request) { CHECK_IS_USER(); messages_manager_->reset_all_notification_settings(); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, const td_api::getMapThumbnailFile &request) { DialogId dialog_id(request.chat_id_); if (!messages_manager_->have_dialog_force(dialog_id)) { dialog_id = DialogId(); } auto r_file_id = file_manager_->get_map_thumbnail_file_id(Location(request.location_), request.zoom_, request.width_, request.height_, request.scale_, dialog_id); if (r_file_id.is_error()) { send_closure(actor_id(this), &Td::send_error, id, r_file_id.move_as_error()); } else { send_closure(actor_id(this), &Td::send_result, id, file_manager_->get_file_object(r_file_id.ok())); } } void Td::on_request(uint64 id, const td_api::getLanguagePackInfo &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(language_pack_manager_, &LanguagePackManager::get_languages, std::move(promise)); } void Td::on_request(uint64 id, td_api::getLanguagePackStrings &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.language_code_); for (auto &key : request.keys_) { CLEAN_INPUT_STRING(key); } CREATE_REQUEST_PROMISE(); send_closure(language_pack_manager_, &LanguagePackManager::get_language_pack_strings, std::move(request.language_code_), std::move(request.keys_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setCustomLanguage &request) { CHECK_IS_USER(); if (request.info_ == nullptr) { return send_error_raw(id, 400, "Language info must not be empty"); } CLEAN_INPUT_STRING(request.info_->code_); CLEAN_INPUT_STRING(request.info_->name_); CLEAN_INPUT_STRING(request.info_->native_name_); CREATE_OK_REQUEST_PROMISE(); send_closure(language_pack_manager_, &LanguagePackManager::set_custom_language, std::move(request.info_->code_), std::move(request.info_->name_), std::move(request.info_->native_name_), std::move(request.strings_), std::move(promise)); } void Td::on_request(uint64 id, td_api::deleteLanguage &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.language_code_); CREATE_OK_REQUEST_PROMISE(); send_closure(language_pack_manager_, &LanguagePackManager::delete_language, std::move(request.language_code_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getOption &request) { CLEAN_INPUT_STRING(request.name_); tl_object_ptr option_value; switch (request.name_[0]) { case 'o': if (request.name_ == "online") { option_value = make_tl_object(is_online_); } break; case 'v': if (request.name_ == "version") { option_value = make_tl_object(TDLIB_VERSION); } break; } if (option_value == nullptr) { option_value = G()->shared_config().get_option_value(request.name_); } send_closure(actor_id(this), &Td::send_result, id, std::move(option_value)); } void Td::on_request(uint64 id, td_api::setOption &request) { CLEAN_INPUT_STRING(request.name_); int32 value_constructor_id = request.value_ == nullptr ? td_api::optionValueEmpty::ID : request.value_->get_id(); auto set_integer_option = [&](Slice name, int32 min = 0, int32 max = std::numeric_limits::max()) { if (request.name_ == name) { if (value_constructor_id != td_api::optionValueInteger::ID && value_constructor_id != td_api::optionValueEmpty::ID) { send_error_raw(id, 3, PSLICE() << "Option \"" << name << "\" must have integer value"); return true; } if (value_constructor_id == td_api::optionValueEmpty::ID) { G()->shared_config().set_option_empty(name); } else { int32 value = static_cast(request.value_.get())->value_; if (value < min || value > max) { send_error_raw(id, 3, PSLICE() << "Option's \"" << name << "\" value " << value << " is outside of a valid range [" << min << ", " << max << "]"); return true; } G()->shared_config().set_option_integer(name, clamp(value, min, max)); } send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); return true; } return false; }; auto set_boolean_option = [&](Slice name) { if (request.name_ == name) { if (value_constructor_id != td_api::optionValueBoolean::ID && value_constructor_id != td_api::optionValueEmpty::ID) { send_error_raw(id, 3, PSLICE() << "Option \"" << name << "\" must have boolean value"); return true; } if (value_constructor_id == td_api::optionValueEmpty::ID) { G()->shared_config().set_option_empty(name); } else { bool value = static_cast(request.value_.get())->value_; G()->shared_config().set_option_boolean(name, value); } send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); return true; } return false; }; auto set_string_option = [&](Slice name, auto check_value) { if (request.name_ == name) { if (value_constructor_id != td_api::optionValueString::ID && value_constructor_id != td_api::optionValueEmpty::ID) { send_error_raw(id, 3, PSLICE() << "Option \"" << name << "\" must have string value"); return true; } if (value_constructor_id == td_api::optionValueEmpty::ID) { G()->shared_config().set_option_empty(name); } else { const string &value = static_cast(request.value_.get())->value_; if (value.empty()) { G()->shared_config().set_option_empty(name); } else { if (check_value(value)) { G()->shared_config().set_option_string(name, value); } else { send_error_raw(id, 3, PSLICE() << "Option \"" << name << "\" can't have specified value"); return true; } } } send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); return true; } return false; }; switch (request.name_[0]) { case 'd': if (set_boolean_option("disable_contact_registered_notifications")) { return; } if (set_boolean_option("disable_top_chats")) { return; } break; case 'i': if (set_boolean_option("ignore_inline_thumbnails")) { return; } if (set_boolean_option("is_emulator")) { return; } break; case 'l': if (set_string_option("language_database_path", [](Slice value) { return true; })) { return; } if (set_string_option("language_pack", LanguagePackManager::check_language_pack_name)) { return; } if (set_string_option("language_code", LanguagePackManager::check_language_code_name)) { return; } break; case 'o': if (request.name_ == "online") { if (value_constructor_id != td_api::optionValueBoolean::ID && value_constructor_id != td_api::optionValueEmpty::ID) { return send_error_raw(id, 3, "Option \"online\" must have boolean value"); } bool is_online = value_constructor_id == td_api::optionValueEmpty::ID || static_cast(request.value_.get())->value_; if (!auth_manager_->is_bot()) { send_closure(G()->state_manager(), &StateManager::on_online, is_online); } if (is_online != is_online_) { is_online_ = is_online; on_online_updated(true, true); } return send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } break; case 'p': if (set_boolean_option("prefer_ipv6")) { send_closure(state_manager_, &StateManager::on_network_updated); return; } break; case 's': if (set_integer_option("session_count", 0, 50)) { return; } if (set_integer_option("storage_max_files_size")) { return; } if (set_integer_option("storage_max_time_from_last_access")) { return; } if (set_integer_option("storage_max_file_count")) { return; } if (set_integer_option("storage_immunity_delay")) { return; } break; case 'X': case 'x': { if (request.name_.size() > 255) { return send_error_raw(id, 3, "Option name is too long"); } switch (value_constructor_id) { case td_api::optionValueBoolean::ID: G()->shared_config().set_option_boolean( request.name_, static_cast(request.value_.get())->value_); break; case td_api::optionValueEmpty::ID: G()->shared_config().set_option_empty(request.name_); break; case td_api::optionValueInteger::ID: G()->shared_config().set_option_integer( request.name_, static_cast(request.value_.get())->value_); break; case td_api::optionValueString::ID: G()->shared_config().set_option_string( request.name_, static_cast(request.value_.get())->value_); break; default: UNREACHABLE(); } return send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } case 'u': if (set_boolean_option("use_pfs")) { return; } if (set_boolean_option("use_quick_ack")) { return; } if (set_boolean_option("use_storage_optimizer")) { return; } break; } return send_error_raw(id, 3, "Option can't be set"); } void Td::on_request(uint64 id, td_api::getInlineQueryResults &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.query_); CLEAN_INPUT_STRING(request.offset_); CREATE_REQUEST(GetInlineQueryResultsRequest, request.bot_user_id_, request.chat_id_, request.user_location_, std::move(request.query_), std::move(request.offset_)); } void Td::on_request(uint64 id, td_api::answerInlineQuery &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.next_offset_); CLEAN_INPUT_STRING(request.switch_pm_text_); CLEAN_INPUT_STRING(request.switch_pm_parameter_); CREATE_OK_REQUEST_PROMISE(); inline_queries_manager_->answer_inline_query( request.inline_query_id_, request.is_personal_, std::move(request.results_), request.cache_time_, request.next_offset_, request.switch_pm_text_, request.switch_pm_parameter_, std::move(promise)); } void Td::on_request(uint64 id, td_api::getCallbackQueryAnswer &request) { CHECK_IS_USER(); CREATE_REQUEST(GetCallbackQueryAnswerRequest, request.chat_id_, request.message_id_, std::move(request.payload_)); } void Td::on_request(uint64 id, td_api::answerCallbackQuery &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.text_); CLEAN_INPUT_STRING(request.url_); CREATE_OK_REQUEST_PROMISE(); callback_queries_manager_->answer_callback_query(request.callback_query_id_, request.text_, request.show_alert_, request.url_, request.cache_time_, std::move(promise)); } void Td::on_request(uint64 id, td_api::answerShippingQuery &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.error_message_); CREATE_OK_REQUEST_PROMISE(); answer_shipping_query(request.shipping_query_id_, std::move(request.shipping_options_), request.error_message_, std::move(promise)); } void Td::on_request(uint64 id, td_api::answerPreCheckoutQuery &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.error_message_); CREATE_OK_REQUEST_PROMISE(); answer_pre_checkout_query(request.pre_checkout_query_id_, request.error_message_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getPaymentForm &request) { CHECK_IS_USER(); CREATE_REQUEST(GetPaymentFormRequest, request.chat_id_, request.message_id_); } void Td::on_request(uint64 id, td_api::validateOrderInfo &request) { CHECK_IS_USER(); CREATE_REQUEST(ValidateOrderInfoRequest, request.chat_id_, request.message_id_, std::move(request.order_info_), request.allow_save_); } void Td::on_request(uint64 id, td_api::sendPaymentForm &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.order_info_id_); CLEAN_INPUT_STRING(request.shipping_option_id_); if (request.credentials_ == nullptr) { return send_error_raw(id, 400, "Input payments credentials must not be empty"); } CREATE_REQUEST(SendPaymentFormRequest, request.chat_id_, request.message_id_, std::move(request.order_info_id_), std::move(request.shipping_option_id_), std::move(request.credentials_)); } void Td::on_request(uint64 id, const td_api::getPaymentReceipt &request) { CHECK_IS_USER(); CREATE_REQUEST(GetPaymentReceiptRequest, request.chat_id_, request.message_id_); } void Td::on_request(uint64 id, const td_api::getSavedOrderInfo &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetSavedOrderInfoRequest); } void Td::on_request(uint64 id, const td_api::deleteSavedOrderInfo &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); delete_saved_order_info(std::move(promise)); } void Td::on_request(uint64 id, const td_api::deleteSavedCredentials &request) { CHECK_IS_USER(); CREATE_OK_REQUEST_PROMISE(); delete_saved_credentials(std::move(promise)); } void Td::on_request(uint64 id, td_api::getPassportElement &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); if (request.type_ == nullptr) { return send_error_raw(id, 400, "Type must not be empty"); } CREATE_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::get_secure_value, std::move(request.password_), get_secure_value_type_td_api(request.type_), std::move(promise)); } void Td::on_request(uint64 id, td_api::getAllPassportElements &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CREATE_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::get_all_secure_values, std::move(request.password_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setPassportElement &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CREATE_REQUEST_PROMISE(); auto r_secure_value = get_secure_value(file_manager_.get(), std::move(request.element_)); if (r_secure_value.is_error()) { return promise.set_error(r_secure_value.move_as_error()); } send_closure(secure_manager_, &SecureManager::set_secure_value, std::move(request.password_), r_secure_value.move_as_ok(), std::move(promise)); } void Td::on_request(uint64 id, const td_api::deletePassportElement &request) { CHECK_IS_USER(); if (request.type_ == nullptr) { return send_error_raw(id, 400, "Type must not be empty"); } CREATE_OK_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::delete_secure_value, get_secure_value_type_td_api(request.type_), std::move(promise)); } void Td::on_request(uint64 id, td_api::setPassportElementErrors &request) { CHECK_IS_BOT(); UserId user_id(request.user_id_); auto input_user = contacts_manager_->get_input_user(user_id); if (input_user == nullptr) { return send_error_raw(id, 400, "User not found"); } CREATE_OK_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::set_secure_value_errors, this, std::move(input_user), std::move(request.errors_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendPhoneNumberVerificationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.phone_number_); send_closure(verify_phone_number_manager_, &PhoneNumberManager::set_phone_number, id, std::move(request.phone_number_), request.allow_flash_call_, request.is_current_phone_number_); } void Td::on_request(uint64 id, const td_api::resendPhoneNumberVerificationCode &request) { CHECK_IS_USER(); send_closure(verify_phone_number_manager_, &PhoneNumberManager::resend_authentication_code, id); } void Td::on_request(uint64 id, td_api::checkPhoneNumberVerificationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.code_); send_closure(verify_phone_number_manager_, &PhoneNumberManager::check_code, id, std::move(request.code_)); } void Td::on_request(uint64 id, td_api::sendEmailAddressVerificationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.email_address_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::send_email_address_verification_code, request.email_address_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::resendEmailAddressVerificationCode &request) { CHECK_IS_USER(); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::resend_email_address_verification_code, std::move(promise)); } void Td::on_request(uint64 id, td_api::checkEmailAddressVerificationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.code_); CREATE_REQUEST_PROMISE(); send_closure(password_manager_, &PasswordManager::check_email_address_verification_code, request.code_, std::move(promise)); } void Td::on_request(uint64 id, td_api::getPassportAuthorizationForm &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.password_); CLEAN_INPUT_STRING(request.public_key_); CLEAN_INPUT_STRING(request.scope_); CLEAN_INPUT_STRING(request.payload_); UserId bot_user_id(request.bot_user_id_); if (!bot_user_id.is_valid()) { return send_error_raw(id, 400, "Bot user identifier invalid"); } if (request.payload_.empty()) { return send_error_raw(id, 400, "Payload must be non-empty"); } CREATE_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::get_passport_authorization_form, std::move(request.password_), bot_user_id, std::move(request.scope_), std::move(request.public_key_), std::move(request.payload_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendPassportAuthorizationForm &request) { CHECK_IS_USER(); for (auto &type : request.types_) { if (type == nullptr) { return send_error_raw(id, 400, "Type must not be empty"); } } CREATE_OK_REQUEST_PROMISE(); send_closure(secure_manager_, &SecureManager::send_passport_authorization_form, request.autorization_form_id_, get_secure_value_types_td_api(request.types_), std::move(promise)); } void Td::on_request(uint64 id, td_api::sendPhoneNumberConfirmationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.phone_number_); CLEAN_INPUT_STRING(request.hash_); send_closure(confirm_phone_number_manager_, &PhoneNumberManager::set_phone_number_and_hash, id, std::move(request.hash_), std::move(request.phone_number_), request.allow_flash_call_, request.is_current_phone_number_); } void Td::on_request(uint64 id, const td_api::resendPhoneNumberConfirmationCode &request) { CHECK_IS_USER(); send_closure(confirm_phone_number_manager_, &PhoneNumberManager::resend_authentication_code, id); } void Td::on_request(uint64 id, td_api::checkPhoneNumberConfirmationCode &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.code_); send_closure(confirm_phone_number_manager_, &PhoneNumberManager::check_code, id, std::move(request.code_)); } void Td::on_request(uint64 id, const td_api::getSupportUser &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetSupportUserRequest); } void Td::on_request(uint64 id, const td_api::getWallpapers &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetWallpapersRequest); } void Td::on_request(uint64 id, td_api::getRecentlyVisitedTMeUrls &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.referrer_); CREATE_REQUEST(GetRecentlyVisitedTMeUrlsRequest, std::move(request.referrer_)); } void Td::on_request(uint64 id, td_api::setBotUpdatesStatus &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.error_message_); create_handler()->send(request.pending_update_count_, request.error_message_); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::sendCustomRequest &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.method_); CLEAN_INPUT_STRING(request.parameters_); CREATE_REQUEST(SendCustomRequestRequest, std::move(request.method_), std::move(request.parameters_)); } void Td::on_request(uint64 id, td_api::answerCustomQuery &request) { CHECK_IS_BOT(); CLEAN_INPUT_STRING(request.data_); CREATE_OK_REQUEST_PROMISE(); create_handler(std::move(promise))->send(request.custom_query_id_, request.data_); } void Td::on_request(uint64 id, const td_api::setAlarm &request) { if (request.seconds_ < 0 || request.seconds_ > 3e9) { return send_error_raw(id, 400, "Wrong parameter seconds specified"); } int64 alarm_id = alarm_id_++; pending_alarms_.emplace(alarm_id, id); alarm_timeout_.set_timeout_in(alarm_id, request.seconds_); } void Td::on_request(uint64 id, td_api::searchHashtags &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.prefix_); CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result> result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(make_tl_object(result.move_as_ok())); } }); send_closure(hashtag_hints_, &HashtagHints::query, std::move(request.prefix_), request.limit_, std::move(query_promise)); } void Td::on_request(uint64 id, td_api::removeRecentHashtag &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.hashtag_); CREATE_OK_REQUEST_PROMISE(); send_closure(hashtag_hints_, &HashtagHints::remove_hashtag, std::move(request.hashtag_), std::move(promise)); } void Td::on_request(uint64 id, td_api::acceptTermsOfService &request) { CHECK_IS_USER(); CLEAN_INPUT_STRING(request.terms_of_service_id_); auto promise = PromiseCreator::lambda([id = id, actor_id = actor_id(this)](Result<> result) mutable { if (result.is_error()) { send_closure(actor_id, &Td::send_error, id, result.move_as_error()); } else { send_closure(actor_id, &Td::send_result, id, td_api::make_object()); send_closure(actor_id, &Td::schedule_get_terms_of_service, 0); } }); accept_terms_of_service(this, std::move(request.terms_of_service_id_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::getCountryCode &request) { CREATE_NO_ARGS_REQUEST(GetCountryCodeRequest); } void Td::on_request(uint64 id, const td_api::getInviteText &request) { CHECK_IS_USER(); CREATE_NO_ARGS_REQUEST(GetInviteTextRequest); } void Td::on_request(uint64 id, td_api::getDeepLinkInfo &request) { CLEAN_INPUT_STRING(request.link_); CREATE_REQUEST_PROMISE(); create_handler(std::move(promise))->send(request.link_); } void Td::on_request(uint64 id, td_api::addProxy &request) { CLEAN_INPUT_STRING(request.server_); CREATE_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::add_proxy, -1, std::move(request.server_), request.port_, request.enable_, std::move(request.type_), std::move(promise)); } void Td::on_request(uint64 id, td_api::editProxy &request) { if (request.proxy_id_ < 0) { return send_error_raw(id, 400, "Proxy identifier invalid"); } CLEAN_INPUT_STRING(request.server_); CREATE_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::add_proxy, request.proxy_id_, std::move(request.server_), request.port_, request.enable_, std::move(request.type_), std::move(promise)); } void Td::on_request(uint64 id, const td_api::enableProxy &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::enable_proxy, request.proxy_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::disableProxy &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::disable_proxy, std::move(promise)); } void Td::on_request(uint64 id, const td_api::removeProxy &request) { CREATE_OK_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::remove_proxy, request.proxy_id_, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getProxies &request) { CREATE_REQUEST_PROMISE(); send_closure(G()->connection_creator(), &ConnectionCreator::get_proxies, std::move(promise)); } void Td::on_request(uint64 id, const td_api::getProxyLink &request) { CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(make_tl_object(result.move_as_ok())); } }); send_closure(G()->connection_creator(), &ConnectionCreator::get_proxy_link, request.proxy_id_, std::move(query_promise)); } void Td::on_request(uint64 id, const td_api::pingProxy &request) { CREATE_REQUEST_PROMISE(); auto query_promise = PromiseCreator::lambda([promise = std::move(promise)](Result result) mutable { if (result.is_error()) { promise.set_error(result.move_as_error()); } else { promise.set_value(make_tl_object(result.move_as_ok())); } }); send_closure(G()->connection_creator(), &ConnectionCreator::ping_proxy, request.proxy_id_, std::move(query_promise)); } void Td::on_request(uint64 id, const td_api::getTextEntities &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } void Td::on_request(uint64 id, td_api::parseTextEntities &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } void Td::on_request(uint64 id, const td_api::getFileMimeType &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } void Td::on_request(uint64 id, const td_api::getFileExtension &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } void Td::on_request(uint64 id, const td_api::cleanFileName &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } void Td::on_request(uint64 id, const td_api::getLanguagePackString &request) { send_closure(actor_id(this), &Td::send_result, id, do_static_request(request)); } template td_api::object_ptr Td::do_static_request(const T &request) { return make_error(400, "Function can't be executed synchronously"); } td_api::object_ptr Td::do_static_request(const td_api::getTextEntities &request) { if (!check_utf8(request.text_)) { return make_error(400, "Text must be encoded in UTF-8"); } auto text_entities = find_entities(request.text_, false); return make_tl_object(get_text_entities_object(text_entities)); } td_api::object_ptr Td::do_static_request(td_api::parseTextEntities &request) { if (!check_utf8(request.text_)) { return make_error(400, "Text must be encoded in UTF-8"); } if (request.parse_mode_ == nullptr) { return make_error(400, "Parse mode must be non-empty"); } auto r_entities = [&]() -> Result> { switch (request.parse_mode_->get_id()) { case td_api::textParseModeHTML::ID: return parse_html(request.text_); case td_api::textParseModeMarkdown::ID: return parse_markdown(request.text_); default: UNREACHABLE(); return Status::Error(500, "Unknown parse mode"); } }(); if (r_entities.is_error()) { return make_error(400, PSLICE() << "Can't parse entities: " << r_entities.error().message()); } return make_tl_object(std::move(request.text_), get_text_entities_object(r_entities.ok())); } td_api::object_ptr Td::do_static_request(const td_api::getFileMimeType &request) { // don't check file name UTF-8 correctness return make_tl_object(MimeType::from_extension(PathView(request.file_name_).extension())); } td_api::object_ptr Td::do_static_request(const td_api::getFileExtension &request) { // don't check MIME type UTF-8 correctness return make_tl_object(MimeType::to_extension(request.mime_type_)); } td_api::object_ptr Td::do_static_request(const td_api::cleanFileName &request) { // don't check file name UTF-8 correctness return make_tl_object(clean_filename(request.file_name_)); } td_api::object_ptr Td::do_static_request(const td_api::getLanguagePackString &request) { return LanguagePackManager::get_language_pack_string(request.language_database_path_, request.language_pack_, request.language_code_, request.key_); } // test void Td::on_request(uint64 id, td_api::testNetwork &request) { create_handler(id)->send(); } void Td::on_request(uint64 id, td_api::testGetDifference &request) { updates_manager_->get_difference("testGetDifference"); send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::testUseUpdate &request) { send_closure(actor_id(this), &Td::send_result, id, nullptr); } void Td::on_request(uint64 id, td_api::testUseError &request) { send_closure(actor_id(this), &Td::send_result, id, nullptr); } void Td::on_request(uint64 id, td_api::testCallEmpty &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object()); } void Td::on_request(uint64 id, td_api::testSquareInt &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(request.x_ * request.x_)); } void Td::on_request(uint64 id, td_api::testCallString &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } void Td::on_request(uint64 id, td_api::testCallBytes &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } void Td::on_request(uint64 id, td_api::testCallVectorInt &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } void Td::on_request(uint64 id, td_api::testCallVectorIntObject &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } void Td::on_request(uint64 id, td_api::testCallVectorString &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } void Td::on_request(uint64 id, td_api::testCallVectorStringObject &request) { send_closure(actor_id(this), &Td::send_result, id, make_tl_object(std::move(request.x_))); } #undef CLEAN_INPUT_STRING #undef CHECK_IS_BOT #undef CHECK_IS_USER #undef CREATE_NO_ARGS_REQUEST #undef CREATE_REQUEST #undef CREATE_REQUEST_PROMISE #undef CREATE_OK_REQUEST_PROMISE constexpr const char *Td::TDLIB_VERSION; } // namespace td