// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024 // // 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/AccessRights.h" #include "td/telegram/ChatManager.h" #include "td/telegram/Dependencies.h" #include "td/telegram/DialogManager.h" #include "td/telegram/Global.h" #include "td/telegram/MessageSender.h" #include "td/telegram/MessagesManager.h" #include "td/telegram/ServerMessageId.h" #include "td/telegram/Td.h" #include "td/telegram/telegram_api.h" #include "td/telegram/UpdatesManager.h" #include "td/telegram/UserManager.h" #include "td/actor/actor.h" #include "td/actor/SleepActor.h" #include "td/utils/algorithm.h" #include "td/utils/buffer.h" #include "td/utils/FlatHashSet.h" #include "td/utils/logging.h" #include "td/utils/misc.h" #include "td/utils/Slice.h" #include "td/utils/Status.h" #include #include #include namespace td { static size_t get_max_reaction_count() { bool is_premium = G()->get_option_boolean("is_premium"); auto option_key = is_premium ? Slice("reactions_user_max_premium") : Slice("reactions_user_max_default"); return static_cast( max(static_cast(1), static_cast(G()->get_option_integer(option_key, is_premium ? 3 : 1)))); } class GetMessagesReactionsQuery final : public Td::ResultHandler { DialogId dialog_id_; vector message_ids_; public: void send(DialogId dialog_id, vector &&message_ids) { dialog_id_ = dialog_id; message_ids_ = std::move(message_ids); auto input_peer = td_->dialog_manager_->get_input_peer(dialog_id_, AccessRights::Read); CHECK(input_peer != nullptr); send_query( G()->net_query_creator().create(telegram_api::messages_getMessagesReactions( std::move(input_peer), MessageId::get_server_message_ids(message_ids_)), {{dialog_id_}})); } 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 GetMessagesReactionsQuery: " << to_string(ptr); if (ptr->get_id() == telegram_api::updates::ID) { auto &updates = static_cast(ptr.get())->updates_; FlatHashSet skipped_message_ids; for (auto message_id : message_ids_) { skipped_message_ids.insert(message_id); } for (const auto &update : updates) { if (update->get_id() == telegram_api::updateMessageReactions::ID) { auto update_message_reactions = static_cast(update.get()); if (DialogId(update_message_reactions->peer_) == dialog_id_) { skipped_message_ids.erase(MessageId(ServerMessageId(update_message_reactions->msg_id_))); } } } for (auto message_id : skipped_message_ids) { td_->messages_manager_->update_message_reactions({dialog_id_, message_id}, nullptr); } } td_->updates_manager_->on_get_updates(std::move(ptr), Promise()); td_->messages_manager_->try_reload_message_reactions(dialog_id_, true); } void on_error(Status status) final { td_->dialog_manager_->on_get_dialog_error(dialog_id_, status, "GetMessagesReactionsQuery"); td_->messages_manager_->try_reload_message_reactions(dialog_id_, true); } }; class SendReactionQuery final : public Td::ResultHandler { Promise promise_; DialogId dialog_id_; public: explicit SendReactionQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(MessageFullId message_full_id, vector reaction_types, bool is_big, bool add_to_recent) { dialog_id_ = message_full_id.get_dialog_id(); auto input_peer = td_->dialog_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_types.empty()) { flags |= telegram_api::messages_sendReaction::REACTION_MASK; if (is_big) { flags |= telegram_api::messages_sendReaction::BIG_MASK; } if (add_to_recent) { flags |= telegram_api::messages_sendReaction::ADD_TO_RECENT_MASK; } } send_query(G()->net_query_creator().create( telegram_api::messages_sendReaction(flags, false /*ignored*/, false /*ignored*/, std::move(input_peer), message_full_id.get_message_id().get_server_message_id().get(), ReactionType::get_input_reactions(reaction_types)), {{dialog_id_}, {message_full_id}})); } 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 SendReactionQuery: " << to_string(ptr); td_->updates_manager_->on_get_updates(std::move(ptr), std::move(promise_)); } void on_error(Status status) final { if (status.message() == "MESSAGE_NOT_MODIFIED") { return promise_.set_value(Unit()); } td_->dialog_manager_->on_get_dialog_error(dialog_id_, status, "SendReactionQuery"); promise_.set_error(std::move(status)); } }; class SendPaidReactionQuery final : public Td::ResultHandler { Promise promise_; DialogId dialog_id_; public: explicit SendPaidReactionQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(MessageFullId message_full_id, int32 star_count, int64 random_id) { dialog_id_ = message_full_id.get_dialog_id(); auto input_peer = td_->dialog_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; send_query(G()->net_query_creator().create( telegram_api::messages_sendPaidReaction(flags, false /*ignored*/, std::move(input_peer), message_full_id.get_message_id().get_server_message_id().get(), star_count, random_id), {{dialog_id_}, {message_full_id}})); } 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 SendPaidReactionQuery: " << to_string(ptr); td_->updates_manager_->on_get_updates(std::move(ptr), std::move(promise_)); } void on_error(Status status) final { if (status.message() == "MESSAGE_NOT_MODIFIED") { return promise_.set_value(Unit()); } td_->dialog_manager_->on_get_dialog_error(dialog_id_, status, "SendPaidReactionQuery"); promise_.set_error(std::move(status)); } }; class GetMessageReactionsListQuery final : public Td::ResultHandler { Promise> promise_; DialogId dialog_id_; MessageId message_id_; ReactionType reaction_type_; string offset_; public: explicit GetMessageReactionsListQuery(Promise> &&promise) : promise_(std::move(promise)) { } void send(MessageFullId message_full_id, ReactionType reaction_type, string offset, int32 limit) { dialog_id_ = message_full_id.get_dialog_id(); message_id_ = message_full_id.get_message_id(); reaction_type_ = std::move(reaction_type); offset_ = std::move(offset); auto input_peer = td_->dialog_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_type_.is_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_type_.get_input_reaction(), offset_, limit), {{message_full_id}})); } 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_->user_manager_->on_get_users(std::move(ptr->users_), "GetMessageReactionsListQuery"); td_->chat_manager_->on_get_chats(std::move(ptr->chats_), "GetMessageReactionsListQuery"); int32 total_count = ptr->count_; auto received_reaction_count = static_cast(ptr->reactions_.size()); if (total_count < received_reaction_count) { LOG(ERROR) << "Receive invalid total_count in " << to_string(ptr); total_count = received_reaction_count; } vector> reactions; FlatHashMap, ReactionTypeHash> recent_reaction_types; for (const auto &reaction : ptr->reactions_) { DialogId dialog_id(reaction->peer_id_); auto reaction_type = ReactionType(reaction->reaction_); if (!dialog_id.is_valid() || (reaction_type_.is_empty() ? reaction_type.is_empty() : reaction_type_ != reaction_type)) { LOG(ERROR) << "Receive unexpected " << to_string(reaction); continue; } if (offset_.empty()) { recent_reaction_types[reaction_type].push_back(dialog_id); } auto message_sender = get_min_message_sender_object(td_, dialog_id, "GetMessageReactionsListQuery"); if (message_sender != nullptr) { reactions.push_back(td_api::make_object( reaction_type.get_reaction_type_object(), std::move(message_sender), reaction->my_, reaction->date_)); } } if (offset_.empty()) { td_->messages_manager_->on_get_message_reaction_list({dialog_id_, message_id_}, reaction_type_, std::move(recent_reaction_types), total_count); } promise_.set_value( td_api::make_object(total_count, std::move(reactions), ptr->next_offset_)); } void on_error(Status status) final { td_->dialog_manager_->on_get_dialog_error(dialog_id_, status, "GetMessageReactionsListQuery"); promise_.set_error(std::move(status)); } }; class ReportReactionQuery final : public Td::ResultHandler { Promise promise_; DialogId dialog_id_; public: explicit ReportReactionQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(DialogId dialog_id, MessageId message_id, DialogId chooser_dialog_id) { dialog_id_ = dialog_id; auto input_peer = td_->dialog_manager_->get_input_peer(dialog_id_, AccessRights::Read); CHECK(input_peer != nullptr); auto chooser_input_peer = td_->dialog_manager_->get_input_peer(chooser_dialog_id, AccessRights::Know); if (chooser_input_peer == nullptr) { return promise_.set_error(Status::Error(400, "Reaction sender is not accessible")); } send_query(G()->net_query_creator().create(telegram_api::messages_reportReaction( std::move(input_peer), message_id.get_server_message_id().get(), std::move(chooser_input_peer)))); } 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()); } promise_.set_value(Unit()); } void on_error(Status status) final { td_->dialog_manager_->on_get_dialog_error(dialog_id_, status, "ReportReactionQuery"); promise_.set_error(std::move(status)); } }; MessageReaction::MessageReaction(ReactionType reaction_type, int32 choose_count, bool is_chosen, DialogId my_recent_chooser_dialog_id, vector &&recent_chooser_dialog_ids, vector> &&recent_chooser_min_channels) : reaction_type_(std::move(reaction_type)) , choose_count_(choose_count) , is_chosen_(is_chosen) , my_recent_chooser_dialog_id_(my_recent_chooser_dialog_id) , recent_chooser_dialog_ids_(std::move(recent_chooser_dialog_ids)) , recent_chooser_min_channels_(std::move(recent_chooser_min_channels)) { if (my_recent_chooser_dialog_id_.is_valid()) { CHECK(td::contains(recent_chooser_dialog_ids_, my_recent_chooser_dialog_id_)); } fix_choose_count(); } void MessageReaction::add_my_recent_chooser_dialog_id(DialogId dialog_id) { CHECK(!my_recent_chooser_dialog_id_.is_valid()); my_recent_chooser_dialog_id_ = dialog_id; add_to_top(recent_chooser_dialog_ids_, MAX_RECENT_CHOOSERS + 1, dialog_id); fix_choose_count(); } bool MessageReaction::remove_my_recent_chooser_dialog_id() { if (my_recent_chooser_dialog_id_.is_valid()) { bool is_removed = td::remove(recent_chooser_dialog_ids_, my_recent_chooser_dialog_id_); CHECK(is_removed); my_recent_chooser_dialog_id_ = DialogId(); return true; } return false; } void MessageReaction::update_from(const MessageReaction &old_reaction) { CHECK(old_reaction.is_chosen()); is_chosen_ = true; auto my_dialog_id = old_reaction.get_my_recent_chooser_dialog_id(); if (my_dialog_id.is_valid() && td::contains(recent_chooser_dialog_ids_, my_dialog_id)) { my_recent_chooser_dialog_id_ = my_dialog_id; } } void MessageReaction::update_recent_chooser_dialog_ids(const MessageReaction &old_reaction) { if (recent_chooser_dialog_ids_.size() != MAX_RECENT_CHOOSERS) { return; } CHECK(is_chosen_ && old_reaction.is_chosen_); CHECK(reaction_type_ == old_reaction.reaction_type_); CHECK(old_reaction.recent_chooser_dialog_ids_.size() == MAX_RECENT_CHOOSERS + 1); for (size_t i = 0; i < MAX_RECENT_CHOOSERS; i++) { if (recent_chooser_dialog_ids_[i] != old_reaction.recent_chooser_dialog_ids_[i]) { return; } } my_recent_chooser_dialog_id_ = old_reaction.my_recent_chooser_dialog_id_; recent_chooser_dialog_ids_ = old_reaction.recent_chooser_dialog_ids_; recent_chooser_min_channels_ = old_reaction.recent_chooser_min_channels_; fix_choose_count(); } void MessageReaction::set_as_chosen(DialogId my_dialog_id, bool have_recent_choosers) { CHECK(!is_chosen_); is_chosen_ = true; choose_count_++; if (have_recent_choosers) { remove_my_recent_chooser_dialog_id(); add_my_recent_chooser_dialog_id(my_dialog_id); } } void MessageReaction::unset_as_chosen() { CHECK(is_chosen_); is_chosen_ = false; choose_count_--; remove_my_recent_chooser_dialog_id(); fix_choose_count(); } void MessageReaction::add_paid_reaction(int32 star_count) { is_chosen_ = true; CHECK(star_count <= std::numeric_limits::max() - choose_count_); choose_count_ += star_count; } void MessageReaction::fix_choose_count() { choose_count_ = max(choose_count_, narrow_cast(recent_chooser_dialog_ids_.size())); } void MessageReaction::set_my_recent_chooser_dialog_id(DialogId my_dialog_id) { if (!my_recent_chooser_dialog_id_.is_valid() || my_recent_chooser_dialog_id_ == my_dialog_id) { return; } td::remove(recent_chooser_dialog_ids_, my_dialog_id); for (auto &dialog_id : recent_chooser_dialog_ids_) { if (dialog_id == my_recent_chooser_dialog_id_) { dialog_id = my_dialog_id; } } CHECK(td::contains(recent_chooser_dialog_ids_, my_dialog_id)); my_recent_chooser_dialog_id_ = my_dialog_id; } td_api::object_ptr MessageReaction::get_message_reaction_object(Td *td, UserId my_user_id, UserId peer_user_id) const { CHECK(!is_empty()); td_api::object_ptr used_sender; vector> recent_choosers; if (my_user_id.is_valid()) { CHECK(peer_user_id.is_valid()); if (is_chosen()) { auto recent_chooser = get_min_message_sender_object(td, DialogId(my_user_id), "get_message_reaction_object"); if (recent_chooser != nullptr) { used_sender = get_min_message_sender_object(td, DialogId(my_user_id), "get_message_reaction_object"); recent_choosers.push_back(std::move(recent_chooser)); } } if (choose_count_ >= (is_chosen() ? 2 : 1)) { auto recent_chooser = get_min_message_sender_object(td, DialogId(peer_user_id), "get_message_reaction_object"); if (recent_chooser != nullptr) { recent_choosers.push_back(std::move(recent_chooser)); } } } else { for (auto dialog_id : recent_chooser_dialog_ids_) { auto recent_chooser = get_min_message_sender_object(td, dialog_id, "get_message_reaction_object"); if (recent_chooser != nullptr) { if (is_chosen() && dialog_id == my_recent_chooser_dialog_id_) { used_sender = get_min_message_sender_object(td, dialog_id, "get_message_reaction_object"); } recent_choosers.push_back(std::move(recent_chooser)); if (recent_choosers.size() == MAX_RECENT_CHOOSERS) { break; } } } } return td_api::make_object(reaction_type_.get_reaction_type_object(), choose_count_, is_chosen_, std::move(used_sender), std::move(recent_choosers)); } bool operator==(const MessageReaction &lhs, const MessageReaction &rhs) { return lhs.reaction_type_ == rhs.reaction_type_ && lhs.choose_count_ == rhs.choose_count_ && lhs.is_chosen_ == rhs.is_chosen_ && lhs.my_recent_chooser_dialog_id_ == rhs.my_recent_chooser_dialog_id_ && lhs.recent_chooser_dialog_ids_ == rhs.recent_chooser_dialog_ids_; } StringBuilder &operator<<(StringBuilder &string_builder, const MessageReaction &reaction) { string_builder << '[' << reaction.reaction_type_ << (reaction.is_chosen_ ? " X " : " x ") << reaction.choose_count_; if (!reaction.recent_chooser_dialog_ids_.empty()) { string_builder << " by " << reaction.recent_chooser_dialog_ids_; if (reaction.my_recent_chooser_dialog_id_.is_valid()) { string_builder << " and my " << reaction.my_recent_chooser_dialog_id_; } } return string_builder << ']'; } td_api::object_ptr UnreadMessageReaction::get_unread_reaction_object(Td *td) const { auto sender_id = get_min_message_sender_object(td, sender_dialog_id_, "get_unread_reaction_object"); if (sender_id == nullptr) { return nullptr; } return td_api::make_object(reaction_type_.get_reaction_type_object(), std::move(sender_id), is_big_); } bool operator==(const UnreadMessageReaction &lhs, const UnreadMessageReaction &rhs) { return lhs.reaction_type_ == rhs.reaction_type_ && lhs.sender_dialog_id_ == rhs.sender_dialog_id_ && lhs.is_big_ == rhs.is_big_; } StringBuilder &operator<<(StringBuilder &string_builder, const UnreadMessageReaction &unread_reaction) { return string_builder << '[' << unread_reaction.reaction_type_ << (unread_reaction.is_big_ ? " BY " : " by ") << unread_reaction.sender_dialog_id_ << ']'; } unique_ptr MessageReactions::get_message_reactions( Td *td, telegram_api::object_ptr &&reactions, bool is_bot) { if (reactions == nullptr || is_bot) { return nullptr; } auto result = make_unique(); result->can_get_added_reactions_ = reactions->can_see_list_; result->is_min_ = reactions->min_; result->are_tags_ = reactions->reactions_as_tags_; DialogId my_dialog_id; for (auto &peer_reaction : reactions->recent_reactions_) { if (peer_reaction->my_) { DialogId dialog_id(peer_reaction->peer_id_); if (!dialog_id.is_valid()) { continue; } if (my_dialog_id.is_valid() && dialog_id != my_dialog_id) { LOG(ERROR) << "Receive my reactions with " << dialog_id << " and " << my_dialog_id; } my_dialog_id = dialog_id; } } FlatHashSet reaction_types; vector> chosen_reaction_order; for (auto &reaction_count : reactions->results_) { auto reaction_type = ReactionType(reaction_count->reaction_); if (reaction_count->count_ <= 0 || reaction_count->count_ >= MessageReaction::MAX_CHOOSE_COUNT || reaction_type.is_empty()) { LOG(ERROR) << "Receive " << reaction_type << " with invalid count " << reaction_count->count_; continue; } if (!reaction_types.insert(reaction_type).second) { LOG(ERROR) << "Receive duplicate " << reaction_type; continue; } FlatHashSet recent_choosers; DialogId my_recent_chooser_dialog_id; vector recent_chooser_dialog_ids; vector> recent_chooser_min_channels; for (auto &peer_reaction : reactions->recent_reactions_) { auto peer_reaction_type = ReactionType(peer_reaction->reaction_); if (peer_reaction_type == reaction_type) { DialogId dialog_id(peer_reaction->peer_id_); if (!dialog_id.is_valid()) { LOG(ERROR) << "Receive invalid " << dialog_id << " as a recent chooser for " << reaction_type; continue; } if (!recent_choosers.insert(dialog_id).second) { LOG(ERROR) << "Receive duplicate " << dialog_id << " as a recent chooser for " << reaction_type; continue; } if (!td->dialog_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->user_manager_->have_min_user(user_id)) { LOG(ERROR) << "Receive unknown " << user_id; continue; } } else if (dialog_type == DialogType::Channel) { auto channel_id = dialog_id.get_channel_id(); auto min_channel = td->chat_manager_->get_min_channel(channel_id); if (min_channel == nullptr) { LOG(ERROR) << "Receive unknown reacted " << channel_id; continue; } recent_chooser_min_channels.emplace_back(channel_id, *min_channel); } else { LOG(ERROR) << "Receive unknown reacted " << dialog_id; continue; } } recent_chooser_dialog_ids.push_back(dialog_id); if (dialog_id == my_dialog_id) { my_recent_chooser_dialog_id = dialog_id; } if (peer_reaction->unread_) { result->unread_reactions_.emplace_back(std::move(peer_reaction_type), dialog_id, peer_reaction->big_); } if (recent_chooser_dialog_ids.size() == MessageReaction::MAX_RECENT_CHOOSERS) { break; } } } bool is_chosen = (reaction_count->flags_ & telegram_api::reactionCount::CHOSEN_ORDER_MASK) != 0; if (is_chosen) { if (reaction_type == ReactionType::paid()) { LOG_IF(ERROR, reaction_count->chosen_order_ != -1) << "Receive paid reaction with order " << reaction_count->chosen_order_; } else { chosen_reaction_order.emplace_back(reaction_count->chosen_order_, reaction_type); } } result->reactions_.push_back({std::move(reaction_type), reaction_count->count_, is_chosen, my_recent_chooser_dialog_id, std::move(recent_chooser_dialog_ids), std::move(recent_chooser_min_channels)}); } if (chosen_reaction_order.size() > 1) { std::sort(chosen_reaction_order.begin(), chosen_reaction_order.end()); result->chosen_reaction_order_ = transform(chosen_reaction_order, [](const std::pair &order) { return order.second; }); } bool was_me = false; for (auto &top_reactor : reactions->top_reactors_) { MessageReactor reactor(std::move(top_reactor)); if (!reactor.is_valid() || (reactions->min_ && reactor.is_me())) { LOG(ERROR) << "Receive " << reactor; continue; } if (reactor.is_me()) { if (was_me) { LOG(ERROR) << "Receive duplicate " << reactor; continue; } was_me = true; } result->top_reactors_.push_back(std::move(reactor)); } MessageReactor::fix_message_reactors(result->top_reactors_, true); return result; } MessageReaction *MessageReactions::get_reaction(const ReactionType &reaction_type) { for (auto &added_reaction : reactions_) { if (added_reaction.get_reaction_type() == reaction_type) { return &added_reaction; } } return nullptr; } const MessageReaction *MessageReactions::get_reaction(const ReactionType &reaction_type) const { for (const auto &added_reaction : reactions_) { if (added_reaction.get_reaction_type() == reaction_type) { return &added_reaction; } } return nullptr; } void MessageReactions::update_from(const MessageReactions &old_reactions) { if (is_min_ && !old_reactions.is_min_) { // chosen reactions were known, keep them is_min_ = false; chosen_reaction_order_ = old_reactions.chosen_reaction_order_; for (const auto &old_reaction : old_reactions.reactions_) { if (old_reaction.is_chosen()) { auto *reaction = get_reaction(old_reaction.get_reaction_type()); if (reaction != nullptr) { reaction->update_from(old_reaction); } } else { td::remove(chosen_reaction_order_, old_reaction.get_reaction_type()); } } unread_reactions_ = old_reactions.unread_reactions_; if (chosen_reaction_order_.size() == 1) { reset_to_empty(chosen_reaction_order_); } // self paid reaction was known, keep it for (auto &reactor : old_reactions.top_reactors_) { if (reactor.is_me()) { // top_reactors_.push_back(reactor); } } MessageReactor::fix_message_reactors(top_reactors_, false); } for (const auto &old_reaction : old_reactions.reactions_) { if (old_reaction.is_chosen() && old_reaction.get_recent_chooser_dialog_ids().size() == MessageReaction::MAX_RECENT_CHOOSERS + 1) { auto *reaction = get_reaction(old_reaction.get_reaction_type()); if (reaction != nullptr && reaction->is_chosen()) { reaction->update_recent_chooser_dialog_ids(old_reaction); } } } } bool MessageReactions::add_my_reaction(const ReactionType &reaction_type, bool is_big, DialogId my_dialog_id, bool have_recent_choosers, bool is_tag) { vector new_chosen_reaction_order = get_chosen_reaction_types(); auto added_reaction = get_reaction(reaction_type); if (added_reaction == nullptr) { vector recent_chooser_dialog_ids; DialogId my_recent_chooser_dialog_id; if (have_recent_choosers) { recent_chooser_dialog_ids.push_back(my_dialog_id); my_recent_chooser_dialog_id = my_dialog_id; } reactions_.push_back( {reaction_type, 1, true, my_recent_chooser_dialog_id, std::move(recent_chooser_dialog_ids), Auto()}); new_chosen_reaction_order.emplace_back(reaction_type); } else if (!added_reaction->is_chosen()) { added_reaction->set_as_chosen(my_dialog_id, have_recent_choosers); new_chosen_reaction_order.emplace_back(reaction_type); } else if (!is_big) { return false; } if (!is_tag) { CHECK(!are_tags_); } else { are_tags_ = true; } auto max_reaction_count = get_max_reaction_count(); while (new_chosen_reaction_order.size() > max_reaction_count) { auto index = new_chosen_reaction_order[0] == reaction_type ? 1 : 0; CHECK(static_cast(index) < new_chosen_reaction_order.size()); bool is_removed = do_remove_my_reaction(new_chosen_reaction_order[index]); CHECK(is_removed); new_chosen_reaction_order.erase(new_chosen_reaction_order.begin() + index); } if (new_chosen_reaction_order.size() == 1) { new_chosen_reaction_order.clear(); } chosen_reaction_order_ = std::move(new_chosen_reaction_order); for (auto &message_reaction : reactions_) { message_reaction.set_my_recent_chooser_dialog_id(my_dialog_id); } return true; } bool MessageReactions::remove_my_reaction(const ReactionType &reaction_type, DialogId my_dialog_id) { if (do_remove_my_reaction(reaction_type)) { if (!chosen_reaction_order_.empty()) { bool is_removed = td::remove(chosen_reaction_order_, reaction_type); CHECK(is_removed); // if the user isn't a Premium user, then max_reaction_count could be reduced from 3 to 1 auto max_reaction_count = get_max_reaction_count(); while (chosen_reaction_order_.size() > max_reaction_count) { is_removed = do_remove_my_reaction(chosen_reaction_order_[0]); CHECK(is_removed); chosen_reaction_order_.erase(chosen_reaction_order_.begin()); } if (chosen_reaction_order_.size() <= 1) { reset_to_empty(chosen_reaction_order_); } } for (auto &message_reaction : reactions_) { message_reaction.set_my_recent_chooser_dialog_id(my_dialog_id); } return true; } return false; } bool MessageReactions::do_remove_my_reaction(const ReactionType &reaction_type) { for (auto it = reactions_.begin(); it != reactions_.end(); ++it) { auto &message_reaction = *it; if (message_reaction.get_reaction_type() == reaction_type) { if (message_reaction.is_chosen()) { message_reaction.unset_as_chosen(); if (message_reaction.is_empty()) { it = reactions_.erase(it); } return true; } break; } } return false; } void MessageReactions::add_my_paid_reaction(int32 star_count) { auto added_reaction = get_reaction(ReactionType::paid()); if (added_reaction == nullptr) { reactions_.push_back({ReactionType::paid(), star_count, true, DialogId(), Auto(), Auto()}); } else { added_reaction->add_paid_reaction(star_count); } } void MessageReactions::sort_reactions(const FlatHashMap &active_reaction_pos) { std::sort(reactions_.begin(), reactions_.end(), [&active_reaction_pos](const MessageReaction &lhs, const MessageReaction &rhs) { if (lhs.get_reaction_type().is_paid_reaction() != rhs.get_reaction_type().is_paid_reaction()) { return lhs.get_reaction_type().is_paid_reaction(); } if (lhs.get_choose_count() != rhs.get_choose_count()) { return lhs.get_choose_count() > rhs.get_choose_count(); } auto lhs_it = active_reaction_pos.find(lhs.get_reaction_type()); auto lhs_pos = lhs_it != active_reaction_pos.end() ? lhs_it->second : active_reaction_pos.size(); auto rhs_it = active_reaction_pos.find(rhs.get_reaction_type()); auto rhs_pos = rhs_it != active_reaction_pos.end() ? rhs_it->second : active_reaction_pos.size(); if (lhs_pos != rhs_pos) { return lhs_pos < rhs_pos; } return lhs.get_reaction_type() < rhs.get_reaction_type(); }); } void MessageReactions::fix_chosen_reaction() { DialogId my_dialog_id; for (auto &reaction : reactions_) { if (!reaction.is_chosen() && reaction.get_my_recent_chooser_dialog_id().is_valid()) { my_dialog_id = reaction.get_my_recent_chooser_dialog_id(); LOG(WARNING) << "Fix recent chosen reaction in " << *this; reaction.remove_my_recent_chooser_dialog_id(); } } if (!my_dialog_id.is_valid()) { return; } for (auto &reaction : reactions_) { if (!reaction.get_reaction_type().is_paid_reaction() && reaction.is_chosen() && !reaction.get_my_recent_chooser_dialog_id().is_valid()) { reaction.add_my_recent_chooser_dialog_id(my_dialog_id); } } } void MessageReactions::fix_my_recent_chooser_dialog_id(DialogId my_dialog_id) { for (auto &reaction : reactions_) { if (!reaction.get_reaction_type().is_paid_reaction() && reaction.is_chosen() && !reaction.get_my_recent_chooser_dialog_id().is_valid() && td::contains(reaction.get_recent_chooser_dialog_ids(), my_dialog_id)) { reaction.my_recent_chooser_dialog_id_ = my_dialog_id; } } } vector MessageReactions::get_chosen_reaction_types() const { if (!chosen_reaction_order_.empty()) { return chosen_reaction_order_; } vector reaction_order; for (const auto &reaction : reactions_) { if (!reaction.get_reaction_type().is_paid_reaction() && reaction.is_chosen()) { reaction_order.push_back(reaction.get_reaction_type()); } } return reaction_order; } bool MessageReactions::are_consistent_with_list( const ReactionType &reaction_type, FlatHashMap, ReactionTypeHash> reaction_types, int32 total_count) const { auto are_consistent = [](const vector &lhs, const vector &rhs) { size_t i = 0; size_t max_i = td::min(lhs.size(), rhs.size()); while (i < max_i && lhs[i] == rhs[i]) { i++; } return i == max_i; }; if (reaction_type.is_empty()) { // received list and total_count for all reactions int32 old_total_count = 0; for (const auto &message_reaction : reactions_) { CHECK(!message_reaction.get_reaction_type().is_empty()); if (!are_consistent(reaction_types[message_reaction.get_reaction_type()], message_reaction.get_recent_chooser_dialog_ids())) { return false; } old_total_count += message_reaction.get_choose_count(); reaction_types.erase(message_reaction.get_reaction_type()); } return old_total_count == total_count && reaction_types.empty(); } // received list and total_count for a single reaction const auto *message_reaction = get_reaction(reaction_type); if (message_reaction == nullptr) { return reaction_types.count(reaction_type) == 0 && total_count == 0; } else { return are_consistent(reaction_types[reaction_type], message_reaction->get_recent_chooser_dialog_ids()) && message_reaction->get_choose_count() == total_count; } } td_api::object_ptr MessageReactions::get_message_reactions_object(Td *td, UserId my_user_id, UserId peer_user_id) const { auto reactions = transform(reactions_, [td, my_user_id, peer_user_id](const MessageReaction &reaction) { return reaction.get_message_reaction_object(td, my_user_id, peer_user_id); }); auto reactors = transform(top_reactors_, [td](const MessageReactor &reactor) { return reactor.get_paid_reactor_object(td); }); return td_api::make_object(std::move(reactions), are_tags_, std::move(reactors)); } void MessageReactions::add_min_channels(Td *td) const { for (const auto &reaction : reactions_) { for (const auto &recent_chooser_min_channel : reaction.get_recent_chooser_min_channels()) { LOG(INFO) << "Add min reacted " << recent_chooser_min_channel.first; td->chat_manager_->add_min_channel(recent_chooser_min_channel.first, recent_chooser_min_channel.second); } } } void MessageReactions::add_dependencies(Dependencies &dependencies) const { for (const auto &reaction : reactions_) { const auto &dialog_ids = reaction.get_recent_chooser_dialog_ids(); for (auto dialog_id : dialog_ids) { dependencies.add_message_sender_dependencies(dialog_id); } } for (const auto &reactor : top_reactors_) { reactor.add_dependencies(dependencies); } } 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 (new_reactions == nullptr) { // remove reactions when they are disabled return true; } // unread_reactions_ and chosen_reaction_order_ are updated independently; compare all other fields return old_reactions->reactions_ != new_reactions->reactions_ || old_reactions->is_min_ != new_reactions->is_min_ || old_reactions->can_get_added_reactions_ != new_reactions->can_get_added_reactions_ || old_reactions->need_polling_ != new_reactions->need_polling_ || old_reactions->are_tags_ != new_reactions->are_tags_ || old_reactions->top_reactors_ != new_reactions->top_reactors_; } bool MessageReactions::need_update_unread_reactions(const MessageReactions *old_reactions, const MessageReactions *new_reactions) { if (old_reactions == nullptr || old_reactions->unread_reactions_.empty()) { return !(new_reactions == nullptr || new_reactions->unread_reactions_.empty()); } return new_reactions == nullptr || old_reactions->unread_reactions_ != new_reactions->unread_reactions_; } StringBuilder &operator<<(StringBuilder &string_builder, const MessageReactions &reactions) { if (reactions.are_tags_) { return string_builder << "MessageTags{" << reactions.reactions_ << '}'; } return string_builder << (reactions.is_min_ ? "Min" : "") << "MessageReactions{" << reactions.reactions_ << " with unread " << reactions.unread_reactions_ << ", reaction order " << reactions.chosen_reaction_order_ << " and can_get_added_reactions = " << reactions.can_get_added_reactions_ << " with paid reactions by " << reactions.top_reactors_ << '}'; } StringBuilder &operator<<(StringBuilder &string_builder, const unique_ptr &reactions) { if (reactions == nullptr) { return string_builder << "null"; } return string_builder << *reactions; } void reload_message_reactions(Td *td, DialogId dialog_id, vector &&message_ids) { if (!td->dialog_manager_->have_input_peer(dialog_id, false, AccessRights::Read) || message_ids.empty()) { create_actor( "RetryReloadMessageReactionsActor", 0.2, PromiseCreator::lambda([actor_id = G()->messages_manager(), dialog_id](Result result) mutable { send_closure(actor_id, &MessagesManager::try_reload_message_reactions, dialog_id, true); })) .release(); return; } for (const auto &message_id : message_ids) { CHECK(message_id.is_valid()); CHECK(message_id.is_server()); } td->create_handler()->send(dialog_id, std::move(message_ids)); } void send_message_reaction(Td *td, MessageFullId message_full_id, vector reaction_types, bool is_big, bool add_to_recent, Promise &&promise) { td->create_handler(std::move(promise)) ->send(message_full_id, std::move(reaction_types), is_big, add_to_recent); } void set_message_reactions(Td *td, MessageFullId message_full_id, vector reaction_types, bool is_big, Promise &&promise) { if (!td->messages_manager_->have_message_force(message_full_id, "set_message_reactions")) { return promise.set_error(Status::Error(400, "Message not found")); } for (const auto &reaction_type : reaction_types) { if (reaction_type.is_empty() || reaction_type.is_paid_reaction()) { return promise.set_error(Status::Error(400, "Invalid reaction type specified")); } } send_message_reaction(td, message_full_id, std::move(reaction_types), is_big, false, std::move(promise)); } void send_paid_message_reaction(Td *td, MessageFullId message_full_id, int32 star_count, int64 random_id, Promise &&promise) { td->create_handler(std::move(promise))->send(message_full_id, star_count, random_id); } void get_message_added_reactions(Td *td, MessageFullId message_full_id, ReactionType reaction_type, string offset, int32 limit, Promise> &&promise) { if (!td->messages_manager_->have_message_force(message_full_id, "get_message_added_reactions")) { return promise.set_error(Status::Error(400, "Message not found")); } auto message_id = message_full_id.get_message_id(); if (message_full_id.get_dialog_id().get_type() == DialogType::SecretChat || !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_ADDED_REACTIONS = 100; // server side limit if (limit > MAX_GET_ADDED_REACTIONS) { limit = MAX_GET_ADDED_REACTIONS; } td->create_handler(std::move(promise)) ->send(message_full_id, std::move(reaction_type), std::move(offset), limit); } void report_message_reactions(Td *td, MessageFullId message_full_id, DialogId chooser_dialog_id, Promise &&promise) { auto dialog_id = message_full_id.get_dialog_id(); TRY_STATUS_PROMISE(promise, td->dialog_manager_->check_dialog_access(dialog_id, false, AccessRights::Read, "report_message_reactions")); if (!td->messages_manager_->have_message_force(message_full_id, "report_message_reactions")) { return promise.set_error(Status::Error(400, "Message not found")); } auto message_id = message_full_id.get_message_id(); if (message_id.is_valid_scheduled()) { return promise.set_error(Status::Error(400, "Can't report reactions on scheduled messages")); } if (!message_id.is_server()) { return promise.set_error(Status::Error(400, "Message reactions can't be reported")); } if (!td->dialog_manager_->have_input_peer(chooser_dialog_id, false, AccessRights::Know)) { return promise.set_error(Status::Error(400, "Reaction sender not found")); } td->create_handler(std::move(promise))->send(dialog_id, message_id, chooser_dialog_id); } vector get_chosen_tags(const unique_ptr &message_reactions) { if (message_reactions == nullptr || !message_reactions->are_tags_) { return {}; } return message_reactions->get_chosen_reaction_types(); } } // namespace td