 ab093d58d6
			
		
	
	
	
	
	ab093d58d6* Move container info classes to separate file * Introduce better `container_info` class * Rename legacy to `container_info_entry` * Conversion * Test * Fixes
		
			
				
	
	
		
			307 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			307 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <nano/lib/tomlconfig.hpp>
 | |
| #include <nano/node/election.hpp>
 | |
| #include <nano/node/node.hpp>
 | |
| #include <nano/node/vote_cache.hpp>
 | |
| #include <nano/node/vote_router.hpp>
 | |
| 
 | |
| #include <ranges>
 | |
| 
 | |
| /*
 | |
|  * entvote_cache_entryry
 | |
|  */
 | |
| 
 | |
| nano::vote_cache_entry::vote_cache_entry (const nano::block_hash & hash) :
 | |
| 	hash_m{ hash }
 | |
| {
 | |
| }
 | |
| 
 | |
| bool nano::vote_cache_entry::vote (std::shared_ptr<nano::vote> const & vote, const nano::uint128_t & rep_weight, std::size_t max_voters)
 | |
| {
 | |
| 	bool updated = vote_impl (vote, rep_weight, max_voters);
 | |
| 	if (updated)
 | |
| 	{
 | |
| 		auto [tally, final_tally] = calculate_tally ();
 | |
| 		tally_m = tally;
 | |
| 		final_tally_m = final_tally;
 | |
| 		last_vote_m = std::chrono::steady_clock::now ();
 | |
| 	}
 | |
| 	return updated;
 | |
| }
 | |
| 
 | |
| bool nano::vote_cache_entry::vote_impl (std::shared_ptr<nano::vote> const & vote, const nano::uint128_t & rep_weight, std::size_t max_voters)
 | |
