Bitcoin Core  31.0.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 <bench/bench.h>
6 #include <interfaces/chain.h>
7 #include <interfaces/wallet.h>
8 #include <kernel/chain.h>
9 #include <kernel/types.h>
10 #include <node/context.h>
11 #include <test/util/mining.h>
12 #include <test/util/setup_common.h>
13 #include <wallet/context.h>
14 #include <wallet/receive.h>
15 #include <wallet/test/util.h>
16 #include <wallet/wallet.h>
17 
18 #include <optional>
19 
20 namespace wallet{
21 
22 static void WalletMigration(benchmark::Bench& bench)
23 {
24  const auto test_setup{MakeNoLogFileContext<TestingSetup>()};
25  const auto loader{MakeWalletLoader(*test_setup->m_node.chain, test_setup->m_args)};
26 
27  // Number of imported watch only addresses
28  int NUM_WATCH_ONLY_ADDR = 20;
29 
30  // Setup legacy wallet
31  std::unique_ptr<CWallet> wallet = std::make_unique<CWallet>(test_setup->m_node.chain.get(), "", CreateMockableWalletDatabase());
32  LegacyDataSPKM* legacy_spkm = wallet->GetOrCreateLegacyDataSPKM();
33  WalletBatch batch{wallet->GetDatabase()};
34 
35  // Write a best block record as migration expects one to exist
36  CBlockLocator loc;
37  batch.WriteBestBlock(loc);
38 
39  // Add watch-only addresses
40  std::vector<CScript> scripts_watch_only;
41  for (int w = 0; w < NUM_WATCH_ONLY_ADDR; ++w) {
42  CKey key = GenerateRandomKey();
43  LOCK(wallet->cs_wallet);
44  const PKHash dest{key.GetPubKey()};
45  const CScript& script = scripts_watch_only.emplace_back(GetScriptForDestination(dest));
46  assert(legacy_spkm->LoadWatchOnly(script));
47  assert(wallet->SetAddressBook(dest, strprintf("watch_%d", w), /*purpose=*/std::nullopt));
48  batch.WriteWatchOnly(script, CKeyMetadata());
49  }
50 
51  // Generate transactions and local addresses
52  for (int j = 0; j < 500; ++j) {
53  CKey key = GenerateRandomKey();
54  CPubKey pubkey = key.GetPubKey();
55  // Load key, scripts and create address book record
56  Assert(legacy_spkm->LoadKey(key, pubkey));
57  CTxDestination dest{PKHash(pubkey)};
58  Assert(wallet->SetAddressBook(dest, strprintf("legacy_%d", j), /*purpose=*/std::nullopt));
59 
61  mtx.vout.emplace_back(COIN, GetScriptForDestination(dest));
62  mtx.vout.emplace_back(COIN, scripts_watch_only.at(j % NUM_WATCH_ONLY_ADDR));
63  mtx.vin.resize(2);
64  wallet->AddToWallet(MakeTransactionRef(mtx), TxStateInactive{}, /*update_wtx=*/nullptr, /*rescanning_old_block=*/true);
65  batch.WriteKey(pubkey, key.GetPrivKey(), CKeyMetadata());
66  }
67 
68  bench.epochs(/*numEpochs=*/1).epochIterations(/*numIters=*/1) // run the migration exactly once
69  .run([&] {
70  auto res{MigrateLegacyToDescriptor(std::move(wallet), /*passphrase=*/"", *loader->context())};
71  assert(res);
72  assert(res->wallet);
73  assert(res->watchonly_wallet);
74  });
75 }
76 
78 
79 } // namespace wallet
is a home for simple enum and struct type definitions that can be used internally by functions in the...
CPrivKey GetPrivKey() const
Convert the private key to a CPrivKey (serialized OpenSSL private key data).
Definition: key.cpp:170
Bench & epochIterations(uint64_t numIters) noexcept
Sets exactly the number of iterations for each epoch.
assert(!tx.IsCoinBase())
Describes a place in the block chain to another node such that if the other node doesn&#39;t have the sam...
Definition: block.h:116
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
CPubKey GetPubKey() const
Compute the public key from a private key.
Definition: key.cpp:183
State of transaction not confirmed or conflicting with a known block and not in the mempool...
Definition: transaction.h:59
Access to the wallet database.
Definition: walletdb.h:192
CKey GenerateRandomKey(bool compressed) noexcept
Definition: key.cpp:475
std::unique_ptr< WalletLoader > MakeWalletLoader(Chain &chain, ArgsManager &args)
Return implementation of ChainClient interface for a wallet loader.
Definition: dummywallet.cpp:59
Bench & run(char const *benchmarkName, Op &&op)
Repeatedly calls op() based on the configuration, and performs measurements.
Definition: nanobench.h:1234
static void WalletMigration(benchmark::Bench &bench)
#define LOCK(cs)
Definition: sync.h:258
An encapsulated public key.
Definition: pubkey.h:33
std::vector< CTxOut > vout
Definition: transaction.h:360
BENCHMARK(WalletBalanceDirty)
util::Result< MigrationResult > MigrateLegacyToDescriptor(const std::string &wallet_name, const SecureString &passphrase, WalletContext &context)
Do all steps to migrate a legacy wallet to a descriptor wallet.
Definition: wallet.cpp:4250
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
static CTransactionRef MakeTransactionRef(Tx &&txIn)
Definition: transaction.h:404
std::unique_ptr< WalletDatabase > CreateMockableWalletDatabase(MockableData records)
Definition: util.cpp:211
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:404
Bench & epochs(size_t numEpochs) noexcept
Controls number of epochs, the number of measurements to perform.
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
Definition: addresstype.h:143
A mutable version of CTransaction.
Definition: transaction.h:357
Main entry point to nanobench&#39;s benchmarking facility.
Definition: nanobench.h:627
An encapsulated private key.
Definition: key.h:35
#define Assert(val)
Identity function.
Definition: check.h:113
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition: amount.h:15