Merge bitcoin/bitcoin#31046: init: Some small chainstate load improvements
Some checks are pending
CI / test each commit (push) Waiting to run
CI / macOS 14 native, arm64, no depends, sqlite only, gui (push) Waiting to run
CI / Win64 native, VS 2022 (push) Waiting to run
CI / ASan + LSan + UBSan + integer, no depends, USDT (push) Waiting to run

31cc5006c3 init: Return fatal failure on snapshot validation failure (Martin Zumsande)
8f1246e833 init: Improve chainstate init db error messages (TheCharlatan)
cd093049dd init: Remove incorrect comment about shutdown condition (MarcoFalke)
635e9f85d7 init: Remove misleading log line when user chooses not to retry (TheCharlatan)
720ce880a3 init: Improve comment describing chainstate load retry behaviour (Martin Zumsande)
baea842ff1 init: Remove unneeded argument for mempool_opts checks (stickies-v)

Pull request description:

  These are mostly followups from #30968, making the code, log lines, error messages, and comments more consistent.

  The last commit is an attempt at improving the error reporting when loading the chainstate. It aims to more cleanly distinguish between errors arising from a specific database, and errors where the culprit may be less clear.

ACKs for top commit:
  achow101:
    ACK 31cc5006c3
  mzumsande:
    Code Review / lightly tested ACK 31cc5006c3
  BrandonOdiwuor:
    Code Review ACK 31cc5006c3.
  stickies-v:
    ACK 31cc5006c3

Tree-SHA512: 59fba4845ee45a3d91bf55807ae6b1c81458463b96bf664c8b1badfac503f6b01efd52a915fc399294e68a3f69985362a5a10a3844fa23f7707145ebe9ad349b
This commit is contained in:
Ava Chow 2024-10-23 18:33:31 -04:00
commit e9b95665ee
No known key found for this signature in database
GPG key ID: 17565732E08E5E41
3 changed files with 27 additions and 21 deletions

View file

