veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
temp_block_tree.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_TEMP_BLOCK_TREE_HPP
7#define ALTINTEGRATION_TEMP_BLOCK_TREE_HPP
8
9#include <memory>
10#include <veriblock/pop/algorithm.hpp>
11#include <veriblock/pop/validation_state.hpp>
12
13namespace altintegration {
14
16template <typename StableBlockTree>
17struct TempBlockTree {
19 template <typename BlockIndexT>
20 struct TempBlockIndex : public BlockIndexT {
21 using base = BlockIndexT;
22 using base::base;
23
24 ~TempBlockIndex() {
25 // if alttree is destroyed before temp block tree, then
26 // `pprev` may point to deallocated block. to solve this, override
27 // destructor of temp block index, as we don't care in which order temp
28 // blocks are destroyed.
29 this->deleteTemporarily();
30 this->pprev = nullptr;
31 this->pnext.clear();
32 }
33 };
34
35 using block_tree_t = StableBlockTree;
36 using block_t = typename block_tree_t::block_t;
37 using index_t = typename block_tree_t::index_t;
38 using wrapped_index_t = TempBlockIndex<index_t>;
39 using prev_block_hash_t = typename StableBlockTree::prev_block_hash_t;
40 using block_index_t =
41 std::unordered_map<prev_block_hash_t, std::unique_ptr<wrapped_index_t>>;
42
43 TempBlockTree(const block_tree_t& tree) : tree_(&tree) {}
44
45 virtual ~TempBlockTree() = default;
46
47 // non-copyable
48 TempBlockTree(const TempBlockTree&) = delete;
49 TempBlockTree& operator=(const TempBlockTree&) = delete;
50
51 // movable
52 TempBlockTree(TempBlockTree&&) = default;
53 TempBlockTree& operator=(TempBlockTree&&) = default;
54
55 template <typename T,
56 typename = typename std::enable_if<
57 std::is_same<T, typename block_t::hash_t>::value ||
58 std::is_same<T, typename block_t::prev_hash_t>::value>::type>
59 const index_t* getTempBlockIndex(const T& hash) const {
60 auto shortHash = tree_->makePrevHash(hash);
61 auto it = temp_blocks_.find(shortHash);
62 return it == temp_blocks_.end() ? nullptr : it->second.get();
63 }
64
66 template <typename T,
67 typename = typename std::enable_if<
68 std::is_same<T, typename block_t::hash_t>::value ||
69 std::is_same<T, typename block_t::prev_hash_t>::value>::type>
70 index_t* getTempBlockIndex(const T& hash) {
71 const auto& t = as_const(*this);
72 return const_cast<index_t*>(t.getTempBlockIndex(hash));
73 }
74
75 template <typename T,
76 typename = typename std::enable_if<
77 std::is_same<T, typename block_t::hash_t>::value ||
78 std::is_same<T, typename block_t::prev_hash_t>::value>::type>
79 const index_t* getBlockIndex(const T& hash) const {
80 auto* index = getTempBlockIndex(hash);
81 return index == nullptr ? tree_->getBlockIndex(hash) : index;
82 }
83
85 template <typename T,
86 typename = typename std::enable_if<
87 std::is_same<T, typename block_t::hash_t>::value ||
88 std::is_same<T, typename block_t::prev_hash_t>::value>::type>
89 index_t* getBlockIndex(const T& hash) {
90 const auto& t = as_const(*this);
91 return const_cast<index_t*>(t.getBlockIndex(hash));
92 }
93
94 bool acceptBlockHeader(const block_t& header, ValidationState& state) {
95 return acceptBlockHeader(std::make_shared<block_t>(header), state);
96 }
97
98 bool acceptBlockHeader(std::shared_ptr<block_t> header,
99 ValidationState& state) {
100 VBK_ASSERT(header);
101 auto* prev = getBlockIndex(header->getPreviousBlock());
102 if (prev == nullptr) {
103 return state.Invalid(
104 block_t::name() + "-bad-prev-block",
105 "can not find previous block: " + HexStr(header->getPreviousBlock()));
106 }
107
108 auto index = insertBlockHeader(std::move(header), prev);
109 VBK_ASSERT(index != nullptr &&
110 "insertBlockHeader should have never returned nullptr");
111
112 return true;
113 }
114
115 const block_tree_t& getStableTree() const { return *tree_; }
116
120 void cleanUpStaleBlocks() {
121 for (auto it = temp_blocks_.begin(); it != temp_blocks_.end();) {
122 auto short_hash = tree_->makePrevHash(it->second->getHash());
123 auto* index = tree_->getBlockIndex(short_hash);
124 it = index ? temp_blocks_.erase(it) : std::next(it);
125 }
126 }
127
128 void clear() { temp_blocks_.clear(); }
129
130 private:
131 index_t* insertBlockHeader(std::shared_ptr<block_t> header, index_t* prev) {
132 auto hash = header->getHash();
133 index_t* current = getBlockIndex(hash);
134 if (current != nullptr) {
135 // it is a duplicate
136 return current;
137 }
138
139 current = doInsertBlockHeader(std::move(header), prev);
140 VBK_ASSERT(current != nullptr);
141
142 return current;
143 }
144
145 index_t* doInsertBlockHeader(std::shared_ptr<block_t> header, index_t* prev) {
146 VBK_ASSERT(header != nullptr);
147 VBK_ASSERT(prev != nullptr);
148 index_t* current = touchBlockIndex(header->getHash());
149 current->setHeader(std::move(header));
150 current->pprev = prev;
151 current->setHeight(current->pprev->getHeight() + 1);
152
153 return current;
154 }
155
156 index_t* touchBlockIndex(const typename block_t::hash_t& hash) {
157 auto shortHash = tree_->makePrevHash(hash);
158 auto it = temp_blocks_.find(shortHash);
159 if (it != temp_blocks_.end()) {
160 return it->second.get();
161 }
162
163 // intentionally do not pass 'prev' here - we don't want 'stable' blocks
164 // have pnext ptr to temp blocks.
165 auto newIndex = make_unique<wrapped_index_t>(0);
166 newIndex->setNull();
167 it = temp_blocks_.emplace(shortHash, std::move(newIndex)).first;
168 return it->second.get();
169 }
170
171 private:
172 block_index_t temp_blocks_;
173 const block_tree_t* tree_;
174};
175
176} // namespace altintegration
177
178#endif
Defines logging helpers.
Definition: block.hpp:14
std::string HexStr(const T itbegin, const T itend)
Convert bytes to hex.
Definition: strutil.hpp:44