tdlight/td/telegram/net/TempAuthKeyWatchdog.h
2024-08-15 15:14:33 +03:00

152 lines
4.2 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)
//
#pragma once
#include "td/telegram/Global.h"
#include "td/telegram/net/NetQuery.h"
#include "td/telegram/net/NetQueryCreator.h"
#include "td/telegram/net/NetQueryDispatcher.h"
#include "td/telegram/telegram_api.h"
#include "td/actor/actor.h"
#include "td/utils/common.h"
#include "td/utils/format.h"
#include "td/utils/logging.h"
#include "td/utils/Time.h"
#include <map>
namespace td {
class TempAuthKeyWatchdog final : public NetQueryCallback {
class RegisteredAuthKeyImpl {
public:
explicit RegisteredAuthKeyImpl(int64 auth_key_id)
: watchdog_(G()->temp_auth_key_watchdog()), auth_key_id_(auth_key_id) {
}
RegisteredAuthKeyImpl(const RegisteredAuthKeyImpl &) = delete;
RegisteredAuthKeyImpl &operator=(const RegisteredAuthKeyImpl &) = delete;
RegisteredAuthKeyImpl(RegisteredAuthKeyImpl &&) = delete;
RegisteredAuthKeyImpl &operator=(RegisteredAuthKeyImpl &&) = delete;
~RegisteredAuthKeyImpl() {
send_closure(watchdog_, &TempAuthKeyWatchdog::unregister_auth_key_id_impl, auth_key_id_);
}
private:
ActorId<TempAuthKeyWatchdog> watchdog_;
int64 auth_key_id_;
};
public:
explicit TempAuthKeyWatchdog(ActorShared<> parent) : parent_(std::move(parent)) {
}
using RegisteredAuthKey = unique_ptr<RegisteredAuthKeyImpl>;
static RegisteredAuthKey register_auth_key_id(int64 id) {
send_closure(G()->temp_auth_key_watchdog(), &TempAuthKeyWatchdog::register_auth_key_id_impl, id);
return make_unique<RegisteredAuthKeyImpl>(id);
}
private:
static constexpr double SYNC_WAIT = 0.1;
static constexpr double SYNC_WAIT_MAX = 1.0;
static constexpr double RESYNC_DELAY = 5.0;
static constexpr int32 MAX_RESYNC_COUNT = 6;
ActorShared<> parent_;
std::map<uint64, uint32> id_count_;
double sync_at_ = 0;
int32 resync_count_ = 0;
bool need_sync_ = false;
bool run_sync_ = false;
void register_auth_key_id_impl(int64 id) {
LOG(INFO) << "Register key " << id;
if (!++id_count_[id]) {
id_count_.erase(id);
}
need_sync();
}
void unregister_auth_key_id_impl(int64 id) {
LOG(INFO) << "Unregister key " << id;
if (!--id_count_[id]) {
id_count_.erase(id);
}
need_sync();
}
void need_sync() {
need_sync_ = true;
resync_count_ = MAX_RESYNC_COUNT;
try_sync();
LOG(DEBUG) << "Need sync temp auth keys";
}
void try_sync() {
if (run_sync_) {
return;
}
if (!need_sync_) {
if (resync_count_ > 0 && id_count_.size() > 1) {
resync_count_--;
need_sync_ = true;
sync_at_ = Time::now() + RESYNC_DELAY;
set_timeout_at(sync_at_);
}
return;
}
auto now = Time::now();
if (sync_at_ == 0) {
sync_at_ = now + SYNC_WAIT_MAX;
}
LOG(DEBUG) << "Set sync timeout";
set_timeout_at(min(sync_at_, now + SYNC_WAIT));
}
void timeout_expired() final {
LOG(DEBUG) << "Sync timeout expired";
CHECK(!run_sync_);
if (!need_sync_) {
LOG(ERROR) << "Do not need sync..";
return;
}
need_sync_ = false;
run_sync_ = true;
sync_at_ = 0;
vector<int64> auth_key_ids;
for (auto &id_count : id_count_) {
auth_key_ids.push_back(id_count.first);
}
if (!G()->close_flag()) {
LOG(WARNING) << "Start auth_dropTempAuthKeys except keys " << format::as_array(auth_key_ids);
auto query = G()->net_query_creator().create_unauth(telegram_api::auth_dropTempAuthKeys(std::move(auth_key_ids)));
G()->net_query_dispatcher().dispatch_with_callback(std::move(query), actor_shared(this));
}
}
void on_result(NetQueryPtr query) final {
run_sync_ = false;
if (query->is_error()) {
if (G()->close_flag()) {
return;
}
LOG(ERROR) << "Receive error for auth_dropTempAuthKeys: " << query->error();
need_sync_ = true;
resync_count_ = MAX_RESYNC_COUNT;
} else {
LOG(INFO) << "Receive OK for auth_dropTempAuthKeys";
}
try_sync();
}
};
} // namespace td