6#ifndef ALTINTEGRATION_TEMP_BLOCK_TREE_HPP
7#define ALTINTEGRATION_TEMP_BLOCK_TREE_HPP
10#include <veriblock/pop/algorithm.hpp>
11#include <veriblock/pop/validation_state.hpp>
16template <
typename StableBlockTree>
19 template <
typename BlockIndexT>
20 struct TempBlockIndex :
public BlockIndexT {
21 using base = BlockIndexT;
29 this->deleteTemporarily();
30 this->pprev =
nullptr;
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;
41 std::unordered_map<prev_block_hash_t, std::unique_ptr<wrapped_index_t>>;
43 TempBlockTree(
const block_tree_t& tree) : tree_(&tree) {}
45 virtual ~TempBlockTree() =
default;
48 TempBlockTree(
const TempBlockTree&) =
delete;
49 TempBlockTree& operator=(
const TempBlockTree&) =
delete;
52 TempBlockTree(TempBlockTree&&) =
default;
53 TempBlockTree& operator=(TempBlockTree&&) =
default;
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();
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));
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;
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));
94 bool acceptBlockHeader(
const block_t& header, ValidationState& state) {
95 return acceptBlockHeader(std::make_shared<block_t>(header), state);
98 bool acceptBlockHeader(std::shared_ptr<block_t> header,
99 ValidationState& state) {
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()));
108 auto index = insertBlockHeader(std::move(header), prev);
109 VBK_ASSERT(index !=
nullptr &&
110 "insertBlockHeader should have never returned nullptr");
115 const block_tree_t& getStableTree()
const {
return *tree_; }
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);
128 void clear() { temp_blocks_.clear(); }
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) {
139 current = doInsertBlockHeader(std::move(header), prev);
140 VBK_ASSERT(current !=
nullptr);
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);
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();
165 auto newIndex = make_unique<wrapped_index_t>(0);
167 it = temp_blocks_.emplace(shortHash, std::move(newIndex)).first;
168 return it->second.get();
172 block_index_t temp_blocks_;
173 const block_tree_t* tree_;
std::string HexStr(const T itbegin, const T itend)
Convert bytes to hex.