Improve authorization errors handling.

This commit is contained in:
levlam 2022-12-12 19:33:35 +03:00
parent ed9f836977
commit 5166914162
2 changed files with 53 additions and 11 deletions

View File

@ -2789,13 +2789,12 @@ class Client::TdOnAuthorizationCallback final : public TdQueryCallback {
bool was_ready = client_->authorization_state_->get_id() != td_api::authorizationStateWaitPhoneNumber::ID; bool was_ready = client_->authorization_state_->get_id() != td_api::authorizationStateWaitPhoneNumber::ID;
if (result->get_id() == td_api::error::ID) { if (result->get_id() == td_api::error::ID) {
auto error = move_object_as<td_api::error>(result); auto error = move_object_as<td_api::error>(result);
if (error->code_ == 429 || error->code_ >= 500 || (error->code_ != 401 && was_ready)) { if (error->code_ != 401 && was_ready) {
// try again // try again
return client_->on_update_authorization_state(); return client_->on_update_authorization_state();
} }
LOG(WARNING) << "Logging out due to " << td::oneline(to_string(error)); client_->log_out(error->code_, error->message_);
client_->log_out(error->message_ == "API_ID_INVALID");
} else if (was_ready) { } else if (was_ready) {
client_->on_update_authorization_state(); client_->on_update_authorization_state();
} }
@ -3966,8 +3965,18 @@ void Client::close() {
} }
} }
void Client::log_out(bool is_api_id_invalid) { void Client::log_out(int32 error_code, Slice error_message) {
is_api_id_invalid_ |= is_api_id_invalid; LOG(WARNING) << "Logging out due to error " << error_code << ": " << error_message;
if (error_message == "API_ID_INVALID") {
is_api_id_invalid_ = true;
} else if (error_code == 429) {
auto retry_after_time = get_retry_after_time(error_message);
if (retry_after_time > 0) {
next_authorization_time_ = td::max(next_authorization_time_, td::Time::now() + retry_after_time);
}
} else if (error_code >= 500) {
next_authorization_time_ = td::max(next_authorization_time_, td::Time::now() + 1);
}
if (!td_client_.empty() && !logging_out_ && !closing_) { if (!td_client_.empty() && !logging_out_ && !closing_) {
do_send_request(make_object<td_api::logOut>(), td::make_unique<TdOnOkCallback>()); do_send_request(make_object<td_api::logOut>(), td::make_unique<TdOnOkCallback>());
} }
@ -4763,7 +4772,7 @@ void Client::on_update_authorization_state() {
case td_api::authorizationStateClosed::ID: case td_api::authorizationStateClosed::ID:
return on_closed(); return on_closed();
default: default:
return log_out(false); // just in case return log_out(500, "Unknown authorization state"); // just in case
} }
} }
@ -5201,7 +5210,18 @@ void Client::finish_closing() {
clear_tqueue(); clear_tqueue();
} }
set_timeout_in(need_close_ ? 0 : 600); if (need_close_) {
return stop();
}
auto timeout = [&] {
if (next_authorization_time_ <= 0.0) {
return 600.0;
}
return td::min(next_authorization_time_ - td::Time::now(), 600.0);
}();
set_timeout_in(timeout);
LOG(INFO) << "Keep client opened for " << timeout << " seconds";
} }
void Client::timeout_expired() { void Client::timeout_expired() {
@ -7209,7 +7229,7 @@ void Client::on_message_send_failed(int64 chat_id, int64 old_message_id, int64 n
void Client::on_cmd(PromisedQueryPtr query) { void Client::on_cmd(PromisedQueryPtr query) {
LOG(DEBUG) << "Process query " << *query; LOG(DEBUG) << "Process query " << *query;
if (!td_client_.empty()) { if (!td_client_.empty() && was_authorized_) {
if (query->method() == "close") { if (query->method() == "close") {
auto retry_after = static_cast<int>(10 * 60 - (td::Time::now() - start_time_)); auto retry_after = static_cast<int>(10 * 60 - (td::Time::now() - start_time_));
if (retry_after > 0 && start_time_ > parameters_->start_time_ + 10 * 60) { if (retry_after > 0 && start_time_ > parameters_->start_time_ + 10 * 60) {
@ -9325,16 +9345,33 @@ void Client::fail_query_conflict(Slice message, PromisedQueryPtr &&query) {
void Client::fail_query_closing(PromisedQueryPtr &&query) { void Client::fail_query_closing(PromisedQueryPtr &&query) {
auto error = get_closing_error(); auto error = get_closing_error();
fail_query(error.code, error.message, std::move(query)); if (error.retry_after > 0) {
query->set_retry_after_error(error.retry_after);
} else {
fail_query(error.code, error.message, std::move(query));
}
} }
Client::ClosingError Client::get_closing_error() { Client::ClosingError Client::get_closing_error() {
ClosingError result; ClosingError result;
result.retry_after = 0;
if (logging_out_) { if (logging_out_) {
result.code = 401;
if (is_api_id_invalid_) { if (is_api_id_invalid_) {
result.code = 401;
result.message = Slice("Unauthorized: invalid api-id/api-hash"); result.message = Slice("Unauthorized: invalid api-id/api-hash");
} else if (next_authorization_time_ > 0.0) {
result.code = 429;
result.retry_after = td::max(static_cast<int>(next_authorization_time_ - td::Time::now()), 0) + 1;
if (result.retry_after != prev_retry_after) {
prev_retry_after = result.retry_after;
retry_after_error_message = PSTRING() << "Too Many Requests: retry after " << result.retry_after;
}
result.message = retry_after_error_message;
} else if (clear_tqueue_) {
result.code = 400;
result.message = Slice("Logged out");
} else { } else {
result.code = 401;
result.message = Slice("Unauthorized"); result.message = Slice("Unauthorized");
} }
} else { } else {

View File

@ -308,7 +308,7 @@ class Client final : public WebhookActor::Callback {
void on_result(td::uint64 id, object_ptr<td_api::Object> result); void on_result(td::uint64 id, object_ptr<td_api::Object> result);
void on_update_authorization_state(); void on_update_authorization_state();
void log_out(bool is_api_id_invalid); void log_out(int32 error_code, Slice error_message);
void on_closed(); void on_closed();
void finish_closing(); void finish_closing();
@ -613,6 +613,7 @@ class Client final : public WebhookActor::Callback {
struct ClosingError { struct ClosingError {
int code; int code;
int retry_after;
Slice message; Slice message;
}; };
ClosingError get_closing_error(); ClosingError get_closing_error();
@ -955,6 +956,10 @@ class Client final : public WebhookActor::Callback {
int64 my_id_ = -1; int64 my_id_ = -1;
int32 authorization_date_ = -1; int32 authorization_date_ = -1;
double next_authorization_time_ = 0;
int32 prev_retry_after = 0;
td::string retry_after_error_message;
int64 group_anonymous_bot_user_id_ = 0; int64 group_anonymous_bot_user_id_ = 0;
int64 channel_bot_user_id_ = 0; int64 channel_bot_user_id_ = 0;