veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
addendorsement.hpp
1// Copyright (c) 2019-2022 Xenios SEZC
2// https://www.veriblock.org
3// Distributed under the MIT software license, see the accompanying
4// file LICENSE or http://www.opensource.org/licenses/mit-license.php.
5
6#ifndef ALTINTEGRATION_ADDENDORSEMENT_HPP
7#define ALTINTEGRATION_ADDENDORSEMENT_HPP
8
9#include <veriblock/pop/blockchain/alt_chain_params.hpp>
10#include <veriblock/pop/blockchain/block_index.hpp>
11#include <veriblock/pop/blockchain/btc_chain_params.hpp>
12#include <veriblock/pop/blockchain/chain.hpp>
13#include <veriblock/pop/blockchain/command.hpp>
14#include <veriblock/pop/blockchain/vbk_chain_params.hpp>
15#include <veriblock/pop/entities/altblock.hpp>
16#include <veriblock/pop/entities/endorsements.hpp>
17#include <veriblock/pop/fmt.hpp>
18
19namespace altintegration {
20
22template <typename ProtectingTree, typename ProtectedTree>
23struct AddEndorsement : public Command {
24 using hash_t = typename ProtectedTree::hash_t;
25 using protected_block_t = typename ProtectedTree::block_t;
26 using protecting_block_t = typename ProtectingTree::block_t;
27 using endorsement_t = typename ProtectedTree::index_t::endorsement_t;
28 using protected_index_t = typename ProtectedTree::index_t;
29
30 ~AddEndorsement() override = default;
31
32 explicit AddEndorsement(ProtectingTree& ing,
33 ProtectedTree& ed,
34 std::shared_ptr<endorsement_t> e)
35 : ing_(&ing), ed_(&ed), e_(std::move(e)) {}
36
37 bool Execute(ValidationState& state) noexcept override {
38 auto* containing = ed_->getBlockIndex(e_->containingHash);
39 if (containing == nullptr) {
40 return state.Invalid(
41 protected_block_t::name() + "-no-containing",
42 format("Can not find containing block in endorsement={}",
43 e_->toPrettyString()));
44 }
45
46 auto* endorsed = ed_->getBlockIndex(e_->endorsedHash);
47 if (endorsed == nullptr) {
48 return state.Invalid(protected_block_t::name() + "-no-endorsed-block",
49 format("Endorsed block={} not found in the tree",
50 HexStr(e_->endorsedHash)));
51 }
52
53 auto actualEndorsed = containing->getAncestor(endorsed->getHeight());
54 if (actualEndorsed == nullptr || endorsed != actualEndorsed) {
55 return state.Invalid(
56 protected_block_t::name() + "-block-differs",
57 format("Endorsed block is on a different chain. Expected: {}, got {}",
58 endorsed->toShortPrettyString(),
59 (actualEndorsed ? actualEndorsed->toShortPrettyString()
60 : "nullptr")));
61 }
62
63 if (containing->getHeight() - endorsed->getHeight() >
64 (int)ed_->getParams().getEndorsementSettlementInterval()) {
65 return state.Invalid(protected_block_t::name() + "-expired",
66 "Endorsement expired");
67 }
68
69 auto* blockOfProof = ing_->getBlockIndex(e_->blockOfProof);
70 if (blockOfProof == nullptr) {
71 return state.Invalid(
72 protected_block_t::name() + "-block-of-proof-not-found",
73 format("Can not find block of proof in SP Chain ({})",
74 HexStr(e_->blockOfProof)));
75 }
76
77 containing->insertContainingEndorsement(e_);
78 endorsed->insertEndorsedBy(e_.get());
79 blockOfProof->insertBlockOfProofEndorsement(e_.get());
80
81 return true;
82 }
83
84 void UnExecute() noexcept override {
85 auto* containing = ed_->getBlockIndex(e_->containingHash);
86 VBK_ASSERT_MSG(
87 containing != nullptr,
88 "failed to roll back AddEndorsement: the containing block does not "
89 "exist %s",
90 e_->toPrettyString());
91
92 auto* endorsed = ed_->getBlockIndex(e_->endorsedHash);
93 VBK_ASSERT_MSG(
94 endorsed != nullptr,
95 "failed to roll back AddEndorsement: the endorsed block does not "
96 "exist %s",
97 e_->toPrettyString());
98
99 auto* blockOfProof = ing_->getBlockIndex(e_->blockOfProof);
100 VBK_ASSERT_MSG(blockOfProof != nullptr,
101 "failed to roll back AddEndorsement: the blockOfProof "
102 "block does not exist %s",
103 e_->toPrettyString());
104
105 // e_ is likely to have different address than one stored in a Block.
106 // make sure that we use correct Endorsement instance, then to remove proper
107 // ptrs from endorsedBy and blockOfProofEndorsements
108 auto Eit = containing->findContainingEndorsement(e_->id);
109 VBK_ASSERT_MSG(Eit != containing->getContainingEndorsements().end(),
110 "state corruption: containing endorsement not found");
111
112 // erase endorsedBy
113 bool p1 = endorsed->eraseLastFromEndorsedBy(Eit->second.get());
114 VBK_ASSERT_MSG(p1,
115 "Failed to remove endorsement %s from endorsedBy in "
116 "AddEndorsement::Unexecute",
117 e_->toPrettyString());
118
119 // erase blockOfProof
120 bool p2 =
121 blockOfProof->eraseLastFromBlockOfProofEndorsement(Eit->second.get());
122 VBK_ASSERT_MSG(p2,
123 "Failed to remove endorsement %s from blockOfProof in "
124 "AddEndorsement::Unexecute",
125 e_->toPrettyString());
126
127 // erase containing, should be removed last
128 containing->removeContainingEndorsement(Eit);
129 }
130
131 private:
132 ProtectingTree* ing_ = nullptr;
133 ProtectedTree* ed_ = nullptr;
134 std::shared_ptr<endorsement_t> e_ = nullptr;
135};
136
137struct AltBlockTree;
138struct VbkBlockTree;
139template <typename Block, typename ChainParams>
140struct BlockTree;
141
143using AddVbkEndorsement =
144 AddEndorsement<BlockTree<BtcBlock, BtcChainParams>, VbkBlockTree>;
145
147using AddAltEndorsement = AddEndorsement<VbkBlockTree, AltBlockTree>;
148
149} // namespace altintegration
150
151#endif // ALTINTEGRATION_ADDENDORSEMENT_HPP
Defines logging helpers.
Definition: block.hpp:14
std::string HexStr(const T itbegin, const T itend)
Convert bytes to hex.
Definition: strutil.hpp:44