@ -1062,9 +1062,7 @@ bool AppInitParameterInteraction(const ArgsManager& args)
if (!blockman_result) { if (!blockman_result) {
return InitError(util::ErrorString(blockman_result)); return InitError(util::ErrorString(blockman_result));
} }
CTxMemPool::Options mempool_opts{ CTxMemPool::Options mempool_opts{};
.check_ratio = chainparams.DefaultConsistencyChecks() ? 1 : 0,
};
auto mempool_result{ApplyArgsManOptions(args, chainparams, mempool_opts)}; auto mempool_result{ApplyArgsManOptions(args, chainparams, mempool_opts)};
if (!mempool_result) { if (!mempool_result) {
return InitError(util::ErrorString(mempool_result)); return InitError(util::ErrorString(mempool_result));
@ -1173,7 +1171,7 @@ bool CheckHostPortOptions(const ArgsManager& args) {
return true; return true;
} }
// A GUI user may opt to retry once if there is a failure during chainstate initialization. // A GUI user may opt to retry once with do_reindex set if there is a failure during chainstate initialization.
// The function therefore has to support re-entry. // The function therefore has to support re-entry.
static ChainstateLoadResult InitAndLoadChainstate( static ChainstateLoadResult InitAndLoadChainstate(
NodeContext& node, NodeContext& node,
@ -1253,7 +1251,7 @@ static ChainstateLoadResult InitAndLoadChainstate(
return f(); return f();
} catch (const std::exception& e) { } catch (const std::exception& e) {
LogError("%s\n", e.what()); LogError("%s\n", e.what());
return std::make_tuple(node::ChainstateLoadStatus::FAILURE, _("Error opening block database")); return std::make_tuple(node::ChainstateLoadStatus::FAILURE, _("Error loading databases"));
} }
}; };
auto [status, error] = catch_exceptions([&] { return LoadChainstate(chainman, cache_sizes, options); }); auto [status, error] = catch_exceptions([&] { return LoadChainstate(chainman, cache_sizes, options); });
@ -1634,11 +1632,10 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info)
if (status == ChainstateLoadStatus::FAILURE && !do_reindex && !ShutdownRequested(node)) { if (status == ChainstateLoadStatus::FAILURE && !do_reindex && !ShutdownRequested(node)) {
// suggest a reindex // suggest a reindex
bool do_retry = uiInterface.ThreadSafeQuestion( bool do_retry = uiInterface.ThreadSafeQuestion(
error + Untranslated(".\n\n") + _("Do you want to rebuild the block database now?"), error + Untranslated(".\n\n") + _("Do you want to rebuild the databases now?"),
error.original + ".\nPlease restart with -reindex or -reindex-chainstate to recover.", error.original + ".\nPlease restart with -reindex or -reindex-chainstate to recover.",
"", CClientUIInterface::MSG_ERROR | CClientUIInterface::BTN_ABORT); "", CClientUIInterface::MSG_ERROR | CClientUIInterface::BTN_ABORT);
if (!do_retry) { if (!do_retry) {
LogError("Aborted block database rebuild. Exiting.\n");
return false; return false;
} }
do_reindex = true; do_reindex = true;
@ -1658,7 +1655,6 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info)
// As LoadBlockIndex can take several minutes, it's possible the user // As LoadBlockIndex can take several minutes, it's possible the user
// requested to kill the GUI during the last operation. If so, exit. // requested to kill the GUI during the last operation. If so, exit.
// As the program has not fully started yet, Shutdown() is possibly overkill.
if (ShutdownRequested(node)) { if (ShutdownRequested(node)) {
LogPrintf("Shutdown requested. Exiting.\n"); LogPrintf("Shutdown requested. Exiting.\n");
return false; return false;

View file

@ -41,12 +41,17 @@ static ChainstateLoadResult CompleteChainstateInitialization(
// new BlockTreeDB tries to delete the existing file, which // new BlockTreeDB tries to delete the existing file, which
// fails if it's still open from the previous loop. Close it first: // fails if it's still open from the previous loop. Close it first:
pblocktree.reset(); pblocktree.reset();
pblocktree = std::make_unique<BlockTreeDB>(DBParams{ try {
.path = chainman.m_options.datadir / "blocks" / "index", pblocktree = std::make_unique<BlockTreeDB>(DBParams{
.cache_bytes = static_cast<size_t>(cache_sizes.block_tree_db), .path = chainman.m_options.datadir / "blocks" / "index",
.memory_only = options.block_tree_db_in_memory, .cache_bytes = static_cast<size_t>(cache_sizes.block_tree_db),
.wipe_data = options.wipe_block_tree_db, .memory_only = options.block_tree_db_in_memory,
.options = chainman.m_options.block_tree_db}); .wipe_data = options.wipe_block_tree_db,
.options = chainman.m_options.block_tree_db});
} catch (dbwrapper_error& err) {
LogError("%s\n", err.what());
return {ChainstateLoadStatus::FAILURE, _("Error opening block database")};
}
if (options.wipe_block_tree_db) { if (options.wipe_block_tree_db) {
pblocktree->WriteReindexing(true); pblocktree->WriteReindexing(true);
@ -107,10 +112,15 @@ static ChainstateLoadResult CompleteChainstateInitialization(
for (Chainstate* chainstate : chainman.GetAll()) { for (Chainstate* chainstate : chainman.GetAll()) {
LogPrintf("Initializing chainstate %s\n", chainstate->ToString()); LogPrintf("Initializing chainstate %s\n", chainstate->ToString());
chainstate->InitCoinsDB( try {
/*cache_size_bytes=*/chainman.m_total_coinsdb_cache * init_cache_fraction, chainstate->InitCoinsDB(
/*in_memory=*/options.coins_db_in_memory, /*cache_size_bytes=*/chainman.m_total_coinsdb_cache * init_cache_fraction,
/*should_wipe=*/options.wipe_chainstate_db); /*in_memory=*/options.coins_db_in_memory,
/*should_wipe=*/options.wipe_chainstate_db);
} catch (dbwrapper_error& err) {
LogError("%s\n", err.what());
return {ChainstateLoadStatus::FAILURE, _("Error opening coins database")};
}
if (options.coins_error_cb) { if (options.coins_error_cb) {
chainstate->CoinsErrorCatcher().AddReadErrCallback(options.coins_error_cb); chainstate->CoinsErrorCatcher().AddReadErrCallback(options.coins_error_cb);
@ -236,7 +246,7 @@ ChainstateLoadResult LoadChainstate(ChainstateManager& chainman, const CacheSize
return {init_status, init_error}; return {init_status, init_error};
} }
} else { } else {
return {ChainstateLoadStatus::FAILURE, _( return {ChainstateLoadStatus::FAILURE_FATAL, _(
"UTXO snapshot failed to validate. " "UTXO snapshot failed to validate. "
"Restart to resume normal initial block download, or try loading a different snapshot.")}; "Restart to resume normal initial block download, or try loading a different snapshot.")};
} }

View file

@ -100,13 +100,13 @@ class InitTest(BitcoinTestFramework):
files_to_delete = { files_to_delete = {
'blocks/index/*.ldb': 'Error opening block database.', 'blocks/index/*.ldb': 'Error opening block database.',
'chainstate/*.ldb': 'Error opening block database.', 'chainstate/*.ldb': 'Error opening coins database.',
'blocks/blk*.dat': 'Error loading block database.', 'blocks/blk*.dat': 'Error loading block database.',
} }
files_to_perturb = { files_to_perturb = {
'blocks/index/*.ldb': 'Error loading block database.', 'blocks/index/*.ldb': 'Error loading block database.',
'chainstate/*.ldb': 'Error opening block database.', 'chainstate/*.ldb': 'Error opening coins database.',
'blocks/blk*.dat': 'Corrupted block database detected.', 'blocks/blk*.dat': 'Corrupted block database detected.',
} }