// // 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/LanguagePackManager.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 void store(StorerT &storer) const { constexpr int32 LAYER = 88; using td::store; // invokeWithLayer#da9b0d0d {X:Type} layer:int query:!X = X; store(static_cast(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(0x785188b8), storer); int32 flags = 0; bool have_proxy = !is_anonymous && options.proxy.type() == Proxy::Type::Mtproto; if (have_proxy) { flags |= 1 << 0; } 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(0x75588b3f), storer); store(Slice(options.proxy.server()), storer); store(options.proxy.port(), 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