From baea842ff184f98d2f07568f0a77e48a34d3cde3 Mon Sep 17 00:00:00 2001 From: stickies-v Date: Mon, 7 Oct 2024 11:02:46 +0200 Subject: [PATCH 1/6] init: Remove unneeded argument for mempool_opts checks --- src/init.cpp | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/init.cpp b/src/init.cpp index 3b8474dd469..d947b1c26e1 100644 --- a/src/init.cpp +++ b/src/init.cpp @@ -1070,9 +1070,7 @@ bool AppInitParameterInteraction(const ArgsManager& args) if (!blockman_result) { return InitError(util::ErrorString(blockman_result)); } - CTxMemPool::Options mempool_opts{ - .check_ratio = chainparams.DefaultConsistencyChecks() ? 1 : 0, - }; + CTxMemPool::Options mempool_opts{}; auto mempool_result{ApplyArgsManOptions(args, chainparams, mempool_opts)}; if (!mempool_result) { return InitError(util::ErrorString(mempool_result)); From 720ce880a355cf59a4f042a504750eb4e3ee68d3 Mon Sep 17 00:00:00 2001 From: Martin Zumsande Date: Mon, 7 Oct 2024 11:05:22 +0200 Subject: [PATCH 2/6] init: Improve comment describing chainstate load retry behaviour --- src/init.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/init.cpp b/src/init.cpp index d947b1c26e1..2d3c118f1f3 100644 --- a/src/init.cpp +++ b/src/init.cpp @@ -1179,7 +1179,7 @@ bool CheckHostPortOptions(const ArgsManager& args) { 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. static ChainstateLoadResult InitAndLoadChainstate( NodeContext& node, From 635e9f85d76c28647120172d9524982ebe36cf3c Mon Sep 17 00:00:00 2001 From: TheCharlatan Date: Mon, 7 Oct 2024 11:11:33 +0200 Subject: [PATCH 3/6] init: Remove misleading log line when user chooses not to retry It is bad, because it is both printed for non-GUI users and does not convey additional information. Co-authored-by: Martin Zumsande --- src/init.cpp | 1 - 1 file changed, 1 deletion(-) diff --git a/src/init.cpp b/src/init.cpp index 2d3c118f1f3..c22370e0407 100644 --- a/src/init.cpp +++ b/src/init.cpp @@ -1643,7 +1643,6 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info) error.original + ".\nPlease restart with -reindex or -reindex-chainstate to recover.", "", CClientUIInterface::MSG_ERROR | CClientUIInterface::BTN_ABORT); if (!do_retry) { - LogError("Aborted block database rebuild. Exiting.\n"); return false; } do_reindex = true; From cd093049dda878e8424fdc1ef828b5f644bd91d4 Mon Sep 17 00:00:00 2001 From: MarcoFalke <*~=`'#}+{/-|&status@721217.xyz> Date: Mon, 7 Oct 2024 11:18:43 +0200 Subject: [PATCH 4/6] init: Remove incorrect comment about shutdown condition Shutdown is indeed called, and it being overkill does not make sense either. --- src/init.cpp | 1 - 1 file changed, 1 deletion(-) diff --git a/src/init.cpp b/src/init.cpp index c22370e0407..883f8fcb89a 100644 --- a/src/init.cpp +++ b/src/init.cpp @@ -1662,7 +1662,6 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info) // As LoadBlockIndex can take several minutes, it's possible the user // 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)) { LogPrintf("Shutdown requested. Exiting.\n"); return false; From 8f1246e833804789ee5d8b59026b49142df5c455 Mon Sep 17 00:00:00 2001 From: TheCharlatan Date: Mon, 7 Oct 2024 12:15:22 +0200 Subject: [PATCH 5/6] init: Improve chainstate init db error messages They should name the correct source of an error, or be generic if no clear source can be ascertained. --- src/init.cpp | 4 ++-- src/node/chainstate.cpp | 30 ++++++++++++++++++++---------- test/functional/feature_init.py | 4 ++-- 3 files changed, 24 insertions(+), 14 deletions(-) diff --git a/src/init.cpp b/src/init.cpp index 883f8fcb89a..7ca9c7b570e 100644 --- a/src/init.cpp +++ b/src/init.cpp @@ -1259,7 +1259,7 @@ static ChainstateLoadResult InitAndLoadChainstate( return f(); } catch (const std::exception& e) { 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); }); @@ -1639,7 +1639,7 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info) if (status == ChainstateLoadStatus::FAILURE && !do_reindex && !ShutdownRequested(node)) { // suggest a reindex 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.", "", CClientUIInterface::MSG_ERROR | CClientUIInterface::BTN_ABORT); if (!do_retry) { diff --git a/src/node/chainstate.cpp b/src/node/chainstate.cpp index d7e6176be1e..b3b215929a6 100644 --- a/src/node/chainstate.cpp +++ b/src/node/chainstate.cpp @@ -41,12 +41,17 @@ static ChainstateLoadResult CompleteChainstateInitialization( // new BlockTreeDB tries to delete the existing file, which // fails if it's still open from the previous loop. Close it first: pblocktree.reset(); - pblocktree = std::make_unique(DBParams{ - .path = chainman.m_options.datadir / "blocks" / "index", - .cache_bytes = static_cast(cache_sizes.block_tree_db), - .memory_only = options.block_tree_db_in_memory, - .wipe_data = options.wipe_block_tree_db, - .options = chainman.m_options.block_tree_db}); + try { + pblocktree = std::make_unique(DBParams{ + .path = chainman.m_options.datadir / "blocks" / "index", + .cache_bytes = static_cast(cache_sizes.block_tree_db), + .memory_only = options.block_tree_db_in_memory, + .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) { pblocktree->WriteReindexing(true); @@ -107,10 +112,15 @@ static ChainstateLoadResult CompleteChainstateInitialization( for (Chainstate* chainstate : chainman.GetAll()) { LogPrintf("Initializing chainstate %s\n", chainstate->ToString()); - chainstate->InitCoinsDB( - /*cache_size_bytes=*/chainman.m_total_coinsdb_cache * init_cache_fraction, - /*in_memory=*/options.coins_db_in_memory, - /*should_wipe=*/options.wipe_chainstate_db); + try { + chainstate->InitCoinsDB( + /*cache_size_bytes=*/chainman.m_total_coinsdb_cache * init_cache_fraction, + /*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) { chainstate->CoinsErrorCatcher().AddReadErrCallback(options.coins_error_cb); diff --git a/test/functional/feature_init.py b/test/functional/feature_init.py index 659d33684e5..a841052efb2 100755 --- a/test/functional/feature_init.py +++ b/test/functional/feature_init.py @@ -97,13 +97,13 @@ class InitStressTest(BitcoinTestFramework): files_to_delete = { '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.', } files_to_perturb = { '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.', } From 31cc5006c3de4dd6a1f7a238684163956604df45 Mon Sep 17 00:00:00 2001 From: Martin Zumsande Date: Tue, 8 Oct 2024 22:33:33 +0200 Subject: [PATCH 6/6] init: Return fatal failure on snapshot validation failure A general reindex won't typically help in this case, and there is already some action being taken with the call to `InvalidateCoinsDBOnDisk`. --- src/node/chainstate.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/node/chainstate.cpp b/src/node/chainstate.cpp index b3b215929a6..fa8d7a386f6 100644 --- a/src/node/chainstate.cpp +++ b/src/node/chainstate.cpp @@ -246,7 +246,7 @@ ChainstateLoadResult LoadChainstate(ChainstateManager& chainman, const CacheSize return {init_status, init_error}; } } else { - return {ChainstateLoadStatus::FAILURE, _( + return {ChainstateLoadStatus::FAILURE_FATAL, _( "UTXO snapshot failed to validate. " "Restart to resume normal initial block download, or try loading a different snapshot.")}; }