// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2022 // // 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/MessageReaction.h" #include "td/telegram/ContactsManager.h" #include "td/telegram/MessagesManager.h" #include "td/telegram/Td.h" #include "td/utils/algorithm.h" #include "td/utils/buffer.h" #include "td/utils/logging.h" #include namespace td { class GetMessageReactionsListQuery final : public Td::ResultHandler { Promise> promise_; DialogId dialog_id_; MessageId message_id_; string reaction_; string offset_; public: explicit GetMessageReactionsListQuery(Promise> &&promise) : promise_(std::move(promise)) { } void send(FullMessageId full_message_id, string reaction, string offset, int32 limit) { dialog_id_ = full_message_id.get_dialog_id(); message_id_ = full_message_id.get_message_id(); reaction_ = std::move(reaction); offset_ = std::move(offset); auto input_peer = td_->messages_manager_->get_input_peer(dialog_id_, AccessRights::Read); if (input_peer == nullptr) { return on_error(Status::Error(400, "Can't access the chat")); } int32 flags = 0; if (!reaction_.empty()) { flags |= telegram_api::messages_getMessageReactionsList::REACTION_MASK; } if (!offset_.empty()) { flags |= telegram_api::messages_getMessageReactionsList::OFFSET_MASK; } send_query(G()->net_query_creator().create(telegram_api::messages_getMessageReactionsList( flags, std::move(input_peer), message_id_.get_server_message_id().get(), reaction_, offset_, limit))); } void on_result(BufferSlice packet) final { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(result_ptr.move_as_error()); } auto ptr = result_ptr.move_as_ok(); LOG(INFO) << "Receive result for GetMessageReactionsListQuery: " << to_string(ptr); td_->contacts_manager_->on_get_users(std::move(ptr->users_), "GetMessageReactionsListQuery"); // td_->contacts_manager_->on_get_chats(std::move(ptr->chats_), "GetMessageReactionsListQuery"); int32 total_count = ptr->count_; if (total_count < static_cast(ptr->reactions_.size())) { LOG(ERROR) << "Receive invalid total_count in " << to_string(ptr); total_count = static_cast(ptr->reactions_.size()); } vector> reactions; for (auto &reaction : ptr->reactions_) { UserId user_id(reaction->user_id_); if (!user_id.is_valid() || (!reaction_.empty() && reaction_ != reaction->reaction_)) { LOG(ERROR) << "Receive unexpected " << to_string(reaction); continue; } reactions.push_back(td_api::make_object( reaction->reaction_, td_api::make_object( td_->contacts_manager_->get_user_id_object(user_id, "GetMessageReactionsListQuery")))); } promise_.set_value( td_api::make_object(total_count, std::move(reactions), ptr->next_offset_)); } void on_error(Status status) final { td_->messages_manager_->on_get_dialog_error(dialog_id_, status, "GetMessageReactionsListQuery"); promise_.set_error(std::move(status)); } }; td_api::object_ptr MessageReaction::get_message_reaction_object(Td *td) const { CHECK(!is_empty()); vector> recent_choosers; for (auto dialog_id : recent_chooser_dialog_ids_) { auto dialog_type = dialog_id.get_type(); if (dialog_type == DialogType::User) { auto user_id = dialog_id.get_user_id(); if (td->contacts_manager_->have_min_user(user_id)) { recent_choosers.push_back(td_api::make_object( td->contacts_manager_->get_user_id_object(user_id, "get_message_reaction_object"))); } else { LOG(ERROR) << "Skip unknown reacted " << user_id; } } else { if (!td->messages_manager_->have_dialog(dialog_id) && (td->messages_manager_->have_dialog_info(dialog_id) || (dialog_type == DialogType::Channel && td->contacts_manager_->have_min_channel(dialog_id.get_channel_id())))) { LOG(INFO) << "Force creation of " << dialog_id; td->messages_manager_->force_create_dialog(dialog_id, "get_message_reaction_object", true); } if (td->messages_manager_->have_dialog(dialog_id)) { recent_choosers.push_back(td_api::make_object(dialog_id.get())); } else { LOG(ERROR) << "Skip unknown reacted " << dialog_id; } } } return td_api::make_object(reaction_, choose_count_, is_chosen_, std::move(recent_choosers)); } bool operator==(const MessageReaction &lhs, const MessageReaction &rhs) { return lhs.reaction_ == rhs.reaction_ && lhs.choose_count_ == rhs.choose_count_ && lhs.is_chosen_ == rhs.is_chosen_ && lhs.recent_chooser_dialog_ids_ == rhs.recent_chooser_dialog_ids_; } StringBuilder &operator<<(StringBuilder &string_builder, const MessageReaction &reaction) { string_builder << '[' << reaction.reaction_ << (reaction.is_chosen_ ? " X " : " x ") << reaction.choose_count_; if (!reaction.recent_chooser_dialog_ids_.empty()) { string_builder << " by " << reaction.recent_chooser_dialog_ids_; } return string_builder << ']'; } unique_ptr MessageReactions::get_message_reactions( Td *td, tl_object_ptr &&reactions, bool is_bot) { if (reactions == nullptr || is_bot) { return nullptr; } auto result = make_unique(); result->can_see_all_choosers_ = reactions->can_see_list_; result->is_min_ = reactions->min_; std::unordered_set reaction_strings; std::unordered_set recent_choosers; for (auto &reaction_count : reactions->results_) { if (reaction_count->count_ <= 0 || reaction_count->count_ >= MessageReaction::MAX_CHOOSE_COUNT) { LOG(ERROR) << "Receive reaction " << reaction_count->reaction_ << " with invalid count " << reaction_count->count_; continue; } if (!reaction_strings.insert(reaction_count->reaction_).second) { LOG(ERROR) << "Receive duplicate reaction " << reaction_count->reaction_; continue; } vector recent_chooser_dialog_ids; vector> recent_chooser_min_channels; for (auto &user_reaction : reactions->recent_reactons_) { if (user_reaction->reaction_ == reaction_count->reaction_) { DialogId dialog_id(UserId(user_reaction->user_id_)); if (!dialog_id.is_valid()) { LOG(ERROR) << "Receive invalid " << dialog_id << " as a recent chooser"; continue; } if (!recent_choosers.insert(dialog_id).second) { LOG(ERROR) << "Receive duplicate " << dialog_id << " as a recent chooser"; continue; } if (!td->messages_manager_->have_dialog_info(dialog_id)) { auto dialog_type = dialog_id.get_type(); if (dialog_type == DialogType::User) { auto user_id = dialog_id.get_user_id(); if (!td->contacts_manager_->have_min_user(user_id)) { LOG(ERROR) << "Have no info about " << user_id; continue; } } else if (dialog_type == DialogType::Channel) { auto channel_id = dialog_id.get_channel_id(); auto min_channel = td->contacts_manager_->get_min_channel(channel_id); if (min_channel == nullptr) { LOG(ERROR) << "Have no info about reacted " << channel_id; continue; } recent_chooser_min_channels.emplace_back(channel_id, *min_channel); } else { LOG(ERROR) << "Have no info about reacted " << dialog_id; continue; } } recent_chooser_dialog_ids.push_back(dialog_id); if (recent_chooser_dialog_ids.size() == MessageReaction::MAX_RECENT_CHOOSERS) { break; } } } result->reactions_.emplace_back(std::move(reaction_count->reaction_), reaction_count->count_, reaction_count->chosen_, std::move(recent_chooser_dialog_ids), std::move(recent_chooser_min_channels)); } return result; } void MessageReactions::update_from(const MessageReactions &old_reactions) { if (old_reactions.has_pending_reaction_) { // we will ignore all updates, received while there is a pending reaction, so there are no reasons to update return; } CHECK(!has_pending_reaction_); if (is_min_ && !old_reactions.is_min_) { // chosen reaction was known, keep it is_min_ = false; for (const auto &old_reaction : old_reactions.reactions_) { if (old_reaction.is_chosen()) { for (auto &reaction : reactions_) { if (reaction.get_reaction() == old_reaction.get_reaction()) { reaction.set_is_chosen(true); } } } } } } bool MessageReactions::need_update_message_reactions(const MessageReactions *old_reactions, const MessageReactions *new_reactions) { if (old_reactions == nullptr) { // add reactions return new_reactions != nullptr; } if (old_reactions->has_pending_reaction_) { // ignore all updates, received while there is a pending reaction return false; } if (new_reactions == nullptr) { // remove reactions when they are disabled return true; } // has_pending_reaction_ and old_chosen_reaction_ don't affect visible state // compare all other fields return old_reactions->reactions_ != new_reactions->reactions_ || old_reactions->is_min_ != new_reactions->is_min_ || old_reactions->can_see_all_choosers_ != new_reactions->can_see_all_choosers_ || old_reactions->need_polling_ != new_reactions->need_polling_; } void get_message_chosen_reactions(Td *td, FullMessageId full_message_id, string reaction, string offset, int32 limit, Promise> &&promise) { if (!td->messages_manager_->have_message_force(full_message_id, "get_message_chosen_reactions")) { return promise.set_error(Status::Error(400, "Message not found")); } auto message_id = full_message_id.get_message_id(); if (!message_id.is_valid() || !message_id.is_server()) { return promise.set_value(td_api::make_object(0, Auto(), string())); } if (limit <= 0) { return promise.set_error(Status::Error(400, "Parameter limit must be positive")); } static constexpr int32 MAX_GET_CHOSEN_REACTIONS = 100; // server side limit if (limit > MAX_GET_CHOSEN_REACTIONS) { limit = MAX_GET_CHOSEN_REACTIONS; } td->create_handler(std::move(promise)) ->send(full_message_id, std::move(reaction), std::move(offset), limit); } } // namespace td