Bitcoin Core  29.1.0
P2P Digital Currency
load.cpp
Go to the documentation of this file.
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2022 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 
6 #include <wallet/load.h>
7 
8 #include <common/args.h>
9 #include <interfaces/chain.h>
10 #include <scheduler.h>
11 #include <util/check.h>
12 #include <util/fs.h>
13 #include <util/string.h>
14 #include <util/translation.h>
15 #include <wallet/context.h>
16 #include <wallet/spend.h>
17 #include <wallet/wallet.h>
18 #include <wallet/walletdb.h>
19 
20 #include <univalue.h>
21 
22 #include <system_error>
23 
24 using util::Join;
25 
26 namespace wallet {
28 {
29  interfaces::Chain& chain = *context.chain;
30  ArgsManager& args = *Assert(context.args);
31 
32  if (args.IsArgSet("-walletdir")) {
33  const fs::path wallet_dir{args.GetPathArg("-walletdir")};
34  std::error_code error;
35  // The canonical path cleans the path, preventing >1 Berkeley environment instances for the same directory
36  // It also lets the fs::exists and fs::is_directory checks below pass on windows, since they return false
37  // if a path has trailing slashes, and it strips trailing slashes.
38  fs::path canonical_wallet_dir = fs::canonical(wallet_dir, error);
39  if (error || !fs::exists(canonical_wallet_dir)) {
40  chain.initError(strprintf(_("Specified -walletdir \"%s\" does not exist"), fs::PathToString(wallet_dir)));
41  return false;
42  } else if (!fs::is_directory(canonical_wallet_dir)) {
43  chain.initError(strprintf(_("Specified -walletdir \"%s\" is not a directory"), fs::PathToString(wallet_dir)));
44  return false;
45  // The canonical path transforms relative paths into absolute ones, so we check the non-canonical version
46  } else if (!wallet_dir.is_absolute()) {
47  chain.initError(strprintf(_("Specified -walletdir \"%s\" is a relative path"), fs::PathToString(wallet_dir)));
48  return false;
49  }
50  args.ForceSetArg("-walletdir", fs::PathToString(canonical_wallet_dir));
51  }
52 
53  LogPrintf("Using wallet directory %s\n", fs::PathToString(GetWalletDir()));
54 
55  chain.initMessage(_("Verifying wallet(s)…"));
56 
57  // For backwards compatibility if an unnamed top level wallet exists in the
58  // wallets directory, include it in the default list of wallets to load.
59  if (!args.IsArgSet("wallet")) {
60  DatabaseOptions options;
61  DatabaseStatus status;
62  ReadDatabaseArgs(args, options);
63  bilingual_str error_string;
64  options.require_existing = true;
65  options.verify = false;
66  if (MakeWalletDatabase("", options, status, error_string)) {
68  wallets.push_back(""); // Default wallet name is ""
69  // Pass write=false because no need to write file and probably
70  // better not to. If unnamed wallet needs to be added next startup
71  // and the setting is empty, this code will just run again.
72  chain.overwriteRwSetting("wallet", std::move(wallets), interfaces::SettingsAction::SKIP_WRITE);
73  }
74  }
75 
76  // Keep track of each wallet absolute path to detect duplicates.
77  std::set<fs::path> wallet_paths;
78 
79  for (const auto& wallet : chain.getSettingsList("wallet")) {
80  if (!wallet.isStr()) {
81  chain.initError(_("Invalid value detected for '-wallet' or '-nowallet'. "
82  "'-wallet' requires a string value, while '-nowallet' accepts only '1' to disable all wallets"));
83  return false;
84  }
85  const auto& wallet_file = wallet.get_str();
86  const fs::path path = fsbridge::AbsPathJoin(GetWalletDir(), fs::PathFromString(wallet_file));
87 
88  if (!wallet_paths.insert(path).second) {
89  chain.initWarning(strprintf(_("Ignoring duplicate -wallet %s."), wallet_file));
90  continue;
91  }
92 
93  DatabaseOptions options;
94  DatabaseStatus status;
95  ReadDatabaseArgs(args, options);
96  options.require_existing = true;
97  options.verify = true;
98  bilingual_str error_string;
99  if (!MakeWalletDatabase(wallet_file, options, status, error_string)) {
100  if (status == DatabaseStatus::FAILED_NOT_FOUND) {
101  chain.initWarning(Untranslated(strprintf("Skipping -wallet path that doesn't exist. %s", error_string.original)));
102  } else {
103  chain.initError(error_string);
104  return false;
105  }
106  }
107  }
108 
109  return true;
110 }
111 
113 {
114  interfaces::Chain& chain = *context.chain;
115  try {
116  std::set<fs::path> wallet_paths;
117  for (const auto& wallet : chain.getSettingsList("wallet")) {
118  if (!wallet.isStr()) {
119  chain.initError(_("Invalid value detected for '-wallet' or '-nowallet'. "
120  "'-wallet' requires a string value, while '-nowallet' accepts only '1' to disable all wallets"));
121  return false;
122  }
123  const auto& name = wallet.get_str();
124  if (!wallet_paths.insert(fs::PathFromString(name)).second) {
125  continue;
126  }
127  DatabaseOptions options;
128  DatabaseStatus status;
129  ReadDatabaseArgs(*context.args, options);
130  options.require_existing = true;
131  options.verify = false; // No need to verify, assuming verified earlier in VerifyWallets()
132  bilingual_str error;
133  std::vector<bilingual_str> warnings;
134  std::unique_ptr<WalletDatabase> database = MakeWalletDatabase(name, options, status, error);
135  if (!database && status == DatabaseStatus::FAILED_NOT_FOUND) {
136  continue;
137  }
138  chain.initMessage(_("Loading wallet…"));
139  std::shared_ptr<CWallet> pwallet = database ? CWallet::Create(context, name, std::move(database), options.create_flags, error, warnings) : nullptr;
140  if (!warnings.empty()) chain.initWarning(Join(warnings, Untranslated("\n")));
141  if (!pwallet) {
142  chain.initError(error);
143  return false;
144  }
145 
146  NotifyWalletLoaded(context, pwallet);
147  AddWallet(context, pwallet);
148  }
149  return true;
150  } catch (const std::runtime_error& e) {
151  chain.initError(Untranslated(e.what()));
152  return false;
153  }
154 }
155 
157 {
158  for (const std::shared_ptr<CWallet>& pwallet : GetWallets(context)) {
159  pwallet->postInitProcess();
160  }
161 
162  // Schedule periodic wallet flushes and tx rebroadcasts
163  if (context.args->GetBoolArg("-flushwallet", DEFAULT_FLUSHWALLET)) {
164  context.scheduler->scheduleEvery([&context] { MaybeCompactWalletDB(context); }, 500ms);
165  }
166  context.scheduler->scheduleEvery([&context] { MaybeResendWalletTxs(context); }, 1min);
167 }
168 
170 {
171  for (const std::shared_ptr<CWallet>& pwallet : GetWallets(context)) {
172  pwallet->Flush();
173  }
174 }
175 
177 {
178  for (const std::shared_ptr<CWallet>& pwallet : GetWallets(context)) {
179  pwallet->Close();
180  }
181 }
182 
184 {
185  auto wallets = GetWallets(context);
186  while (!wallets.empty()) {
187  auto wallet = wallets.back();
188  wallets.pop_back();
189  std::vector<bilingual_str> warnings;
190  RemoveWallet(context, wallet, /* load_on_start= */ std::nullopt, warnings);
191  WaitForDeleteWallet(std::move(wallet));
192  }
193 }
194 } // namespace wallet
static path PathFromString(const std::string &string)
Convert byte string to path object.
Definition: fs.h:174
fs::path GetPathArg(std::string arg, const fs::path &default_value={}) const
Return path argument or default value.
Definition: args.cpp:272
void push_back(UniValue val)
Definition: univalue.cpp:104
bool IsArgSet(const std::string &strArg) const
Return true if the given argument has been manually set.
Definition: args.cpp:371
void ReadDatabaseArgs(const ArgsManager &args, DatabaseOptions &options)
Definition: db.cpp:154
fs::path AbsPathJoin(const fs::path &base, const fs::path &path)
Helper function for joining two paths.
Definition: fs.cpp:36
void FlushWallets(WalletContext &context)
Flush all wallets in preparation for shutdown.
Definition: load.cpp:169
Bilingual messages:
Definition: translation.h:24
static const bool DEFAULT_FLUSHWALLET
Overview of wallet database classes:
Definition: walletdb.h:42
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
virtual std::vector< common::SettingsValue > getSettingsList(const std::string &arg)=0
Get list of settings values.
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
Definition: translation.h:82
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
Definition: args.cpp:507
void ForceSetArg(const std::string &strArg, const std::string &strValue)
Definition: args.cpp:546
fs::path GetWalletDir()
Get the path of the wallet directory.
Definition: walletutil.cpp:13
consteval auto _(util::TranslatedLiteral str)
Definition: translation.h:79
CScheduler * scheduler
Definition: context.h:38
bool LoadWallets(WalletContext &context)
Load wallet databases.
Definition: load.cpp:112
void MaybeCompactWalletDB(WalletContext &context)
Compacts BDB state so that wallet.dat is self-contained (if there are changes)
Definition: walletdb.cpp:1305
void MaybeResendWalletTxs(WalletContext &context)
Called periodically by the schedule thread.
Definition: wallet.cpp:2165
void StopWallets(WalletContext &context)
Stop all wallets. Wallets will be flushed first.
Definition: load.cpp:176
void StartWallets(WalletContext &context)
Complete startup of wallets.
Definition: load.cpp:156
bool VerifyWallets(WalletContext &context)
Responsible for reading and validating the -wallet arguments and verifying the wallet database...
Definition: load.cpp:27
void WaitForDeleteWallet(std::shared_ptr< CWallet > &&wallet)
Explicitly delete the wallet.
Definition: wallet.cpp:252
static std::string PathToString(const path &path)
Convert path object to a byte string.
Definition: fs.h:151
ArgsManager & args
Definition: bitcoind.cpp:277
uint64_t create_flags
Definition: db.h:195
const char * name
Definition: rest.cpp:49
virtual bool overwriteRwSetting(const std::string &name, common::SettingsValue value, SettingsAction action=SettingsAction::WRITE)=0
Replace a setting in <datadir>/settings.json with a new value.
void NotifyWalletLoaded(WalletContext &context, const std::shared_ptr< CWallet > &wallet)
Definition: wallet.cpp:220
virtual void initWarning(const bilingual_str &message)=0
Send init warning.
bool RemoveWallet(WalletContext &context, const std::shared_ptr< CWallet > &wallet, std::optional< bool > load_on_start, std::vector< bilingual_str > &warnings)
Definition: wallet.cpp:161
DatabaseStatus
Definition: db.h:205
std::vector< std::shared_ptr< CWallet > > GetWallets(WalletContext &context)
Definition: wallet.cpp:191
std::string original
Definition: translation.h:25
Interface giving clients (wallet processes, maybe other analysis tools in the future) ability to acce...
Definition: chain.h:128
virtual void initMessage(const std::string &message)=0
Send init message.
WalletContext struct containing references to state shared between CWallet instances, like the reference to the chain interface, and the list of opened wallets.
Definition: context.h:36
bool require_existing
Definition: db.h:192
interfaces::Chain * chain
Definition: context.h:37
void UnloadWallets(WalletContext &context)
Close all wallets.
Definition: load.cpp:183
static bool exists(const path &p)
Definition: fs.h:89
#define LogPrintf(...)
Definition: logging.h:361
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
Definition: fs.h:32
bool verify
Check data integrity on load.
Definition: db.h:199
ArgsManager * args
Definition: context.h:39
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
Definition: string.h:192
virtual void initError(const bilingual_str &message)=0
Send init error.
bool AddWallet(WalletContext &context, const std::shared_ptr< CWallet > &wallet)
Definition: wallet.cpp:149
void scheduleEvery(Function f, std::chrono::milliseconds delta) EXCLUSIVE_LOCKS_REQUIRED(!newTaskMutex)
Repeat f until the scheduler is stopped.
Definition: scheduler.cpp:108
std::unique_ptr< WalletDatabase > MakeWalletDatabase(const std::string &name, const DatabaseOptions &options, DatabaseStatus &status, bilingual_str &error_string)
Definition: wallet.cpp:2992
#define Assert(val)
Identity function.
Definition: check.h:85
static std::shared_ptr< CWallet > Create(WalletContext &context, const std::string &name, std::unique_ptr< WalletDatabase > database, uint64_t wallet_creation_flags, bilingual_str &error, std::vector< bilingual_str > &warnings)
Definition: wallet.cpp:3003