This repository has been archived on 2020-05-25. You can view files and clone it, but cannot push or open issues or pull requests.
tdlib-fork/td/telegram/net/MtprotoHeader.cpp

97 lines
3.0 KiB
C++
Raw Normal View History

//
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2019
//
// 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/MtprotoHeader.h"
#include "td/telegram/JsonValue.h"
#include "td/telegram/LanguagePackManager.h"
#include "td/tl/tl_object_store.h"
#include "td/utils/tl_helpers.h"
namespace td {
namespace {
class HeaderStorer {
public:
HeaderStorer(const MtprotoHeader::Options &options, bool is_anonymous)
: options(options), is_anonymous(is_anonymous) {
}
template <class StorerT>
void store(StorerT &storer) const {
constexpr int32 LAYER = 98;
using td::store;
// invokeWithLayer#da9b0d0d {X:Type} layer:int query:!X = X;
store(static_cast<int32>(0xda9b0d0d), storer);
store(LAYER, storer);
// initConnection#785188b8 {X:Type} flags:# api_id:int device_model:string system_version:string app_version:string
// system_lang_code:string lang_pack:string lang_code:string proxy:flags.0?InputClientProxy query:!X = X;
store(static_cast<int32>(0x785188b8), storer);
int32 flags = 0;
bool have_proxy = !is_anonymous && options.proxy.type() == Proxy::Type::Mtproto;
if (have_proxy) {
flags |= 1 << 0;
}
if (!options.parameters.empty()) {
flags |= 1 << 1;
}
if (options.is_emulator) {
flags |= 1 << 10;
}
store(flags, storer);
store(options.api_id, storer);
if (is_anonymous) {
store(Slice("n/a"), storer);
store(Slice("n/a"), storer);
} else {
store(options.device_model, storer);
store(options.system_version, storer);
}
store(options.application_version, storer);
store(options.system_language_code, storer);
if (is_anonymous || options.language_pack.empty() ||
LanguagePackManager::is_custom_language_code(options.language_code)) {
store(Slice(), storer);
store(Slice(), storer);
} else {
store(options.language_pack, storer);
if (options.language_code.empty()) {
store(Slice("en"), storer);
} else {
store(options.language_code, storer);
}
}
if (have_proxy) {
// inputClientProxy#75588b3f address:string port:int = InputClientProxy;
store(static_cast<int32>(0x75588b3f), storer);
store(Slice(options.proxy.server()), storer);
store(options.proxy.port(), storer);
}
if (!options.parameters.empty()) {
auto parameters_copy = options.parameters;
auto json_value = get_input_json_value(parameters_copy).move_as_ok();
CHECK(json_value != nullptr);
TlStoreBoxedUnknown<TlStoreObject>::store(json_value, storer);
}
}
private:
const MtprotoHeader::Options &options;
bool is_anonymous;
};
} // namespace
string MtprotoHeader::gen_header(const MtprotoHeader::Options &options, bool is_anonymous) {
HeaderStorer storer(options, is_anonymous);
return serialize(storer);
}
} // namespace td