Bitcoin Core 31.99.0
P2P Digital Currency
wallet_migration.cpp
Go to the documentation of this file.
1// Copyright (c) 2024-present The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or https://www.opensource.org/licenses/mit-license.php.
4
5#include <addresstype.h>
6#include <bench/bench.h>
7#include <consensus/amount.h>
8#include <interfaces/wallet.h>
9#include <key.h>
10#include <primitives/block.h>
12#include <pubkey.h>
13#include <script/script.h>
14#include <sync.h>
16#include <tinyformat.h>
17#include <util/check.h>
18#include <util/result.h>
19#include <wallet/db.h>
21#include <wallet/test/util.h>
22#include <wallet/transaction.h>
23#include <wallet/wallet.h>
24#include <wallet/walletdb.h>
25
26#include <algorithm>
27#include <cstddef>
28#include <memory>
29#include <optional>
30#include <string>
31#include <utility>
32#include <vector>
33
34namespace wallet{
35
37{
38 const auto test_setup{MakeNoLogFileContext<TestingSetup>()};
39 const auto loader{MakeWalletLoader(*test_setup->m_node.chain, test_setup->m_args)};
40
41 // Number of imported watch only addresses
42 int NUM_WATCH_ONLY_ADDR = 20;
43
44 // Add watch-only addresses
45 std::vector<std::pair<CScript, CTxDestination>> scripts_watch_only;
46 for (int w = 0; w < NUM_WATCH_ONLY_ADDR; ++w) {
47 CKey key = GenerateRandomKey();
48 const PKHash dest{key.GetPubKey()};
49 scripts_watch_only.emplace_back(GetScriptForDestination(dest), dest);
50 }
51
52 // Generate transactions and local addresses
53 std::vector<CKey> keys(500);
55
56 std::unique_ptr<CWallet> wallet;
57 size_t i = 0;
58 bench.epochs(/*numEpochs=*/1) // run the migration exactly once
59 .setup([&] {
60 // Setup legacy wallet
61 wallet = std::make_unique<CWallet>(test_setup->m_node.chain.get(), std::string(i++, 'A'), CreateMockableWalletDatabase());
62 LegacyDataSPKM* legacy_spkm = wallet->GetOrCreateLegacyDataSPKM();
63 WalletBatch batch{wallet->GetDatabase()};
64
65 LOCK(wallet->cs_wallet);
66
67 // Write a best block record as migration expects one to exist
68 CBlockLocator loc;
69 batch.WriteBestBlock(loc);
70
71 // Add watch-only addresses
72 for (size_t w = 0; w < scripts_watch_only.size(); ++w) {
73 const auto& [script, dest] = scripts_watch_only.at(w);
74 assert(legacy_spkm->LoadWatchOnly(script));
75 assert(wallet->SetAddressBook(dest, strprintf("watch_%d", w), /*purpose=*/std::nullopt));
76 batch.WriteWatchOnly(script, CKeyMetadata());
77 }
78
79 // Generate transactions and local addresses
80 for (size_t j = 0; j < keys.size(); ++j) {
81 const CKey& key = keys.at(j);
82 // Load key, scripts and create address book record
83 CPubKey pubkey = key.GetPubKey();
84 Assert(legacy_spkm->LoadKey(key, pubkey));
85 CTxDestination dest{PKHash(pubkey)};
86 Assert(wallet->SetAddressBook(dest, strprintf("legacy_%d", j), /*purpose=*/std::nullopt));
87
89 mtx.vout.emplace_back(COIN, GetScriptForDestination(dest));
90 mtx.vout.emplace_back(COIN, scripts_watch_only.at(j % NUM_WATCH_ONLY_ADDR).first);
91 mtx.vin.resize(2);
92 wallet->AddToWallet(MakeTransactionRef(mtx), TxStateInactive{}, /*update_wtx=*/nullptr, /*rescanning_old_block=*/true);
93 batch.WriteKey(pubkey, key.GetPrivKey(), CKeyMetadata());
94 }
95 })
96 .run([&] {
97 auto res{MigrateLegacyToDescriptor(std::move(wallet), /*passphrase=*/"", *loader->context())};
98 assert(res);
99 assert(res->wallet);
100 assert(res->watchonly_wallet);
101 });
102}
103
105
106} // namespace wallet
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
Definition: addresstype.h:143
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition: amount.h:15
#define Assert(val)
Identity function.
Definition: check.h:116
An encapsulated private key.
Definition: key.h:37
CPrivKey GetPrivKey() const
Convert the private key to a CPrivKey (serialized OpenSSL private key data).
Definition: key.cpp:169
CPubKey GetPubKey() const
Compute the public key from a private key.
Definition: key.cpp:182
An encapsulated public key.
Definition: pubkey.h:32
Main entry point to nanobench's benchmarking facility.
Definition: nanobench.h:649
Bench & epochs(size_t numEpochs) noexcept
Controls number of epochs, the number of measurements to perform.
detail::SetupRunner< SetupOp > setup(SetupOp setupOp)
Configure an untimed setup step per epoch (forces single-iteration epochs).
Definition: nanobench.h:1302
bool LoadKey(const CKey &key, const CPubKey &pubkey)
Adds a key to the store, without saving it to disk (used by LoadWallet)
bool LoadWatchOnly(const CScript &dest)
Adds a watch-only address to the store, without saving it to disk (used by LoadWallet)
Access to the wallet database.
Definition: walletdb.h:197
CKey GenerateRandomKey(bool compressed) noexcept
Definition: key.cpp:352
std::unique_ptr< WalletLoader > MakeWalletLoader(Chain &chain, ArgsManager &args)
Return implementation of ChainClient interface for a wallet loader.
Definition: dummywallet.cpp:59
std::unique_ptr< WalletDatabase > CreateMockableWalletDatabase()
Definition: util.cpp:122
util::Result< MigrationResult > MigrateLegacyToDescriptor(const std::string &wallet_name, const SecureString &passphrase, WalletContext &context, bool load_wallet)
Do all steps to migrate a legacy wallet to a descriptor wallet.
Definition: wallet.cpp:4276
static void WalletMigration(benchmark::Bench &bench)
BENCHMARK(CoinSelection)
static CTransactionRef MakeTransactionRef(Tx &&txIn)
Definition: transaction.h:404
static RPCMethod generate()
Definition: mining.cpp:288
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
Definition: block.h:117
A mutable version of CTransaction.
Definition: transaction.h:358
std::vector< CTxOut > vout
Definition: transaction.h:360
std::vector< CTxIn > vin
Definition: transaction.h:359
State of transaction not confirmed or conflicting with a known block and not in the mempool.
Definition: transaction.h:59
#define LOCK(cs)
Definition: sync.h:268
std::vector< uint16_t > keys
Definition: dbwrapper.cpp:377
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
assert(!tx.IsCoinBase())