/* 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-2020 Telegram Systems LLP */ #include "fabric.h" #include "collator-impl.h" #include "validator/db/rootdb.hpp" #include "validator/block-handle.hpp" #include "apply-block.hpp" #include "accept-block.hpp" #include "shard.hpp" #include "block.hpp" #include "proof.hpp" #include "signature-set.hpp" #include "external-message.hpp" #include "ihr-message.hpp" #include "validate-query.hpp" #include "check-proof.hpp" #include "top-shard-descr.hpp" #include "ton/ton-io.hpp" #include "liteserver.hpp" #include "validator/fabric.h" #include "liteserver-cache.hpp" namespace ton { namespace validator { td::actor::ActorOwn create_db_actor(td::actor::ActorId manager, std::string db_root_, td::Ref opts) { return td::actor::create_actor("db", manager, db_root_, opts); } td::actor::ActorOwn create_liteserver_cache_actor(td::actor::ActorId manager, std::string db_root) { return td::actor::create_actor("cache"); } td::Result> create_block(BlockIdExt block_id, td::BufferSlice data) { auto res = BlockQ::create(block_id, std::move(data)); if (res.is_error()) { return res.move_as_error(); } else { return td::Ref{res.move_as_ok()}; } } td::Result> create_block(ReceivedBlock data) { return create_block(data.id, std::move(data.data)); } td::Result> create_proof(BlockIdExt masterchain_block_id, td::BufferSlice proof) { return Ref{true, masterchain_block_id, std::move(proof)}; } td::Result> create_proof_link(BlockIdExt block_id, td::BufferSlice proof_link) { return Ref{true, block_id, std::move(proof_link)}; } td::Result> create_signature_set(td::BufferSlice sig_set) { return BlockSignatureSetQ::fetch(std::move(sig_set)); } td::Result> create_shard_state(BlockIdExt block_id, td::BufferSlice data) { auto res = ShardStateQ::fetch(block_id, std::move(data)); if (res.is_error()) { return res.move_as_error(); } else { return td::Ref{res.move_as_ok()}; } } td::Result> create_shard_state(BlockIdExt block_id, td::Ref root_cell) { auto res = ShardStateQ::fetch(block_id, {}, std::move(root_cell)); if (res.is_error()) { return res.move_as_error(); } else { return td::Ref{res.move_as_ok()}; } } td::Result create_block_handle(td::BufferSlice data) { return ton::validator::BlockHandleImpl::create(data.as_slice()); } td::Result create_block_handle(td::Slice data) { return ton::validator::BlockHandleImpl::create(data); } td::Result create_temp_block_handle(td::BufferSlice data) { return ton::validator::BlockHandleImpl::create(std::move(data)); } BlockHandle create_empty_block_handle(BlockIdExt id) { return ton::validator::BlockHandleImpl::create_empty(id); } td::Ref create_signature_set(std::vector sig_set) { return td::Ref{true, std::move(sig_set)}; } td::Result> create_ext_message(td::BufferSlice data, block::SizeLimitsConfig::ExtMsgLimits limits) { TRY_RESULT(res, ExtMessageQ::create_ext_message(std::move(data), limits)); return std::move(res); } void run_check_external_message(Ref message, td::actor::ActorId manager, td::Promise> promise) { ExtMessageQ::run_message(std::move(message), std::move(manager), std::move(promise)); } td::Result> create_ihr_message(td::BufferSlice data) { TRY_RESULT(res, IhrMessageQ::create_ihr_message(std::move(data))); return std::move(res); } void run_accept_block_query(BlockIdExt id, td::Ref data, std::vector prev, td::Ref validator_set, td::Ref signatures, td::Ref approve_signatures, bool send_broadcast, td::actor::ActorId manager, td::Promise promise) { td::actor::create_actor(PSTRING() << "accept" << id.id.to_str(), id, std::move(data), prev, std::move(validator_set), std::move(signatures), std::move(approve_signatures), send_broadcast, manager, std::move(promise)) .release(); } void run_fake_accept_block_query(BlockIdExt id, td::Ref data, std::vector prev, td::Ref validator_set, td::actor::ActorId manager, td::Promise promise) { td::actor::create_actor("fakeaccept", AcceptBlockQuery::IsFake(), id, std::move(data), std::move(prev), std::move(validator_set), std::move(manager), std::move(promise)) .release(); } void run_hardfork_accept_block_query(BlockIdExt id, td::Ref data, td::actor::ActorId manager, td::Promise promise) { td::actor::create_actor("fork/accept", AcceptBlockQuery::ForceFork(), id, std::move(data), std::move(manager), std::move(promise)) .release(); } void run_apply_block_query(BlockIdExt id, td::Ref block, BlockIdExt masterchain_block_id, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise) { td::actor::create_actor(PSTRING() << "apply " << id, id, std::move(block), masterchain_block_id, manager, timeout, std::move(promise)) .release(); } void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise, bool skip_check_signatures) { td::actor::create_actor("checkproof", id, std::move(proof), manager, timeout, std::move(promise), skip_check_signatures) .release(); } void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise, td::Ref rel_key_block_proof, bool skip_check_signatures) { td::actor::create_actor("checkproof/key", id, std::move(proof), manager, timeout, std::move(promise), skip_check_signatures, std::move(rel_key_block_proof)) .release(); } void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise, td::Ref rel_mc_state, bool skip_check_signatures) { td::actor::create_actor("checkproof/st", id, std::move(proof), manager, timeout, std::move(promise), skip_check_signatures, std::move(rel_mc_state)) .release(); } void run_check_proof_link_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise) { td::actor::create_actor("checkprooflink", id, std::move(proof), manager, timeout, std::move(promise)) .release(); } void run_validate_query(ShardIdFull shard, BlockIdExt min_masterchain_block_id, std::vector prev, BlockCandidate candidate, td::Ref validator_set, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise, bool is_fake) { BlockSeqno seqno = 0; for (auto& p : prev) { if (p.seqno() > seqno) { seqno = p.seqno(); } } static std::atomic idx; td::actor::create_actor(PSTRING() << (is_fake ? "fakevalidate" : "validateblock") << shard.to_str() << ":" << (seqno + 1) << "#" << idx.fetch_add(1), shard, min_masterchain_block_id, std::move(prev), std::move(candidate), std::move(validator_set), std::move(manager), timeout, std::move(promise), is_fake) .release(); } void run_collate_query(ShardIdFull shard, const BlockIdExt& min_masterchain_block_id, std::vector prev, Ed25519_PublicKey creator, td::Ref validator_set, td::Ref collator_opts, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise) { BlockSeqno seqno = 0; for (auto& p : prev) { if (p.seqno() > seqno) { seqno = p.seqno(); } } td::actor::create_actor(PSTRING() << "collate" << shard.to_str() << ":" << (seqno + 1), shard, false, min_masterchain_block_id, std::move(prev), std::move(validator_set), creator, std::move(collator_opts), std::move(manager), timeout, std::move(promise)) .release(); } void run_collate_hardfork(ShardIdFull shard, const BlockIdExt& min_masterchain_block_id, std::vector prev, td::actor::ActorId manager, td::Timestamp timeout, td::Promise promise) { BlockSeqno seqno = 0; for (auto& p : prev) { if (p.seqno() > seqno) { seqno = p.seqno(); } } td::actor::create_actor(PSTRING() << "collate" << shard.to_str() << ":" << (seqno + 1), shard, true, min_masterchain_block_id, std::move(prev), td::Ref{}, Ed25519_PublicKey{Bits256::zero()}, td::Ref{true}, std::move(manager), timeout, std::move(promise)) .release(); } void run_liteserver_query(td::BufferSlice data, td::actor::ActorId manager, td::actor::ActorId cache, td::Promise promise) { LiteQuery::run_query(std::move(data), std::move(manager), std::move(cache), std::move(promise)); } void run_fetch_account_state(WorkchainId wc, StdSmcAddress addr, td::actor::ActorId manager, td::Promise,UnixTime,LogicalTime,std::unique_ptr>> promise) { LiteQuery::fetch_account_state(wc, addr, std::move(manager), std::move(promise)); } void run_validate_shard_block_description(td::BufferSlice data, BlockHandle masterchain_block, td::Ref masterchain_state, td::actor::ActorId manager, td::Timestamp timeout, td::Promise> promise, bool is_fake) { auto id = masterchain_block->id(); td::actor::create_actor("topshardfetch", std::move(data), id, std::move(masterchain_block), std::move(masterchain_state), manager, timeout, is_fake, std::move(promise)) .release(); } } // namespace validator } // namespace ton