tdlight/td/telegram/OrderedMessage.cpp

338 lines
12 KiB
C++
Raw Normal View History

//
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023
//
// 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/OrderedMessage.h"
#include "td/utils/logging.h"
namespace td {
2023-05-03 12:29:41 +03:00
void OrderedMessages::insert(MessageId message_id, bool auto_attach, bool have_previous, bool have_next,
MessageId old_last_message_id, const char *source) {
bool is_attached = false;
if (auto_attach) {
CHECK(have_previous && have_next);
auto attach_info = auto_attach_message(message_id, old_last_message_id, source);
have_previous = attach_info.have_previous_;
have_next = attach_info.have_next_;
is_attached = have_previous || have_next;
}
if (!is_attached && !have_previous && !have_next) {
auto it = get_iterator(message_id);
if (*it != nullptr && (*it)->have_next_) {
// need to drop a connection between messages
auto previous_message = *it;
CHECK(previous_message->message_id_ < message_id);
++it;
auto next_message = *it;
CHECK(next_message != nullptr);
CHECK(next_message->message_id_ > message_id);
next_message->have_previous_ = false;
previous_message->have_next_ = false;
}
}
auto random_y = static_cast<int32>(static_cast<uint32>(message_id.get() * 2101234567u));
2023-05-02 18:04:54 +03:00
unique_ptr<OrderedMessage> *v = &messages_;
2023-05-02 23:27:37 +03:00
while (*v != nullptr && (*v)->random_y_ >= random_y) {
if ((*v)->message_id_.get() < message_id.get()) {
v = &(*v)->right_;
} else if ((*v)->message_id_ == message_id) {
UNREACHABLE();
} else {
2023-05-02 23:27:37 +03:00
v = &(*v)->left_;
}
}
auto message = make_unique<OrderedMessage>();
2023-05-02 23:27:37 +03:00
message->message_id_ = message_id;
message->random_y_ = random_y;
2023-05-02 23:27:37 +03:00
unique_ptr<OrderedMessage> *left = &message->left_;
unique_ptr<OrderedMessage> *right = &message->right_;
unique_ptr<OrderedMessage> cur = std::move(*v);
while (cur != nullptr) {
2023-05-02 23:27:37 +03:00
if (cur->message_id_.get() < message_id.get()) {
*left = std::move(cur);
2023-05-02 23:27:37 +03:00
left = &((*left)->right_);
cur = std::move(*left);
} else {
*right = std::move(cur);
2023-05-02 23:27:37 +03:00
right = &((*right)->left_);
cur = std::move(*right);
}
}
CHECK(*left == nullptr);
CHECK(*right == nullptr);
*v = std::move(message);
2023-05-03 00:25:21 +03:00
2023-05-03 12:29:41 +03:00
if (!is_attached) {
2023-05-03 00:25:21 +03:00
if (have_next) {
CHECK(!have_previous);
2023-05-03 12:29:41 +03:00
attach_message_to_next(message_id, source);
2023-05-03 00:25:21 +03:00
} else if (have_previous) {
2023-05-03 12:29:41 +03:00
attach_message_to_previous(message_id, source);
2023-05-03 00:25:21 +03:00
}
} else {
(*v)->have_previous_ = have_previous;
(*v)->have_next_ = have_next;
}
}
2023-05-03 01:34:58 +03:00
void OrderedMessages::erase(MessageId message_id, bool only_from_memory) {
2023-05-02 18:04:54 +03:00
unique_ptr<OrderedMessage> *v = &messages_;
while (*v != nullptr) {
2023-05-02 23:27:37 +03:00
if ((*v)->message_id_.get() < message_id.get()) {
v = &(*v)->right_;
} else if ((*v)->message_id_.get() > message_id.get()) {
v = &(*v)->left_;
} else {
break;
}
}
2023-05-03 01:34:58 +03:00
CHECK(*v != nullptr);
if ((*v)->have_previous_ && (only_from_memory || !(*v)->have_next_)) {
auto it = get_iterator(message_id);
2023-05-03 01:34:58 +03:00
CHECK(*it == v->get());
--it;
OrderedMessage *prev_m = *it;
CHECK(prev_m != nullptr);
prev_m->have_next_ = false;
}
if ((*v)->have_next_ && (only_from_memory || !(*v)->have_previous_)) {
auto it = get_iterator(message_id);
2023-05-03 01:34:58 +03:00
CHECK(*it == v->get());
++it;
OrderedMessage *next_m = *it;
CHECK(next_m != nullptr);
next_m->have_previous_ = false;
}
unique_ptr<OrderedMessage> result = std::move(*v);
2023-05-02 23:27:37 +03:00
unique_ptr<OrderedMessage> left = std::move(result->left_);
unique_ptr<OrderedMessage> right = std::move(result->right_);
while (left != nullptr || right != nullptr) {
2023-05-02 23:27:37 +03:00
if (left == nullptr || (right != nullptr && right->random_y_ > left->random_y_)) {
*v = std::move(right);
2023-05-02 23:27:37 +03:00
v = &((*v)->left_);
right = std::move(*v);
} else {
*v = std::move(left);
2023-05-02 23:27:37 +03:00
v = &((*v)->right_);
left = std::move(*v);
}
}
CHECK(*v == nullptr);
}
void OrderedMessages::attach_message_to_previous(MessageId message_id, const char *source) {
CHECK(message_id.is_valid());
auto it = get_iterator(message_id);
OrderedMessage *ordered_message = *it;
CHECK(ordered_message != nullptr);
2023-05-02 23:27:37 +03:00
CHECK(ordered_message->message_id_ == message_id);
if (ordered_message->have_previous_) {
return;
}
2023-05-02 23:27:37 +03:00
ordered_message->have_previous_ = true;
--it;
LOG_CHECK(*it != nullptr) << message_id << ' ' << source;
2023-05-02 23:27:37 +03:00
LOG(INFO) << "Attach " << message_id << " to the previous " << (*it)->message_id_ << " from " << source;
if ((*it)->have_next_) {
ordered_message->have_next_ = true;
} else {
2023-05-02 23:27:37 +03:00
(*it)->have_next_ = true;
}
}
void OrderedMessages::attach_message_to_next(MessageId message_id, const char *source) {
CHECK(message_id.is_valid());
auto it = get_iterator(message_id);
OrderedMessage *ordered_message = *it;
CHECK(ordered_message != nullptr);
2023-05-02 23:27:37 +03:00
CHECK(ordered_message->message_id_ == message_id);
if (ordered_message->have_next_) {
return;
}
2023-05-02 23:27:37 +03:00
ordered_message->have_next_ = true;
++it;
LOG_CHECK(*it != nullptr) << message_id << ' ' << source;
2023-05-02 23:27:37 +03:00
LOG(INFO) << "Attach " << message_id << " to the next " << (*it)->message_id_ << " from " << source;
if ((*it)->have_previous_) {
ordered_message->have_previous_ = true;
} else {
2023-05-02 23:27:37 +03:00
(*it)->have_previous_ = true;
}
}
OrderedMessages::AttachInfo OrderedMessages::auto_attach_message(MessageId message_id, MessageId last_message_id,
const char *source) {
auto it = get_iterator(message_id);
OrderedMessage *previous_message = *it;
if (previous_message != nullptr) {
2023-05-02 23:27:37 +03:00
auto previous_message_id = previous_message->message_id_;
CHECK(previous_message_id < message_id);
2023-05-02 23:27:37 +03:00
if (previous_message->have_next_ || (last_message_id.is_valid() && previous_message_id >= last_message_id)) {
if (message_id.is_server() && previous_message_id.is_server() && previous_message->have_next_) {
++it;
auto next_message = *it;
2023-05-03 12:22:47 +03:00
CHECK(next_message != nullptr);
if (next_message->message_id_.is_server()) {
LOG(ERROR) << "Attach " << message_id << " before " << next_message->message_id_ << " and after "
2023-05-03 12:29:41 +03:00
<< previous_message_id << " from " << source;
}
}
2023-05-03 12:29:41 +03:00
LOG(INFO) << "Attach " << message_id << " to the previous " << previous_message_id << " from " << source;
2023-05-02 23:27:37 +03:00
auto have_next = previous_message->have_next_;
previous_message->have_next_ = true;
return {true, have_next};
}
}
if (!message_id.is_yet_unsent()) {
// message may be attached to the next message if there is no previous message
OrderedMessage *cur = messages_.get();
OrderedMessage *next_message = nullptr;
while (cur != nullptr) {
2023-05-02 23:27:37 +03:00
if (cur->message_id_ < message_id) {
cur = cur->right_.get();
} else {
next_message = cur;
2023-05-02 23:27:37 +03:00
cur = cur->left_.get();
}
}
if (next_message != nullptr) {
2023-05-02 23:27:37 +03:00
CHECK(!next_message->have_previous_);
2023-05-03 12:29:41 +03:00
LOG(INFO) << "Attach " << message_id << " to the next " << next_message->message_id_ << " from " << source;
2023-05-02 23:27:37 +03:00
auto have_previous = next_message->have_previous_;
return {have_previous, true};
}
}
2023-05-03 12:29:41 +03:00
LOG(INFO) << "Can't auto-attach " << message_id << " from " << source;
return {false, false};
}
void OrderedMessages::do_find_older_messages(const OrderedMessage *ordered_message, MessageId max_message_id,
vector<MessageId> &message_ids) {
if (ordered_message == nullptr) {
return;
}
2023-05-02 23:27:37 +03:00
do_find_older_messages(ordered_message->left_.get(), max_message_id, message_ids);
2023-05-02 23:27:37 +03:00
if (ordered_message->message_id_ <= max_message_id) {
message_ids.push_back(ordered_message->message_id_);
2023-05-02 23:27:37 +03:00
do_find_older_messages(ordered_message->right_.get(), max_message_id, message_ids);
}
}
vector<MessageId> OrderedMessages::find_older_messages(MessageId max_message_id) const {
vector<MessageId> message_ids;
do_find_older_messages(messages_.get(), max_message_id, message_ids);
return message_ids;
}
void OrderedMessages::do_find_newer_messages(const OrderedMessage *ordered_message, MessageId min_message_id,
vector<MessageId> &message_ids) {
if (ordered_message == nullptr) {
return;
}
2023-05-02 23:27:37 +03:00
if (ordered_message->message_id_ > min_message_id) {
do_find_newer_messages(ordered_message->left_.get(), min_message_id, message_ids);
2023-05-02 23:27:37 +03:00
message_ids.push_back(ordered_message->message_id_);
}
2023-05-02 23:27:37 +03:00
do_find_newer_messages(ordered_message->right_.get(), min_message_id, message_ids);
}
vector<MessageId> OrderedMessages::find_newer_messages(MessageId min_message_id) const {
vector<MessageId> message_ids;
do_find_newer_messages(messages_.get(), min_message_id, message_ids);
return message_ids;
}
MessageId OrderedMessages::do_find_message_by_date(const OrderedMessage *ordered_message, int32 date,
const std::function<int32(MessageId)> &get_message_date) {
if (ordered_message == nullptr) {
return MessageId();
}
2023-05-02 23:27:37 +03:00
auto message_date = get_message_date(ordered_message->message_id_);
if (message_date > date) {
2023-05-02 23:27:37 +03:00
return do_find_message_by_date(ordered_message->left_.get(), date, get_message_date);
}
2023-05-02 23:27:37 +03:00
auto message_id = do_find_message_by_date(ordered_message->right_.get(), date, get_message_date);
if (message_id.is_valid()) {
return message_id;
}
2023-05-02 23:27:37 +03:00
return ordered_message->message_id_;
}
MessageId OrderedMessages::find_message_by_date(int32 date,
const std::function<int32(MessageId)> &get_message_date) const {
return do_find_message_by_date(messages_.get(), date, get_message_date);
}
void OrderedMessages::do_find_messages_by_date(const OrderedMessage *ordered_message, int32 min_date, int32 max_date,
const std::function<int32(MessageId)> &get_message_date,
vector<MessageId> &message_ids) {
if (ordered_message == nullptr) {
return;
}
2023-05-02 23:27:37 +03:00
auto message_date = get_message_date(ordered_message->message_id_);
if (message_date >= min_date) {
2023-05-02 23:27:37 +03:00
do_find_messages_by_date(ordered_message->left_.get(), min_date, max_date, get_message_date, message_ids);
if (message_date <= max_date) {
2023-05-02 23:27:37 +03:00
message_ids.push_back(ordered_message->message_id_);
}
}
if (message_date <= max_date) {
2023-05-02 23:27:37 +03:00
do_find_messages_by_date(ordered_message->right_.get(), min_date, max_date, get_message_date, message_ids);
}
}
vector<MessageId> OrderedMessages::find_messages_by_date(
int32 min_date, int32 max_date, const std::function<int32(MessageId)> &get_message_date) const {
vector<MessageId> message_ids;
do_find_messages_by_date(messages_.get(), min_date, max_date, get_message_date, message_ids);
return message_ids;
}
void OrderedMessages::do_traverse_messages(const OrderedMessage *ordered_message,
const std::function<bool(MessageId)> &need_scan_older,
const std::function<bool(MessageId)> &need_scan_newer) {
if (ordered_message == nullptr) {
return;
}
2023-05-02 23:27:37 +03:00
if (need_scan_older(ordered_message->message_id_)) {
do_traverse_messages(ordered_message->left_.get(), need_scan_older, need_scan_newer);
}
2023-05-02 23:27:37 +03:00
if (need_scan_newer(ordered_message->message_id_)) {
do_traverse_messages(ordered_message->right_.get(), need_scan_older, need_scan_newer);
}
}
void OrderedMessages::traverse_messages(const std::function<bool(MessageId)> &need_scan_older,
const std::function<bool(MessageId)> &need_scan_newer) const {
do_traverse_messages(messages_.get(), need_scan_older, need_scan_newer);
}
} // namespace td