tdlight/td/telegram/MessageForwardInfo.cpp
2024-01-11 22:48:22 +03:00

200 lines
7.3 KiB
C++

//
// 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/MessageForwardInfo.h"
#include "td/telegram/Dependencies.h"
#include "td/telegram/DialogManager.h"
#include "td/telegram/MessagesManager.h"
#include "td/telegram/Td.h"
#include "td/utils/logging.h"
namespace td {
bool LastForwardedMessageInfo::is_empty() const {
return dialog_id_ == DialogId() && message_id_ == MessageId();
}
bool LastForwardedMessageInfo::validate() {
if (is_empty()) {
return true;
}
if (!dialog_id_.is_valid() || !message_id_.is_valid()) {
*this = {};
return false;
}
return true;
}
void LastForwardedMessageInfo::add_dependencies(Dependencies &dependencies) const {
dependencies.add_dialog_and_dependencies(dialog_id_);
}
void LastForwardedMessageInfo::add_min_user_ids(vector<UserId> &user_ids) const {
if (dialog_id_.get_type() == DialogType::User) {
user_ids.push_back(dialog_id_.get_user_id());
}
}
void LastForwardedMessageInfo::add_min_channel_ids(vector<ChannelId> &channel_ids) const {
if (dialog_id_.get_type() == DialogType::Channel) {
channel_ids.push_back(dialog_id_.get_channel_id());
}
}
td_api::object_ptr<td_api::forwardSource> LastForwardedMessageInfo::get_forward_source_object(Td *td) const {
if (is_empty()) {
return nullptr;
}
return td_api::make_object<td_api::forwardSource>(
td->messages_manager_->get_chat_id_object(dialog_id_, "forwardSource"), message_id_.get());
}
bool operator==(const LastForwardedMessageInfo &lhs, const LastForwardedMessageInfo &rhs) {
return lhs.dialog_id_ == rhs.dialog_id_ && lhs.message_id_ == rhs.message_id_;
}
bool operator!=(const LastForwardedMessageInfo &lhs, const LastForwardedMessageInfo &rhs) {
return !(lhs == rhs);
}
StringBuilder &operator<<(StringBuilder &string_builder, const LastForwardedMessageInfo &last_message_info) {
if (last_message_info.is_empty()) {
return string_builder;
}
return string_builder << MessageFullId(last_message_info.dialog_id_, last_message_info.message_id_);
}
unique_ptr<MessageForwardInfo> MessageForwardInfo::get_message_forward_info(
Td *td, telegram_api::object_ptr<telegram_api::messageFwdHeader> &&forward_header) {
if (forward_header == nullptr) {
return nullptr;
}
auto date = forward_header->date_;
if (date <= 0) {
LOG(ERROR) << "Wrong date in message forward header: " << oneline(to_string(forward_header));
return nullptr;
}
LastForwardedMessageInfo last_message_info;
if (forward_header->saved_from_peer_ != nullptr) {
last_message_info = LastForwardedMessageInfo(DialogId(forward_header->saved_from_peer_),
MessageId(ServerMessageId(forward_header->saved_from_msg_id_)));
if (last_message_info.is_empty() || !last_message_info.validate()) {
LOG(ERROR) << "Receive wrong last message in message forward header: " << oneline(to_string(forward_header));
} else {
Dependencies dependencies;
last_message_info.add_dependencies(dependencies);
for (auto dialog_id : dependencies.get_dialog_ids()) {
td->dialog_manager_->force_create_dialog(dialog_id, "get_message_forward_info", true);
}
}
}
bool is_imported = forward_header->imported_;
auto psa_type = std::move(forward_header->psa_type_);
auto r_origin = MessageOrigin::get_message_origin(td, std::move(forward_header));
if (r_origin.is_error()) {
return nullptr;
}
return td::make_unique<MessageForwardInfo>(r_origin.move_as_ok(), date, std::move(last_message_info),
std::move(psa_type), is_imported);
}
unique_ptr<MessageForwardInfo> MessageForwardInfo::copy_message_forward_info(
Td *td, const MessageForwardInfo &forward_info, LastForwardedMessageInfo &&last_message_info) {
last_message_info.validate();
auto result = make_unique<MessageForwardInfo>(forward_info);
result->last_message_info_ = std::move(last_message_info);
result->origin_.hide_sender_if_needed(td);
return result;
}
td_api::object_ptr<td_api::messageForwardInfo> MessageForwardInfo::get_message_forward_info_object(Td *td) const {
if (is_imported_) {
return nullptr;
}
return td_api::make_object<td_api::messageForwardInfo>(origin_.get_message_origin_object(td), date_,
last_message_info_.get_forward_source_object(td), psa_type_);
}
td_api::object_ptr<td_api::messageImportInfo> MessageForwardInfo::get_message_import_info_object() const {
if (!is_imported_) {
return nullptr;
}
return td_api::make_object<td_api::messageImportInfo>(origin_.get_sender_name(), date_);
}
void MessageForwardInfo::add_dependencies(Dependencies &dependencies) const {
origin_.add_dependencies(dependencies);
last_message_info_.add_dependencies(dependencies);
}
void MessageForwardInfo::add_min_user_ids(vector<UserId> &user_ids) const {
origin_.add_user_ids(user_ids);
last_message_info_.add_min_user_ids(user_ids);
}
void MessageForwardInfo::add_min_channel_ids(vector<ChannelId> &channel_ids) const {
origin_.add_channel_ids(channel_ids);
last_message_info_.add_min_channel_ids(channel_ids);
}
bool MessageForwardInfo::need_change_warning(const MessageForwardInfo *lhs, const MessageForwardInfo *rhs,
MessageId message_id) {
// it should be already checked that *lhs != *rhs
if (lhs == nullptr || rhs == nullptr || lhs->is_imported_ || rhs->is_imported_) {
return true;
}
if (!message_id.is_scheduled() && !message_id.is_yet_unsent()) {
return true;
}
// yet unsent or scheduled messages can change sender name or author signature when being sent
return !lhs->origin_.has_sender_signature() && !rhs->origin_.has_sender_signature();
}
bool operator==(const MessageForwardInfo &lhs, const MessageForwardInfo &rhs) {
return lhs.origin_ == rhs.origin_ && lhs.date_ == rhs.date_ && lhs.last_message_info_ == rhs.last_message_info_ &&
lhs.psa_type_ == rhs.psa_type_ && lhs.is_imported_ == rhs.is_imported_;
}
bool operator!=(const MessageForwardInfo &lhs, const MessageForwardInfo &rhs) {
return !(lhs == rhs);
}
bool operator==(const unique_ptr<MessageForwardInfo> &lhs, const unique_ptr<MessageForwardInfo> &rhs) {
if (lhs == nullptr) {
return rhs == nullptr;
}
return rhs != nullptr && *lhs == *rhs;
}
bool operator!=(const unique_ptr<MessageForwardInfo> &lhs, const unique_ptr<MessageForwardInfo> &rhs) {
return !(lhs == rhs);
}
StringBuilder &operator<<(StringBuilder &string_builder, const MessageForwardInfo &forward_info) {
string_builder << "MessageForwardInfo[" << (forward_info.is_imported_ ? "imported " : "") << forward_info.origin_;
if (!forward_info.psa_type_.empty()) {
string_builder << ", psa_type " << forward_info.psa_type_;
}
if (!forward_info.last_message_info_.is_empty()) {
string_builder << ", from " << forward_info.last_message_info_;
}
return string_builder << " at " << forward_info.date_ << ']';
}
StringBuilder &operator<<(StringBuilder &string_builder, const unique_ptr<MessageForwardInfo> &forward_info) {
if (forward_info == nullptr) {
return string_builder << "[null]";
}
return string_builder << *forward_info;
}
} // namespace td