2018-12-31 22:04:05 +03:00
|
|
|
//
|
2023-01-01 00:28:08 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// 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/net/NetQuery.h"
|
|
|
|
|
2022-02-05 23:28:43 +03:00
|
|
|
#include "td/telegram/ChainId.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/telegram/Global.h"
|
2019-02-19 21:15:08 +03:00
|
|
|
#include "td/telegram/telegram_api.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2022-02-05 23:28:43 +03:00
|
|
|
#include "td/utils/algorithm.h"
|
2018-12-20 00:57:56 +03:00
|
|
|
#include "td/utils/as.h"
|
2018-06-26 02:43:11 +03:00
|
|
|
#include "td/utils/misc.h"
|
2021-05-17 15:21:11 +03:00
|
|
|
#include "td/utils/SliceBuilder.h"
|
2022-06-02 16:19:16 +03:00
|
|
|
#include "td/utils/Time.h"
|
2018-06-26 02:43:11 +03:00
|
|
|
|
2022-02-08 17:30:06 +03:00
|
|
|
#include <algorithm>
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
namespace td {
|
2018-04-24 19:21:47 +03:00
|
|
|
|
2020-10-05 18:07:23 +03:00
|
|
|
int VERBOSITY_NAME(net_query) = VERBOSITY_NAME(INFO);
|
|
|
|
|
2022-04-17 23:21:56 +03:00
|
|
|
void NetQuery::debug(string state, bool may_be_lost) {
|
|
|
|
may_be_lost_ = may_be_lost;
|
|
|
|
VLOG(net_query) << *this << " " << tag("state", state);
|
|
|
|
{
|
|
|
|
auto guard = lock();
|
|
|
|
auto &data = get_data_unsafe();
|
|
|
|
data.state_ = std::move(state);
|
|
|
|
data.state_timestamp_ = Time::now();
|
|
|
|
data.state_change_count_++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NetQuery::NetQuery(State state, uint64 id, BufferSlice &&query, BufferSlice &&answer, DcId dc_id, Type type,
|
|
|
|
AuthFlag auth_flag, GzipFlag gzip_flag, int32 tl_constructor, int32 total_timeout_limit,
|
|
|
|
NetQueryStats *stats, vector<ChainId> chain_ids)
|
|
|
|
: state_(state)
|
|
|
|
, type_(type)
|
|
|
|
, auth_flag_(auth_flag)
|
|
|
|
, gzip_flag_(gzip_flag)
|
|
|
|
, dc_id_(dc_id)
|
|
|
|
, status_()
|
|
|
|
, id_(id)
|
|
|
|
, query_(std::move(query))
|
|
|
|
, answer_(std::move(answer))
|
|
|
|
, tl_constructor_(tl_constructor)
|
|
|
|
, total_timeout_limit_(total_timeout_limit) {
|
|
|
|
CHECK(id_ != 0);
|
2022-02-06 00:16:16 +03:00
|
|
|
chain_ids_ = transform(chain_ids, [](ChainId chain_id) { return chain_id.get() == 0 ? 1 : chain_id.get(); });
|
2022-02-05 23:28:43 +03:00
|
|
|
td::unique(chain_ids_);
|
2022-04-17 23:21:56 +03:00
|
|
|
|
|
|
|
auto &data = get_data_unsafe();
|
2022-08-18 20:21:40 +03:00
|
|
|
data.my_id_ = G()->get_option_integer("my_id");
|
2022-04-17 23:21:56 +03:00
|
|
|
data.start_timestamp_ = data.state_timestamp_ = Time::now();
|
|
|
|
LOG(INFO) << *this;
|
|
|
|
if (stats) {
|
|
|
|
nq_counter_ = stats->register_query(this);
|
|
|
|
}
|
2022-02-05 23:28:43 +03:00
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
void NetQuery::on_net_write(size_t size) {
|
2023-03-13 17:26:23 +03:00
|
|
|
const auto &callbacks = G()->get_net_stats_file_callbacks();
|
|
|
|
if (static_cast<size_t>(file_type_) < callbacks.size()) {
|
|
|
|
callbacks[file_type_]->on_write(size);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
2018-04-19 20:21:26 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
void NetQuery::on_net_read(size_t size) {
|
2023-03-13 17:26:23 +03:00
|
|
|
const auto &callbacks = G()->get_net_stats_file_callbacks();
|
|
|
|
if (static_cast<size_t>(file_type_) < callbacks.size()) {
|
|
|
|
callbacks[file_type_]->on_read(size);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
2018-04-19 20:21:26 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
int32 NetQuery::tl_magic(const BufferSlice &buffer_slice) {
|
|
|
|
auto slice = buffer_slice.as_slice();
|
|
|
|
if (slice.size() < 4) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return as<int32>(slice.begin());
|
|
|
|
}
|
|
|
|
|
2018-06-08 21:42:04 +03:00
|
|
|
void NetQuery::set_error(Status status, string source) {
|
2021-05-31 21:51:48 +03:00
|
|
|
if (status.code() == Error::Resend || status.code() == Error::Canceled || status.code() == Error::ResendInvokeAfter) {
|
2018-06-08 21:42:04 +03:00
|
|
|
return set_error_impl(Status::Error(200, PSLICE() << status), std::move(source));
|
|
|
|
}
|
|
|
|
|
2018-06-25 00:52:17 +03:00
|
|
|
if (begins_with(status.message(), "INPUT_METHOD_INVALID")) {
|
2023-02-21 11:21:07 +03:00
|
|
|
LOG(ERROR) << "Receive INPUT_METHOD_INVALID for query " << format::as_hex_dump<4>(query_.as_slice());
|
2018-06-08 21:42:04 +03:00
|
|
|
}
|
|
|
|
if (status.message() == "BOT_METHOD_INVALID") {
|
2019-02-19 21:15:08 +03:00
|
|
|
auto id = tl_constructor();
|
2020-04-28 16:25:56 +03:00
|
|
|
if (id != telegram_api::help_getNearestDc::ID && id != telegram_api::help_getAppConfig::ID) {
|
2019-02-19 21:15:08 +03:00
|
|
|
LOG(ERROR) << "Receive BOT_METHOD_INVALID for query " << format::as_hex(id);
|
|
|
|
}
|
2018-06-08 21:42:04 +03:00
|
|
|
}
|
2018-06-21 22:32:56 +03:00
|
|
|
if (status.message() == "MSG_WAIT_FAILED" && status.code() != 400) {
|
|
|
|
status = Status::Error(400, "MSG_WAIT_FAILED");
|
|
|
|
}
|
2018-06-08 21:42:04 +03:00
|
|
|
set_error_impl(std::move(status), std::move(source));
|
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
} // namespace td
|