mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			718 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			718 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|     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 <http://www.gnu.org/licenses/>.
 | |
| 
 | |
|     Copyright 2017-2020 Telegram Systems LLP
 | |
| */
 | |
| #include "full-node.hpp"
 | |
| #include "ton/ton-io.hpp"
 | |
| #include "td/actor/MultiPromise.h"
 | |
| #include "full-node.h"
 | |
| #include "common/delay.h"
 | |
| 
 | |
| namespace ton {
 | |
| 
 | |
| namespace validator {
 | |
| 
 | |
| namespace fullnode {
 | |
| 
 | |
| static const double INACTIVE_SHARD_TTL = (double)overlay::Overlays::overlay_peer_ttl() + 60.0;
 | |
| 
 | |
| void FullNodeImpl::add_permanent_key(PublicKeyHash key, td::Promise<td::Unit> promise) {
 | |
|   if (local_keys_.count(key)) {
 | |
|     promise.set_value(td::Unit());
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   local_keys_.insert(key);
 | |
| 
 | |
|   if (!sign_cert_by_.is_zero()) {
 | |
|     promise.set_value(td::Unit());
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   for (auto &x : all_validators_) {
 | |
|     if (x == key) {
 | |
|       sign_cert_by_ = key;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (auto &shard : shards_) {
 | |
|     if (!shard.second.actor.empty()) {
 | |
|       td::actor::send_closure(shard.second.actor, &FullNodeShard::update_validators, all_validators_, sign_cert_by_);
 | |
|     }
 | |
|   }
 | |
|   // create_private_block_overlay(key);
 | |
|   promise.set_value(td::Unit());
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::del_permanent_key(PublicKeyHash key, td::Promise<td::Unit> promise) {
 | |
|   if (!local_keys_.count(key)) {
 | |
|     promise.set_value(td::Unit());
 | |
|     return;
 | |
|   }
 | |
|   local_keys_.erase(key);
 | |
|   if (sign_cert_by_ != key) {
 | |
|     promise.set_value(td::Unit());
 | |
|     return;
 | |
|   }
 | |
|   sign_cert_by_ = PublicKeyHash::zero();
 | |
| 
 | |
|   for (auto &x : all_validators_) {
 | |
|     if (local_keys_.count(x)) {
 | |
|       sign_cert_by_ = x;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (auto &shard : shards_) {
 | |
|     if (!shard.second.actor.empty()) {
 | |
|       td::actor::send_closure(shard.second.actor, &FullNodeShard::update_validators, all_validators_, sign_cert_by_);
 | |
|     }
 | |
|   }
 | |
|   // private_block_overlays_.erase(key);
 | |
|   promise.set_value(td::Unit());
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::add_collator_adnl_id(adnl::AdnlNodeIdShort id) {
 | |
|   ++local_collator_nodes_[id];
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::del_collator_adnl_id(adnl::AdnlNodeIdShort id) {
 | |
|   if (--local_collator_nodes_[id] == 0) {
 | |
|     local_collator_nodes_.erase(id);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::sign_shard_overlay_certificate(ShardIdFull shard_id, PublicKeyHash signed_key, td::uint32 expiry_at,
 | |
|                                                   td::uint32 max_size, td::Promise<td::BufferSlice> promise) {
 | |
|   auto it = shards_.find(shard_id);
 | |
|   if(it == shards_.end() || it->second.actor.empty()) {
 | |
|     promise.set_error(td::Status::Error(ErrorCode::error, "shard not found"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(it->second.actor, &FullNodeShard::sign_overlay_certificate, signed_key, expiry_at, max_size,
 | |
|                           std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::import_shard_overlay_certificate(ShardIdFull shard_id, PublicKeyHash signed_key,
 | |
|                                                     std::shared_ptr<ton::overlay::Certificate> cert,
 | |
|                                                     td::Promise<td::Unit> promise) {
 | |
|   auto it = shards_.find(shard_id);
 | |
|   if(it == shards_.end() || it->second.actor.empty()) {
 | |
|     promise.set_error(td::Status::Error(ErrorCode::error, "shard not found"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(it->second.actor, &FullNodeShard::import_overlay_certificate, signed_key, cert,
 | |
|                           std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::update_adnl_id(adnl::AdnlNodeIdShort adnl_id, td::Promise<td::Unit> promise) {
 | |
|   adnl_id_ = adnl_id;
 | |
| 
 | |
|   td::MultiPromise mp;
 | |
|   auto ig = mp.init_guard();
 | |
|   ig.add_promise(std::move(promise));
 | |
| 
 | |
|   for (auto &s : shards_) {
 | |
|     if (!s.second.actor.empty()) {
 | |
|       td::actor::send_closure(s.second.actor, &FullNodeShard::update_adnl_id, adnl_id, ig.get_promise());
 | |
|     }
 | |
|   }
 | |
|   local_id_ = adnl_id_.pubkey_hash();
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::set_config(FullNodeConfig config) {
 | |
|   config_ = config;
 | |
|   for (auto& s : shards_) {
 | |
|     if (!s.second.actor.empty()) {
 | |
|       td::actor::send_closure(s.second.actor, &FullNodeShard::set_config, config);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::initial_read_complete(BlockHandle top_handle) {
 | |
|   auto P = td::PromiseCreator::lambda([SelfId = actor_id(this)](td::Result<td::Unit> R) {
 | |
|     R.ensure();
 | |
|     td::actor::send_closure(SelfId, &FullNodeImpl::sync_completed);
 | |
|   });
 | |
|   auto it = shards_.find(ShardIdFull{masterchainId});
 | |
|   CHECK(it != shards_.end() && !it->second.actor.empty());
 | |
|   td::actor::send_closure(it->second.actor, &FullNodeShard::set_handle, top_handle, std::move(P));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::update_shard_configuration(td::Ref<MasterchainState> state, std::set<ShardIdFull> shards_to_monitor,
 | |
|                                               std::set<ShardIdFull> temporary_shards) {
 | |
|   CHECK(shards_to_monitor.count(ShardIdFull(masterchainId)));
 | |
|   std::set<ShardIdFull> new_shards;
 | |
|   std::map<ShardIdFull, FullNodeShardMode> new_active;
 | |
|   new_shards.insert(ShardIdFull(masterchainId));
 | |
|   std::set<WorkchainId> workchains;
 | |
|   auto cut_shard = [&](ShardIdFull shard) -> ShardIdFull {
 | |
|     int min_split = state->monitor_min_split_depth(shard.workchain);
 | |
|     return min_split < shard.pfx_len() ? shard_prefix(shard, min_split) : shard;
 | |
|   };
 | |
|   auto set_active = [&](ShardIdFull shard, FullNodeShardMode mode) {
 | |
|     while (new_active.emplace(shard, mode).second && shard.pfx_len() > 0) {
 | |
|       shard = shard_parent(shard);
 | |
|     }
 | |
|   };
 | |
|   for (auto &info : state->get_shards()) {
 | |
|     workchains.insert(info->shard().workchain);
 | |
|     new_shards.insert(cut_shard(info->shard()));
 | |
|   }
 | |
|   for (const auto &wpair : state->get_workchain_list()) {
 | |
|     ton::WorkchainId wc = wpair.first;
 | |
|     const block::WorkchainInfo *winfo = wpair.second.get();
 | |
|     if (workchains.count(wc) == 0 && winfo->active && winfo->enabled_since <= state->get_unix_time()) {
 | |
|       new_shards.insert(ShardIdFull(wc));
 | |
|     }
 | |
|   }
 | |
|   for (ShardIdFull shard : shards_to_monitor) {
 | |
|     set_active(cut_shard(shard), FullNodeShardMode::active);
 | |
|   }
 | |
|   for (ShardIdFull shard : temporary_shards) {
 | |
|     set_active(cut_shard(shard), FullNodeShardMode::active_temp);
 | |
|   }
 | |
| 
 | |
|   auto info_set_mode = [&](ShardIdFull shard, ShardInfo& info, FullNodeShardMode mode) {
 | |
|     if (info.mode == mode) {
 | |
|       return;
 | |
|     }
 | |
|     if (info.actor.empty()) {
 | |
|       add_shard_actor(shard, mode);
 | |
|       return;
 | |
|     }
 | |
|     info.mode = mode;
 | |
|     td::actor::send_closure(info.actor, &FullNodeShard::set_mode, mode);
 | |
|     info.delete_at =
 | |
|         mode != FullNodeShardMode::inactive ? td::Timestamp::never() : td::Timestamp::in(INACTIVE_SHARD_TTL);
 | |
|   };
 | |
| 
 | |
|   for (auto shard : new_shards) {
 | |
|     auto &info = shards_[shard];
 | |
|     info.exists = true;
 | |
|   }
 | |
| 
 | |
|   for (auto& p : shards_) {
 | |
|     ShardIdFull shard = p.first;
 | |
|     ShardInfo &info = p.second;
 | |
|     info.exists = new_shards.count(shard);
 | |
|     auto it = new_active.find(shard);
 | |
|     info_set_mode(shard, info, it == new_active.end() ? FullNodeShardMode::inactive : it->second);
 | |
|   }
 | |
| 
 | |
|   for (const auto& s : new_active) {
 | |
|     info_set_mode(s.first, shards_[s.first], s.second);
 | |
|   }
 | |
| 
 | |
|   auto it = shards_.begin();
 | |
|   while (it != shards_.end()) {
 | |
|     if (it->second.mode == FullNodeShardMode::inactive && it->second.delete_at && it->second.delete_at.is_in_past()) {
 | |
|       it->second.actor.reset();
 | |
|       it->second.delete_at = td::Timestamp::never();
 | |
|     }
 | |
|     if (!it->second.exists && it->second.actor.empty()) {
 | |
|       it = shards_.erase(it);
 | |
|     } else {
 | |
|       ++it;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   std::set<adnl::AdnlNodeIdShort> my_adnl_ids;
 | |
|   for (const auto &p : local_collator_nodes_) {
 | |
|     my_adnl_ids.insert(p.first);
 | |
|   }
 | |
|   for (auto key : local_keys_) {
 | |
|     auto it = current_validators_.find(key);
 | |
|     if (it != current_validators_.end()) {
 | |
|       my_adnl_ids.insert(it->second);
 | |
|     }
 | |
|   }
 | |
|   private_block_overlays_.update_overlays(state, std::move(my_adnl_ids), zero_state_file_hash_, keyring_, adnl_, rldp_,
 | |
|                                           rldp2_, overlays_, validator_manager_);
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::add_shard_actor(ShardIdFull shard, FullNodeShardMode mode) {
 | |
|   ShardInfo &info = shards_[shard];
 | |
|   if (!info.actor.empty()) {
 | |
|     return;
 | |
|   }
 | |
|   info.actor = FullNodeShard::create(shard, local_id_, adnl_id_, zero_state_file_hash_, config_, keyring_, adnl_, rldp_,
 | |
|                                      rldp2_, overlays_, validator_manager_, client_, mode);
 | |
|   info.mode = mode;
 | |
|   info.delete_at = mode != FullNodeShardMode::inactive ? td::Timestamp::never() : td::Timestamp::in(INACTIVE_SHARD_TTL);
 | |
|   if (all_validators_.size() > 0) {
 | |
|     td::actor::send_closure(info.actor, &FullNodeShard::update_validators, all_validators_, sign_cert_by_);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::sync_completed() {
 | |
|   td::actor::send_closure(validator_manager_, &ValidatorManagerInterface::sync_complete, [](td::Unit) {});
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::send_ihr_message(AccountIdPrefixFull dst, td::BufferSlice data) {
 | |
|   auto shard = get_shard(dst);
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping OUT ihr message to unknown shard";
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::send_ihr_message, std::move(data));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::send_ext_message(AccountIdPrefixFull dst, td::BufferSlice data) {
 | |
|   auto shard = get_shard(dst);
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping OUT ext message to unknown shard";
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::send_external_message, std::move(data));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::send_shard_block_info(BlockIdExt block_id, CatchainSeqno cc_seqno, td::BufferSlice data) {
 | |
|   auto shard = get_shard(ShardIdFull{masterchainId});
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping OUT shard block info message to unknown shard";
 | |
|     return;
 | |
|   }
 | |
|   auto private_overlay = private_block_overlays_.choose_overlay(ShardIdFull(masterchainId));
 | |
|   if (!private_overlay.empty()) {
 | |
|     td::actor::send_closure(private_overlay, &FullNodePrivateOverlayV2::send_shard_block_info, block_id, cc_seqno,
 | |
|                             data.clone());
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::send_shard_block_info, block_id, cc_seqno, std::move(data));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::send_broadcast(BlockBroadcast broadcast) {
 | |
|   auto shard = get_shard(broadcast.block_id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping OUT broadcast to unknown shard";
 | |
|     return;
 | |
|   }
 | |
|   auto private_overlay = private_block_overlays_.choose_overlay(broadcast.block_id.shard_full());
 | |
|   if (!private_overlay.empty()) {
 | |
|     td::actor::send_closure(private_overlay, &FullNodePrivateOverlayV2::send_broadcast, broadcast.clone());
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::send_broadcast, std::move(broadcast));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_block(BlockIdExt id, td::uint32 priority, td::Timestamp timeout,
 | |
|                                   td::Promise<ReceivedBlock> promise) {
 | |
|   auto shard = get_shard(id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download block query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_block, id, priority, timeout, std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_zero_state(BlockIdExt id, td::uint32 priority, td::Timestamp timeout,
 | |
|                                        td::Promise<td::BufferSlice> promise) {
 | |
|   auto shard = get_shard(id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download state query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_zero_state, id, priority, timeout, std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_persistent_state(BlockIdExt id, BlockIdExt masterchain_block_id, td::uint32 priority,
 | |
|                                              td::Timestamp timeout, td::Promise<td::BufferSlice> promise) {
 | |
|   auto shard = get_shard(id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download state diff query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_persistent_state, id, masterchain_block_id, priority, timeout,
 | |
|                           std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_block_proof(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout,
 | |
|                                         td::Promise<td::BufferSlice> promise) {
 | |
|   auto shard = get_shard(block_id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download proof query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_block_proof, block_id, priority, timeout, std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_block_proof_link(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout,
 | |
|                                              td::Promise<td::BufferSlice> promise) {
 | |
|   auto shard = get_shard(block_id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download proof link query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_block_proof_link, block_id, priority, timeout,
 | |
|                           std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::get_next_key_blocks(BlockIdExt block_id, td::Timestamp timeout,
 | |
|                                        td::Promise<std::vector<BlockIdExt>> promise) {
 | |
|   auto shard = get_shard(block_id.shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download proof link query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::get_next_key_blocks, block_id, timeout, std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_archive(BlockSeqno masterchain_seqno, std::string tmp_dir, td::Timestamp timeout,
 | |
|                                     td::Promise<std::string> promise) {
 | |
|   auto shard = get_shard(ShardIdFull{masterchainId});
 | |
|   CHECK(!shard.empty());
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_archive, masterchain_seqno, std::move(tmp_dir), timeout,
 | |
|                           std::move(promise));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::download_out_msg_queue_proof(ShardIdFull dst_shard, std::vector<BlockIdExt> blocks,
 | |
|                                                 block::ImportedMsgQueueLimits limits, td::Timestamp timeout,
 | |
|                                                 td::Promise<std::vector<td::Ref<OutMsgQueueProof>>> promise) {
 | |
|   if (blocks.empty()) {
 | |
|     promise.set_value({});
 | |
|     return;
 | |
|   }
 | |
|   // All blocks are expected to have the same minsplit shard prefix
 | |
|   auto shard = get_shard(blocks[0].shard_full());
 | |
|   if (shard.empty()) {
 | |
|     VLOG(FULL_NODE_WARNING) << "dropping download msg queue query to unknown shard";
 | |
|     promise.set_error(td::Status::Error(ErrorCode::notready, "shard not ready"));
 | |
|     return;
 | |
|   }
 | |
|   td::actor::send_closure(shard, &FullNodeShard::download_out_msg_queue_proof, dst_shard, std::move(blocks), limits,
 | |
|                           timeout, std::move(promise));
 | |
| }
 | |
| 
 | |
| td::actor::ActorId<FullNodeShard> FullNodeImpl::get_shard(ShardIdFull shard) {
 | |
|   ShardIdFull shard0 = shard;
 | |
|   while (true) {
 | |
|     auto it = shards_.find(shard);
 | |
|     if (it != shards_.end() && it->second.exists) {
 | |
|       if (it->second.actor.empty()) {
 | |
|         add_shard_actor(shard, FullNodeShardMode::inactive);
 | |
|       }
 | |
|       if (it->second.mode == FullNodeShardMode::inactive) {
 | |
|         it->second.delete_at = td::Timestamp::in(INACTIVE_SHARD_TTL);
 | |
|       }
 | |
|       return it->second.actor.get();
 | |
|     }
 | |
|     if (shard.pfx_len() == 0) {
 | |
|       break;
 | |
|     }
 | |
|     shard = shard_parent(shard);
 | |
|   }
 | |
|   shard = shard0;
 | |
|   auto it = shards_.find(shard);
 | |
|   if (it == shards_.end()) {
 | |
|     it = shards_.emplace(shard = ShardIdFull(shard.workchain), ShardInfo{}).first;
 | |
|   }
 | |
|   auto &info = it->second;
 | |
|   if (info.actor.empty()) {
 | |
|     add_shard_actor(shard, FullNodeShardMode::inactive);
 | |
|   }
 | |
|   if (info.mode == FullNodeShardMode::inactive) {
 | |
|     info.delete_at = td::Timestamp::in(INACTIVE_SHARD_TTL);
 | |
|   }
 | |
|   return info.actor.get();
 | |
| }
 | |
| 
 | |
| td::actor::ActorId<FullNodeShard> FullNodeImpl::get_shard(AccountIdPrefixFull dst) {
 | |
|   return get_shard(shard_prefix(dst, max_shard_pfx_len));
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::got_key_block_proof(td::Ref<ProofLink> proof) {
 | |
|   auto R = proof->get_key_block_config();
 | |
|   R.ensure();
 | |
|   auto config = R.move_as_ok();
 | |
| 
 | |
|   PublicKeyHash l = PublicKeyHash::zero();
 | |
|   std::vector<PublicKeyHash> keys;
 | |
|   std::map<PublicKeyHash, adnl::AdnlNodeIdShort> current_validators;
 | |
|   for (td::int32 i = -1; i <= 1; i++) {
 | |
|     auto r = config->get_total_validator_set(i < 0 ? i : 1 - i);
 | |
|     if (r.not_null()) {
 | |
|       auto vec = r->export_vector();
 | |
|       for (auto &el : vec) {
 | |
|         auto key = ValidatorFullId{el.key}.compute_short_id();
 | |
|         keys.push_back(key);
 | |
|         if (local_keys_.count(key)) {
 | |
|           l = key;
 | |
|         }
 | |
|         if (i == 1) {
 | |
|           current_validators[key] = adnl::AdnlNodeIdShort{el.addr.is_zero() ? key.bits256_value() : el.addr};
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (current_validators != current_validators_) {
 | |
|     current_validators_ = std::move(current_validators);
 | |
|     // update_private_block_overlays();
 | |
|   }
 | |
| 
 | |
|   if (keys == all_validators_) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   all_validators_ = keys;
 | |
|   sign_cert_by_ = l;
 | |
|   CHECK(all_validators_.size() > 0);
 | |
| 
 | |
|   for (auto &shard : shards_) {
 | |
|     if (!shard.second.actor.empty()) {
 | |
|       td::actor::send_closure(shard.second.actor, &FullNodeShard::update_validators, all_validators_, sign_cert_by_);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::got_zero_block_state(td::Ref<ShardState> state) {
 | |
|   auto m = td::Ref<MasterchainState>{std::move(state)};
 | |
| 
 | |
|   PublicKeyHash l = PublicKeyHash::zero();
 | |
|   std::vector<PublicKeyHash> keys;
 | |
|   std::map<PublicKeyHash, adnl::AdnlNodeIdShort> current_validators;
 | |
|   for (td::int32 i = -1; i <= 1; i++) {
 | |
|     auto r = m->get_total_validator_set(i < 0 ? i : 1 - i);
 | |
|     if (r.not_null()) {
 | |
|       auto vec = r->export_vector();
 | |
|       for (auto &el : vec) {
 | |
|         auto key = ValidatorFullId{el.key}.compute_short_id();
 | |
|         keys.push_back(key);
 | |
|         if (local_keys_.count(key)) {
 | |
|           l = key;
 | |
|         }
 | |
|         if (i == 1) {
 | |
|           current_validators[key] = adnl::AdnlNodeIdShort{el.addr.is_zero() ? key.bits256_value() : el.addr};
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (current_validators != current_validators_) {
 | |
|     current_validators_ = std::move(current_validators);
 | |
|     // update_private_block_overlays();
 | |
|   }
 | |
| 
 | |
|   if (keys == all_validators_) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   all_validators_ = keys;
 | |
|   sign_cert_by_ = l;
 | |
|   CHECK(all_validators_.size() > 0);
 | |
| 
 | |
|   for (auto &shard : shards_) {
 | |
|     if (!shard.second.actor.empty()) {
 | |
|       td::actor::send_closure(shard.second.actor, &FullNodeShard::update_validators, all_validators_, sign_cert_by_);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::new_key_block(BlockHandle handle) {
 | |
|   if (handle->id().seqno() == 0) {
 | |
|     auto P = td::PromiseCreator::lambda([SelfId = actor_id(this)](td::Result<td::Ref<ShardState>> R) {
 | |
|       if (R.is_error()) {
 | |
|         VLOG(FULL_NODE_WARNING) << "failed to get zero state: " << R.move_as_error();
 | |
|       } else {
 | |
|         td::actor::send_closure(SelfId, &FullNodeImpl::got_zero_block_state, R.move_as_ok());
 | |
|       }
 | |
|     });
 | |
|     td::actor::send_closure(validator_manager_, &ValidatorManagerInterface::get_shard_state_from_db, handle,
 | |
|                             std::move(P));
 | |
|   } else {
 | |
|     CHECK(handle->is_key_block());
 | |
|     auto P = td::PromiseCreator::lambda([SelfId = actor_id(this)](td::Result<td::Ref<ProofLink>> R) {
 | |
|       if (R.is_error()) {
 | |
|         VLOG(FULL_NODE_WARNING) << "failed to get key block proof: " << R.move_as_error();
 | |
|       } else {
 | |
|         td::actor::send_closure(SelfId, &FullNodeImpl::got_key_block_proof, R.move_as_ok());
 | |
|       }
 | |
|     });
 | |
|     td::actor::send_closure(validator_manager_, &ValidatorManagerInterface::get_block_proof_link_from_db, handle,
 | |
|                             std::move(P));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::start_up() {
 | |
|   if (local_id_.is_zero()) {
 | |
|     if (adnl_id_.is_zero()) {
 | |
|       auto pk = ton::PrivateKey{ton::privkeys::Ed25519::random()};
 | |
|       local_id_ = pk.compute_short_id();
 | |
| 
 | |
|       td::actor::send_closure(keyring_, &ton::keyring::Keyring::add_key, std::move(pk), true, [](td::Unit) {});
 | |
|     } else {
 | |
|       local_id_ = adnl_id_.pubkey_hash();
 | |
|     }
 | |
|   }
 | |
|   class Callback : public ValidatorManagerInterface::Callback {
 | |
|    public:
 | |
|     void initial_read_complete(BlockHandle handle) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::initial_read_complete, handle);
 | |
|     }
 | |
|     void update_shard_configuration(td::Ref<MasterchainState> state, std::set<ShardIdFull> shards_to_monitor,
 | |
|                                     std::set<ShardIdFull> temporary_shards) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::update_shard_configuration, std::move(state),
 | |
|                               std::move(shards_to_monitor), std::move(temporary_shards));
 | |
|     }
 | |
|     void send_ihr_message(AccountIdPrefixFull dst, td::BufferSlice data) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::send_ihr_message, dst, std::move(data));
 | |
|     }
 | |
|     void send_ext_message(AccountIdPrefixFull dst, td::BufferSlice data) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::send_ext_message, dst, std::move(data));
 | |
|     }
 | |
|     void send_shard_block_info(BlockIdExt block_id, CatchainSeqno cc_seqno, td::BufferSlice data) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::send_shard_block_info, block_id, cc_seqno, std::move(data));
 | |
|     }
 | |
|     void send_broadcast(BlockBroadcast broadcast) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::send_broadcast, std::move(broadcast));
 | |
|     }
 | |
|     void download_block(BlockIdExt id, td::uint32 priority, td::Timestamp timeout,
 | |
|                         td::Promise<ReceivedBlock> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_block, id, priority, timeout, std::move(promise));
 | |
|     }
 | |
|     void download_zero_state(BlockIdExt id, td::uint32 priority, td::Timestamp timeout,
 | |
|                              td::Promise<td::BufferSlice> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_zero_state, id, priority, timeout, std::move(promise));
 | |
|     }
 | |
|     void download_persistent_state(BlockIdExt id, BlockIdExt masterchain_block_id, td::uint32 priority,
 | |
|                                    td::Timestamp timeout, td::Promise<td::BufferSlice> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_persistent_state, id, masterchain_block_id, priority,
 | |
|                               timeout, std::move(promise));
 | |
|     }
 | |
|     void download_block_proof(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout,
 | |
|                               td::Promise<td::BufferSlice> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_block_proof, block_id, priority, timeout,
 | |
|                               std::move(promise));
 | |
|     }
 | |
|     void download_block_proof_link(BlockIdExt block_id, td::uint32 priority, td::Timestamp timeout,
 | |
|                                    td::Promise<td::BufferSlice> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_block_proof_link, block_id, priority, timeout,
 | |
|                               std::move(promise));
 | |
|     }
 | |
|     void get_next_key_blocks(BlockIdExt block_id, td::Timestamp timeout,
 | |
|                              td::Promise<std::vector<BlockIdExt>> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::get_next_key_blocks, block_id, timeout, std::move(promise));
 | |
|     }
 | |
|     void download_archive(BlockSeqno masterchain_seqno, std::string tmp_dir, td::Timestamp timeout,
 | |
|                           td::Promise<std::string> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_archive, masterchain_seqno, std::move(tmp_dir), timeout,
 | |
|                               std::move(promise));
 | |
|     }
 | |
|     void download_out_msg_queue_proof(ShardIdFull dst_shard, std::vector<BlockIdExt> blocks,
 | |
|                                       block::ImportedMsgQueueLimits limits, td::Timestamp timeout,
 | |
|                                       td::Promise<std::vector<td::Ref<OutMsgQueueProof>>> promise) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::download_out_msg_queue_proof, dst_shard, std::move(blocks), limits,
 | |
|                               timeout, std::move(promise));
 | |
|     }
 | |
| 
 | |
|     void new_key_block(BlockHandle handle) override {
 | |
|       td::actor::send_closure(id_, &FullNodeImpl::new_key_block, std::move(handle));
 | |
|     }
 | |
| 
 | |
|     Callback(td::actor::ActorId<FullNodeImpl> id) : id_(id) {
 | |
|     }
 | |
| 
 | |
|    private:
 | |
|     td::actor::ActorId<FullNodeImpl> id_;
 | |
|   };
 | |
| 
 | |
|   td::actor::send_closure(validator_manager_, &ValidatorManagerInterface::install_callback,
 | |
|                           std::make_unique<Callback>(actor_id(this)), std::move(started_promise_));
 | |
| }
 | |
| 
 | |
| /* void FullNodeImpl::update_private_block_overlays() {
 | |
|   private_block_overlays_.clear();
 | |
|   if (local_keys_.empty()) {
 | |
|     return;
 | |
|   }
 | |
|   for (const auto &key : local_keys_) {
 | |
|     create_private_block_overlay(key);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void FullNodeImpl::create_private_block_overlay(PublicKeyHash key) {
 | |
|   CHECK(local_keys_.count(key));
 | |
|   if (current_validators_.count(key)) {
 | |
|     std::vector<adnl::AdnlNodeIdShort> nodes;
 | |
|     for (const auto &p : current_validators_) {
 | |
|       nodes.push_back(p.second);
 | |
|     }
 | |
|     private_block_overlays_[key] = td::actor::create_actor<FullNodePrivateOverlay>(
 | |
|         "BlocksPrivateOverlay", current_validators_[key], std::move(nodes), zero_state_file_hash_, config_, keyring_,
 | |
|         adnl_, rldp_, rldp2_, overlays_, validator_manager_);
 | |
|   }
 | |
| } */
 | |
| 
 | |
| FullNodeImpl::FullNodeImpl(PublicKeyHash local_id, adnl::AdnlNodeIdShort adnl_id, FileHash zero_state_file_hash,
 | |
|                            FullNodeConfig config, td::actor::ActorId<keyring::Keyring> keyring,
 | |
|                            td::actor::ActorId<adnl::Adnl> adnl, td::actor::ActorId<rldp::Rldp> rldp,
 | |
|                            td::actor::ActorId<rldp2::Rldp> rldp2, td::actor::ActorId<dht::Dht> dht,
 | |
|                            td::actor::ActorId<overlay::Overlays> overlays,
 | |
|                            td::actor::ActorId<ValidatorManagerInterface> validator_manager,
 | |
|                            td::actor::ActorId<adnl::AdnlExtClient> client, std::string db_root,
 | |
|                            td::Promise<td::Unit> started_promise)
 | |
|     : local_id_(local_id)
 | |
|     , adnl_id_(adnl_id)
 | |
|     , zero_state_file_hash_(zero_state_file_hash)
 | |
|     , keyring_(keyring)
 | |
|     , adnl_(adnl)
 | |
|     , rldp_(rldp)
 | |
|     , rldp2_(rldp2)
 | |
|     , dht_(dht)
 | |
|     , overlays_(overlays)
 | |
|     , validator_manager_(validator_manager)
 | |
|     , client_(client)
 | |
|     , db_root_(db_root)
 | |
|     , started_promise_(std::move(started_promise))
 | |
|     , config_(config) {
 | |
|   add_shard_actor(ShardIdFull{masterchainId}, FullNodeShardMode::active);
 | |
| }
 | |
| 
 | |
| td::actor::ActorOwn<FullNode> FullNode::create(
 | |
|     ton::PublicKeyHash local_id, adnl::AdnlNodeIdShort adnl_id, FileHash zero_state_file_hash, FullNodeConfig config,
 | |
|     td::actor::ActorId<keyring::Keyring> keyring, td::actor::ActorId<adnl::Adnl> adnl,
 | |
|     td::actor::ActorId<rldp::Rldp> rldp, td::actor::ActorId<rldp2::Rldp> rldp2, td::actor::ActorId<dht::Dht> dht,
 | |
|     td::actor::ActorId<overlay::Overlays> overlays, td::actor::ActorId<ValidatorManagerInterface> validator_manager,
 | |
|     td::actor::ActorId<adnl::AdnlExtClient> client, std::string db_root, td::Promise<td::Unit> started_promise) {
 | |
|   return td::actor::create_actor<FullNodeImpl>("fullnode", local_id, adnl_id, zero_state_file_hash, config, keyring,
 | |
|                                                adnl, rldp, rldp2, dht, overlays, validator_manager, client, db_root,
 | |
|                                                std::move(started_promise));
 | |
| }
 | |
| 
 | |
| FullNodeConfig::FullNodeConfig(const tl_object_ptr<ton_api::engine_validator_fullNodeConfig> &obj)
 | |
|     : ext_messages_broadcast_disabled_(obj->ext_messages_broadcast_disabled_) {
 | |
| }
 | |
| 
 | |
| tl_object_ptr<ton_api::engine_validator_fullNodeConfig> FullNodeConfig::tl() const {
 | |
|   return create_tl_object<ton_api::engine_validator_fullNodeConfig>(ext_messages_broadcast_disabled_);
 | |
| }
 | |
| bool FullNodeConfig::operator==(const FullNodeConfig &rhs) const {
 | |
|   return ext_messages_broadcast_disabled_ == rhs.ext_messages_broadcast_disabled_;
 | |
| }
 | |
| bool FullNodeConfig::operator!=(const FullNodeConfig &rhs) const {
 | |
|   return !(*this == rhs);
 | |
| }
 | |
| 
 | |
| }  // namespace fullnode
 | |
| 
 | |
| }  // namespace validator
 | |
| 
 | |
| }  // namespace ton
 |