2019-04-02 17:11:26 -04:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
2020-04-16 13:14:08 -04:00
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
2019-04-02 17:11:26 -04:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include <wallet/load.h>
|
|
|
|
|
|
|
|
#include <interfaces/chain.h>
|
|
|
|
#include <scheduler.h>
|
2019-10-06 17:52:05 -04:00
|
|
|
#include <util/string.h>
|
2019-04-02 17:11:26 -04:00
|
|
|
#include <util/system.h>
|
2019-06-17 10:56:52 +03:00
|
|
|
#include <util/translation.h>
|
2019-04-02 17:11:26 -04:00
|
|
|
#include <wallet/wallet.h>
|
|
|
|
|
|
|
|
bool VerifyWallets(interfaces::Chain& chain, const std::vector<std::string>& wallet_files)
|
|
|
|
{
|
|
|
|
if (gArgs.IsArgSet("-walletdir")) {
|
|
|
|
fs::path wallet_dir = gArgs.GetArg("-walletdir", "");
|
|
|
|
boost::system::error_code error;
|
|
|
|
// The canonical path cleans the path, preventing >1 Berkeley environment instances for the same directory
|
|
|
|
fs::path canonical_wallet_dir = fs::canonical(wallet_dir, error);
|
|
|
|
if (error || !fs::exists(wallet_dir)) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(strprintf(_("Specified -walletdir \"%s\" does not exist"), wallet_dir.string()));
|
2019-04-02 17:11:26 -04:00
|
|
|
return false;
|
|
|
|
} else if (!fs::is_directory(wallet_dir)) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(strprintf(_("Specified -walletdir \"%s\" is not a directory"), wallet_dir.string()));
|
2019-04-02 17:11:26 -04:00
|
|
|
return false;
|
|
|
|
// The canonical path transforms relative paths into absolute ones, so we check the non-canonical version
|
|
|
|
} else if (!wallet_dir.is_absolute()) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(strprintf(_("Specified -walletdir \"%s\" is a relative path"), wallet_dir.string()));
|
2019-04-02 17:11:26 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
gArgs.ForceSetArg("-walletdir", canonical_wallet_dir.string());
|
|
|
|
}
|
|
|
|
|
|
|
|
LogPrintf("Using wallet directory %s\n", GetWalletDir().string());
|
|
|
|
|
2019-06-28 21:09:58 +03:00
|
|
|
chain.initMessage(_("Verifying wallet(s)...").translated);
|
2019-04-02 17:11:26 -04:00
|
|
|
|
|
|
|
// Keep track of each wallet absolute path to detect duplicates.
|
|
|
|
std::set<fs::path> wallet_paths;
|
|
|
|
|
|
|
|
for (const auto& wallet_file : wallet_files) {
|
|
|
|
WalletLocation location(wallet_file);
|
|
|
|
|
|
|
|
if (!wallet_paths.insert(location.GetPath()).second) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(strprintf(_("Error loading wallet %s. Duplicate -wallet filename specified."), wallet_file));
|
2019-04-02 17:11:26 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-08-19 18:12:35 -04:00
|
|
|
bilingual_str error_string;
|
|
|
|
std::vector<bilingual_str> warnings;
|
2020-05-08 17:10:59 -04:00
|
|
|
bool verify_success = CWallet::Verify(chain, location, error_string, warnings);
|
2020-05-09 14:46:01 +03:00
|
|
|
if (!warnings.empty()) chain.initWarning(Join(warnings, Untranslated("\n")));
|
2019-08-19 18:12:35 -04:00
|
|
|
if (!verify_success) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(error_string);
|
2019-08-19 18:12:35 -04:00
|
|
|
return false;
|
|
|
|
}
|
2019-04-02 17:11:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LoadWallets(interfaces::Chain& chain, const std::vector<std::string>& wallet_files)
|
|
|
|
{
|
2020-03-28 02:14:08 +00:00
|
|
|
try {
|
|
|
|
for (const std::string& walletFile : wallet_files) {
|
2019-08-19 18:12:35 -04:00
|
|
|
bilingual_str error;
|
|
|
|
std::vector<bilingual_str> warnings;
|
2020-03-28 02:14:08 +00:00
|
|
|
std::shared_ptr<CWallet> pwallet = CWallet::CreateWalletFromFile(chain, WalletLocation(walletFile), error, warnings);
|
2020-05-09 14:46:01 +03:00
|
|
|
if (!warnings.empty()) chain.initWarning(Join(warnings, Untranslated("\n")));
|
2020-03-28 02:14:08 +00:00
|
|
|
if (!pwallet) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(error);
|
2020-03-28 02:14:08 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
AddWallet(pwallet);
|
2019-04-02 17:11:26 -04:00
|
|
|
}
|
2020-03-28 02:14:08 +00:00
|
|
|
return true;
|
|
|
|
} catch (const std::runtime_error& e) {
|
2020-04-11 18:48:04 +03:00
|
|
|
chain.initError(Untranslated(e.what()));
|
2020-03-28 02:14:08 +00:00
|
|
|
return false;
|
2019-04-02 17:11:26 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void StartWallets(CScheduler& scheduler)
|
|
|
|
{
|
|
|
|
for (const std::shared_ptr<CWallet>& pwallet : GetWallets()) {
|
|
|
|
pwallet->postInitProcess();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Schedule periodic wallet flushes and tx rebroadcasts
|
2020-03-06 18:06:50 -05:00
|
|
|
scheduler.scheduleEvery(MaybeCompactWalletDB, std::chrono::milliseconds{500});
|
|
|
|
scheduler.scheduleEvery(MaybeResendWalletTxs, std::chrono::milliseconds{1000});
|
2019-04-02 17:11:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void FlushWallets()
|
|
|
|
{
|
|
|
|
for (const std::shared_ptr<CWallet>& pwallet : GetWallets()) {
|
|
|
|
pwallet->Flush(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void StopWallets()
|
|
|
|
{
|
|
|
|
for (const std::shared_ptr<CWallet>& pwallet : GetWallets()) {
|
|
|
|
pwallet->Flush(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UnloadWallets()
|
|
|
|
{
|
|
|
|
auto wallets = GetWallets();
|
|
|
|
while (!wallets.empty()) {
|
|
|
|
auto wallet = wallets.back();
|
|
|
|
wallets.pop_back();
|
|
|
|
RemoveWallet(wallet);
|
|
|
|
UnloadWallet(std::move(wallet));
|
|
|
|
}
|
|
|
|
}
|