veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
pop_state_machine.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_POP_STATE_MACHINE_HPP
7#define ALTINTEGRATION_POP_STATE_MACHINE_HPP
8
9#include <functional>
10#include <veriblock/pop/assert.hpp>
11#include <veriblock/pop/blockchain/chain.hpp>
12#include <veriblock/pop/blockchain/chain_slice.hpp>
13#include <veriblock/pop/reversed_range.hpp>
14#include <veriblock/pop/storage/payloads_provider.hpp>
15#include <veriblock/pop/trace.hpp>
16
17namespace altintegration {
18
19namespace internal {
20
21template <typename index_t>
22void assertBlockCanBeApplied(index_t& index) {
23 VBK_ASSERT_MSG(!index.isRoot(), "cannot apply the root block");
24
25 VBK_ASSERT_MSG(index.pprev->hasFlags(BLOCK_ACTIVE),
26 "state corruption: tried to apply a block that follows an "
27 "unapplied block %s",
28 index.pprev->toPrettyString());
29
30 VBK_ASSERT_MSG(!index.hasFlags(BLOCK_ACTIVE),
31 "state corruption: tried to apply an already applied block %s",
32 index.toPrettyString());
33 // an expensive check; might want to disable it eventually
34 VBK_ASSERT_MSG_DEBUG(
35 index.allDescendantsUnapplied(),
36 "state corruption: found an unapplied block that has some of its "
37 "descendants applied %s",
38 index.toPrettyString());
39
40 VBK_ASSERT(!index.hasFlags(BLOCK_FAILED_CHILD) &&
41 "state corruption: attempted to apply a block that has an "
42 "invalid ancestor");
43}
44
45template <typename index_t>
46void assertBlockCanBeUnapplied(index_t& index) {
47 VBK_ASSERT_MSG(!index.isRoot(), "cannot unapply the root block");
48 VBK_ASSERT_MSG(!index.finalized, "cannot unapply finalized block");
49 VBK_ASSERT_MSG(
50 index.hasFlags(BLOCK_ACTIVE),
51 "state corruption: tried to unapply an already unapplied block %s",
52 index.toPrettyString());
53 VBK_ASSERT_MSG(index.pprev->hasFlags(BLOCK_ACTIVE),
54 "state corruption: tried to unapply a block that follows an "
55 "unapplied block %s",
56 index.pprev->toPrettyString());
57 // an expensive check; might want to disable it eventually
58 VBK_ASSERT_MSG_DEBUG(
59 index.allDescendantsUnapplied(),
60 "state corruption: tried to unapply a block before unapplying "
61 "its applied descendants %s",
62 index.toPrettyString());
63}
64
65} // namespace internal
66
68template <typename ProtectingBlockTree,
69 typename ProtectedTree,
70 typename ProtectedIndex,
71 typename ProtectedChainParams>
72struct PopStateMachine {
73 using index_t = ProtectedIndex;
74 using block_t = typename index_t::block_t;
75 using endorsement_t = typename index_t::endorsement_t;
76 using height_t = typename ProtectedIndex::height_t;
77 using command_group_store_t = typename ProtectedTree::command_group_store_t;
78
79 PopStateMachine(ProtectedTree& ed, ProtectingBlockTree& ing)
80 : ed_(ed), ing_(ing), commandGroupStore_(ed_.getCommandGroupStore()) {}
81
83 VBK_CHECK_RETURN bool applyBlock(index_t& index, ValidationState& state) {
84 VBK_TRACE_ZONE_SCOPED;
85
86 internal::assertBlockCanBeApplied(index);
87
88 if (!index.isValid()) {
89 return state.Invalid(
90 index_t::block_t::name() + "-marked-invalid",
91 format("block {} is marked as invalid and cannot be applied",
92 index.toPrettyString()));
93 }
94
95 VBK_ASSERT_MSG(index.isValid(BLOCK_CONNECTED),
96 "attempted to apply an unconnected block %s",
97 index.toPrettyString());
98
99 if (index.hasPayloads()) {
100 auto cgroups = commandGroupStore_.getCommands(index, state);
101 // TODO:
102 VBK_ASSERT_MSG(
103 cgroups, "support for payload pre-validation is not implemented yet");
104
105 for (auto cgroup = cgroups->cbegin(); cgroup != cgroups->cend();
106 ++cgroup) {
107 VBK_LOG_DEBUG("Applying payload %s from block %s",
108 HexStr((*cgroup)->id),
109 index.toShortPrettyString());
110
111 CommandGroup& cg = *(*cgroup);
112
113 if (!cg.execute(state)) {
114 VBK_LOG_ERROR("Invalid %s command in block %s: %s",
115 index_t::block_t::name(),
116 index.toPrettyString(),
117 state.toString());
118
119 // unexecute executed command groups in the reverse order
120 for (auto r_group = std::reverse_iterator<decltype(cgroup)>(cgroup);
121 r_group != cgroups->rend();
122 ++r_group) {
123 (*r_group)->unExecute();
124 }
125
126 ed_.invalidateSubtree(index, BLOCK_FAILED_POP, /*do fr=*/false);
127
128 return state.Invalid(index_t::block_t::name() + "-bad-command");
129 }
130
131 } // end for
132 }
133
134 // if the applied block count equals the size of the chain between the root
135 // and index, we have just applied a block on top of the only applied chain,
136 // so it is fully valid
137 auto appliedChainBlockCount =
138 ed_.getRoot().getHeight() +
139 (typename block_t::height_t)ed_.appliedBlockCount;
140
141 if (index.pprev->isValid(BLOCK_CAN_BE_APPLIED) &&
142 index.getHeight() == appliedChainBlockCount) {
143 index.raiseValidity(BLOCK_CAN_BE_APPLIED);
144 } else {
145 // this block is applied together with the other chain during POP FR
147 }
148
149 index.setFlag(BLOCK_ACTIVE);
150 ++ed_.appliedBlockCount;
151
152 return true;
153 }
154
160 void unapplyBlock(index_t& index) {
161 VBK_TRACE_ZONE_SCOPED;
162
163 internal::assertBlockCanBeUnapplied(index);
164
165 if (index.hasPayloads()) {
166 ValidationState state;
167 auto cgroups = commandGroupStore_.getCommands(index, state);
168 // TODO:
169 VBK_ASSERT_MSG(
170 cgroups, "support for payload pre-validation is not implemented yet");
171
172 for (const auto& cgroup : reverse_iterate(*cgroups)) {
173 VBK_LOG_DEBUG("Unapplying payload %s from block %s",
174 HexStr(cgroup->id),
175 index.toShortPrettyString());
176 cgroup->unExecute();
177 }
178 }
179
180 index.unsetFlag(BLOCK_ACTIVE);
181 VBK_ASSERT(ed_.getBestChain().blocksCount() > 0);
182 VBK_ASSERT(ed_.appliedBlockCount > 0);
183 --ed_.appliedBlockCount;
184 }
185
195 VBK_CHECK_RETURN index_t& unapplyWhile(
196 index_t& from,
197 index_t& to,
198 const std::function<bool(index_t& index)>& pred) {
199 VBK_TRACE_ZONE_SCOPED;
200
201 if (&from == &to) {
202 return to;
203 }
204
205 VBK_LOG_DEBUG("Unapply %d blocks from=%s, to=%s",
206 from.getHeight() - to.getHeight(),
207 from.toPrettyString(),
208 to.toPrettyString());
209
210 for (auto* current = &from; current != &to; current = current->getPrev()) {
211 VBK_ASSERT_MSG(current != nullptr,
212 "reached the genesis or first bootstrap block");
213 VBK_ASSERT_MSG(current->getHeight() > to.getHeight(),
214 "[from, to) is not a chain, detected at %s",
215 current->toPrettyString());
216
217 if (!pred(*current)) {
218 return *current;
219 }
220
221 unapplyBlock(*current);
222 }
223
224 return to;
225 }
226
228 VBK_CHECK_RETURN index_t& unapplyWhile(
229 Chain<index_t>& chain, const std::function<bool(index_t& index)>& pred) {
230 return unapplyWhile(*chain.tip(), *chain.first(), pred);
231 }
232
234 void unapply(index_t& from, index_t& to) {
235 VBK_TRACE_ZONE_SCOPED;
236
237 auto pred = [](index_t&) -> bool { return true; };
238 auto& firstUnprocessed = unapplyWhile(from, to, pred);
239 VBK_ASSERT(&firstUnprocessed == &to);
240 }
241
244 void unapply(Chain<index_t>& chain) {
245 VBK_ASSERT(!chain.empty());
246 unapply(*chain.tip(), *chain.first());
247 }
248
251 void unapply(ChainSlice<index_t>& chain) {
252 VBK_ASSERT(!chain.empty());
253 unapply(*chain.tip(), *chain.first());
254 }
255
258 VBK_CHECK_RETURN bool apply(index_t& from,
259 index_t& to,
260 ValidationState& state) {
261 VBK_TRACE_ZONE_SCOPED;
262
263 if (&from == &to) {
264 // already applied this block
265 return true;
266 }
267
268 if (!to.isValid()) {
269 return state.Invalid(
270 index_t::block_t::name() + "-marked-invalid",
271 format("block {} is marked as invalid and cannot be applied",
272 to.toPrettyString()));
273 }
274
275 VBK_ASSERT(from.getHeight() < to.getHeight());
276 // exclude 'from' by adding 1
277 Chain<index_t> chain(from.getHeight() + 1, &to);
278 VBK_ASSERT(chain.first());
279 VBK_ASSERT(chain.first()->pprev == &from);
280
281 VBK_LOG_DEBUG("Applying %d blocks from=%s, to=%s",
282 chain.blocksCount(),
283 from.toPrettyString(),
284 to.toPrettyString());
285
286 for (auto* index : chain) {
287 if (!applyBlock(*index, state)) {
288 // rollback the previously applied slice of the chain
289 unapply(*index->pprev, from);
290 return false;
291 }
292 }
293
294 // this subchain is valid
295 return true;
296 }
297
301 VBK_CHECK_RETURN bool apply(Chain<index_t>& chain, ValidationState& state) {
302 return apply(*chain.first(), *chain.tip(), state);
303 }
304
305 // effectively unapplies [from; genesis) and applies (genesis; to]
306 // assumes and requires that [from; genesis) is applied
307 // optimization: avoids applying/unapplying (genesis; fork_point(from, to)]
308 // atomic: either changes the state to 'to' or leaves it unchanged
309 VBK_CHECK_RETURN bool setState(index_t& from,
310 index_t& to,
311 ValidationState& state) {
312 VBK_TRACE_ZONE_SCOPED;
313
314 if (&from == &to) {
315 // already at this state
316 return true;
317 }
318
319 const auto* forkBlock = getForkBlock(from, to);
320 VBK_ASSERT_MSG(forkBlock, "Fork Block must exist!");
321
322 auto& fork = as_mut(*forkBlock);
323 unapply(from, fork);
324 if (!apply(fork, to, state)) {
325 // attempted to switch to an invalid block, rollback
326 bool success = apply(fork, from, state);
327 VBK_ASSERT_MSG(success,
328 "state corruption: failed to rollback the state: %s",
329 state.toString());
330
331 return false;
332 }
333
334 return true;
335 }
336
337 ProtectingBlockTree& tree() { return ing_; }
338 const ProtectingBlockTree& tree() const { return ing_; }
339 const ProtectedChainParams& params() const { return ed_.getParams(); }
340
341 private:
342 ProtectedTree& ed_;
343 ProtectingBlockTree& ing_;
344 command_group_store_t& commandGroupStore_;
345};
346
347} // namespace altintegration
348
349#endif // ALTINTEGRATION_POP_STATE_MACHINE_HPP
Defines logging helpers.
Definition: block.hpp:14
@ BLOCK_CONNECTED
the block is connected via connectBlock, which means that this block and all its ancestors are at lea...
@ BLOCK_CAN_BE_APPLIED_MAYBE_WITH_OTHER_CHAIN
the block has been successfully applied, but may not be fully valid, because it may connect to the "o...
@ BLOCK_CAN_BE_APPLIED
the chain with the block at its tip is fully valid, so if we do SetState on this block,...
@ BLOCK_FAILED_CHILD
block is state{lessly,fully} valid and the altchain did not report it as invalid, but some of the anc...
@ BLOCK_ACTIVE
the block is currently applied via SetState.
@ BLOCK_FAILED_POP
block failed state{less,ful} validation due to its payloads
std::string HexStr(const T itbegin, const T itend)
Convert bytes to hex.
Definition: strutil.hpp:44
reverse_range< T > reverse_iterate(T &x)
This is an overloaded member function, provided for convenience. It differs from the above function o...
const BlockIndex< Block > * getForkBlock(const BlockIndex< Block > &a, const BlockIndex< Block > &b)
getForkBlock assumes that: the block tree is not malformed the fork block(worst case: genesis/bootstr...