/* This file is part of TON Blockchain Library. TON Blockchain Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. TON Blockchain Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with TON Blockchain Library. If not, see . Copyright 2017-2019 Telegram Systems LLP */ #pragma once #include "full-node-shard.h" namespace ton { namespace validator { namespace fullnode { struct Neighbour { adnl::AdnlNodeIdShort adnl_id; td::uint32 proto_version = 0; td::uint64 capabilities = 0; td::Clocks::Duration roundtrip = 0; td::Clocks::Duration roundtrip_relax_at = 0; double roundtrip_weight = 0; double unreliability = 0; Neighbour(adnl::AdnlNodeIdShort adnl_id) : adnl_id(std::move(adnl_id)) { } void update_proto_version(const ton_api::tonNode_capabilities &q); void query_success(td::Clocks::Duration t); void query_failed(); void update_roundtrip(td::Clocks::Duration t); static Neighbour zero; }; class FullNodeShardImpl : public FullNodeShard { public: WorkchainId get_workchain() const override { return shard_.workchain; } ShardId get_shard() const override { return shard_.shard; } ShardIdFull get_shard_full() const override { return shard_; } static constexpr td::uint32 download_next_priority() { return 1; } static constexpr td::uint32 proto_version() { return 1; } static constexpr td::uint64 proto_capabilities() { return 0; } static constexpr td::uint32 max_neighbours() { return 16; } static constexpr double stop_unreliability() { return 5.0; } static constexpr double fail_unreliability() { return 10.0; } void create_overlay(); void update_adnl_id(adnl::AdnlNodeIdShort adnl_id, td::Promise promise) override; //td::Result fetch_block(td::BufferSlice data); void prevalidate_block(BlockIdExt block_id, td::BufferSlice data, td::BufferSlice proof, td::Promise promise); void try_get_next_block(td::Timestamp timestamp, td::Promise promise); void got_next_block(td::Result block); void get_next_block(); template void process_query(adnl::AdnlNodeIdShort src, T &query, td::Promise promise) { promise.set_error(td::Status::Error(ErrorCode::error, "unknown query")); } void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_getNextBlockDescription &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_prepareBlockProof &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadBlockProof &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadBlockProofLink &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_prepareBlock &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadBlock &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadBlockFull &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadNextBlockFull &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_prepareZeroState &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_preparePersistentState &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_getNextKeyBlockIds &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadZeroState &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadPersistentState &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_downloadPersistentStateSlice &query, td::Promise promise); void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_getCapabilities &query, td::Promise promise); // void process_query(adnl::AdnlNodeIdShort src, ton_api::tonNode_prepareNextKeyBlockProof &query, // td::Promise promise); void receive_query(adnl::AdnlNodeIdShort src, td::BufferSlice query, td::Promise promise); void process_broadcast(PublicKeyHash src, ton_api::tonNode_blockBroadcast &query); void process_broadcast(PublicKeyHash src, ton_api::tonNode_ihrMessageBroadcast &query); void process_broadcast(PublicKeyHash src, ton_api::tonNode_externalMessageBroadcast &query); void process_broadcast(PublicKeyHash src, ton_api::tonNode_newShardBlockBroadcast &query); void receive_broadcast(PublicKeyHash src, td::BufferSlice query); void send_ihr_message(td::BufferSlice data) override; void send_external_message(td::BufferSlice data) override; void send_shard_block_info(BlockIdExt block_id, CatchainSeqno cc_seqno, td::BufferSlice data) override; void send_broadcast(BlockBroadcast broadcast) override; void download_block(BlockIdExt id, td::uint32 priority, td::Timestamp timeout, td::Promise promise) override; void download_zero_state(BlockIdExt id, td::uint32 priority, td::Timestamp timeout, td::Promise promise) override; void download_persistent_state(BlockIdExt id, BlockIdExt masterchain_block_id, td::uint32 priority, td::Timestamp timeout, td::Promise promise) override; void download_block_proof(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout, td::Promise promise) override; void download_block_proof_link(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout, td::Promise promise) override; void get_next_key_blocks(BlockIdExt block_id, td::Timestamp timeout, td::Promise> promise) override; void set_handle(BlockHandle handle, td::Promise promise) override; void start_up() override; void alarm() override; void update_validators(std::vector public_key_hashes, PublicKeyHash local_hash) override; void sign_new_certificate(PublicKeyHash sign_by); void signed_new_certificate(ton::overlay::Certificate cert); void ping_neighbours(); void reload_neighbours(); void got_neighbours(std::vector res); void update_neighbour_stats(adnl::AdnlNodeIdShort adnl_id, td::Clocks::Duration t, bool success); void got_neighbour_capabilities(adnl::AdnlNodeIdShort adnl_id, td::Clocks::Duration t, td::BufferSlice data); const Neighbour &choose_neighbour() const; template td::Promise create_neighbour_promise(const Neighbour &x, td::Promise p) { return td::PromiseCreator::lambda([id = x.adnl_id, SelfId = actor_id(this), p = std::move(p), ts = td::Time::now()](td::Result R) mutable { if (R.is_error() && R.error().code() != ErrorCode::notready && R.error().code() != ErrorCode::cancelled) { td::actor::send_closure(SelfId, &FullNodeShardImpl::update_neighbour_stats, id, td::Time::now() - ts, false); } else { td::actor::send_closure(SelfId, &FullNodeShardImpl::update_neighbour_stats, id, td::Time::now() - ts, true); } p.set_result(std::move(R)); }); } FullNodeShardImpl(ShardIdFull shard, PublicKeyHash local_id, adnl::AdnlNodeIdShort adnl_id, FileHash zero_state_file_hash, td::actor::ActorId keyring, td::actor::ActorId adnl, td::actor::ActorId rldp, td::actor::ActorId overlays, td::actor::ActorId validator_manager, td::actor::ActorId client); private: bool use_new_download() const { return false; } ShardIdFull shard_; BlockHandle handle_; td::Promise promise_; PublicKeyHash local_id_; adnl::AdnlNodeIdShort adnl_id_; FileHash zero_state_file_hash_; td::actor::ActorId keyring_; td::actor::ActorId adnl_; td::actor::ActorId rldp_; td::actor::ActorId overlays_; td::actor::ActorId validator_manager_; td::actor::ActorId client_; td::uint32 attempt_ = 0; overlay::OverlayIdFull overlay_id_full_; overlay::OverlayIdShort overlay_id_; PublicKeyHash sign_cert_by_ = PublicKeyHash::zero(); td::Timestamp update_certificate_at_; td::Timestamp sync_completed_at_; std::shared_ptr cert_; overlay::OverlayPrivacyRules rules_; std::map neighbours_; td::Timestamp reload_neighbours_at_; td::Timestamp ping_neighbours_at_; adnl::AdnlNodeIdShort last_pinged_neighbour_ = adnl::AdnlNodeIdShort::zero(); }; } // namespace fullnode } // namespace validator } // namespace ton