blockchain_utilities: add tool to track running counts of locked outputs

This commit is contained in:
jeffro256 2024-10-17 03:25:05 -05:00
parent 9866a0e902
commit b35d7c6cf1
No known key found for this signature in database
GPG Key ID: 6F79797A6E392442
8 changed files with 539 additions and 38 deletions

View File

@ -1743,7 +1743,7 @@ public:
* *
* @return false if the function returns false for any output, otherwise true * @return false if the function returns false for any output, otherwise true
*/ */
virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)> f) const = 0; virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx, uint64_t unlock_time)> f) const = 0;
virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const = 0; virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const = 0;
/** /**

View File

@ -3696,7 +3696,7 @@ bool BlockchainLMDB::for_all_transactions(std::function<bool(const crypto::hash&
return fret; return fret;
} }
bool BlockchainLMDB::for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)> f) const bool BlockchainLMDB::for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx, uint64_t unlock_time)> f) const
{ {
LOG_PRINT_L3("BlockchainLMDB::" << __func__); LOG_PRINT_L3("BlockchainLMDB::" << __func__);
check_open(); check_open();
@ -3720,7 +3720,7 @@ bool BlockchainLMDB::for_all_outputs(std::function<bool(uint64_t amount, const c
uint64_t amount = *(const uint64_t*)k.mv_data; uint64_t amount = *(const uint64_t*)k.mv_data;
outkey *ok = (outkey *)v.mv_data; outkey *ok = (outkey *)v.mv_data;
tx_out_index toi = get_output_tx_and_index_from_global(ok->output_id); tx_out_index toi = get_output_tx_and_index_from_global(ok->output_id);
if (!f(amount, toi.first, ok->data.height, toi.second)) { if (!f(amount, toi.first, ok->data.height, toi.second, ok->data.unlock_time)) {
fret = false; fret = false;
break; break;
} }

View File

@ -306,7 +306,7 @@ public:
virtual bool for_all_key_images(std::function<bool(const crypto::key_image&)>) const; virtual bool for_all_key_images(std::function<bool(const crypto::key_image&)>) const;
virtual bool for_blocks_range(const uint64_t& h1, const uint64_t& h2, std::function<bool(uint64_t, const crypto::hash&, const cryptonote::block&)>) const; virtual bool for_blocks_range(const uint64_t& h1, const uint64_t& h2, std::function<bool(uint64_t, const crypto::hash&, const cryptonote::block&)>) const;
virtual bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const; virtual bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const;
virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)> f) const; virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx, uint64_t unlock_time)> f) const;
virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const; virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const;
virtual bool for_all_alt_blocks(std::function<bool(const crypto::hash &blkid, const alt_block_data_t &data, const cryptonote::blobdata_ref *blob)> f, bool include_blob = false) const; virtual bool for_all_alt_blocks(std::function<bool(const crypto::hash &blkid, const alt_block_data_t &data, const cryptonote::blobdata_ref *blob)> f, bool include_blob = false) const;

View File

@ -120,7 +120,7 @@ public:
virtual bool for_all_key_images(std::function<bool(const crypto::key_image&)>) const override { return true; } virtual bool for_all_key_images(std::function<bool(const crypto::key_image&)>) const override { return true; }
virtual bool for_blocks_range(const uint64_t&, const uint64_t&, std::function<bool(uint64_t, const crypto::hash&, const cryptonote::block&)>) const override { return true; } virtual bool for_blocks_range(const uint64_t&, const uint64_t&, std::function<bool(uint64_t, const crypto::hash&, const cryptonote::block&)>) const override { return true; }
virtual bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const override { return true; } virtual bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const override { return true; }
virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)> f) const override { return true; } virtual bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx, uint64_t unlock_time)> f) const override { return true; }
virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const override { return true; } virtual bool for_all_outputs(uint64_t amount, const std::function<bool(uint64_t height)> &f) const override { return true; }
virtual bool is_read_only() const override { return false; } virtual bool is_read_only() const override { return false; }
virtual std::map<uint64_t, std::tuple<uint64_t, uint64_t, uint64_t>> get_output_histogram(const std::vector<uint64_t> &amounts, bool unlocked, uint64_t recent_cutoff, uint64_t min_count) const override { return std::map<uint64_t, std::tuple<uint64_t, uint64_t, uint64_t>>(); } virtual std::map<uint64_t, std::tuple<uint64_t, uint64_t, uint64_t>> get_output_histogram(const std::vector<uint64_t> &amounts, bool unlocked, uint64_t recent_cutoff, uint64_t min_count) const override { return std::map<uint64_t, std::tuple<uint64_t, uint64_t, uint64_t>>(); }

View File

@ -26,6 +26,8 @@
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
# THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
################################################################################
set(blockchain_import_sources set(blockchain_import_sources
blockchain_import.cpp blockchain_import.cpp
bootstrap_file.cpp bootstrap_file.cpp
@ -41,6 +43,8 @@ set(blockchain_import_private_headers
monero_private_headers(blockchain_import monero_private_headers(blockchain_import
${blockchain_import_private_headers}) ${blockchain_import_private_headers})
####################
set(blockchain_export_sources set(blockchain_export_sources
blockchain_export.cpp blockchain_export.cpp
bootstrap_file.cpp bootstrap_file.cpp
@ -56,6 +60,7 @@ set(blockchain_export_private_headers
monero_private_headers(blockchain_export monero_private_headers(blockchain_export
${blockchain_export_private_headers}) ${blockchain_export_private_headers})
####################
set(blockchain_blackball_sources set(blockchain_blackball_sources
blockchain_blackball.cpp blockchain_blackball.cpp
@ -70,6 +75,8 @@ set(blockchain_blackball_private_headers
monero_private_headers(blockchain_blackball monero_private_headers(blockchain_blackball
${blockchain_blackball_private_headers}) ${blockchain_blackball_private_headers})
####################
set(blockchain_usage_sources set(blockchain_usage_sources
blockchain_usage.cpp blockchain_usage.cpp
) )
@ -79,7 +86,7 @@ set(blockchain_usage_private_headers)
monero_private_headers(blockchain_usage monero_private_headers(blockchain_usage
${blockchain_usage_private_headers}) ${blockchain_usage_private_headers})
####################
set(blockchain_prune_known_spent_data_sources set(blockchain_prune_known_spent_data_sources
blockchain_prune_known_spent_data.cpp blockchain_prune_known_spent_data.cpp
@ -90,7 +97,7 @@ set(blockchain_prune_known_spent_data_private_headers)
monero_private_headers(blockchain_prune_known_spent_data monero_private_headers(blockchain_prune_known_spent_data
${blockchain_prune_known_spent_data_private_headers}) ${blockchain_prune_known_spent_data_private_headers})
####################
set(blockchain_prune_sources set(blockchain_prune_sources
blockchain_prune.cpp blockchain_prune.cpp
@ -101,7 +108,7 @@ set(blockchain_prune_private_headers)
monero_private_headers(blockchain_prune monero_private_headers(blockchain_prune
${blockchain_prune_private_headers}) ${blockchain_prune_private_headers})
####################
set(blockchain_ancestry_sources set(blockchain_ancestry_sources
blockchain_ancestry.cpp blockchain_ancestry.cpp
@ -112,7 +119,7 @@ set(blockchain_ancestry_private_headers)
monero_private_headers(blockchain_ancestry monero_private_headers(blockchain_ancestry
${blockchain_ancestry_private_headers}) ${blockchain_ancestry_private_headers})
####################
set(blockchain_depth_sources set(blockchain_depth_sources
blockchain_depth.cpp blockchain_depth.cpp
@ -123,6 +130,8 @@ set(blockchain_depth_private_headers)
monero_private_headers(blockchain_depth monero_private_headers(blockchain_depth
${blockchain_depth_private_headers}) ${blockchain_depth_private_headers})
####################
set(blockchain_stats_sources set(blockchain_stats_sources
blockchain_stats.cpp blockchain_stats.cpp
) )
@ -132,6 +141,18 @@ set(blockchain_stats_private_headers)
monero_private_headers(blockchain_stats monero_private_headers(blockchain_stats
${blockchain_stats_private_headers}) ${blockchain_stats_private_headers})
####################
set(blockchain_find_locked_sources
blockchain_find_locked.cpp
)
set(blockchain_find_locked_private_headers)
monero_private_headers(blockchain_find_locked
${blockchain_find_locked_private_headers})
################################################################################
monero_add_executable(blockchain_import monero_add_executable(blockchain_import
${blockchain_import_sources} ${blockchain_import_sources}
@ -160,6 +181,8 @@ set_property(TARGET blockchain_import
OUTPUT_NAME "monero-blockchain-import") OUTPUT_NAME "monero-blockchain-import")
install(TARGETS blockchain_import DESTINATION bin) install(TARGETS blockchain_import DESTINATION bin)
####################
monero_add_executable(blockchain_export monero_add_executable(blockchain_export
${blockchain_export_sources} ${blockchain_export_sources}
${blockchain_export_private_headers}) ${blockchain_export_private_headers})
@ -181,6 +204,8 @@ set_property(TARGET blockchain_export
OUTPUT_NAME "monero-blockchain-export") OUTPUT_NAME "monero-blockchain-export")
install(TARGETS blockchain_export DESTINATION bin) install(TARGETS blockchain_export DESTINATION bin)
####################
monero_add_executable(blockchain_blackball monero_add_executable(blockchain_blackball
${blockchain_blackball_sources} ${blockchain_blackball_sources}
${blockchain_blackball_private_headers}) ${blockchain_blackball_private_headers})
@ -203,6 +228,7 @@ set_property(TARGET blockchain_blackball
OUTPUT_NAME "monero-blockchain-mark-spent-outputs") OUTPUT_NAME "monero-blockchain-mark-spent-outputs")
install(TARGETS blockchain_blackball DESTINATION bin) install(TARGETS blockchain_blackball DESTINATION bin)
####################
monero_add_executable(blockchain_usage monero_add_executable(blockchain_usage
${blockchain_usage_sources} ${blockchain_usage_sources}
@ -225,6 +251,8 @@ set_property(TARGET blockchain_usage
OUTPUT_NAME "monero-blockchain-usage") OUTPUT_NAME "monero-blockchain-usage")
install(TARGETS blockchain_usage DESTINATION bin) install(TARGETS blockchain_usage DESTINATION bin)
####################
monero_add_executable(blockchain_ancestry monero_add_executable(blockchain_ancestry
${blockchain_ancestry_sources} ${blockchain_ancestry_sources}
${blockchain_ancestry_private_headers}) ${blockchain_ancestry_private_headers})
@ -246,6 +274,8 @@ set_property(TARGET blockchain_ancestry
OUTPUT_NAME "monero-blockchain-ancestry") OUTPUT_NAME "monero-blockchain-ancestry")
install(TARGETS blockchain_ancestry DESTINATION bin) install(TARGETS blockchain_ancestry DESTINATION bin)
####################
monero_add_executable(blockchain_depth monero_add_executable(blockchain_depth
${blockchain_depth_sources} ${blockchain_depth_sources}
${blockchain_depth_private_headers}) ${blockchain_depth_private_headers})
@ -267,6 +297,8 @@ set_property(TARGET blockchain_depth
OUTPUT_NAME "monero-blockchain-depth") OUTPUT_NAME "monero-blockchain-depth")
install(TARGETS blockchain_depth DESTINATION bin) install(TARGETS blockchain_depth DESTINATION bin)
####################
monero_add_executable(blockchain_stats monero_add_executable(blockchain_stats
${blockchain_stats_sources} ${blockchain_stats_sources}
${blockchain_stats_private_headers}) ${blockchain_stats_private_headers})
@ -288,6 +320,31 @@ set_property(TARGET blockchain_stats
OUTPUT_NAME "monero-blockchain-stats") OUTPUT_NAME "monero-blockchain-stats")
install(TARGETS blockchain_stats DESTINATION bin) install(TARGETS blockchain_stats DESTINATION bin)
####################
monero_add_executable(blockchain_find_locked
${blockchain_find_locked_sources}
${blockchain_find_locked_private_headers})
target_link_libraries(blockchain_find_locked
PRIVATE
cryptonote_core
blockchain_db
version
epee
${Boost_FILESYSTEM_LIBRARY}
${Boost_SYSTEM_LIBRARY}
${Boost_THREAD_LIBRARY}
${CMAKE_THREAD_LIBS_INIT}
${EXTRA_LIBRARIES})
set_property(TARGET blockchain_find_locked
PROPERTY
OUTPUT_NAME "monero-blockchain-find-locked")
install(TARGETS blockchain_find_locked DESTINATION bin)
####################
monero_add_executable(blockchain_prune_known_spent_data monero_add_executable(blockchain_prune_known_spent_data
${blockchain_prune_known_spent_data_sources} ${blockchain_prune_known_spent_data_sources}
${blockchain_prune_known_spent_data_private_headers}) ${blockchain_prune_known_spent_data_private_headers})
@ -310,6 +367,8 @@ set_property(TARGET blockchain_prune_known_spent_data
OUTPUT_NAME "monero-blockchain-prune-known-spent-data") OUTPUT_NAME "monero-blockchain-prune-known-spent-data")
install(TARGETS blockchain_prune_known_spent_data DESTINATION bin) install(TARGETS blockchain_prune_known_spent_data DESTINATION bin)
####################
monero_add_executable(blockchain_prune monero_add_executable(blockchain_prune
${blockchain_prune_sources} ${blockchain_prune_sources}
${blockchain_prune_private_headers}) ${blockchain_prune_private_headers})
@ -331,3 +390,5 @@ target_link_libraries(blockchain_prune
${Boost_THREAD_LIBRARY} ${Boost_THREAD_LIBRARY}
${CMAKE_THREAD_LIBS_INIT} ${CMAKE_THREAD_LIBS_INIT}
${EXTRA_LIBRARIES}) ${EXTRA_LIBRARIES})
################################################################################

View File

@ -0,0 +1,469 @@
// Copyright (c) 2024, The Monero Project
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without modification, are
// permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this list of
// conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
// of conditions and the following disclaimer in the documentation and/or other
// materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its contributors may be
// used to endorse or promote products derived from this software without specific
// prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include <filesystem>
#include "common/command_line.h"
#include "cryptonote_core/blockchain_and_pool.h"
#include "cryptonote_core/cryptonote_core.h"
#include "version.h"
#undef MONERO_DEFAULT_LOG_CATEGORY
#define MONERO_DEFAULT_LOG_CATEGORY "bcutil"
namespace po = boost::program_options;
using namespace epee;
using namespace cryptonote;
static std::atomic<bool> stop_requested = false;
enum class unlock_time_type_t
{
Standard, // coinbase or unlock_time == 0
NonStandardBlock, // non-coinbase and 0 < unlock_time < CRYPTONOTE_MAX_BLOCK_NUMBER
NonStandardUnix // non-coinbase and unlock_time >= CRYPTONOTE_MAX_BLOCK_NUMBER
};
struct report_slot_entry_t
{
uint32_t num_standard_locks_opened;
uint32_t num_nonstandard_blocktime_locks_opened;
uint32_t num_nonstandard_unixtime_locks_opened;
uint32_t num_total_locks_opened;
uint32_t num_standard_locks_closed;
uint32_t num_nonstandard_blocktime_locks_closed;
uint32_t num_nonstandard_unixtime_locks_closed;
uint32_t num_total_locks_closed;
uint32_t standard_locks_balance;
uint32_t nonstandard_blocktime_locks_balance;
uint32_t nonstandard_unixtime_locks_balance;
uint32_t total_locks_balance;
void tally_totals()
{
num_total_locks_opened = num_standard_locks_opened
+ num_nonstandard_blocktime_locks_opened
+ num_nonstandard_unixtime_locks_opened;
num_total_locks_closed = num_standard_locks_closed
+ num_nonstandard_blocktime_locks_closed
+ num_nonstandard_unixtime_locks_closed;
total_locks_balance = standard_locks_balance
+ nonstandard_blocktime_locks_balance
+ nonstandard_unixtime_locks_balance;
}
void update_balances(uint32_t &standard_locks_balance_inout,
uint32_t &nonstandard_blocktime_locks_balance_inout,
uint32_t &nonstandard_unixtime_locks_balance_inout,
uint32_t &total_locks_balance_inout)
{
this->standard_locks_balance = standard_locks_balance_inout =
(standard_locks_balance_inout + this->num_standard_locks_opened - this->num_standard_locks_closed);
this->nonstandard_blocktime_locks_balance = nonstandard_blocktime_locks_balance_inout =
(nonstandard_blocktime_locks_balance_inout + this->num_nonstandard_blocktime_locks_opened -
this->num_nonstandard_blocktime_locks_closed);
this->nonstandard_unixtime_locks_balance = nonstandard_unixtime_locks_balance_inout =
(nonstandard_unixtime_locks_balance_inout + this->num_nonstandard_unixtime_locks_opened -
this->num_nonstandard_unixtime_locks_closed);
this->total_locks_balance = total_locks_balance_inout =
(total_locks_balance_inout + this->num_total_locks_opened - this->num_total_locks_closed);
}
};
const std::string report_slot_entry_header = "num standard locks opened,num nonstandard blocktime locks opened,"
"num nonstandard unixtime locks opened,num total locks opened,num standard locks closed,"
"num nonstandard blocktime locks closed,num nonstandard unixtime locks closed,num total locks closed,"
"standard locks balance,nonstandard blocktime locks balance,nonstandard unixtime locks balance,total locks balance";
std::ostream& operator<<(std::ostream &os, const report_slot_entry_t &report_slot_entry)
{
os << report_slot_entry.num_standard_locks_opened;
os << ',';
os << report_slot_entry.num_nonstandard_blocktime_locks_opened;
os << ',';
os << report_slot_entry.num_nonstandard_unixtime_locks_opened;
os << ',';
os << report_slot_entry.num_total_locks_opened;
os << ',';
os << report_slot_entry.num_standard_locks_closed;
os << ',';
os << report_slot_entry.num_nonstandard_blocktime_locks_closed;
os << ',';
os << report_slot_entry.num_nonstandard_unixtime_locks_closed;
os << ',';
os << report_slot_entry.num_total_locks_closed;
os << ',';
os << report_slot_entry.standard_locks_balance;
os << ',';
os << report_slot_entry.nonstandard_blocktime_locks_balance;
os << ',';
os << report_slot_entry.nonstandard_unixtime_locks_balance;
os << ',';
os << report_slot_entry.total_locks_balance;
//os << ',';
return os;
}
static uint32_t get_first_spendable_block_index(
const uint32_t unlock_time,
const uint32_t block_included_in_chain)
{
uint32_t unlock_block_index = 0;
static_assert(CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE > 0, "unexpected default spendable age");
const uint32_t default_block_index = block_included_in_chain + CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE;
if (unlock_time < CRYPTONOTE_MAX_BLOCK_NUMBER)
{
unlock_block_index = unlock_time;
}
else // unix timelock
{
// Interpret the unlock_time as time
// TODO: hardcode correct times for each network and take in nettype
const uint32_t hf_v15_time = 1656629118;
const uint32_t hf_v15_height = 2689608;
// Use the last hard fork's time and block combo to convert the time-based timelock into an unlock block
// TODO: consider taking into account 60s block times when that was consensus
if (hf_v15_time > unlock_time)
{
const uint32_t seconds_since_unlock = hf_v15_time - unlock_time;
const uint32_t blocks_since_unlock = seconds_since_unlock / DIFFICULTY_TARGET_V2;
unlock_block_index = hf_v15_height > blocks_since_unlock
? (hf_v15_height - blocks_since_unlock)
: 0;
}
else
{
const uint32_t seconds_until_unlock = unlock_time - hf_v15_time;
const uint32_t blocks_until_unlock = seconds_until_unlock / DIFFICULTY_TARGET_V2;
unlock_block_index = hf_v15_height + blocks_until_unlock;
}
}
// Can't unlock earlier than the default unlock block
return std::max(unlock_block_index, default_block_index);
}
static uint32_t get_fcmp_tree_insertion_block_index(
const uint32_t unlock_time,
const uint32_t block_included_in_chain,
const bool is_coinbase,
const uint32_t ignore_after_block_index,
const uint32_t fcmp_activation_index)
{
const uint32_t effective_unlock_time =
(is_coinbase || block_included_in_chain <= ignore_after_block_index)
? unlock_time : 0;
const uint32_t first_spendable_block_index = std::min<uint32_t>(
get_first_spendable_block_index(effective_unlock_time, block_included_in_chain),
CRYPTONOTE_MAX_BLOCK_NUMBER);
if (first_spendable_block_index <= fcmp_activation_index)
return block_included_in_chain;
else
return first_spendable_block_index;
}
static unlock_time_type_t classify_unlock_time(const uint32_t unlock_time, const bool is_coinbase)
{
if (unlock_time == 0 || is_coinbase) return unlock_time_type_t::Standard;
else if (unlock_time < CRYPTONOTE_MAX_BLOCK_NUMBER) return unlock_time_type_t::NonStandardBlock;
else return unlock_time_type_t::NonStandardUnix;
}
static bool is_coinbase_tx(epee::span<const char> tx_prefix_blob)
{
uint32_t v;
int nread;
// consume version
nread = tools::read_varint(tx_prefix_blob.data(), tx_prefix_blob.cend(), v);
if (nread < 0)
return false;
tx_prefix_blob.remove_prefix(nread);
// consume unlock_time
nread = tools::read_varint(tx_prefix_blob.data(), tx_prefix_blob.cend(), v);
if (nread < 0)
return false;
tx_prefix_blob.remove_prefix(nread);
// consume size of vin and expect 1
nread = tools::read_varint(tx_prefix_blob.data(), tx_prefix_blob.cend(), v);
tx_prefix_blob.remove_prefix(nread);
if (v != 1 || nread < 0)
return false;
// consume first vin type and expect txin_gen
if (tx_prefix_blob.empty())
return false;
v = static_cast<unsigned char>(tx_prefix_blob[0]);
tx_prefix_blob.remove_prefix(1);
if (v != 0xff)
return false;
return true;
}
int main(int argc, char* argv[])
{
epee::string_tools::set_module_name_and_folder(argv[0]);
uint32_t log_level = 0;
tools::on_startup();
tools::signal_handler::install([](int type) {
stop_requested.store(true);
});
po::options_description desc_cmd_only("Command line options");
po::options_description desc_cmd_sett("Command line options and settings options");
const command_line::arg_descriptor<std::string> arg_log_level = {"log-level", "0-4 or categories", ""};
const command_line::arg_descriptor<uint32_t> arg_ignore_index = {"ignore-index", "ignore non-standard unlock times of outputs created after this block index", CRYPTONOTE_MAX_BLOCK_NUMBER};
const command_line::arg_descriptor<uint32_t> arg_fcmp_index = {"fcmp-index",
"if the calculated first spendable block index is less than or equal to this block index, simulate inserting immediately", 0};
const command_line::arg_descriptor<std::string> arg_output_file = {"output-file", "path to CSV output file", "blockchain-find-locked-output.csv"};
command_line::add_arg(desc_cmd_sett, cryptonote::arg_data_dir);
command_line::add_arg(desc_cmd_sett, cryptonote::arg_testnet_on);
command_line::add_arg(desc_cmd_sett, cryptonote::arg_stagenet_on);
command_line::add_arg(desc_cmd_sett, arg_log_level);
command_line::add_arg(desc_cmd_sett, arg_ignore_index);
command_line::add_arg(desc_cmd_sett, arg_fcmp_index);
command_line::add_arg(desc_cmd_sett, arg_output_file);
command_line::add_arg(desc_cmd_only, command_line::arg_help);
po::options_description desc_options("Allowed options");
desc_options.add(desc_cmd_only).add(desc_cmd_sett);
po::variables_map vm;
bool r = command_line::handle_error_helper(desc_options, [&]()
{
auto parser = po::command_line_parser(argc, argv).options(desc_options);
po::store(parser.run(), vm);
po::notify(vm);
return true;
});
if (! r)
return 1;
if (command_line::get_arg(vm, command_line::arg_help))
{
std::cout << "Monero '" << MONERO_RELEASE_NAME << "' (v" << MONERO_VERSION_FULL << ")" << ENDL << ENDL;
std::cout << desc_options << std::endl;
return 1;
}
mlog_configure(mlog_get_default_log_path("monero-blockchain-ancestry.log"), true);
if (!command_line::is_arg_defaulted(vm, arg_log_level))
mlog_set_log(command_line::get_arg(vm, arg_log_level).c_str());
else
mlog_set_log(std::string(std::to_string(log_level) + ",bcutil:INFO").c_str());
LOG_PRINT_L0("Starting...");
std::string opt_data_dir = command_line::get_arg(vm, cryptonote::arg_data_dir);
const bool opt_testnet = command_line::get_arg(vm, cryptonote::arg_testnet_on);
const bool opt_stagenet = command_line::get_arg(vm, cryptonote::arg_stagenet_on);
network_type net_type = opt_testnet ? TESTNET : opt_stagenet ? STAGENET : MAINNET;
const uint32_t opt_ignore_index = command_line::get_arg(vm, arg_ignore_index);
const uint32_t opt_fcmp_index = command_line::get_arg(vm, arg_fcmp_index);
const std::string opt_output_file = command_line::get_arg(vm, arg_output_file);
if (std::filesystem::exists(opt_output_file))
{
LOG_ERROR("Output file already exists");
throw std::runtime_error("Output file already exists");
}
// If we wanted to use the memory pool, we would set up a fake_core.
// Use Blockchain instead of lower-level BlockchainDB for ona main reason:
// Blockchain has the init() method for easy setup
//
// cannot match blockchain_storage setup above with just one line,
// e.g.
// Blockchain* core_storage = new Blockchain(NULL);
// because unlike blockchain_storage constructor, which takes a pointer to
// tx_memory_pool, Blockchain's constructor takes tx_memory_pool object.
LOG_PRINT_L0("Initializing source blockchain (BlockchainDB)");
std::unique_ptr<BlockchainAndPool> core_storage = std::make_unique<BlockchainAndPool>();
BlockchainDB *db = new_db();
if (db == NULL)
{
LOG_ERROR("Failed to initialize a database");
throw std::runtime_error("Failed to initialize a database");
}
LOG_PRINT_L0("database: LMDB");
const std::string filename = (std::filesystem::path(opt_data_dir) / db->get_db_name()).string();
LOG_PRINT_L0("Loading blockchain from folder " << filename << " ...");
try
{
db->open(filename, DBF_RDONLY);
}
catch (const std::exception& e)
{
LOG_PRINT_L0("Error opening database: " << e.what());
return 1;
}
r = core_storage->blockchain.init(db, net_type);
CHECK_AND_ASSERT_MES(r, 1, "Failed to initialize source blockchain storage");
LOG_PRINT_L0("Source blockchain storage initialized OK");
std::map<uint32_t, report_slot_entry_t> report_by_block;
LOG_PRINT_L0("Opening output file: " << opt_output_file);
std::ofstream report_ofs;
report_ofs.open(opt_output_file);
CHECK_AND_ASSERT_THROW_MES(!report_ofs.fail(),
"Could not open file '" << opt_output_file << "' for writing");
// get number of blocks and resize standard_cache_updates accordingly
const uint32_t db_height = db->height();
LOG_PRINT_L0("Blockchain height: " << db_height);
LOG_PRINT_L0("Starting main output iteration loop...");
uint32_t last_print_height = 0;
std::cout << "0 / " << db_height << "\r"; std::cout.flush();
// perform main data processing on all outputs
cryptonote::blobdata pruned_txblob;
db->for_all_outputs([&](const rct::xmr_amount amount,
const crypto::hash &tx_hash,
const uint32_t block_included_in_chain,
const size_t tx_idx,
const uint32_t unlock_time) -> bool
{
// determine if this output is a coinbase output
const bool is_potential_coinbase = unlock_time == block_included_in_chain + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW;
bool is_coinbase = false;
if (is_potential_coinbase)
{
CHECK_AND_ASSERT_THROW_MES(db->get_pruned_tx_blob(tx_hash, pruned_txblob),
"Pruned tx fetch fail");
is_coinbase = is_coinbase_tx(epee::to_span(pruned_txblob));
if (!is_coinbase)
MDEBUG("fake coinbase at txid " << tx_hash);
}
// calculate tree insertion index and unlock time type, sanity checking
const uint32_t tree_insertion_index = get_fcmp_tree_insertion_block_index(unlock_time,
block_included_in_chain,
is_coinbase,
opt_ignore_index,
opt_fcmp_index);
CHECK_AND_ASSERT_THROW_MES(tree_insertion_index >= block_included_in_chain,
"get_fcmp_tree_insertion_block_index() trying to insert before creation");
// get references to relevant entries
report_slot_entry_t &open_entry = report_by_block[block_included_in_chain];
report_slot_entry_t &close_entry = report_by_block[tree_insertion_index];
// depending on unlock_time_type, increment columns
const unlock_time_type_t unlock_time_type = classify_unlock_time(unlock_time, is_coinbase);
switch (unlock_time_type)
{
case unlock_time_type_t::Standard:
++open_entry.num_standard_locks_opened;
++close_entry.num_standard_locks_closed;
break;
case unlock_time_type_t::NonStandardBlock:
++open_entry.num_nonstandard_blocktime_locks_opened;
++close_entry.num_nonstandard_blocktime_locks_closed;
break;
case unlock_time_type_t::NonStandardUnix:
++open_entry.num_nonstandard_unixtime_locks_opened;
++close_entry.num_nonstandard_unixtime_locks_closed;
break;
}
// print progress
if (amount == 0 && block_included_in_chain % 1000 == 0 && block_included_in_chain > last_print_height)
{
std::cout << block_included_in_chain << " / " << db_height << "\r"; std::cout.flush();
last_print_height = block_included_in_chain;
}
// goto next output if interrupt signal not received
return !stop_requested.load();
}
);
LOG_PRINT_L0("Finished main output iteration loop. Cumulating results...");
uint32_t standard_locks_balance = 0;
uint32_t nonstandard_blocktime_locks_balance = 0;
uint32_t nonstandard_unixtime_locks_balance = 0;
uint32_t total_locks_balance = 0;
for (auto &p : report_by_block)
{
p.second.tally_totals();
p.second.update_balances(standard_locks_balance,
nonstandard_blocktime_locks_balance,
nonstandard_unixtime_locks_balance,
total_locks_balance);
}
LOG_PRINT_L0("Writing report to CSV file...");
report_ofs << "block_index," << report_slot_entry_header << "\n"; // header row
for (const auto &p : report_by_block)
{
report_ofs << p.first << ','; // block index col
report_ofs << p.second; // data columns
report_ofs << '\n'; // newline
}
report_ofs << '\n';
CHECK_AND_ASSERT_THROW_MES(!report_ofs.fail(), "writing CSV to output file failed");
LOG_PRINT_L0("Saved report.... Done!");
return 0;
}

View File

@ -5592,16 +5592,6 @@ bool Blockchain::for_all_transactions(std::function<bool(const crypto::hash&, co
return m_db->for_all_transactions(f, pruned); return m_db->for_all_transactions(f, pruned);
} }
bool Blockchain::for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)> f) const
{
return m_db->for_all_outputs(f);
}
bool Blockchain::for_all_outputs(uint64_t amount, std::function<bool(uint64_t height)> f) const
{
return m_db->for_all_outputs(amount, f);
}
void Blockchain::invalidate_block_template_cache() void Blockchain::invalidate_block_template_cache()
{ {
MDEBUG("Invalidating block template cache"); MDEBUG("Invalidating block template cache");

View File

@ -978,25 +978,6 @@ namespace cryptonote
*/ */
bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const; bool for_all_transactions(std::function<bool(const crypto::hash&, const cryptonote::transaction&)>, bool pruned) const;
/**
* @brief perform a check on all outputs in the blockchain
*
* @param std::function the check to perform, pass/fail
*
* @return false if any output fails the check, otherwise true
*/
bool for_all_outputs(std::function<bool(uint64_t amount, const crypto::hash &tx_hash, uint64_t height, size_t tx_idx)>) const;
/**
* @brief perform a check on all outputs of a given amount in the blockchain
*
* @param amount the amount to iterate through
* @param std::function the check to perform, pass/fail
*
* @return false if any output fails the check, otherwise true
*/
bool for_all_outputs(uint64_t amount, std::function<bool(uint64_t height)>) const;
/** /**
* @brief get a reference to the BlockchainDB in use by Blockchain * @brief get a reference to the BlockchainDB in use by Blockchain
* *