From e7a17c660978c7d7dc62af521aaf11b64fc9642e Mon Sep 17 00:00:00 2001 From: clemahieu Date: Sun, 19 Jan 2020 23:01:35 -0600 Subject: [PATCH 1/3] Move vote replay code in to rep_crawler since this is a very infrequently needed codepath and it only really needs to be done once and can be done during rep discovery. --- nano/core_test/ledger.cpp | 19 +++++++++--------- nano/core_test/node.cpp | 39 ++++++++++++++---------------------- nano/node/repcrawler.cpp | 10 +++++++++ nano/node/vote_processor.cpp | 25 +++-------------------- nano/node/vote_processor.hpp | 2 +- 5 files changed, 38 insertions(+), 57 deletions(-) diff --git a/nano/core_test/ledger.cpp b/nano/core_test/ledger.cpp index b9ffb56497..8ec09d5b5b 100644 --- a/nano/core_test/ledger.cpp +++ b/nano/core_test/ledger.cpp @@ -742,11 +742,10 @@ TEST (votes, check_signature) } auto vote1 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 1, send1)); vote1->signature.bytes[0] ^= 1; - auto transaction (node1.store.tx_begin_read ()); - ASSERT_EQ (nano::vote_code::invalid, node1.vote_processor.vote_blocking (transaction, vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); + ASSERT_EQ (nano::vote_code::invalid, node1.vote_processor.vote_blocking (vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); vote1->signature.bytes[0] ^= 1; - ASSERT_EQ (nano::vote_code::vote, node1.vote_processor.vote_blocking (transaction, vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); - ASSERT_EQ (nano::vote_code::replay, node1.vote_processor.vote_blocking (transaction, vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); + ASSERT_EQ (nano::vote_code::vote, node1.vote_processor.vote_blocking (vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); + ASSERT_EQ (nano::vote_code::replay, node1.vote_processor.vote_blocking (vote1, std::make_shared (node1.network.udp_channels, nano::endpoint (boost::asio::ip::address_v6 (), 0), node1.network_params.protocol.protocol_version))); } TEST (votes, add_one) @@ -878,13 +877,13 @@ TEST (votes, add_old) votes1 = node1.active.roots.find (send1->qualified_root ())->election; } auto channel (std::make_shared (node1.network.udp_channels, node1.network.endpoint (), node1.network_params.protocol.protocol_version)); - node1.vote_processor.vote_blocking (transaction, vote1, channel); + node1.vote_processor.vote_blocking (vote1, channel); nano::keypair key2; auto send2 (std::make_shared (genesis.hash (), key2.pub, 0, nano::test_genesis_key.prv, nano::test_genesis_key.pub, 0)); node1.work_generate_blocking (*send2); auto vote2 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 1, send2)); votes1->last_votes[nano::test_genesis_key.pub].time = std::chrono::steady_clock::now () - std::chrono::seconds (20); - node1.vote_processor.vote_blocking (transaction, vote2, channel); + node1.vote_processor.vote_blocking (vote2, channel); ASSERT_EQ (2, votes1->last_votes_size ()); ASSERT_NE (votes1->last_votes.end (), votes1->last_votes.find (nano::test_genesis_key.pub)); ASSERT_EQ (send1->hash (), votes1->last_votes[nano::test_genesis_key.pub].hash); @@ -919,12 +918,12 @@ TEST (votes, add_old_different_account) ASSERT_EQ (1, votes2->last_votes_size ()); auto vote1 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 2, send1)); auto channel (std::make_shared (node1.network.udp_channels, node1.network.endpoint (), node1.network_params.protocol.protocol_version)); - auto vote_result1 (node1.vote_processor.vote_blocking (transaction, vote1, channel)); + auto vote_result1 (node1.vote_processor.vote_blocking (vote1, channel)); ASSERT_EQ (nano::vote_code::vote, vote_result1); ASSERT_EQ (2, votes1->last_votes.size ()); ASSERT_EQ (1, votes2->last_votes.size ()); auto vote2 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 1, send2)); - auto vote_result2 (node1.vote_processor.vote_blocking (transaction, vote2, channel)); + auto vote_result2 (node1.vote_processor.vote_blocking (vote2, channel)); ASSERT_EQ (nano::vote_code::vote, vote_result2); ASSERT_EQ (2, votes1->last_votes.size ()); ASSERT_EQ (2, votes2->last_votes.size ()); @@ -957,12 +956,12 @@ TEST (votes, add_cooldown) } auto vote1 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 1, send1)); auto channel (std::make_shared (node1.network.udp_channels, node1.network.endpoint (), node1.network_params.protocol.protocol_version)); - node1.vote_processor.vote_blocking (transaction, vote1, channel); + node1.vote_processor.vote_blocking (vote1, channel); nano::keypair key2; auto send2 (std::make_shared (genesis.hash (), key2.pub, 0, nano::test_genesis_key.prv, nano::test_genesis_key.pub, 0)); node1.work_generate_blocking (*send2); auto vote2 (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 2, send2)); - node1.vote_processor.vote_blocking (transaction, vote2, channel); + node1.vote_processor.vote_blocking (vote2, channel); ASSERT_EQ (2, votes1->last_votes.size ()); ASSERT_NE (votes1->last_votes.end (), votes1->last_votes.find (nano::test_genesis_key.pub)); ASSERT_EQ (send1->hash (), votes1->last_votes[nano::test_genesis_key.pub].hash); diff --git a/nano/core_test/node.cpp b/nano/core_test/node.cpp index f25a524c6f..43f3800c1c 100644 --- a/nano/core_test/node.cpp +++ b/nano/core_test/node.cpp @@ -2221,34 +2221,31 @@ TEST (node, send_callback) // This helps representatives continue from their last sequence number if their node is reinitialized and the old sequence number is lost TEST (node, vote_replay) { - nano::system system (2); + nano::system system (1); auto & node1 (*system.nodes[0]); - auto & node2 (*system.nodes[1]); nano::keypair key; - auto open (std::make_shared (0, 1, key.pub, key.prv, key.pub, 0)); - node1.work_generate_blocking (*open); + nano::genesis genesis; for (auto i (0); i < 11000; ++i) { - auto transaction (node2.store.tx_begin_read ()); - auto vote (node2.store.vote_generate (transaction, nano::test_genesis_key.pub, nano::test_genesis_key.prv, open)); + auto transaction (node1.store.tx_begin_read ()); + auto vote (node1.store.vote_generate (transaction, nano::test_genesis_key.pub, nano::test_genesis_key.prv, genesis.open)); } + auto node2 = system.add_node (); { - auto transaction (node1.store.tx_begin_read ()); - nano::lock_guard lock (node1.store.get_cache_mutex ()); - auto vote (node1.store.vote_current (transaction, nano::test_genesis_key.pub)); + auto transaction (node2->store.tx_begin_read ()); + nano::lock_guard lock (node2->store.get_cache_mutex ()); + auto vote (node2->store.vote_current (transaction, nano::test_genesis_key.pub)); ASSERT_EQ (nullptr, vote); } - system.wallet (0)->insert_adhoc (nano::test_genesis_key.prv); - auto block (system.wallet (0)->send_action (nano::test_genesis_key.pub, key.pub, nano::Gxrb_ratio)); - ASSERT_NE (nullptr, block); + system.wallet (1)->insert_adhoc (nano::test_genesis_key.prv); auto done (false); - system.deadline_set (20s); + system.deadline_set (2s); while (!done) { auto ec = system.poll (); - auto transaction (node1.store.tx_begin_read ()); - nano::lock_guard lock (node1.store.get_cache_mutex ()); - auto vote (node1.store.vote_current (transaction, nano::test_genesis_key.pub)); + auto transaction (node2->store.tx_begin_read ()); + nano::lock_guard lock (node2->store.get_cache_mutex ()); + auto vote (node2->store.vote_current (transaction, nano::test_genesis_key.pub)); done = vote && (vote->sequence >= 10000); ASSERT_NO_ERROR (ec); } @@ -2925,10 +2922,7 @@ TEST (node, fork_invalid_block_signature_vote_by_hash) std::vector vote_blocks; vote_blocks.push_back (send2->hash ()); auto vote (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 0, vote_blocks)); - { - auto transaction (node1.store.tx_begin_read ()); - node1.vote_processor.vote_blocking (transaction, vote, std::make_shared (node1.network.udp_channels, node1.network.endpoint (), node1.network_params.protocol.protocol_version)); - } + node1.vote_processor.vote_blocking (vote, std::make_shared (node1.network.udp_channels, node1.network.endpoint (), node1.network_params.protocol.protocol_version)); while (node1.block (send1->hash ())) { ASSERT_NO_ERROR (system.poll ()); @@ -3094,10 +3088,7 @@ TEST (node, confirm_back) std::vector vote_blocks; vote_blocks.push_back (send2->hash ()); auto vote (std::make_shared (nano::test_genesis_key.pub, nano::test_genesis_key.prv, 0, vote_blocks)); - { - auto transaction (node.store.tx_begin_read ()); - node.vote_processor.vote_blocking (transaction, vote, std::make_shared (node.network.udp_channels, node.network.endpoint (), node.network_params.protocol.protocol_version)); - } + node.vote_processor.vote_blocking (vote, std::make_shared (node.network.udp_channels, node.network.endpoint (), node.network_params.protocol.protocol_version)); system.deadline_set (10s); while (!node.active.empty ()) { diff --git a/nano/node/repcrawler.cpp b/nano/node/repcrawler.cpp index 076dc81556..eec2320429 100644 --- a/nano/node/repcrawler.cpp +++ b/nano/node/repcrawler.cpp @@ -38,6 +38,7 @@ void nano::rep_crawler::validate () nano::lock_guard lock (active_mutex); responses_l.swap (responses); } + auto transaction (node.store.tx_begin_read ()); auto minimum = node.minimum_principal_weight (); for (auto const & i : responses_l) { @@ -85,6 +86,15 @@ void nano::rep_crawler::validate () } } } + // This tries to assist rep nodes that have lost track of their highest sequence number by replaying our highest known vote back to them + // Only do this if the sequence number is significantly different to account for network reordering + // Amplify attack considerations: We're sending out a confirm_ack in response to a confirm_ack for no net traffic increase + auto max_vote (node.store.vote_max (transaction, vote)); + if (max_vote->sequence > vote->sequence + 10000) + { + nano::confirm_ack confirm (max_vote); + channel->send (confirm); // this is non essential traffic as it will be resolicited if not received + } } } diff --git a/nano/node/vote_processor.cpp b/nano/node/vote_processor.cpp index b0b6e8bbd9..3a356a9ae6 100644 --- a/nano/node/vote_processor.cpp +++ b/nano/node/vote_processor.cpp @@ -70,19 +70,9 @@ void nano::vote_processor::process_loop () is_active = true; lock.unlock (); verify_votes (votes_l); + for (auto & i : votes_l) { - auto transaction (store.tx_begin_read ()); - uint64_t count (1); - for (auto & i : votes_l) - { - vote_blocking (transaction, i.first, i.second, true); - // Free active_transactions mutex each 100 processed votes - if (count % 100 == 0) - { - transaction.refresh (); - } - count++; - } + vote_blocking (i.first, i.second, true); } lock.lock (); is_active = false; @@ -193,22 +183,13 @@ void nano::vote_processor::verify_votes (std::deque vote_a, std::shared_ptr channel_a, bool validated) +nano::vote_code nano::vote_processor::vote_blocking (std::shared_ptr vote_a, std::shared_ptr channel_a, bool validated) { auto result (nano::vote_code::invalid); if (validated || !vote_a->validate ()) { result = active.vote (vote_a); observers.vote.notify (vote_a, channel_a, result); - // This tries to assist rep nodes that have lost track of their highest sequence number by replaying our highest known vote back to them - // Only do this if the sequence number is significantly different to account for network reordering - // Amplify attack considerations: We're sending out a confirm_ack in response to a confirm_ack for no net traffic increase - auto max_vote (store.vote_max (transaction_a, vote_a)); - if (max_vote->sequence > vote_a->sequence + 10000) - { - nano::confirm_ack confirm (max_vote); - channel_a->send (confirm); // this is non essential traffic as it will be resolicited if not received - } } std::string status; switch (result) diff --git a/nano/node/vote_processor.hpp b/nano/node/vote_processor.hpp index cea715fe49..8c2f390dc8 100644 --- a/nano/node/vote_processor.hpp +++ b/nano/node/vote_processor.hpp @@ -35,7 +35,7 @@ class vote_processor final explicit vote_processor (nano::signature_checker & checker_a, nano::active_transactions & active_a, nano::block_store & store_a, nano::node_observers & observers_a, nano::stat & stats_a, nano::node_config & config_a, nano::logger_mt & logger_a, nano::online_reps & online_reps_a, nano::ledger & ledger_a, nano::network_params & network_params_a); void vote (std::shared_ptr, std::shared_ptr); /** Note: node.active.mutex lock is required */ - nano::vote_code vote_blocking (nano::transaction const &, std::shared_ptr, std::shared_ptr, bool = false); + nano::vote_code vote_blocking (std::shared_ptr, std::shared_ptr, bool = false); void verify_votes (std::deque, std::shared_ptr>> &); void flush (); void calculate_weights (); From 48aef317bc498dfb7b4a77e27ab74c43cb363e4b Mon Sep 17 00:00:00 2001 From: clemahieu Date: Mon, 20 Jan 2020 00:22:12 -0600 Subject: [PATCH 2/3] Directly calling vote_blocking instead of creating a new deque of valid votes. --- nano/node/node.cpp | 2 +- nano/node/vote_processor.cpp | 15 ++++----------- nano/node/vote_processor.hpp | 5 ++--- 3 files changed, 7 insertions(+), 15 deletions(-) diff --git a/nano/node/node.cpp b/nano/node/node.cpp index 202ba198ce..b6bcb308f7 100644 --- a/nano/node/node.cpp +++ b/nano/node/node.cpp @@ -129,7 +129,7 @@ bootstrap_initiator (*this), bootstrap (config.peering_port, *this), application_path (application_path_a), port_mapping (*this), -vote_processor (checker, active, store, observers, stats, config, logger, online_reps, ledger, network_params), +vote_processor (checker, active, observers, stats, config, logger, online_reps, ledger, network_params), rep_crawler (*this), warmed_up (0), block_processor (*this, write_database_queue), diff --git a/nano/node/vote_processor.cpp b/nano/node/vote_processor.cpp index 3a356a9ae6..b6fe77d193 100644 --- a/nano/node/vote_processor.cpp +++ b/nano/node/vote_processor.cpp @@ -15,10 +15,9 @@ #include -nano::vote_processor::vote_processor (nano::signature_checker & checker_a, nano::active_transactions & active_a, nano::block_store & store_a, nano::node_observers & observers_a, nano::stat & stats_a, nano::node_config & config_a, nano::logger_mt & logger_a, nano::online_reps & online_reps_a, nano::ledger & ledger_a, nano::network_params & network_params_a) : +nano::vote_processor::vote_processor (nano::signature_checker & checker_a, nano::active_transactions & active_a, nano::node_observers & observers_a, nano::stat & stats_a, nano::node_config & config_a, nano::logger_mt & logger_a, nano::online_reps & online_reps_a, nano::ledger & ledger_a, nano::network_params & network_params_a) : checker (checker_a), active (active_a), -store (store_a), observers (observers_a), stats (stats_a), config (config_a), @@ -54,7 +53,7 @@ void nano::vote_processor::process_loop () { if (!votes.empty ()) { - std::deque, std::shared_ptr>> votes_l; + decltype (votes) votes_l; votes_l.swap (votes); log_this_iteration = false; @@ -70,10 +69,6 @@ void nano::vote_processor::process_loop () is_active = true; lock.unlock (); verify_votes (votes_l); - for (auto & i : votes_l) - { - vote_blocking (i.first, i.second, true); - } lock.lock (); is_active = false; @@ -145,7 +140,7 @@ void nano::vote_processor::vote (std::shared_ptr vote_a, std::shared } } -void nano::vote_processor::verify_votes (std::deque, std::shared_ptr>> & votes_a) +void nano::vote_processor::verify_votes (decltype (votes) const & votes_a) { auto size (votes_a.size ()); std::vector messages; @@ -168,18 +163,16 @@ void nano::vote_processor::verify_votes (std::deque result; auto i (0); for (auto & vote : votes_a) { assert (verifications[i] == 1 || verifications[i] == 0); if (verifications[i] == 1) { - result.push_back (vote); + vote_blocking (vote.first, vote.second, true); } ++i; } - votes_a.swap (result); } // node.active.mutex lock required diff --git a/nano/node/vote_processor.hpp b/nano/node/vote_processor.hpp index 8c2f390dc8..0a8e3ef2cc 100644 --- a/nano/node/vote_processor.hpp +++ b/nano/node/vote_processor.hpp @@ -32,11 +32,11 @@ namespace transport class vote_processor final { public: - explicit vote_processor (nano::signature_checker & checker_a, nano::active_transactions & active_a, nano::block_store & store_a, nano::node_observers & observers_a, nano::stat & stats_a, nano::node_config & config_a, nano::logger_mt & logger_a, nano::online_reps & online_reps_a, nano::ledger & ledger_a, nano::network_params & network_params_a); + explicit vote_processor (nano::signature_checker & checker_a, nano::active_transactions & active_a, nano::node_observers & observers_a, nano::stat & stats_a, nano::node_config & config_a, nano::logger_mt & logger_a, nano::online_reps & online_reps_a, nano::ledger & ledger_a, nano::network_params & network_params_a); void vote (std::shared_ptr, std::shared_ptr); /** Note: node.active.mutex lock is required */ nano::vote_code vote_blocking (std::shared_ptr, std::shared_ptr, bool = false); - void verify_votes (std::deque, std::shared_ptr>> &); + void verify_votes (std::deque, std::shared_ptr>> const &); void flush (); void calculate_weights (); void stop (); @@ -46,7 +46,6 @@ class vote_processor final nano::signature_checker & checker; nano::active_transactions & active; - nano::block_store & store; nano::node_observers & observers; nano::stat & stats; nano::node_config & config; From db75908e91e56d8e2ad9b08a251207a684379f40 Mon Sep 17 00:00:00 2001 From: clemahieu Date: Thu, 23 Jan 2020 22:20:30 -0600 Subject: [PATCH 3/3] Fixing timeout for possibly slow test situations. --- nano/core_test/node.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/nano/core_test/node.cpp b/nano/core_test/node.cpp index 43f3800c1c..d68880847c 100644 --- a/nano/core_test/node.cpp +++ b/nano/core_test/node.cpp @@ -2239,7 +2239,7 @@ TEST (node, vote_replay) } system.wallet (1)->insert_adhoc (nano::test_genesis_key.prv); auto done (false); - system.deadline_set (2s); + system.deadline_set (20s); while (!done) { auto ec = system.poll ();