6#ifndef ALTINTEGRATION_POP_STATE_MACHINE_HPP
7#define ALTINTEGRATION_POP_STATE_MACHINE_HPP
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>
21template <
typename index_t>
22void assertBlockCanBeApplied(index_t& index) {
23 VBK_ASSERT_MSG(!index.isRoot(),
"cannot apply the root block");
26 "state corruption: tried to apply a block that follows an "
28 index.pprev->toPrettyString());
31 "state corruption: tried to apply an already applied block %s",
32 index.toPrettyString());
35 index.allDescendantsUnapplied(),
36 "state corruption: found an unapplied block that has some of its "
37 "descendants applied %s",
38 index.toPrettyString());
41 "state corruption: attempted to apply a block that has an "
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");
51 "state corruption: tried to unapply an already unapplied block %s",
52 index.toPrettyString());
54 "state corruption: tried to unapply a block that follows an "
56 index.pprev->toPrettyString());
59 index.allDescendantsUnapplied(),
60 "state corruption: tried to unapply a block before unapplying "
61 "its applied descendants %s",
62 index.toPrettyString());
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;
79 PopStateMachine(ProtectedTree& ed, ProtectingBlockTree& ing)
80 : ed_(ed), ing_(ing), commandGroupStore_(ed_.getCommandGroupStore()) {}
83 VBK_CHECK_RETURN
bool applyBlock(index_t& index, ValidationState& state) {
84 VBK_TRACE_ZONE_SCOPED;
86 internal::assertBlockCanBeApplied(index);
88 if (!index.isValid()) {
90 index_t::block_t::name() +
"-marked-invalid",
91 format(
"block {} is marked as invalid and cannot be applied",
92 index.toPrettyString()));
96 "attempted to apply an unconnected block %s",
97 index.toPrettyString());
99 if (index.hasPayloads()) {
100 auto cgroups = commandGroupStore_.getCommands(index, state);
103 cgroups,
"support for payload pre-validation is not implemented yet");
105 for (
auto cgroup = cgroups->cbegin(); cgroup != cgroups->cend();
107 VBK_LOG_DEBUG(
"Applying payload %s from block %s",
109 index.toShortPrettyString());
111 CommandGroup& cg = *(*cgroup);
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(),
120 for (
auto r_group = std::reverse_iterator<
decltype(cgroup)>(cgroup);
121 r_group != cgroups->rend();
123 (*r_group)->unExecute();
128 return state.Invalid(index_t::block_t::name() +
"-bad-command");
137 auto appliedChainBlockCount =
138 ed_.getRoot().getHeight() +
139 (
typename block_t::height_t)ed_.appliedBlockCount;
142 index.getHeight() == appliedChainBlockCount) {
150 ++ed_.appliedBlockCount;
160 void unapplyBlock(index_t& index) {
161 VBK_TRACE_ZONE_SCOPED;
163 internal::assertBlockCanBeUnapplied(index);
165 if (index.hasPayloads()) {
166 ValidationState state;
167 auto cgroups = commandGroupStore_.getCommands(index, state);
170 cgroups,
"support for payload pre-validation is not implemented yet");
173 VBK_LOG_DEBUG(
"Unapplying payload %s from block %s",
175 index.toShortPrettyString());
181 VBK_ASSERT(ed_.getBestChain().blocksCount() > 0);
182 VBK_ASSERT(ed_.appliedBlockCount > 0);
183 --ed_.appliedBlockCount;
195 VBK_CHECK_RETURN index_t& unapplyWhile(
198 const std::function<
bool(index_t& index)>& pred) {
199 VBK_TRACE_ZONE_SCOPED;
205 VBK_LOG_DEBUG(
"Unapply %d blocks from=%s, to=%s",
206 from.getHeight() - to.getHeight(),
207 from.toPrettyString(),
208 to.toPrettyString());
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());
217 if (!pred(*current)) {
221 unapplyBlock(*current);
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);
234 void unapply(index_t& from, index_t& to) {
235 VBK_TRACE_ZONE_SCOPED;
237 auto pred = [](index_t&) ->
bool {
return true; };
238 auto& firstUnprocessed = unapplyWhile(from, to, pred);
239 VBK_ASSERT(&firstUnprocessed == &to);
244 void unapply(Chain<index_t>& chain) {
245 VBK_ASSERT(!chain.empty());
246 unapply(*chain.tip(), *chain.first());
251 void unapply(ChainSlice<index_t>& chain) {
252 VBK_ASSERT(!chain.empty());
253 unapply(*chain.tip(), *chain.first());
258 VBK_CHECK_RETURN
bool apply(index_t& from,
260 ValidationState& state) {
261 VBK_TRACE_ZONE_SCOPED;
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()));
275 VBK_ASSERT(from.getHeight() < to.getHeight());
277 Chain<index_t> chain(from.getHeight() + 1, &to);
278 VBK_ASSERT(chain.first());
279 VBK_ASSERT(chain.first()->pprev == &from);
281 VBK_LOG_DEBUG(
"Applying %d blocks from=%s, to=%s",
283 from.toPrettyString(),
284 to.toPrettyString());
286 for (
auto* index : chain) {
287 if (!applyBlock(*index, state)) {
289 unapply(*index->pprev, from);
301 VBK_CHECK_RETURN
bool apply(Chain<index_t>& chain, ValidationState& state) {
302 return apply(*chain.first(), *chain.tip(), state);
309 VBK_CHECK_RETURN
bool setState(index_t& from,
311 ValidationState& state) {
312 VBK_TRACE_ZONE_SCOPED;
320 VBK_ASSERT_MSG(forkBlock,
"Fork Block must exist!");
322 auto& fork = as_mut(*forkBlock);
324 if (!apply(fork, to, state)) {
326 bool success = apply(fork, from, state);
327 VBK_ASSERT_MSG(success,
328 "state corruption: failed to rollback the state: %s",
337 ProtectingBlockTree& tree() {
return ing_; }
338 const ProtectingBlockTree& tree()
const {
return ing_; }
339 const ProtectedChainParams& params()
const {
return ed_.getParams(); }
343 ProtectingBlockTree& ing_;
344 command_group_store_t& commandGroupStore_;
@ 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.
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...