Check for MAX_CONCURRENTLY_SENT_CHAT_MESSAGES as early as possible.

This commit is contained in:
levlam 2022-11-22 21:31:04 +03:00
parent c8405e95f3
commit 1d5dc9e5e8
2 changed files with 59 additions and 29 deletions

View File

@ -2836,11 +2836,13 @@ class Client::TdOnGetUserProfilePhotosCallback final : public TdQueryCallback {
class Client::TdOnSendMessageCallback final : public TdQueryCallback { class Client::TdOnSendMessageCallback final : public TdQueryCallback {
public: public:
TdOnSendMessageCallback(Client *client, PromisedQueryPtr query) : client_(client), query_(std::move(query)) { TdOnSendMessageCallback(Client *client, int64 chat_id, PromisedQueryPtr query)
: client_(client), chat_id_(chat_id), query_(std::move(query)) {
} }
void on_result(object_ptr<td_api::Object> result) final { void on_result(object_ptr<td_api::Object> result) final {
if (result->get_id() == td_api::error::ID) { if (result->get_id() == td_api::error::ID) {
client_->decrease_yet_unsent_message_count(chat_id_, 1);
return fail_query_with_error(std::move(query_), move_object_as<td_api::error>(result)); return fail_query_with_error(std::move(query_), move_object_as<td_api::error>(result));
} }
@ -2851,21 +2853,27 @@ class Client::TdOnSendMessageCallback final : public TdQueryCallback {
private: private:
Client *client_; Client *client_;
int64 chat_id_;
PromisedQueryPtr query_; PromisedQueryPtr query_;
}; };
class Client::TdOnSendMessageAlbumCallback final : public TdQueryCallback { class Client::TdOnSendMessageAlbumCallback final : public TdQueryCallback {
public: public:
TdOnSendMessageAlbumCallback(Client *client, PromisedQueryPtr query) : client_(client), query_(std::move(query)) { TdOnSendMessageAlbumCallback(Client *client, int64 chat_id, std::size_t message_count, PromisedQueryPtr query)
: client_(client), chat_id_(chat_id), message_count_(message_count), query_(std::move(query)) {
} }
void on_result(object_ptr<td_api::Object> result) final { void on_result(object_ptr<td_api::Object> result) final {
if (result->get_id() == td_api::error::ID) { if (result->get_id() == td_api::error::ID) {
if (message_count_ > 0) {
client_->decrease_yet_unsent_message_count(chat_id_, static_cast<int32>(message_count_));
}
return fail_query_with_error(std::move(query_), move_object_as<td_api::error>(result)); return fail_query_with_error(std::move(query_), move_object_as<td_api::error>(result));
} }
CHECK(result->get_id() == td_api::messages::ID); CHECK(result->get_id() == td_api::messages::ID);
auto messages = move_object_as<td_api::messages>(result); auto messages = move_object_as<td_api::messages>(result);
CHECK(messages->messages_.size() == message_count_);
auto query_id = client_->get_send_message_query_id(std::move(query_), true); auto query_id = client_->get_send_message_query_id(std::move(query_), true);
for (auto &message : messages->messages_) { for (auto &message : messages->messages_) {
client_->on_sent_message(std::move(message), query_id); client_->on_sent_message(std::move(message), query_id);
@ -2874,6 +2882,8 @@ class Client::TdOnSendMessageAlbumCallback final : public TdQueryCallback {
private: private:
Client *client_; Client *client_;
int64 chat_id_;
std::size_t message_count_;
PromisedQueryPtr query_; PromisedQueryPtr query_;
}; };
@ -4062,7 +4072,7 @@ void Client::send(PromisedQueryPtr query) {
query->set_stat_actor(stat_actor_); query->set_stat_actor(stat_actor_);
if (!parameters_->local_mode_ && !is_local_method(query->method())) { if (!parameters_->local_mode_ && !is_local_method(query->method())) {
const BotStatActor *stat = stat_actor_.get_actor_unsafe(); const BotStatActor *stat = stat_actor_.get_actor_unsafe();
if (stat->get_active_request_count() > 10000) { if (stat->get_active_request_count() > 5000) {
LOG(INFO) << "Fail a query, because there are too many active queries: " << *query; LOG(INFO) << "Fail a query, because there are too many active queries: " << *query;
return query->set_retry_after_error(60); return query->set_retry_after_error(60);
} }
@ -5140,7 +5150,7 @@ void Client::on_closed() {
} }
on_message_send_failed(chat_id, message_id, 0, Status::Error(http_status_code, description)); on_message_send_failed(chat_id, message_id, 0, Status::Error(http_status_code, description));
} }
CHECK(yet_unsent_message_count_.empty()); yet_unsent_message_count_.clear();
while (!pending_bot_resolve_queries_.empty()) { while (!pending_bot_resolve_queries_.empty()) {
auto it = pending_bot_resolve_queries_.begin(); auto it = pending_bot_resolve_queries_.begin();
@ -7083,6 +7093,16 @@ td::Result<td::int64> Client::get_user_id(const Query *query, Slice field_name)
return user_id; return user_id;
} }
void Client::decrease_yet_unsent_message_count(int64 chat_id, int32 count) {
auto count_it = yet_unsent_message_count_.find(chat_id);
CHECK(count_it != yet_unsent_message_count_.end());
CHECK(count_it->second >= count);
count_it->second -= count;
if (count_it->second == 0) {
yet_unsent_message_count_.erase(count_it);
}
}
td::int64 Client::extract_yet_unsent_message_query_id(int64 chat_id, int64 message_id, td::int64 Client::extract_yet_unsent_message_query_id(int64 chat_id, int64 message_id,
bool *is_reply_to_message_deleted) { bool *is_reply_to_message_deleted) {
auto yet_unsent_message_it = yet_unsent_messages_.find({chat_id, message_id}); auto yet_unsent_message_it = yet_unsent_messages_.find({chat_id, message_id});
@ -7096,12 +7116,7 @@ td::int64 Client::extract_yet_unsent_message_query_id(int64 chat_id, int64 messa
yet_unsent_messages_.erase(yet_unsent_message_it); yet_unsent_messages_.erase(yet_unsent_message_it);
auto count_it = yet_unsent_message_count_.find(chat_id); decrease_yet_unsent_message_count(chat_id, 1);
CHECK(count_it != yet_unsent_message_count_.end());
CHECK(count_it->second > 0);
if (--count_it->second == 0) {
yet_unsent_message_count_.erase(count_it);
}
if (reply_to_message_id > 0) { if (reply_to_message_id > 0) {
auto it = yet_unsent_reply_message_ids_.find({chat_id, reply_to_message_id}); auto it = yet_unsent_reply_message_ids_.find({chat_id, reply_to_message_id});
@ -7329,6 +7344,16 @@ td::Status Client::process_get_user_profile_photos_query(PromisedQueryPtr &query
} }
td::Status Client::process_send_message_query(PromisedQueryPtr &query) { td::Status Client::process_send_message_query(PromisedQueryPtr &query) {
auto r_chat_id = td::to_integer_safe<int64>(query->arg("chat_id"));
if (r_chat_id.is_ok()) {
// fast path
auto it = yet_unsent_message_count_.find(r_chat_id.ok());
if (it != yet_unsent_message_count_.end() && it->second >= MAX_CONCURRENTLY_SENT_CHAT_MESSAGES) {
query->set_retry_after_error(60);
return Status::OK();
}
}
TRY_RESULT(input_message_text, get_input_message_text(query.get())); TRY_RESULT(input_message_text, get_input_message_text(query.get()));
do_send_message(std::move(input_message_text), std::move(query)); do_send_message(std::move(input_message_text), std::move(query));
return Status::OK(); return Status::OK();
@ -7622,20 +7647,23 @@ td::Status Client::process_send_media_group_query(PromisedQueryPtr &query) {
input_message_contents = std::move(input_message_contents), input_message_contents = std::move(input_message_contents),
reply_markup = std::move(reply_markup)](int64 chat_id, int64 reply_to_message_id, reply_markup = std::move(reply_markup)](int64 chat_id, int64 reply_to_message_id,
PromisedQueryPtr query) mutable { PromisedQueryPtr query) mutable {
auto on_message_thread_checked = auto on_message_thread_checked = [this, disable_notification, protect_content,
[this, disable_notification, protect_content, input_message_contents = std::move(input_message_contents), input_message_contents = std::move(input_message_contents),
reply_markup = std::move(reply_markup)](int64 chat_id, int64 message_thread_id, reply_markup = std::move(reply_markup)](
int64 reply_to_message_id, PromisedQueryPtr query) mutable { int64 chat_id, int64 message_thread_id, int64 reply_to_message_id,
auto it = yet_unsent_message_count_.find(chat_id); PromisedQueryPtr query) mutable {
if (it != yet_unsent_message_count_.end() && it->second > MAX_CONCURRENTLY_SENT_CHAT_MESSAGES) { auto &count = yet_unsent_message_count_[chat_id];
if (count >= MAX_CONCURRENTLY_SENT_CHAT_MESSAGES) {
return query->set_retry_after_error(60); return query->set_retry_after_error(60);
} }
auto message_count = input_message_contents.size();
count += static_cast<int32>(message_count);
send_request(make_object<td_api::sendMessageAlbum>( send_request(
chat_id, message_thread_id, reply_to_message_id, make_object<td_api::sendMessageAlbum>(chat_id, message_thread_id, reply_to_message_id,
get_message_send_options(disable_notification, protect_content), get_message_send_options(disable_notification, protect_content),
std::move(input_message_contents), false), std::move(input_message_contents), false),
td::make_unique<TdOnSendMessageAlbumCallback>(this, std::move(query))); td::make_unique<TdOnSendMessageAlbumCallback>(this, chat_id, message_count, std::move(query)));
}; };
check_message_thread(chat_id, message_thread_id, reply_to_message_id, std::move(query), check_message_thread(chat_id, message_thread_id, reply_to_message_id, std::move(query),
std::move(on_message_thread_checked)); std::move(on_message_thread_checked));
@ -9206,16 +9234,17 @@ void Client::do_send_message(object_ptr<td_api::InputMessageContent> input_messa
[this, disable_notification, protect_content, input_message_content = std::move(input_message_content), [this, disable_notification, protect_content, input_message_content = std::move(input_message_content),
reply_markup = std::move(reply_markup)](int64 chat_id, int64 message_thread_id, reply_markup = std::move(reply_markup)](int64 chat_id, int64 message_thread_id,
int64 reply_to_message_id, PromisedQueryPtr query) mutable { int64 reply_to_message_id, PromisedQueryPtr query) mutable {
auto it = yet_unsent_message_count_.find(chat_id); auto &count = yet_unsent_message_count_[chat_id];
if (it != yet_unsent_message_count_.end() && it->second > MAX_CONCURRENTLY_SENT_CHAT_MESSAGES) { if (count >= MAX_CONCURRENTLY_SENT_CHAT_MESSAGES) {
return query->set_retry_after_error(60); return query->set_retry_after_error(60);
} }
count++;
send_request( send_request(
make_object<td_api::sendMessage>(chat_id, message_thread_id, reply_to_message_id, make_object<td_api::sendMessage>(chat_id, message_thread_id, reply_to_message_id,
get_message_send_options(disable_notification, protect_content), get_message_send_options(disable_notification, protect_content),
std::move(reply_markup), std::move(input_message_content)), std::move(reply_markup), std::move(input_message_content)),
td::make_unique<TdOnSendMessageCallback>(this, std::move(query))); td::make_unique<TdOnSendMessageCallback>(this, chat_id, std::move(query)));
}; };
check_message_thread(chat_id, message_thread_id, reply_to_message_id, std::move(query), check_message_thread(chat_id, message_thread_id, reply_to_message_id, std::move(query),
std::move(on_message_thread_checked)); std::move(on_message_thread_checked));
@ -9252,7 +9281,6 @@ void Client::on_sent_message(object_ptr<td_api::message> &&message, int64 query_
yet_unsent_message.send_message_query_id = query_id; yet_unsent_message.send_message_query_id = query_id;
auto emplace_result = yet_unsent_messages_.emplace(yet_unsent_message_id, yet_unsent_message); auto emplace_result = yet_unsent_messages_.emplace(yet_unsent_message_id, yet_unsent_message);
CHECK(emplace_result.second); CHECK(emplace_result.second);
yet_unsent_message_count_[chat_id]++;
auto &query = *pending_send_message_queries_[query_id]; auto &query = *pending_send_message_queries_[query_id];
query.awaited_message_count++; query.awaited_message_count++;

View File

@ -71,7 +71,7 @@ class Client final : public WebhookActor::Callback {
static constexpr int32 MAX_CERTIFICATE_FILE_SIZE = 3 << 20; static constexpr int32 MAX_CERTIFICATE_FILE_SIZE = 3 << 20;
static constexpr int32 MAX_DOWNLOAD_FILE_SIZE = 20 << 20; static constexpr int32 MAX_DOWNLOAD_FILE_SIZE = 20 << 20;
static constexpr int32 MAX_CONCURRENTLY_SENT_CHAT_MESSAGES = 1000; // some unreasonably big value static constexpr int32 MAX_CONCURRENTLY_SENT_CHAT_MESSAGES = 250; // some unreasonably big value
static constexpr std::size_t MIN_PENDING_UPDATES_WARNING = 200; static constexpr std::size_t MIN_PENDING_UPDATES_WARNING = 200;
@ -474,6 +474,8 @@ class Client final : public WebhookActor::Callback {
static td::Result<int64> get_user_id(const Query *query, Slice field_name = Slice("user_id")); static td::Result<int64> get_user_id(const Query *query, Slice field_name = Slice("user_id"));
void decrease_yet_unsent_message_count(int64 chat_id, int32 count);
int64 extract_yet_unsent_message_query_id(int64 chat_id, int64 message_id, bool *is_reply_to_message_deleted); int64 extract_yet_unsent_message_query_id(int64 chat_id, int64 message_id, bool *is_reply_to_message_deleted);
void on_message_send_succeeded(object_ptr<td_api::message> &&message, int64 old_message_id); void on_message_send_succeeded(object_ptr<td_api::message> &&message, int64 old_message_id);