| {
 | |
| 	auto const representative = vote->account;
 | |
| 
 | |
| 	if (auto existing = voters.find (representative); existing != voters.end ())
 | |
| 	{
 | |
| 		// We already have a vote from this rep
 | |
| 		// Update timestamp if newer but tally remains unchanged as we already counted this rep weight
 | |
| 		// It is not essential to keep tally up to date if rep voting weight changes, elections do tally calculations independently, so in the worst case scenario only our queue ordering will be a bit off
 | |
| 		if (vote->timestamp () > existing->vote->timestamp ())
 | |
| 		{
 | |
| 			bool was_final = existing->vote->is_final ();
 | |
| 			voters.modify (existing, [&vote, &rep_weight] (auto & existing) {
 | |
| 				existing.vote = vote;
 | |
| 				existing.weight = rep_weight;
 | |
| 			});
 | |
| 			return !was_final && vote->is_final (); // Tally changed only if the vote became final
 | |
| 		}
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		auto should_add = [&, this] () {
 | |
| 			if (voters.size () < max_voters)
 | |
| 			{
 | |
| 				return true;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				release_assert (!voters.empty ());
 | |
| 				auto const & min_weight = voters.get<tag_weight> ().begin ()->weight;
 | |
| 				return rep_weight > min_weight;
 | |
| 			}
 | |
| 		};
 | |
| 
 | |
| 		// Vote from a new representative, add it to the list and update tally
 | |
| 		if (should_add ())
 | |
| 		{
 | |
| 			voters.insert ({ representative, rep_weight, vote });
 | |
| 
 | |
| 			// If we have reached the maximum number of voters, remove the lowest weight voter
 | |
| 			if (voters.size () >= max_voters)
 | |
| 			{
 | |
| 				release_assert (!voters.empty ());
 | |
| 				voters.get<tag_weight> ().erase (voters.get<tag_weight> ().begin ());
 | |
| 			}
 | |
| 
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 	return false; // Tally unchanged
 | |
| }
 | |
| 
 | |
| std::size_t nano::vote_cache_entry::size () const
 | |
| {
 | |
| 	return voters.size ();
 | |
| }
 | |
| 
 | |
| auto nano::vote_cache_entry::calculate_tally () const -> std::pair<nano::uint128_t, nano::uint128_t>
 | |
| {
 | |
| 	nano::uint128_t tally{ 0 }, final_tally{ 0 };
 | |
| 	for (auto const & voter : voters)
 | |
| 	{
 | |
| 		tally += voter.weight;
 | |
| 		final_tally += voter.vote->is_final () ? voter.weight : 0;
 | |
| 	}
 | |
| 	return { tally, final_tally };
 | |
| }
 | |
| 
 | |
| std::vector<std::shared_ptr<nano::vote>> nano::vote_cache_entry::votes () const
 | |
| {
 | |
| 	auto r = voters | std::views::transform ([] (auto const & item) { return item.vote; });
 | |
| 	return { r.begin (), r.end () };
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * vote_cache
 | |
|  */
 | |
| 
 | |
| nano::vote_cache::vote_cache (vote_cache_config const & config_a, nano::stats & stats_a) :
 | |
| 	config{ config_a },
 | |
| 	stats{ stats_a }
 | |
| {
 | |
| }
 | |
| 
 | |
| void nano::vote_cache::insert (std::shared_ptr<nano::vote> const & vote, std::unordered_map<nano::block_hash, nano::vote_code> const & results)
 | |
| {
 | |
| 	// Results map should be empty or have the same hashes as the vote
 | |
| 	debug_assert (results.empty () || std::all_of (vote->hashes.begin (), vote->hashes.end (), [&results] (auto const & hash) { return results.find (hash) != results.end (); }));
 | |
| 
 | |
| 	auto const representative = vote->account;
 | |
| 	auto const rep_weight = rep_weight_query (representative);
 | |
| 
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 
 | |
| 	// Cache votes with a corresponding active election (indicated by `vote_code::vote`) in case that election gets dropped
 | |
| 	auto filter = [] (auto code) {
 | |
| 		return code == nano::vote_code::vote || code == nano::vote_code::indeterminate;
 | |
| 	};
 | |
| 
 | |
| 	// If results map is empty, insert all hashes (meant for testing)
 | |
| 	if (results.empty ())
 | |
| 	{
 | |
| 		for (auto const & hash : vote->hashes)
 | |
| 		{
 | |
| 			insert_impl (vote, hash, rep_weight);
 | |
| 		}
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		for (auto const & [hash, code] : results)
 | |
| 		{
 | |
| 			if (filter (code))
 | |
| 			{
 | |
| 				insert_impl (vote, hash, rep_weight);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void nano::vote_cache::insert_impl (std::shared_ptr<nano::vote> const & vote, nano::block_hash const & hash, nano::uint128_t const & rep_weight)
 | |
| {
 | |
| 	debug_assert (!mutex.try_lock ());
 | |
| 	debug_assert (std::any_of (vote->hashes.begin (), vote->hashes.end (), [&hash] (auto const & vote_hash) { return vote_hash == hash; }));
 | |
| 
 | |
| 	if (auto existing = cache.find (hash); existing != cache.end ())
 | |
| 	{
 | |
| 		stats.inc (nano::stat::type::vote_cache, nano::stat::detail::update);
 | |
| 
 | |
| 		cache.modify (existing, [this, &vote, &rep_weight] (entry & ent) {
 | |
| 			ent.vote (vote, rep_weight, config.max_voters);
 | |
| 		});
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		stats.inc (nano::stat::type::vote_cache, nano::stat::detail::insert);
 | |
| 
 | |
| 		entry cache_entry{ hash };
 | |
| 		cache_entry.vote (vote, rep_weight, config.max_voters);
 | |
| 		cache.insert (cache_entry);
 | |
| 
 | |
| 		// Remove the oldest entry if we have reached the capacity limit
 | |
| 		if (cache.size () > config.max_size)
 | |
| 		{
 | |
| 			cache.get<tag_sequenced> ().pop_front ();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool nano::vote_cache::empty () const
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 	return cache.empty ();
 | |
| }
 | |
| 
 | |
| std::size_t nano::vote_cache::size () const
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 	return cache.size ();
 | |
| }
 | |
| 
 | |
| std::vector<std::shared_ptr<nano::vote>> nano::vote_cache::find (const nano::block_hash & hash) const
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 
 | |
| 	auto & cache_by_hash = cache.get<tag_hash> ();
 | |
| 	if (auto existing = cache_by_hash.find (hash); existing != cache_by_hash.end ())
 | |
| 	{
 | |
| 		return existing->votes ();
 | |
| 	}
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| bool nano::vote_cache::erase (const nano::block_hash & hash)
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 
 | |
| 	bool result = false;
 | |
| 	auto & cache_by_hash = cache.get<tag_hash> ();
 | |
| 	if (auto existing = cache_by_hash.find (hash); existing != cache_by_hash.end ())
 | |
| 	{
 | |
| 		cache_by_hash.erase (existing);
 | |
| 		result = true;
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void nano::vote_cache::clear ()
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 	cache.clear ();
 | |
| }
 | |
| 
 | |
| std::deque<nano::vote_cache::top_entry> nano::vote_cache::top (const nano::uint128_t & min_tally)
 | |
| {
 | |
| 	stats.inc (nano::stat::type::vote_cache, nano::stat::detail::top);
 | |
| 
 | |
| 	std::deque<top_entry> results;
 | |
| 	{
 | |
| 		nano::lock_guard<nano::mutex> lock{ mutex };
 | |
| 
 | |
| 		if (cleanup_interval.elapsed (config.age_cutoff / 2))
 | |
| 		{
 | |
| 			cleanup ();
 | |
| 		}
 | |
| 
 | |
| 		for (auto & entry : cache.get<tag_tally> ())
 | |
| 		{
 | |
| 			auto tally = entry.tally ();
 | |
| 			if (tally < min_tally)
 | |
| 			{
 | |
| 				break;
 | |
| 			}
 | |
| 			results.push_back ({ entry.hash (), tally, entry.final_tally () });
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Sort by final tally then by normal tally, descending
 | |
| 	std::sort (results.begin (), results.end (), [] (auto const & a, auto const & b) {
 | |
| 		if (a.final_tally == b.final_tally)
 | |
| 		{
 | |
| 			return a.tally > b.tally;
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			return a.final_tally > b.final_tally;
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	return results;
 | |
| }
 | |
| 
 | |
| void nano::vote_cache::cleanup ()
 | |
| {
 | |
| 	debug_assert (!mutex.try_lock ());
 | |
| 
 | |
| 	stats.inc (nano::stat::type::vote_cache, nano::stat::detail::cleanup);
 | |
| 
 | |
| 	auto const cutoff = std::chrono::steady_clock::now () - config.age_cutoff;
 | |
| 
 | |
| 	erase_if (cache, [cutoff] (auto const & entry) {
 | |
| 		return entry.last_vote () < cutoff;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| nano::container_info nano::vote_cache::container_info () const
 | |
| {
 | |
| 	nano::lock_guard<nano::mutex> guard{ mutex };
 | |
| 
 | |
| 	nano::container_info info;
 | |
| 	info.put ("cache", cache);
 | |
| 	return info;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * vote_cache_config
 | |
|  */
 | |
| 
 | |
| nano::error nano::vote_cache_config::serialize (nano::tomlconfig & toml) const
 | |
| {
 | |
| 	toml.put ("max_size", max_size, "Maximum number of blocks to cache votes for. \ntype:uint64");
 | |
| 	toml.put ("max_voters", max_voters, "Maximum number of voters to cache per block. \ntype:uint64");
 | |
| 	toml.put ("age_cutoff", age_cutoff.count (), "Maximum age of votes to keep in cache. \ntype:seconds");
 | |
| 
 | |
| 	return toml.get_error ();
 | |
| }
 | |
| 
 | |
| nano::error nano::vote_cache_config::deserialize (nano::tomlconfig & toml)
 | |
| {
 | |
| 	toml.get ("max_size", max_size);
 | |
| 	toml.get ("max_voters", max_voters);
 | |
| 
 | |
| 	auto age_cutoff_l = age_cutoff.count ();
 | |
| 	toml.get ("age_cutoff", age_cutoff_l);
 | |
| 	age_cutoff = std::chrono::seconds{ age_cutoff_l };
 | |
| 
 | |
| 	return toml.get_error ();
 | |
| }
 |