From c3c7cb61b69d25cdd5fcfa25debd7ed0e024be83 Mon Sep 17 00:00:00 2001 From: Colin LeMahieu Date: Thu, 14 Mar 2024 16:13:43 +0000 Subject: [PATCH] WIP- Storing unconfirmed blocks in memory. # Conflicts: # nano/secure/ledger_view_unconfirmed.cpp --- nano/core_test/bootstrap.cpp | 8 +- nano/core_test/confirming_set.cpp | 4 +- nano/core_test/ledger.cpp | 97 +++-- nano/core_test/ledger_confirm.cpp | 49 +-- nano/core_test/node.cpp | 5 +- nano/node/active_transactions.cpp | 2 +- nano/node/active_transactions.hpp | 2 +- nano/node/backlog_population.cpp | 43 +-- nano/node/backlog_population.hpp | 7 +- nano/node/bootstrap/bootstrap_frontier.cpp | 11 +- nano/node/bootstrap/bootstrap_server.cpp | 7 +- nano/node/node.cpp | 16 +- nano/node/scheduler/priority.cpp | 5 +- nano/secure/account_iterator.cpp | 2 + nano/secure/ledger.cpp | 408 ++++++++++----------- nano/secure/ledger.hpp | 20 +- nano/secure/ledger_set_any.cpp | 82 ++++- nano/secure/ledger_set_confirmed.cpp | 84 ++--- nano/secure/ledger_set_confirmed.hpp | 8 + nano/store/lmdb/block.cpp | 13 +- nano/test_common/system.cpp | 4 + 21 files changed, 444 insertions(+), 433 deletions(-) diff --git a/nano/core_test/bootstrap.cpp b/nano/core_test/bootstrap.cpp index 91d548da70..76ae773141 100644 --- a/nano/core_test/bootstrap.cpp +++ b/nano/core_test/bootstrap.cpp @@ -56,7 +56,9 @@ TEST (bulk_pull, end_not_owned) nano::test::system system (1); nano::keypair key2; system.wallet (0)->insert_adhoc (nano::dev::genesis_key.prv); - ASSERT_NE (nullptr, system.wallet (0)->send_action (nano::dev::genesis_key.pub, key2.pub, 100)); + auto block = system.wallet (0)->send_action (nano::dev::genesis_key.pub, key2.pub, 100); + ASSERT_NE (nullptr, block); + system.nodes[0]->ledger.confirm (system.nodes[0]->store.tx_begin_write (), block->hash ()); nano::block_hash latest (system.nodes[0]->latest (nano::dev::genesis_key.pub)); nano::block_builder builder; auto open = builder @@ -127,6 +129,7 @@ TEST (bulk_pull, ascending_one_hash) .build (); node.work_generate_blocking (*block1); ASSERT_EQ (nano::block_status::progress, node.process (block1)); + node.ledger.confirm (node.store.tx_begin_write (), block1->hash ()); auto socket = std::make_shared (node, nano::transport::socket::endpoint_type_t::server); auto connection = std::make_shared (socket, system.nodes[0]); auto req = std::make_unique (nano::dev::network_params.network); @@ -159,6 +162,7 @@ TEST (bulk_pull, ascending_two_account) .build (); node.work_generate_blocking (*block1); ASSERT_EQ (nano::block_status::progress, node.process (block1)); + node.ledger.confirm (node.store.tx_begin_write (), block1->hash ()); auto socket = std::make_shared (node, nano::transport::socket::endpoint_type_t::server); auto connection = std::make_shared (socket, system.nodes[0]); auto req = std::make_unique (nano::dev::network_params.network); @@ -262,7 +266,7 @@ TEST (bulk_pull, count_limit) .work (*system.work.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, node0->process (receive1)); - + node0->ledger.confirm (node0->store.tx_begin_write (), receive1->hash ()); auto connection (std::make_shared (std::make_shared (*node0, nano::transport::socket::endpoint_type_t::server), node0)); auto req = std::make_unique (nano::dev::network_params.network); req->start = receive1->hash (); diff --git a/nano/core_test/confirming_set.cpp b/nano/core_test/confirming_set.cpp index 490957e1f9..3d8d5edbb7 100644 --- a/nano/core_test/confirming_set.cpp +++ b/nano/core_test/confirming_set.cpp @@ -276,6 +276,6 @@ TEST (confirmation_callback, election_winner_details_clearing_node_process_confi node->active.add_election_winner_details (send->hash (), nullptr); nano::election_status election; election.winner = send; - node->process_confirmed (election, 1000000); - ASSERT_EQ (0, node->active.election_winner_details_size ()); + node->process_confirmed (election); + ASSERT_TIMELY_EQ (5s, 0, node->active.election_winner_details_size ()); } diff --git a/nano/core_test/ledger.cpp b/nano/core_test/ledger.cpp index 18468f5b63..b2cdf3ff0a 100644 --- a/nano/core_test/ledger.cpp +++ b/nano/core_test/ledger.cpp @@ -71,6 +71,7 @@ TEST (ledger, genesis_balance) // Frontier time should have been updated when genesis balance was added ASSERT_GE (nano::seconds_since_epoch (), info->modified); ASSERT_LT (nano::seconds_since_epoch () - info->modified, 10); + ASSERT_EQ (1, ledger.account_count ()); // Genesis block should be confirmed by default nano::confirmation_height_info confirmation_height_info; ASSERT_FALSE (store.confirmation_height.get (transaction, nano::dev::genesis_key.pub, confirmation_height_info)); @@ -283,7 +284,6 @@ TEST (ledger, process_receive) ASSERT_TRUE (pending1); ASSERT_EQ (nano::dev::genesis_key.pub, pending1->source); ASSERT_EQ (25, pending1->amount.number ()); - ASSERT_EQ (store.account.count (transaction), ledger.account_count ()); } TEST (ledger, rollback_receiver) @@ -1637,7 +1637,6 @@ TEST (ledger, fail_open_fork_previous) .work (*pool.generate (key1.pub)) .build (); ASSERT_EQ (nano::block_status::fork, ledger.process (transaction, block4)); - ASSERT_EQ (store.account.count (transaction), ledger.account_count ()); } TEST (ledger, fail_open_account_mismatch) @@ -2296,6 +2295,7 @@ TEST (ledger, bootstrap_rep_weight) .work (*pool.generate (info1->head)) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send)); + ledger.confirm (transaction, send->hash ()); } ASSERT_EQ (3, ledger.block_count ()); ASSERT_EQ (0, ledger.weight (key2.pub)); @@ -2599,7 +2599,7 @@ TEST (ledger, state_open) ASSERT_EQ (nano::Gxrb_ratio, ledger.any.balance (transaction, open1->hash ())); ASSERT_EQ (nano::Gxrb_ratio, ledger.any.amount (transaction, open1->hash ())); ASSERT_EQ (nano::dev::constants.genesis_amount, ledger.weight (nano::dev::genesis_key.pub)); - ASSERT_EQ (ledger.account_count (), store.account.count (transaction)); + // ASSERT_EQ (ledger.cache.account_count, store.account.count (transaction)); ASSERT_EQ (1, open2->sideband ().height); ASSERT_FALSE (open2->is_send ()); ASSERT_TRUE (open2->is_receive ()); @@ -2929,7 +2929,6 @@ TEST (ledger, state_state_open_fork) .build (); ASSERT_EQ (nano::block_status::fork, ledger.process (transaction, open2)); ASSERT_EQ (open1->root (), open2->root ()); - ASSERT_EQ (store.account.count (transaction), ledger.account_count ()); } TEST (ledger, state_open_previous_fail) @@ -3291,7 +3290,7 @@ TEST (ledger, state_rollback_received_send) ASSERT_EQ (nano::dev::constants.genesis_amount, ledger.weight (nano::dev::genesis_key.pub)); ASSERT_FALSE (ledger.any.balance (transaction, key.pub)); ASSERT_EQ (0, ledger.weight (key.pub)); - ASSERT_EQ (store.account.count (transaction), ledger.account_count ()); + ASSERT_EQ (1, ledger.account_count ()); } TEST (ledger, state_rep_change_rollback) @@ -3943,7 +3942,6 @@ TEST (ledger, epoch_blocks_receive_upgrade) ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, epoch4)); ASSERT_EQ (nano::epoch::epoch_2, epoch4->sideband ().details.epoch); ASSERT_EQ (nano::epoch::epoch_0, epoch4->sideband ().source_epoch); // Not used for epoch blocks - ASSERT_EQ (store.account.count (transaction), ledger.account_count ()); } TEST (ledger, epoch_blocks_fork) @@ -4691,9 +4689,9 @@ TEST (ledger, dependents_confirmed) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, receive2)); ASSERT_FALSE (ledger.dependents_confirmed (transaction, *receive2)); - ledger.confirm (transaction, receive1->hash ()); - ASSERT_FALSE (ledger.dependents_confirmed (transaction, *receive2)); ledger.confirm (transaction, send2->hash ()); + ASSERT_FALSE (ledger.dependents_confirmed (transaction, *receive2)); + ledger.confirm (transaction, receive1->hash ()); ASSERT_TRUE (ledger.dependents_confirmed (transaction, *receive2)); } @@ -4793,10 +4791,10 @@ TEST (ledger, cache) auto pruned_count = i; auto cache_check = [&, i] (nano::ledger const & ledger) { - ASSERT_EQ (account_count, ledger.account_count ()); + // ASSERT_EQ (account_count, ledger.account_count ()); ASSERT_EQ (block_count, ledger.block_count ()); ASSERT_EQ (cemented_count, ledger.cemented_count ()); - ASSERT_EQ (genesis_weight, ledger.cache.rep_weights.representation_get (nano::dev::genesis_key.pub)); + ASSERT_EQ (genesis_weight, ledger.weight (nano::dev::genesis_key.pub)); ASSERT_EQ (pruned_count, ledger.pruned_count ()); }; @@ -4828,7 +4826,7 @@ TEST (ledger, cache) ++block_count; --genesis_weight; cache_check (ledger); - cache_check (nano::ledger (store, stats, nano::dev::constants)); + // cache_check (nano::ledger (store, stats, nano::dev::constants)); { auto transaction (store.tx_begin_write ()); @@ -4838,7 +4836,7 @@ TEST (ledger, cache) ++block_count; ++account_count; cache_check (ledger); - cache_check (nano::ledger (store, stats, nano::dev::constants)); + // cache_check (nano::ledger (store, stats, nano::dev::constants)); { auto transaction (store.tx_begin_write ()); @@ -4848,7 +4846,7 @@ TEST (ledger, cache) ++cemented_count; cache_check (ledger); - cache_check (nano::ledger (store, stats, nano::dev::constants)); + // cache_check (nano::ledger (store, stats, nano::dev::constants)); { auto transaction (store.tx_begin_write ()); @@ -4893,10 +4891,7 @@ TEST (ledger, pruning_action) .work (*pool.generate (nano::dev::genesis->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send1)); - ASSERT_TRUE (store->block.exists (transaction, send1->hash ())); - auto send1_stored (store->block.get (transaction, send1->hash ())); - ASSERT_NE (nullptr, send1_stored); - ASSERT_EQ (*send1, *send1_stored); + ASSERT_TRUE (ledger.any.exists (transaction, send1->hash ())); ASSERT_TRUE (ledger.any.get (transaction, nano::pending_key{ nano::dev::genesis_key.pub, send1->hash () })); auto send2 = builder .state () @@ -4909,7 +4904,7 @@ TEST (ledger, pruning_action) .work (*pool.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); // Pruning action ledger.confirm (transaction, send1->hash ()); ASSERT_EQ (1, ledger.pruning_action (transaction, send1->hash (), 1)); @@ -4922,8 +4917,8 @@ TEST (ledger, pruning_action) ASSERT_TRUE (ledger.any.exists_or_pruned (transaction, send1->hash ())); ledger.pruning = true; ASSERT_TRUE (store->pruned.exists (transaction, send1->hash ())); - ASSERT_TRUE (store->block.exists (transaction, nano::dev::genesis->hash ())); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, nano::dev::genesis->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); // Receiving pruned block auto receive1 = builder .state () @@ -4936,23 +4931,17 @@ TEST (ledger, pruning_action) .work (*pool.generate (send2->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, receive1)); - ASSERT_TRUE (store->block.exists (transaction, receive1->hash ())); - auto receive1_stored (store->block.get (transaction, receive1->hash ())); - ASSERT_NE (nullptr, receive1_stored); - ASSERT_EQ (*receive1, *receive1_stored); + ASSERT_TRUE (ledger.any.exists (transaction, receive1->hash ())); ASSERT_FALSE (ledger.any.get (transaction, nano::pending_key{ nano::dev::genesis_key.pub, send1->hash () })); - ASSERT_EQ (4, receive1_stored->sideband ().height); - ASSERT_FALSE (receive1_stored->is_send ()); - ASSERT_TRUE (receive1_stored->is_receive ()); - ASSERT_FALSE (receive1_stored->sideband ().details.is_epoch); // Middle block pruning - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); ledger.confirm (transaction, send2->hash ()); ASSERT_EQ (1, ledger.pruning_action (transaction, send2->hash (), 1)); ASSERT_TRUE (store->pruned.exists (transaction, send2->hash ())); - ASSERT_FALSE (store->block.exists (transaction, send2->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send2->hash ())); ASSERT_EQ (store->account.count (transaction), ledger.account_count ()); ASSERT_EQ (store->pruned.count (transaction), ledger.pruned_count ()); + ledger.confirm (transaction, receive1->hash ()); ASSERT_EQ (store->block.count (transaction), ledger.block_count () - ledger.pruned_count ()); } @@ -5058,15 +5047,15 @@ TEST (ledger, pruning_source_rollback) .work (*pool.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); ledger.confirm (transaction, send1->hash ()); // Pruning action ASSERT_EQ (2, ledger.pruning_action (transaction, send1->hash (), 1)); - ASSERT_FALSE (store->block.exists (transaction, send1->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send1->hash ())); ASSERT_TRUE (store->pruned.exists (transaction, send1->hash ())); - ASSERT_FALSE (store->block.exists (transaction, epoch1->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, epoch1->hash ())); ASSERT_TRUE (store->pruned.exists (transaction, epoch1->hash ())); - ASSERT_TRUE (store->block.exists (transaction, nano::dev::genesis->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, nano::dev::genesis->hash ())); auto info = ledger.any.get (transaction, nano::pending_key (nano::dev::genesis_key.pub, send1->hash ())); ASSERT_TRUE (info); ASSERT_EQ (nano::dev::genesis_key.pub, info->source); @@ -5091,7 +5080,7 @@ TEST (ledger, pruning_source_rollback) ASSERT_FALSE (ledger.rollback (transaction, receive1->hash ())); auto info2 = ledger.any.get (transaction, nano::pending_key (nano::dev::genesis_key.pub, send1->hash ())); ASSERT_TRUE (info2); - ASSERT_NE (nano::dev::genesis_key.pub, info2->source); // Tradeoff to not store pruned blocks accounts + ASSERT_EQ (nano::dev::genesis_key.pub, info2->source); ASSERT_EQ (nano::Gxrb_ratio, info2->amount.number ()); ASSERT_EQ (nano::epoch::epoch_1, info2->epoch); // Process receive block again @@ -5133,7 +5122,7 @@ TEST (ledger, pruning_source_rollback_legacy) .work (*pool.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); ASSERT_TRUE (ledger.any.get (transaction, nano::pending_key{ key1.pub, send2->hash () })); auto send3 = builder .send () @@ -5144,16 +5133,16 @@ TEST (ledger, pruning_source_rollback_legacy) .work (*pool.generate (send2->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send3)); - ASSERT_TRUE (store->block.exists (transaction, send3->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send3->hash ())); ASSERT_TRUE (ledger.any.get (transaction, nano::pending_key{ nano::dev::genesis_key.pub, send3->hash () })); ledger.confirm (transaction, send2->hash ()); // Pruning action ASSERT_EQ (2, ledger.pruning_action (transaction, send2->hash (), 1)); - ASSERT_FALSE (store->block.exists (transaction, send2->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send2->hash ())); ASSERT_TRUE (store->pruned.exists (transaction, send2->hash ())); - ASSERT_FALSE (store->block.exists (transaction, send1->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send1->hash ())); ASSERT_TRUE (store->pruned.exists (transaction, send1->hash ())); - ASSERT_TRUE (store->block.exists (transaction, nano::dev::genesis->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, nano::dev::genesis->hash ())); auto info1 = ledger.any.get (transaction, nano::pending_key (nano::dev::genesis_key.pub, send1->hash ())); ASSERT_TRUE (info1); ASSERT_EQ (nano::dev::genesis_key.pub, info1->source); @@ -5180,7 +5169,7 @@ TEST (ledger, pruning_source_rollback_legacy) ASSERT_FALSE (ledger.rollback (transaction, receive1->hash ())); auto info3 = ledger.any.get (transaction, nano::pending_key (nano::dev::genesis_key.pub, send1->hash ())); ASSERT_TRUE (info3); - ASSERT_NE (nano::dev::genesis_key.pub, info3->source); // Tradeoff to not store pruned blocks accounts + ASSERT_EQ (nano::dev::genesis_key.pub, info3->source); ASSERT_EQ (nano::Gxrb_ratio, info3->amount.number ()); ASSERT_EQ (nano::epoch::epoch_0, info3->epoch); // Process receive block again @@ -5205,7 +5194,7 @@ TEST (ledger, pruning_source_rollback_legacy) ASSERT_FALSE (ledger.rollback (transaction, open1->hash ())); auto info4 = ledger.any.get (transaction, nano::pending_key (key1.pub, send2->hash ())); ASSERT_TRUE (info4); - ASSERT_NE (nano::dev::genesis_key.pub, info4->source); // Tradeoff to not store pruned blocks accounts + ASSERT_EQ (nano::dev::genesis_key.pub, info4->source); ASSERT_EQ (nano::Gxrb_ratio, info4->amount.number ()); ASSERT_EQ (nano::epoch::epoch_0, info4->epoch); // Process open block again @@ -5281,7 +5270,7 @@ TEST (ledger, pruning_legacy_blocks) .work (*pool.generate (open1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send3)); - ledger.confirm (transaction, open1->hash ()); + ledger.confirm (transaction, send3->hash ()); // Pruning action ASSERT_EQ (3, ledger.pruning_action (transaction, change1->hash (), 2)); ASSERT_EQ (1, ledger.pruning_action (transaction, open1->hash (), 1)); @@ -5325,7 +5314,7 @@ TEST (ledger, pruning_safe_functions) .work (*pool.generate (nano::dev::genesis->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send1)); - ASSERT_TRUE (store->block.exists (transaction, send1->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send1->hash ())); auto send2 = builder .state () .account (nano::dev::genesis_key.pub) @@ -5337,15 +5326,15 @@ TEST (ledger, pruning_safe_functions) .work (*pool.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); ledger.confirm (transaction, send1->hash ()); // Pruning action ASSERT_EQ (1, ledger.pruning_action (transaction, send1->hash (), 1)); - ASSERT_FALSE (store->block.exists (transaction, send1->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send1->hash ())); ASSERT_TRUE (ledger.any.exists_or_pruned (transaction, send1->hash ())); // true for pruned ASSERT_TRUE (store->pruned.exists (transaction, send1->hash ())); - ASSERT_TRUE (store->block.exists (transaction, nano::dev::genesis->hash ())); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, nano::dev::genesis->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); // Safe ledger actions ASSERT_FALSE (ledger.any.balance (transaction, send1->hash ())); ASSERT_EQ (nano::dev::constants.genesis_amount - nano::Gxrb_ratio * 2, ledger.any.balance (transaction, send2->hash ()).value ()); @@ -5377,7 +5366,7 @@ TEST (ledger, hash_root_random) .work (*pool.generate (nano::dev::genesis->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send1)); - ASSERT_TRUE (store->block.exists (transaction, send1->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send1->hash ())); auto send2 = builder .state () .account (nano::dev::genesis_key.pub) @@ -5389,14 +5378,14 @@ TEST (ledger, hash_root_random) .work (*pool.generate (send1->hash ())) .build (); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); - ledger.confirm (transaction, send1->hash ()); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); + ledger.confirm (transaction, send2->hash ()); // Pruning action ASSERT_EQ (1, ledger.pruning_action (transaction, send1->hash (), 1)); - ASSERT_FALSE (store->block.exists (transaction, send1->hash ())); + ASSERT_FALSE (ledger.any.exists (transaction, send1->hash ())); ASSERT_TRUE (store->pruned.exists (transaction, send1->hash ())); - ASSERT_TRUE (store->block.exists (transaction, nano::dev::genesis->hash ())); - ASSERT_TRUE (store->block.exists (transaction, send2->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, nano::dev::genesis->hash ())); + ASSERT_TRUE (ledger.any.exists (transaction, send2->hash ())); // Test random block including pruned bool done (false); auto iteration (0); diff --git a/nano/core_test/ledger_confirm.cpp b/nano/core_test/ledger_confirm.cpp index e742199a24..86df7778aa 100644 --- a/nano/core_test/ledger_confirm.cpp +++ b/nano/core_test/ledger_confirm.cpp @@ -44,8 +44,8 @@ TEST (ledger_confirm, single) ASSERT_FALSE (confirmed.exists (transaction, send1->hash ())); node->ledger.confirm (transaction, send1->hash ()); ASSERT_TRUE (confirmed.exists (transaction, send1->hash ())); - ASSERT_EQ (2, node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().height); - ASSERT_EQ (send1->hash (), node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().frontier); + ASSERT_EQ (2, node->ledger.confirmed.height (transaction, nano::dev::genesis_key.pub)); + ASSERT_EQ (send1->hash (), confirmed.head (transaction, nano::dev::genesis_key.pub)); // Rollbacks should fail as these blocks have been cemented ASSERT_TRUE (node->ledger.rollback (transaction, latest1)); @@ -200,17 +200,17 @@ TEST (ledger_confirm, multiple_accounts) ASSERT_TRUE (node->ledger.confirmed.exists (transaction, receive3->hash ())); ASSERT_EQ (4, node->ledger.any.get (transaction, nano::dev::genesis_key.pub).value ().block_count); - ASSERT_EQ (4, node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().height); - ASSERT_EQ (send3->hash (), node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().frontier); + ASSERT_EQ (4, node->ledger.confirmed.height (transaction, nano::dev::genesis_key.pub)); + ASSERT_EQ (send3->hash (), node->ledger.confirmed.head (transaction, nano::dev::genesis_key.pub)); ASSERT_EQ (3, node->ledger.any.get (transaction, key1.pub).value ().block_count); - ASSERT_EQ (2, node->store.confirmation_height.get (transaction, key1.pub).value ().height); - ASSERT_EQ (send4->hash (), node->store.confirmation_height.get (transaction, key1.pub).value ().frontier); + ASSERT_EQ (2, node->ledger.confirmed.height (transaction, key1.pub)); + ASSERT_EQ (send4->hash (), node->ledger.confirmed.head (transaction, key1.pub)); ASSERT_EQ (4, node->ledger.any.get (transaction, key2.pub).value ().block_count); - ASSERT_EQ (3, node->store.confirmation_height.get (transaction, key2.pub).value ().height); - ASSERT_EQ (send6->hash (), node->store.confirmation_height.get (transaction, key2.pub).value ().frontier); + ASSERT_EQ (3, node->ledger.confirmed.height (transaction, key2.pub)); + ASSERT_EQ (send6->hash (), node->ledger.confirmed.head (transaction, key2.pub)); ASSERT_EQ (2, node->ledger.any.get (transaction, key3.pub).value ().block_count); - ASSERT_EQ (2, node->store.confirmation_height.get (transaction, key3.pub).value ().height); - ASSERT_EQ (receive3->hash (), node->store.confirmation_height.get (transaction, key3.pub).value ().frontier); + ASSERT_EQ (2, node->ledger.confirmed.height (transaction, key3.pub)); + ASSERT_EQ (receive3->hash (), node->ledger.confirmed.head (transaction, key3.pub)); // The accounts for key1 and key2 have 1 more block in the chain than is confirmed. // So this can be rolled back, but the one before that cannot. Check that this is the case @@ -350,12 +350,12 @@ TEST (ledger_confirm, send_receive_between_2_accounts) ASSERT_TRUE (node->ledger.confirmed.exists (transaction, receive4->hash ())); ASSERT_EQ (7, node->ledger.any.get (transaction, nano::dev::genesis_key.pub).value ().block_count); - ASSERT_EQ (6, node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().height); - ASSERT_EQ (send5->hash (), node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().frontier); + ASSERT_EQ (6, node->ledger.confirmed.height (transaction, nano::dev::genesis_key.pub)); + ASSERT_EQ (send5->hash (), node->ledger.confirmed.head (transaction, nano::dev::genesis_key.pub)); ASSERT_EQ (5, node->ledger.any.get (transaction, key1.pub).value ().block_count); - ASSERT_EQ (5, node->store.confirmation_height.get (transaction, key1.pub).value ().height); - ASSERT_EQ (receive4->hash (), node->store.confirmation_height.get (transaction, key1.pub).value ().frontier); + ASSERT_EQ (5, node->ledger.confirmed.height (transaction, key1.pub)); + ASSERT_EQ (receive4->hash (), node->ledger.confirmed.head (transaction, key1.pub)); } TEST (ledger_confirm, send_receive_self) @@ -441,8 +441,8 @@ TEST (ledger_confirm, send_receive_self) ASSERT_TRUE (node->ledger.confirmed.exists (transaction, receive3->hash ())); ASSERT_EQ (8, node->ledger.any.get (transaction, nano::dev::genesis_key.pub).value ().block_count); - ASSERT_EQ (7, node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().height); - ASSERT_EQ (receive3->hash (), node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().frontier); + ASSERT_EQ (7, node->ledger.confirmed.height (transaction, nano::dev::genesis_key.pub)); + ASSERT_EQ (receive3->hash (), node->ledger.confirmed.head (transaction, nano::dev::genesis_key.pub)); ASSERT_EQ (7, node->ledger.cemented_count ()); } @@ -668,16 +668,16 @@ TEST (ledger_confirm, all_block_types) ASSERT_TRUE (node->ledger.confirmed.exists (transaction, state_send2->hash ())); nano::confirmation_height_info confirmation_height_info; ASSERT_LE (4, node->ledger.any.get (transaction, nano::dev::genesis_key.pub).value ().block_count); - ASSERT_EQ (3, node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().height); - ASSERT_EQ (send1->hash (), node->store.confirmation_height.get (transaction, nano::dev::genesis_key.pub).value ().frontier); + ASSERT_EQ (3, node->ledger.confirmed.height (transaction, nano::dev::genesis_key.pub)); + ASSERT_EQ (send1->hash (), node->ledger.confirmed.head (transaction, nano::dev::genesis_key.pub)); ASSERT_LE (7, node->ledger.any.get (transaction, key1.pub).value ().block_count); - ASSERT_EQ (6, node->store.confirmation_height.get (transaction, key1.pub).value ().height); - ASSERT_EQ (state_send1->hash (), node->store.confirmation_height.get (transaction, key1.pub).value ().frontier); + ASSERT_EQ (6, node->ledger.confirmed.height (transaction, key1.pub)); + ASSERT_EQ (state_send1->hash (), node->ledger.confirmed.head (transaction, key1.pub)); ASSERT_EQ (8, node->ledger.any.get (transaction, key2.pub).value ().block_count); - ASSERT_EQ (7, node->store.confirmation_height.get (transaction, key2.pub).value ().height); - ASSERT_EQ (state_send2->hash (), node->store.confirmation_height.get (transaction, key2.pub).value ().frontier); + ASSERT_EQ (7, node->ledger.confirmed.height (transaction, key2.pub)); + ASSERT_EQ (state_send2->hash (), node->ledger.confirmed.head (transaction, key2.pub)); } // This test ensures a block that's cemented cannot be rolled back by the node @@ -755,7 +755,7 @@ TEST (ledger_confirm, observers) ASSERT_EQ (2, node1->ledger.cemented_count ()); } -TEST (ledger_confirm, election_winner_details_clearing_node_process_confirmed) +TEST (ledger_confirm, DISABLED_election_winner_details_clearing_node_process_confirmed) { // Make sure election_winner_details is also cleared if the block never enters the confirmation height processor from node::process_confirmed nano::test::system system (1); @@ -774,7 +774,7 @@ TEST (ledger_confirm, election_winner_details_clearing_node_process_confirmed) node->active.add_election_winner_details (send->hash (), nullptr); nano::election_status election; election.winner = send; - node->process_confirmed (election, 1000000); + node->process_confirmed (election); ASSERT_EQ (0, node->active.election_winner_details_size ()); } @@ -846,6 +846,7 @@ TEST (ledger_confirm, pruned_source) ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send1)); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, open1)); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send2)); + ledger.confirm (transaction, send2->hash ()); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, send3)); ASSERT_EQ (nano::block_status::progress, ledger.process (transaction, open2)); ledger.confirm (transaction, send2->hash ()); diff --git a/nano/core_test/node.cpp b/nano/core_test/node.cpp index fcfc9c28b1..3de1f92a34 100644 --- a/nano/core_test/node.cpp +++ b/nano/core_test/node.cpp @@ -147,7 +147,7 @@ TEST (node, send_single) system.wallet (0)->insert_adhoc (nano::dev::genesis_key.prv); system.wallet (1)->insert_adhoc (key2.prv); ASSERT_NE (nullptr, system.wallet (0)->send_action (nano::dev::genesis_key.pub, key2.pub, system.nodes[0]->config.receive_minimum.number ())); - ASSERT_EQ (std::numeric_limits::max () - system.nodes[0]->config.receive_minimum.number (), system.nodes[0]->balance (nano::dev::genesis_key.pub)); + ASSERT_TIMELY_EQ (5s, std::numeric_limits::max () - system.nodes[0]->config.receive_minimum.number (), system.nodes[0]->balance (nano::dev::genesis_key.pub)); ASSERT_TRUE (system.nodes[0]->balance (key2.pub).is_zero ()); ASSERT_TIMELY (10s, !system.nodes[0]->balance (key2.pub).is_zero ()); } @@ -159,7 +159,7 @@ TEST (node, send_single_observing_peer) system.wallet (0)->insert_adhoc (nano::dev::genesis_key.prv); system.wallet (1)->insert_adhoc (key2.prv); ASSERT_NE (nullptr, system.wallet (0)->send_action (nano::dev::genesis_key.pub, key2.pub, system.nodes[0]->config.receive_minimum.number ())); - ASSERT_EQ (std::numeric_limits::max () - system.nodes[0]->config.receive_minimum.number (), system.nodes[0]->balance (nano::dev::genesis_key.pub)); + ASSERT_TIMELY_EQ (5s, std::numeric_limits::max () - system.nodes[0]->config.receive_minimum.number (), system.nodes[0]->balance (nano::dev::genesis_key.pub)); ASSERT_TRUE (system.nodes[0]->balance (key2.pub).is_zero ()); ASSERT_TIMELY (10s, std::all_of (system.nodes.begin (), system.nodes.end (), [&] (std::shared_ptr const & node_a) { return !node_a->balance (key2.pub).is_zero (); })); } @@ -1574,6 +1574,7 @@ TEST (node, unconfirmed_send) ASSERT_TIMELY_EQ (5s, node2.get_confirmation_height (node2.store.tx_begin_read (), key2.pub), 1); ASSERT_EQ (node2.balance (key2.pub), 2 * nano::Mxrb_ratio); auto recv1 = node2.ledger.find_receive_block_by_send_hash (node2.store.tx_begin_read (), key2.pub, send1->hash ()); + ASSERT_NE (nullptr, recv1); // create send2 to send from node2 to node1 and save it to node2's ledger without triggering an election (node1 does not hear about it) auto send2 = nano::state_block_builder{} diff --git a/nano/node/active_transactions.cpp b/nano/node/active_transactions.cpp index 38ba0fb38f..b5c951ff6a 100644 --- a/nano/node/active_transactions.cpp +++ b/nano/node/active_transactions.cpp @@ -97,7 +97,7 @@ void nano::active_transactions::block_cemented_callback (std::shared_ptrget_status (); votes = election->votes_with_weight (); } - if (confirming_set.exists (block->hash ())) + if (node.confirming_set.exists (block->hash ())) { status.type = nano::election_status_type::active_confirmed_quorum; } diff --git a/nano/node/active_transactions.hpp b/nano/node/active_transactions.hpp index 7fded5f623..076fbce40a 100644 --- a/nano/node/active_transactions.hpp +++ b/nano/node/active_transactions.hpp @@ -171,7 +171,7 @@ class active_transactions final bool empty () const; std::size_t size () const; bool publish (std::shared_ptr const &); - void block_cemented_callback (std::shared_ptr const &); + void block_cemented_callback (std::shared_ptr const & block); void block_already_cemented_callback (nano::block_hash const &); /** diff --git a/nano/node/backlog_population.cpp b/nano/node/backlog_population.cpp index 524e9d83d0..941a8706f6 100644 --- a/nano/node/backlog_population.cpp +++ b/nano/node/backlog_population.cpp @@ -2,13 +2,15 @@ #include #include #include +#include +#include #include #include #include -nano::backlog_population::backlog_population (const config & config_a, nano::store::component & store_a, nano::stats & stats_a) : +nano::backlog_population::backlog_population (const config & config_a, nano::ledger & ledger, nano::stats & stats_a) : config_m{ config_a }, - store{ store_a }, + ledger{ ledger }, stats{ stats_a } { } @@ -90,22 +92,22 @@ void nano::backlog_population::populate_backlog (nano::unique_lock lock.unlock (); { - auto transaction = store.tx_begin_read (); + auto transaction = ledger.store.tx_begin_read (); auto count = 0u; - auto i = store.account.begin (transaction, next); - auto const end = store.account.end (); + auto i = ledger.unconfirmed.account_upper_bound (transaction, next); + auto const end = ledger.unconfirmed.account_end (); for (; i != end && count < chunk_size; ++i, ++count, ++total) { transaction.refresh_if_needed (); stats.inc (nano::stat::type::backlog, nano::stat::detail::total); - auto const & account = i->first; - activate (transaction, account); - next = account.number () + 1; + next = i->first; + stats.inc (nano::stat::type::backlog, nano::stat::detail::activated); + activate_callback.notify (transaction, next); } - done = store.account.begin (transaction, next) == end; + done = i == end; } lock.lock (); @@ -114,26 +116,3 @@ void nano::backlog_population::populate_backlog (nano::unique_lock condition.wait_for (lock, std::chrono::milliseconds{ 1000 / config_m.frequency }); } } - -void nano::backlog_population::activate (store::transaction const & transaction, nano::account const & account) -{ - debug_assert (!activate_callback.empty ()); - - auto const maybe_account_info = store.account.get (transaction, account); - if (!maybe_account_info) - { - return; - } - auto const account_info = *maybe_account_info; - - auto const maybe_conf_info = store.confirmation_height.get (transaction, account); - auto const conf_info = maybe_conf_info.value_or (nano::confirmation_height_info{}); - - // If conf info is empty then it means then it means nothing is confirmed yet - if (conf_info.height < account_info.block_count) - { - stats.inc (nano::stat::type::backlog, nano::stat::detail::activated); - - activate_callback.notify (transaction, account); - } -} diff --git a/nano/node/backlog_population.hpp b/nano/node/backlog_population.hpp index a81f3b3058..92ff99912f 100644 --- a/nano/node/backlog_population.hpp +++ b/nano/node/backlog_population.hpp @@ -10,13 +10,13 @@ namespace nano::store { -class component; class transaction; } namespace nano { class account_info; class election_scheduler; +class ledger; class stats; class backlog_population final @@ -34,7 +34,7 @@ class backlog_population final unsigned frequency; }; - backlog_population (const config &, store::component &, nano::stats &); + backlog_population (const config &, nano::ledger & ledger, nano::stats &); ~backlog_population (); void start (); @@ -54,7 +54,7 @@ class backlog_population final callback_t activate_callback; private: // Dependencies - nano::store::component & store; + nano::ledger & ledger; nano::stats & stats; config config_m; @@ -64,7 +64,6 @@ class backlog_population final bool predicate () const; void populate_backlog (nano::unique_lock & lock); - void activate (store::transaction const &, nano::account const &); /** This is a manual trigger, the ongoing backlog population does not use this. * It can be triggered even when backlog population (frontiers confirmation) is disabled. */ diff --git a/nano/node/bootstrap/bootstrap_frontier.cpp b/nano/node/bootstrap/bootstrap_frontier.cpp index f37d7b90ce..4afbfa38a0 100644 --- a/nano/node/bootstrap/bootstrap_frontier.cpp +++ b/nano/node/bootstrap/bootstrap_frontier.cpp @@ -5,6 +5,7 @@ #include #include #include +#include #include @@ -392,15 +393,9 @@ void nano::frontier_req_server::next () } else { - for (auto i (node->store.confirmation_height.begin (transaction, current.number () + 1)), n (node->store.confirmation_height.end ()); i != n && accounts.size () != max_size; ++i) + for (auto i (node->ledger.confirmed.account_upper_bound (transaction, current.number ())), n (node->ledger.confirmed.account_end ()); i != n && accounts.size () != max_size; ++i) { - nano::confirmation_height_info const & info (i->second); - nano::block_hash const & confirmed_frontier (info.frontier); - if (!confirmed_frontier.is_zero ()) - { - nano::account const & account (i->first); - accounts.emplace_back (account, confirmed_frontier); - } + accounts.emplace_back (i->first, i->second.head); } } diff --git a/nano/node/bootstrap/bootstrap_server.cpp b/nano/node/bootstrap/bootstrap_server.cpp index 8484d4b387..20466fe123 100644 --- a/nano/node/bootstrap/bootstrap_server.cpp +++ b/nano/node/bootstrap/bootstrap_server.cpp @@ -5,6 +5,7 @@ #include #include #include +#include #include #include #include @@ -303,11 +304,11 @@ nano::asc_pull_ack nano::bootstrap_server::process (const store::transaction & t response_payload.account_head = account_info->head; response_payload.account_block_count = account_info->block_count; - auto conf_info = store.confirmation_height.get (transaction, target); + auto conf_info = ledger.confirmed.get (transaction, target); if (conf_info) { - response_payload.account_conf_frontier = conf_info->frontier; - response_payload.account_conf_height = conf_info->height; + response_payload.account_conf_frontier = conf_info.value ().head; + response_payload.account_conf_height = conf_info.value ().block_count; } } // If account is missing the response payload will contain all 0 fields, except for the target diff --git a/nano/node/node.cpp b/nano/node/node.cpp index 237faa7fa1..06524eb73f 100644 --- a/nano/node/node.cpp +++ b/nano/node/node.cpp @@ -22,6 +22,7 @@ #include #include #include +#include #include #include @@ -195,7 +196,7 @@ nano::node::node (std::shared_ptr io_ctx_a, std::filesy scheduler{ *scheduler_impl }, aggregator (config, stats, generator, final_generator, history, ledger, wallets, active), wallets (wallets_store.init_error (), *this), - backlog{ nano::backlog_population_config (config), store, stats }, + backlog{ nano::backlog_population_config (config), ledger, stats }, ascendboot{ config, block_processor, ledger, network, stats }, websocket{ config.websocket_config, observers, wallets, ledger, io_ctx, logger }, epoch_upgrader{ *this, ledger, store, network_params, logger }, @@ -944,11 +945,11 @@ bool nano::node::collect_ledger_pruning_targets (std::deque & uint64_t read_operations (0); bool finish_transaction (false); auto const transaction (store.tx_begin_read ()); - for (auto i (store.confirmation_height.begin (transaction, last_account_a)), n (store.confirmation_height.end ()); i != n && !finish_transaction;) + for (auto i (ledger.confirmed.account_upper_bound (transaction, last_account_a)), n (ledger.confirmed.account_end ()); i != n && !finish_transaction;) { ++read_operations; auto const & account (i->first); - nano::block_hash hash (i->second.frontier); + nano::block_hash hash (i->second.head); uint64_t depth (0); while (!hash.is_zero () && depth < max_depth_a) { @@ -981,7 +982,7 @@ bool nano::node::collect_ledger_pruning_targets (std::deque & read_operations += depth; if (read_operations >= batch_read_size_a) { - last_account_a = account.number () + 1; + last_account_a = account.number (); finish_transaction = true; } else @@ -1266,8 +1267,7 @@ void nano::node::process_confirmed (nano::election_status const & status_a, uint if (auto block_l = ledger.any.get (ledger.store.tx_begin_read (), hash)) { logger.trace (nano::log::type::node, nano::log::detail::process_confirmed, nano::log::arg{ "block", block_l }); - - confirming_set.add (block_l->hash ()); + confirming_set.add (hash); } else if (iteration_a < num_iters) { @@ -1345,9 +1345,7 @@ void nano::node::bootstrap_block (const nano::block_hash & hash) /** Convenience function to easily return the confirmation height of an account. */ uint64_t nano::node::get_confirmation_height (store::transaction const & transaction_a, nano::account & account_a) { - nano::confirmation_height_info info; - store.confirmation_height.get (transaction_a, account_a, info); - return info.height; + return ledger.confirmed.height (transaction_a, account_a); } nano::account nano::node::get_node_id () const diff --git a/nano/node/scheduler/priority.cpp b/nano/node/scheduler/priority.cpp index 9e970c6876..5e96928490 100644 --- a/nano/node/scheduler/priority.cpp +++ b/nano/node/scheduler/priority.cpp @@ -49,7 +49,10 @@ bool nano::scheduler::priority::activate (nano::account const & account, store:: { return false; } - auto block = node.ledger.any.get (transaction, node.ledger.any.successor (transaction, { head.is_zero () ? static_cast (account) : head, head }).value ()); + nano::qualified_root root{ head.is_zero () ? static_cast (account) : head, head }; + auto successor = node.ledger.any.successor (transaction, root); + auto block = node.ledger.any.get (transaction, successor.value ()); + debug_assert (block != nullptr); if (!node.ledger.dependents_confirmed (transaction, *block)) { return false; diff --git a/nano/secure/account_iterator.cpp b/nano/secure/account_iterator.cpp index 0665545abb..ed68b2264a 100644 --- a/nano/secure/account_iterator.cpp +++ b/nano/secure/account_iterator.cpp @@ -1,6 +1,8 @@ #include #include +#include #include template class nano::account_iterator; +template class nano::account_iterator; template class nano::account_iterator; diff --git a/nano/secure/ledger.cpp b/nano/secure/ledger.cpp index 39258f6846..2bc6b149ab 100644 --- a/nano/secure/ledger.cpp +++ b/nano/secure/ledger.cpp @@ -30,159 +30,6 @@ namespace { -/** - * Roll back the visited block - */ -class rollback_visitor : public nano::block_visitor -{ -public: - rollback_visitor (nano::store::write_transaction const & transaction_a, nano::ledger & ledger_a, std::vector> & list_a) : - transaction (transaction_a), - ledger (ledger_a), - list (list_a) - { - } - virtual ~rollback_visitor () = default; - void send_block (nano::send_block const & block_a) override - { - auto hash (block_a.hash ()); - nano::pending_key key (block_a.hashables.destination, hash); - auto pending = ledger.store.pending.get (transaction, key); - while (!error && !pending.has_value ()) - { - error = ledger.rollback (transaction, ledger.any.head (transaction, block_a.hashables.destination), list); - pending = ledger.store.pending.get (transaction, key); - } - if (!error) - { - auto info = ledger.any.get (transaction, pending.value ().source); - debug_assert (info); - ledger.store.pending.del (transaction, key); - ledger.cache.rep_weights.representation_add (transaction, info->representative, pending.value ().amount.number ()); - nano::account_info new_info (block_a.hashables.previous, info->representative, info->open_block, ledger.any.balance (transaction, block_a.hashables.previous).value (), nano::seconds_since_epoch (), info->block_count - 1, nano::epoch::epoch_0); - ledger.update_account (transaction, pending.value ().source, *info, new_info); - ledger.store.block.del (transaction, hash); - ledger.store.block.successor_clear (transaction, block_a.hashables.previous); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::send); - } - } - void receive_block (nano::receive_block const & block_a) override - { - auto hash (block_a.hash ()); - auto amount = ledger.any.amount (transaction, hash).value (); - auto destination_account = block_a.account (); - // Pending account entry can be incorrect if source block was pruned. But it's not affecting correct ledger processing - auto source_account = ledger.any.account (transaction, block_a.hashables.source); - auto info = ledger.any.get (transaction, destination_account); - debug_assert (info); - ledger.cache.rep_weights.representation_add (transaction, info->representative, 0 - amount); - nano::account_info new_info (block_a.hashables.previous, info->representative, info->open_block, ledger.any.balance (transaction, block_a.hashables.previous).value (), nano::seconds_since_epoch (), info->block_count - 1, nano::epoch::epoch_0); - ledger.update_account (transaction, destination_account, *info, new_info); - ledger.store.block.del (transaction, hash); - ledger.store.pending.put (transaction, nano::pending_key (destination_account, block_a.hashables.source), { source_account.value_or (0), amount, nano::epoch::epoch_0 }); - ledger.store.block.successor_clear (transaction, block_a.hashables.previous); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::receive); - } - void open_block (nano::open_block const & block_a) override - { - auto hash (block_a.hash ()); - auto amount = ledger.any.amount (transaction, hash).value (); - auto destination_account = block_a.account (); - auto source_account = ledger.any.account (transaction, block_a.hashables.source); - ledger.cache.rep_weights.representation_add (transaction, block_a.representative_field ().value (), 0 - amount); - nano::account_info new_info; - ledger.update_account (transaction, destination_account, new_info, new_info); - ledger.store.block.del (transaction, hash); - ledger.store.pending.put (transaction, nano::pending_key (destination_account, block_a.hashables.source), { source_account.value_or (0), amount, nano::epoch::epoch_0 }); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::open); - } - void change_block (nano::change_block const & block_a) override - { - auto hash (block_a.hash ()); - auto rep_block (ledger.representative (transaction, block_a.hashables.previous)); - auto account = block_a.account (); - auto info = ledger.any.get (transaction, account); - debug_assert (info); - auto balance = ledger.any.balance (transaction, block_a.hashables.previous).value (); - auto block = ledger.store.block.get (transaction, rep_block); - release_assert (block != nullptr); - auto representative = block->representative_field ().value (); - ledger.cache.rep_weights.representation_add_dual (transaction, block_a.hashables.representative, 0 - balance, representative, balance); - ledger.store.block.del (transaction, hash); - nano::account_info new_info (block_a.hashables.previous, representative, info->open_block, info->balance, nano::seconds_since_epoch (), info->block_count - 1, nano::epoch::epoch_0); - ledger.update_account (transaction, account, *info, new_info); - ledger.store.block.successor_clear (transaction, block_a.hashables.previous); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::change); - } - void state_block (nano::state_block const & block_a) override - { - auto hash (block_a.hash ()); - nano::block_hash rep_block_hash (0); - if (!block_a.hashables.previous.is_zero ()) - { - rep_block_hash = ledger.representative (transaction, block_a.hashables.previous); - } - nano::uint128_t balance = ledger.any.balance (transaction, block_a.hashables.previous).value_or (0); - auto is_send (block_a.hashables.balance < balance); - nano::account representative{}; - if (!rep_block_hash.is_zero ()) - { - // Move existing representation & add in amount delta - auto block (ledger.store.block.get (transaction, rep_block_hash)); - debug_assert (block != nullptr); - representative = block->representative_field ().value (); - ledger.cache.rep_weights.representation_add_dual (transaction, representative, balance, block_a.hashables.representative, 0 - block_a.hashables.balance.number ()); - } - else - { - // Add in amount delta only - ledger.cache.rep_weights.representation_add (transaction, block_a.hashables.representative, 0 - block_a.hashables.balance.number ()); - } - - auto info = ledger.any.get (transaction, block_a.hashables.account); - debug_assert (info); - - if (is_send) - { - nano::pending_key key (block_a.hashables.link.as_account (), hash); - while (!error && !ledger.any.get (transaction, key)) - { - error = ledger.rollback (transaction, ledger.any.head (transaction, block_a.hashables.link.as_account ()), list); - } - ledger.store.pending.del (transaction, key); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::send); - } - else if (!block_a.hashables.link.is_zero () && !ledger.is_epoch_link (block_a.hashables.link)) - { - // Pending account entry can be incorrect if source block was pruned. But it's not affecting correct ledger processing - auto source_account = ledger.any.account (transaction, block_a.hashables.link.as_block_hash ()); - nano::pending_info pending_info (source_account.value_or (0), block_a.hashables.balance.number () - balance, block_a.sideband ().source_epoch); - ledger.store.pending.put (transaction, nano::pending_key (block_a.hashables.account, block_a.hashables.link.as_block_hash ()), pending_info); - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::receive); - } - - debug_assert (!error); - auto previous_version (ledger.version (transaction, block_a.hashables.previous)); - nano::account_info new_info (block_a.hashables.previous, representative, info->open_block, balance, nano::seconds_since_epoch (), info->block_count - 1, previous_version); - ledger.update_account (transaction, block_a.hashables.account, *info, new_info); - - auto previous (ledger.store.block.get (transaction, block_a.hashables.previous)); - if (previous != nullptr) - { - ledger.store.block.successor_clear (transaction, block_a.hashables.previous); - } - else - { - ledger.stats.inc (nano::stat::type::rollback, nano::stat::detail::open); - } - ledger.store.block.del (transaction, hash); - } - nano::store::write_transaction const & transaction; - nano::ledger & ledger; - std::vector> & list; - bool error{ false }; -}; - /** * Determine the representative for this block */ @@ -272,7 +119,7 @@ nano::ledger::~ledger () void nano::ledger::initialize (nano::generate_cache_flags const & generate_cache_flags_a) { - if (generate_cache_flags_a.reps || generate_cache_flags_a.account_count || generate_cache_flags_a.block_count) + if (generate_cache_flags_a.reps || generate_cache_flags_a.account_count || generate_cache_flags_a.block_count || generate_cache_flags_a.cemented_count) { store.account.for_each_par ( [this] (store::read_transaction const & /*unused*/, store::iterator i, store::iterator n) { @@ -285,6 +132,7 @@ void nano::ledger::initialize (nano::generate_cache_flags const & generate_cache ++account_count_l; } this->cache.block_count += block_count_l; + this->cache.cemented_count += block_count_l; this->cache.account_count += account_count_l; }); @@ -299,19 +147,6 @@ void nano::ledger::initialize (nano::generate_cache_flags const & generate_cache }); } - if (generate_cache_flags_a.cemented_count) - { - store.confirmation_height.for_each_par ( - [this] (store::read_transaction const & /*unused*/, store::iterator i, store::iterator n) { - uint64_t cemented_count_l (0); - for (; i != n; ++i) - { - cemented_count_l += i->second.height; - } - this->cache.cemented_count += cemented_count_l; - }); - } - auto transaction (store.tx_begin_read ()); cache.pruned_count = store.pruned.count (transaction); } @@ -332,6 +167,7 @@ nano::uint128_t nano::ledger::account_receivable (store::transaction const & tra std::deque> nano::ledger::confirm (nano::store::write_transaction const & transaction, nano::block_hash const & hash) { + std::lock_guard lock{ unconfirmed.mutex }; std::deque> result; std::stack stack; stack.push (hash); @@ -354,7 +190,10 @@ std::deque> nano::ledger::confirm (nano::store::wri if (!confirmed.exists_or_pruned (transaction, hash)) { result.push_back (block); - confirm (transaction, *block); + auto existing = unconfirmed.block.find (hash); + release_assert (existing != unconfirmed.block.end ()); + auto delta = std::move (existing->second); + confirm (transaction, delta); } } else @@ -365,17 +204,9 @@ std::deque> nano::ledger::confirm (nano::store::wri return result; } -void nano::ledger::confirm (nano::store::write_transaction const & transaction, nano::block const & block) -{ - debug_assert ((!store.confirmation_height.get (transaction, block.account ()) && block.sideband ().height == 1) || store.confirmation_height.get (transaction, block.account ()).value ().height + 1 == block.sideband ().height); - confirmation_height_info info{ block.sideband ().height, block.hash () }; - store.confirmation_height.put (transaction, block.account (), info); - ++cache.cemented_count; - stats.inc (nano::stat::type::confirmation_height, nano::stat::detail::blocks_confirmed); -} - nano::block_status nano::ledger::process (store::write_transaction const & transaction_a, std::shared_ptr block_a) { + std::lock_guard lock{ unconfirmed.mutex }; debug_assert (!constants.work.validate_entry (*block_a) || constants.genesis == nano::dev::genesis); nano::block_check_context ctx{ *this, block_a }; auto code = ctx.check (transaction_a); @@ -447,7 +278,7 @@ std::pair nano::ledger::hash_root_random (st } // Vote weight of an account -nano::uint128_t nano::ledger::weight (nano::account const & account_a) +nano::uint128_t nano::ledger::weight (nano::account const & account_a) const { if (check_bootstrap_weights.load ()) { @@ -464,51 +295,131 @@ nano::uint128_t nano::ledger::weight (nano::account const & account_a) check_bootstrap_weights = false; } } + { + std::lock_guard lock{ unconfirmed.mutex }; + if (auto it = unconfirmed.weight.find (account_a); it != unconfirmed.weight.end ()) + { + return it->second.number (); + } + } return cache.rep_weights.representation_get (account_a); } -nano::uint128_t nano::ledger::weight_exact (store::transaction const & txn_a, nano::account const & representative_a) +nano::uint128_t nano::ledger::weight_exact (store::transaction const & txn_a, nano::account const & representative_a) const { return store.rep_weight.get (txn_a, representative_a); } // Rollback blocks until `block_a' doesn't exist or it tries to penetrate the confirmation height -bool nano::ledger::rollback (store::write_transaction const & transaction_a, nano::block_hash const & block_a, std::vector> & list_a) +bool nano::ledger::rollback (store::write_transaction const & transaction, nano::block_hash const & target, std::vector> & list_a) { - debug_assert (any.exists (transaction_a, block_a)); - auto account_l = any.account (transaction_a, block_a).value (); - auto block_account_height (any.height (transaction_a, block_a)); - rollback_visitor rollback (transaction_a, *this, list_a); - auto error (false); - while (!error && any.exists (transaction_a, block_a)) - { - nano::confirmation_height_info confirmation_height_info; - store.confirmation_height.get (transaction_a, account_l, confirmation_height_info); - if (block_account_height > confirmation_height_info.height) - { - auto info = any.get (transaction_a, account_l); - debug_assert (info); - auto block_l = any.get (transaction_a, info->head); - list_a.push_back (block_l); - block_l->visit (rollback); - error = rollback.error; - if (!error) + std::stack stack; + stack.push (target); + while (!stack.empty ()) + { + auto hash = stack.top (); + auto block = any.get (transaction, hash); + if (block != nullptr) + { + if (confirmed.exists_or_pruned (transaction, hash)) + { + return true; + } + if (unconfirmed.successor.count (hash) != 0) { - --cache.block_count; + auto head = unconfirmed.account.at (block->account ()).head; + debug_assert (head != block->hash ()); + stack.push (head); + } + if (block->is_send ()) + { + auto destination = block->destination (); + if (unconfirmed.received.count ({ destination, hash }) != 0) + { + auto head = unconfirmed.account.at (destination).head; + stack.push (head); + } + } + if (hash == stack.top ()) + { + list_a.push_back (block); + stack.pop (); + rollback (transaction, *block); + } + else + { + // Rollback dependencies were added } } else { - error = true; + stack.pop (); } } - return error; + return false; } -bool nano::ledger::rollback (store::write_transaction const & transaction_a, nano::block_hash const & block_a) +void nano::ledger::rollback (store::write_transaction const & transaction, nano::block const & block) +{ + debug_assert (unconfirmed.account.at (block.account ()).head == block.hash ()); + auto existing = unconfirmed.block.find (unconfirmed.account.at (block.account ()).head); + auto delta = std::move (existing->second); + unconfirmed.block.erase (existing); + if (delta.block->previous ().is_zero ()) + { + stats.inc (nano::stat::type::rollback, nano::stat::detail::open); + [[maybe_unused]] auto erased = unconfirmed.account.erase (block.account ()); + debug_assert (erased == 1); + } + else + { + [[maybe_unused]] auto erased = unconfirmed.successor.erase (delta.block->previous ()); + debug_assert (erased == 1); + if (auto existing = unconfirmed.block.find (block.previous ()); existing != unconfirmed.block.end ()) + { + unconfirmed.account[block.account ()] = unconfirmed.block.at (block.previous ()).head; + } + else + { + [[maybe_unused]] auto erased = unconfirmed.account.erase (block.account ()); + debug_assert (erased == 1); + debug_assert (unconfirmed.accounts_updated > 0); + --unconfirmed.accounts_updated; + } + } + auto const & [receivable_key, receivable_info] = delta.receivable; + if (receivable_key) + { + if (receivable_info) + { + // key + info = send + [[maybe_unused]] auto erased = unconfirmed.receivable.erase (receivable_key.value ()); + debug_assert (erased == 1); + stats.inc (nano::stat::type::rollback, nano::stat::detail::send); + } + else + { + // key + null = receive + [[maybe_unused]] auto erased = unconfirmed.received.erase (receivable_key.value ()); + debug_assert (erased == 1); + stats.inc (nano::stat::type::rollback, nano::stat::detail::receive); + } + } + else + { + stats.inc (nano::stat::type::rollback, nano::stat::detail::change); + } + unconfirmed.weight_add (delta.head.representative, 0 - block.balance ().number (), weight_exact (transaction, delta.head.representative)); + if (delta.weight.first) + { + unconfirmed.weight_add (delta.weight.first.value (), delta.weight.second.value (), weight_exact (transaction, delta.weight.first.value ())); + } +} + +bool nano::ledger::rollback (store::write_transaction const & transaction, nano::block_hash const & hash) { std::vector> rollback_list; - return rollback (transaction_a, block_a, rollback_list); + return rollback (transaction, hash, rollback_list); } // Return latest root for account, account number if there are no blocks for this account. @@ -631,12 +542,12 @@ std::shared_ptr nano::ledger::find_receive_block_by_send_hash (stor debug_assert (send_block_hash != 0); // get the cemented frontier - nano::confirmation_height_info info; - if (store.confirmation_height.get (transaction, destination, info)) + std::optional info; + if (info = confirmed.get (transaction, destination); !info) { return nullptr; } - auto possible_receive_block = any.get (transaction, info.frontier); + auto possible_receive_block = any.get (transaction, info.value ().head); // walk down the chain until the source field of a receive block matches the send block hash while (possible_receive_block != nullptr) @@ -691,23 +602,27 @@ void nano::ledger::update_account (store::write_transaction const & transaction_ void nano::ledger::track (store::write_transaction const & transaction, nano::block_delta const & delta) { auto & block = *delta.block; - store.block.put (transaction, delta.block->hash (), block); - ++cache.block_count; - store.account.put (transaction, block.account (), delta.head); - if (block.previous ().is_zero ()) + unconfirmed.block[block.hash ()] = delta; + if (!block.previous ().is_zero ()) { - ++cache.account_count; + debug_assert (unconfirmed.successor.count (block.previous ()) == 0); + unconfirmed.successor[block.previous ()] = block.hash (); + if (unconfirmed.account.count (block.account ()) == 0) + { + ++unconfirmed.accounts_updated; + } } - auto const &[receivable_key, receivable_info] = delta.receivable; + unconfirmed.account[block.account ()] = delta.head; + auto const & [receivable_key, receivable_info] = delta.receivable; if (receivable_key) { if (receivable_info) { - store.pending.put (transaction, receivable_key.value (), receivable_info.value ()); + unconfirmed.receivable[receivable_key.value ()] = receivable_info.value (); } else { - store.pending.del (transaction, receivable_key.value ()); + unconfirmed.received.insert (receivable_key.value ()); } } if (delta.weight.first) @@ -929,6 +844,67 @@ nano::epoch nano::ledger::version (store::transaction const & transaction, nano: return version (*block_l); } +void nano::ledger::confirm (nano::store::write_transaction const & transaction, nano::block_delta const & delta) +{ + auto & block = *delta.block; + auto account_l = block.account (); + auto hash = block.hash (); + store.block.put (transaction, hash, block); + stats.inc (nano::stat::type::confirmation_height, nano::stat::detail::blocks_confirmed); + ++cache.cemented_count; + ++cache.block_count; + [[maybe_unused]] auto erased = unconfirmed.block.erase (hash); + debug_assert (erased == 1); + if (!block.previous ().is_zero ()) + { + [[maybe_unused]] auto erased = unconfirmed.successor.erase (block.previous ()); + debug_assert (erased == 1); + } + auto info = store.account.get (transaction, account_l); + auto representative = block.representative_field () ? block.representative_field ().value () : info.value ().representative; + if (unconfirmed.account[account_l].head == hash) + { + [[maybe_unused]] auto erased = unconfirmed.account.erase (account_l); + debug_assert (erased == 1); + if (!block.previous ().is_zero ()) + { + debug_assert (unconfirmed.accounts_updated > 0); + --unconfirmed.accounts_updated; + } + } + else + { + if (block.previous ().is_zero ()) + { + ++unconfirmed.accounts_updated; + } + } + if (block.sideband ().details.is_send) + { + auto destination_l = block.destination (); + nano::pending_key key{ destination_l, hash }; + auto amount = delta.receivable.second.value ().amount; + nano::pending_info value{ account_l, amount, block.sideband ().details.epoch }; + store.pending.put (transaction, key, value); + [[maybe_unused]] auto erased = unconfirmed.receivable.erase (key); + debug_assert (erased); + debug_assert (info.has_value ()); + } + else if (block.sideband ().details.is_receive) + { + auto source_l = block.source (); + nano::pending_key key{ account_l, source_l }; + store.pending.del (transaction, key); + [[maybe_unused]] auto erased = unconfirmed.received.erase (key); + debug_assert (erased); + } + store.account.put (transaction, account_l, delta.head); + if (block.previous ().is_zero ()) + { + ++cache.account_count; + } +} + uint64_t nano::ledger::cemented_count () const { return cache.cemented_count; @@ -936,12 +912,12 @@ uint64_t nano::ledger::cemented_count () const uint64_t nano::ledger::block_count () const { - return cache.block_count; + return cache.block_count + unconfirmed.block_size (); } uint64_t nano::ledger::account_count () const { - return cache.account_count; + return cache.account_count + unconfirmed.account_size (); } uint64_t nano::ledger::pruned_count () const diff --git a/nano/secure/ledger.hpp b/nano/secure/ledger.hpp index 15579f1cf1..d4bf493928 100644 --- a/nano/secure/ledger.hpp +++ b/nano/secure/ledger.hpp @@ -11,6 +11,13 @@ #include #include +namespace nano +{ +class block_check_context; +class confirmed_set; +class stats; +} + namespace nano::store { class component; @@ -20,6 +27,7 @@ class write_transaction; namespace nano { +class backlog_population; class block; class block_delta; enum class block_status; @@ -34,6 +42,9 @@ class stats; class ledger final { + friend class backlog_population; + friend class block_check_context; + friend class ledger_set_any; template friend class receivable_iterator; @@ -47,9 +58,9 @@ class ledger final * If the weight is below the cache limit it returns 0. * During bootstrap it returns the preconfigured bootstrap weights. */ - nano::uint128_t weight (nano::account const &); + nano::uint128_t weight (nano::account const &) const; /* Returns the exact vote weight for the given representative by doing a database lookup */ - nano::uint128_t weight_exact (store::transaction const &, nano::account const &); + nano::uint128_t weight_exact (store::transaction const &, nano::account const &) const; std::shared_ptr forked_block (store::transaction const &, nano::block const &); nano::root latest_root (store::transaction const &, nano::account const &); nano::block_hash representative (store::transaction const &, nano::block_hash const &); @@ -86,13 +97,14 @@ class ledger final nano::stats & stats; std::unordered_map bootstrap_weights; uint64_t bootstrap_weight_max_blocks{ 1 }; - std::atomic check_bootstrap_weights; + mutable std::atomic check_bootstrap_weights; bool pruning{ false }; private: void initialize (nano::generate_cache_flags const &); void track (store::write_transaction const & transaction, nano::block_delta const & delta); - void confirm (nano::store::write_transaction const & transaction, nano::block const & block); + void confirm (nano::store::write_transaction const & transaction, nano::block_delta const & delta); + void rollback (store::write_transaction const & transaction, nano::block const & block); std::unique_ptr any_impl; std::unique_ptr confirmed_impl; diff --git a/nano/secure/ledger_set_any.cpp b/nano/secure/ledger_set_any.cpp index 57cdb3f661..353d886ea6 100644 --- a/nano/secure/ledger_set_any.cpp +++ b/nano/secure/ledger_set_any.cpp @@ -1,5 +1,6 @@ #include #include +#include #include #include #include @@ -33,12 +34,29 @@ auto nano::ledger_set_any::account_end () const -> account_iterator // Returns the next receivable entry equal or greater than 'key' auto nano::ledger_set_any::account_lower_bound (store::transaction const & transaction, nano::account const & account) const -> account_iterator { + std::lock_guard lock{ ledger.unconfirmed.mutex }; + auto mem = ledger.unconfirmed.account.lower_bound (account); auto disk = ledger.store.account.begin (transaction, account); - if (disk == ledger.store.account.end ()) + std::optional> mem_val; + if (mem != ledger.unconfirmed.account.end ()) { - return account_iterator{}; + mem_val = *mem; } - return account_iterator{ transaction, *this, *disk }; + std::optional> disk_val; + if (disk != ledger.store.account.end ()) + { + disk_val = *disk; + } + if (!mem_val) + { + return account_iterator{ transaction, *this, disk_val }; + } + if (!disk_val) + { + return account_iterator{ transaction, *this, mem_val }; + } + auto lower = mem_val.value ().first.number () <= disk_val.value ().first.number () ? mem_val : disk_val; + return account_iterator{ transaction, *this, lower }; } auto nano::ledger_set_any::account_upper_bound (store::transaction const & transaction, nano::account const & account) const -> account_iterator @@ -93,30 +111,38 @@ std::optional nano::ledger_set_any::balance (store::transaction bool nano::ledger_set_any::exists (store::transaction const & transaction, nano::block_hash const & hash) const { - return ledger.store.block.exists (transaction, hash); + std::lock_guard lock{ ledger.unconfirmed.mutex }; + return ledger.unconfirmed.block.count (hash) == 1 || ledger.store.block.exists (transaction, hash); } bool nano::ledger_set_any::exists_or_pruned (store::transaction const & transaction, nano::block_hash const & hash) const { - if (ledger.store.pruned.exists (transaction, hash)) - { - return true; - } - return ledger.store.block.exists (transaction, hash); + return exists (transaction, hash) || ledger.store.pruned.exists (transaction, hash); } std::optional nano::ledger_set_any::get (store::transaction const & transaction, nano::account const & account) const { - return ledger.store.account.get (transaction, account); + std::lock_guard lock{ ledger.unconfirmed.mutex }; + return ledger.unconfirmed.account.count (account) == 1 ? ledger.unconfirmed.account.at (account) : ledger.store.account.get (transaction, account); } std::shared_ptr nano::ledger_set_any::get (store::transaction const & transaction, nano::block_hash const & hash) const { - return ledger.store.block.get (transaction, hash); + std::lock_guard lock{ ledger.unconfirmed.mutex }; + return ledger.unconfirmed.block.count (hash) == 1 ? ledger.unconfirmed.block.at (hash).block : ledger.store.block.get (transaction, hash); } std::optional nano::ledger_set_any::get (store::transaction const & transaction, nano::pending_key const & key) const { + std::lock_guard lock{ ledger.unconfirmed.mutex }; + if (ledger.unconfirmed.received.count (key) != 0) + { + return std::nullopt; + } + if (ledger.unconfirmed.receivable.count (key) != 0) + { + return ledger.unconfirmed.receivable.at (key); + } return ledger.store.pending.get (transaction, key); } @@ -158,12 +184,36 @@ bool nano::ledger_set_any::receivable_any (store::transaction const & transactio std::optional> nano::ledger_set_any::receivable_lower_bound (store::transaction const & transaction, nano::account const & account, nano::block_hash const & hash) const { - auto result = ledger.store.pending.begin (transaction, { account, hash }); - if (result == ledger.store.pending.end ()) + std::lock_guard lock{ ledger.unconfirmed.mutex }; + auto mem = ledger.unconfirmed.receivable.lower_bound ({ account, hash }); + while (mem != ledger.unconfirmed.receivable.end () && ledger.unconfirmed.received.count (mem->first) != 0) { - return std::nullopt; + ++mem; + } + auto disk = ledger.store.pending.begin (transaction, { account, hash }); + while (disk != ledger.store.pending.end () && ledger.unconfirmed.received.count (disk->first) != 0) + { + ++disk; + } + std::optional> mem_val; + if (mem != ledger.unconfirmed.receivable.end ()) + { + mem_val = *mem; + } + std::optional> disk_val; + if (disk != ledger.store.pending.end ()) + { + disk_val = *disk; + } + if (!mem_val) + { + return disk_val; + } + if (!disk_val) + { + return mem_val; } - return *result; + return mem_val.value ().first < disk_val.value ().first ? mem_val : disk_val; } auto nano::ledger_set_any::receivable_end () const -> receivable_iterator @@ -195,7 +245,7 @@ std::optional nano::ledger_set_any::successor (store::transact { if (!root.previous ().is_zero ()) { - return ledger.store.block.successor (transaction, root.previous ()); + return ledger.unconfirmed.successor.count (root.previous ()) == 1 ? ledger.unconfirmed.successor.at (root.previous ()) : ledger.store.block.successor (transaction, root.previous ()); } else { diff --git a/nano/secure/ledger_set_confirmed.cpp b/nano/secure/ledger_set_confirmed.cpp index c6fe0b0ed3..23edea3d31 100644 --- a/nano/secure/ledger_set_confirmed.cpp +++ b/nano/secure/ledger_set_confirmed.cpp @@ -11,6 +11,33 @@ nano::ledger_set_confirmed::ledger_set_confirmed (nano::ledger const & ledger) : { } +auto nano::ledger_set_confirmed::account_begin (store::transaction const & transaction) const -> account_iterator +{ + return account_upper_bound (transaction, 0); +} + +auto nano::ledger_set_confirmed::account_end () const -> account_iterator +{ + return account_iterator{}; +} + +// Returns the next receivable entry equal or greater than 'key' +auto nano::ledger_set_confirmed::account_lower_bound (store::transaction const & transaction, nano::account const & account) const -> account_iterator +{ + auto existing = ledger.store.account.begin (transaction, account); + if (existing == ledger.store.account.end ()) + { + return account_iterator{ transaction, *this, std::nullopt }; + } + return account_iterator{ transaction, *this, *existing }; +} + +// Returns the next receivable entry for an account greater than 'account' +auto nano::ledger_set_confirmed::account_upper_bound (store::transaction const & transaction, nano::account const & account) const -> account_iterator +{ + return account_lower_bound (transaction, account.number () + 1); +} + std::optional nano::ledger_set_confirmed::balance (store::transaction const & transaction, nano::account const & account_a) const { auto block = get (transaction, head (transaction, account_a)); @@ -38,82 +65,37 @@ std::optional nano::ledger_set_confirmed::balance (store::trans bool nano::ledger_set_confirmed::exists (store::transaction const & transaction, nano::block_hash const & hash) const { - auto block = ledger.store.block.get (transaction, hash); - if (!block) - { - return false; - } - auto info = ledger.store.confirmation_height.get (transaction, block->account ()); - if (!info) - { - return false; - } - return block->sideband ().height <= info.value ().height; + return ledger.store.block.exists (transaction, hash); } bool nano::ledger_set_confirmed::exists_or_pruned (store::transaction const & transaction, nano::block_hash const & hash) const { - if (ledger.store.pruned.exists (transaction, hash)) - { - return true; - } - auto block = ledger.store.block.get (transaction, hash); - if (!block) - { - return false; - } - auto info = ledger.store.confirmation_height.get (transaction, block->account ()); - if (!info) - { - return false; - } - return block->sideband ().height <= info.value ().height; + return ledger.store.pruned.exists (transaction, hash) || exists (transaction, hash); } std::optional nano::ledger_set_confirmed::get (store::transaction const & transaction, nano::account const & account) const { - auto info = ledger.store.confirmation_height.get (transaction, account); - if (!info) - { - return std::nullopt; - } - debug_assert (false); return ledger.store.account.get (transaction, account); } std::shared_ptr nano::ledger_set_confirmed::get (store::transaction const & transaction, nano::block_hash const & hash) const { - auto block = ledger.store.block.get (transaction, hash); - if (!block) - { - return nullptr; - } - auto info = ledger.store.confirmation_height.get (transaction, block->account ()); - if (!info) - { - return nullptr; - } - return block->sideband ().height <= info.value ().height ? block : nullptr; + return ledger.store.block.get (transaction, hash); } std::optional nano::ledger_set_confirmed::get (store::transaction const & transaction, nano::pending_key const & key) const { - auto result = ledger.store.pending.get (transaction, key); - if (!result && !exists_or_pruned (transaction, key.hash)) - { - return std::nullopt; - } - return result; + return ledger.store.pending.get (transaction, key); } nano::block_hash nano::ledger_set_confirmed::head (store::transaction const & transaction, nano::account const & account) const { - auto info = ledger.store.confirmation_height.get (transaction, account); + auto info = get (transaction, account); if (!info) { return 0; } - return info.value ().frontier; + return info.value ().head; } uint64_t nano::ledger_set_confirmed::height (store::transaction const & transaction, nano::account const & account) const diff --git a/nano/secure/ledger_set_confirmed.hpp b/nano/secure/ledger_set_confirmed.hpp index 2845cb0acc..1539c86324 100644 --- a/nano/secure/ledger_set_confirmed.hpp +++ b/nano/secure/ledger_set_confirmed.hpp @@ -1,5 +1,6 @@ #pragma once +#include #include #include @@ -22,10 +23,17 @@ namespace nano class ledger_set_confirmed { public: + using account_iterator = nano::account_iterator; using receivable_iterator = nano::receivable_iterator; ledger_set_confirmed (nano::ledger const & ledger); + account_iterator account_begin (store::transaction const & transaction) const; + account_iterator account_end () const; + // Returns the next receivable entry equal or greater than 'key' + account_iterator account_lower_bound (store::transaction const & transaction, nano::account const & account) const; + // Returns the next receivable entry for an account greater than 'account' + account_iterator account_upper_bound (store::transaction const & transaction, nano::account const & account) const; std::optional balance (store::transaction const & transaction, nano::account const & account) const; std::optional balance (store::transaction const & transaction, nano::block_hash const & hash) const; bool exists (store::transaction const & transaction, nano::block_hash const & hash) const; diff --git a/nano/store/lmdb/block.cpp b/nano/store/lmdb/block.cpp index 9d3e308abb..269c376c8a 100644 --- a/nano/store/lmdb/block.cpp +++ b/nano/store/lmdb/block.cpp @@ -37,9 +37,16 @@ void nano::store::lmdb::block::put (store::write_transaction const & transaction block.sideband ().serialize (stream, block.type ()); } raw_put (transaction, vector, hash); - block_predecessor_mdb_set predecessor (transaction, *this); - block.visit (predecessor); - debug_assert (block.previous ().is_zero () || successor (transaction, block.previous ()) == hash); + if (exists (transaction, block.previous ())) + { + block_predecessor_mdb_set predecessor (transaction, *this); + block.visit (predecessor); + debug_assert (block.previous ().is_zero () || successor (transaction, block.previous ()) == hash); + } + else + { + // Pruned + } } void nano::store::lmdb::block::raw_put (store::write_transaction const & transaction_a, std::vector const & data, nano::block_hash const & hash_a) diff --git a/nano/test_common/system.cpp b/nano/test_common/system.cpp index b61e84017d..c3bf0e0baf 100644 --- a/nano/test_common/system.cpp +++ b/nano/test_common/system.cpp @@ -320,6 +320,10 @@ std::shared_ptr nano::test::upgrade_epoch (nano::work_pool & { error = ledger_a.process (transaction, epoch) != nano::block_status::progress; } + if (!error) + { + ledger_a.confirm (transaction, epoch->hash ()); + } return !error ? std::move (epoch) : nullptr; }