Check message_id of loaded from database messages.

This commit is contained in:
levlam 2021-10-01 16:43:02 +03:00
parent 4d180ff0e1
commit 4a185caa74
2 changed files with 44 additions and 20 deletions

View File

@ -22276,20 +22276,41 @@ void MessagesManager::preload_older_messages(const Dialog *d, MessageId min_mess
} }
} }
unique_ptr<MessagesManager::Message> MessagesManager::parse_message(DialogId dialog_id, const BufferSlice &value, unique_ptr<MessagesManager::Message> MessagesManager::parse_message(DialogId dialog_id, MessageId expected_message_id,
bool is_scheduled) { const BufferSlice &value, bool is_scheduled) {
auto m = make_unique<Message>(); auto m = make_unique<Message>();
auto status = log_event_parse(*m, value.as_slice()); auto status = log_event_parse(*m, value.as_slice());
bool is_message_id_valid = is_scheduled ? m->message_id.is_valid_scheduled() : m->message_id.is_valid(); bool is_message_id_valid = [&] {
if (is_scheduled) {
if (!expected_message_id.is_valid_scheduled()) {
return false;
}
if (m->message_id == expected_message_id) {
return true;
}
return m->message_id.is_valid_scheduled() && expected_message_id.is_scheduled_server() &&
m->message_id.is_scheduled_server() &&
m->message_id.get_scheduled_server_message_id() == expected_message_id.get_scheduled_server_message_id();
} else {
if (!expected_message_id.is_valid()) {
return false;
}
return m->message_id == expected_message_id;
}
}();
if (status.is_error() || !is_message_id_valid) { if (status.is_error() || !is_message_id_valid) {
// can't happen unless database is broken, but has been seen in the wild // can't happen unless the database is broken, but has been seen in the wild
LOG(ERROR) << "Receive invalid message from database: " << m->message_id << ' ' << status << ' ' LOG(ERROR) << "Receive invalid message from database: " << expected_message_id << ' ' << m->message_id << ' '
<< format::as_hex_dump<4>(value.as_slice()); << status << ' ' << format::as_hex_dump<4>(value.as_slice());
if (!is_scheduled && dialog_id.get_type() != DialogType::SecretChat && m->message_id.is_valid() && if (!is_scheduled && dialog_id.get_type() != DialogType::SecretChat) {
m->message_id.is_server()) {
// trying to repair the message // trying to repair the message
get_message_from_server({dialog_id, m->message_id}, Auto(), "parse_message"); if (expected_message_id.is_valid() && expected_message_id.is_server()) {
get_message_from_server({dialog_id, expected_message_id}, Auto(), "parse_message");
}
if (m->message_id.is_valid() && m->message_id.is_server()) {
get_message_from_server({dialog_id, m->message_id}, Auto(), "parse_message");
}
} }
return nullptr; return nullptr;
} }
@ -22365,7 +22386,7 @@ void MessagesManager::on_get_history_from_database(DialogId dialog_id, MessageId
if (!d->first_database_message_id.is_valid() && !d->have_full_history) { if (!d->first_database_message_id.is_valid() && !d->have_full_history) {
break; break;
} }
auto message = parse_message(dialog_id, message_slice.data, false); auto message = parse_message(dialog_id, message_slice.message_id, message_slice.data, false);
if (message == nullptr) { if (message == nullptr) {
if (d->have_full_history) { if (d->have_full_history) {
d->have_full_history = false; d->have_full_history = false;
@ -22806,7 +22827,7 @@ void MessagesManager::on_get_scheduled_messages_from_database(DialogId dialog_id
Dependencies dependencies; Dependencies dependencies;
vector<MessageId> added_message_ids; vector<MessageId> added_message_ids;
for (auto &message_slice : messages) { for (auto &message_slice : messages) {
auto message = parse_message(dialog_id, message_slice.data, true); auto message = parse_message(dialog_id, message_slice.message_id, message_slice.data, true);
if (message == nullptr) { if (message == nullptr) {
continue; continue;
} }
@ -32128,23 +32149,25 @@ MessagesManager::Message *MessagesManager::get_message_force(Dialog *d, MessageI
MessagesManager::Message *MessagesManager::on_get_message_from_database(const MessagesDbMessage &message, MessagesManager::Message *MessagesManager::on_get_message_from_database(const MessagesDbMessage &message,
bool is_scheduled, const char *source) { bool is_scheduled, const char *source) {
return on_get_message_from_database(get_dialog_force(message.dialog_id, source), message.dialog_id, message.data, return on_get_message_from_database(get_dialog_force(message.dialog_id, source), message.dialog_id,
is_scheduled, source); message.message_id, message.data, is_scheduled, source);
} }
MessagesManager::Message *MessagesManager::on_get_message_from_database(Dialog *d, const MessagesDbDialogMessage &value, MessagesManager::Message *MessagesManager::on_get_message_from_database(Dialog *d,
const MessagesDbDialogMessage &message,
bool is_scheduled, const char *source) { bool is_scheduled, const char *source) {
return on_get_message_from_database(d, d->dialog_id, value.data, is_scheduled, source); return on_get_message_from_database(d, d->dialog_id, message.message_id, message.data, is_scheduled, source);
} }
MessagesManager::Message *MessagesManager::on_get_message_from_database(Dialog *d, DialogId dialog_id, MessagesManager::Message *MessagesManager::on_get_message_from_database(Dialog *d, DialogId dialog_id,
MessageId expected_message_id,
const BufferSlice &value, bool is_scheduled, const BufferSlice &value, bool is_scheduled,
const char *source) { const char *source) {
if (value.empty()) { if (value.empty()) {
return nullptr; return nullptr;
} }
auto m = parse_message(dialog_id, value, is_scheduled); auto m = parse_message(dialog_id, expected_message_id, value, is_scheduled);
if (m == nullptr) { if (m == nullptr) {
return nullptr; return nullptr;
} }

View File

@ -2660,11 +2660,11 @@ class MessagesManager final : public Actor {
Message *on_get_message_from_database(const MessagesDbMessage &message, bool is_scheduled, const char *source); Message *on_get_message_from_database(const MessagesDbMessage &message, bool is_scheduled, const char *source);
Message *on_get_message_from_database(Dialog *d, const MessagesDbDialogMessage &value, bool is_scheduled, Message *on_get_message_from_database(Dialog *d, const MessagesDbDialogMessage &message, bool is_scheduled,
const char *source); const char *source);
Message *on_get_message_from_database(Dialog *d, DialogId dialog_id, const BufferSlice &value, bool is_scheduled, Message *on_get_message_from_database(Dialog *d, DialogId dialog_id, MessageId message_id, const BufferSlice &value,
const char *source); bool is_scheduled, const char *source);
void get_dialog_message_by_date_from_server(const Dialog *d, int32 date, int64 random_id, bool after_database_search, void get_dialog_message_by_date_from_server(const Dialog *d, int32 date, int64 random_id, bool after_database_search,
Promise<Unit> &&promise); Promise<Unit> &&promise);
@ -3010,7 +3010,8 @@ class MessagesManager final : public Actor {
string get_message_search_text(const Message *m) const; string get_message_search_text(const Message *m) const;
unique_ptr<Message> parse_message(DialogId dialog_id, const BufferSlice &value, bool is_scheduled); unique_ptr<Message> parse_message(DialogId dialog_id, MessageId expected_message_id, const BufferSlice &value,
bool is_scheduled);
unique_ptr<Dialog> parse_dialog(DialogId dialog_id, const BufferSlice &value, const char *source); unique_ptr<Dialog> parse_dialog(DialogId dialog_id, const BufferSlice &value, const char *source);