Bitcoin Core Fuzz Coverage Report for wallet_tx_can_be_bumped

Coverage Report

Created: 2025-11-19 11:20

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/Users/brunogarcia/projects/bitcoin-core-dev/src/wallet/scriptpubkeyman.h
Line
Count
Source
1
// Copyright (c) 2019-present The Bitcoin Core developers
2
// Distributed under the MIT software license, see the accompanying
3
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5
#ifndef BITCOIN_WALLET_SCRIPTPUBKEYMAN_H
6
#define BITCOIN_WALLET_SCRIPTPUBKEYMAN_H
7
8
#include <addresstype.h>
9
#include <common/messages.h>
10
#include <common/signmessage.h>
11
#include <common/types.h>
12
#include <logging.h>
13
#include <musig.h>
14
#include <node/types.h>
15
#include <psbt.h>
16
#include <script/descriptor.h>
17
#include <script/script.h>
18
#include <script/signingprovider.h>
19
#include <util/result.h>
20
#include <util/time.h>
21
#include <wallet/crypter.h>
22
#include <wallet/types.h>
23
#include <wallet/walletdb.h>
24
#include <wallet/walletutil.h>
25
26
#include <boost/signals2/signal.hpp>
27
28
#include <functional>
29
#include <optional>
30
#include <unordered_map>
31
32
enum class OutputType;
33
34
namespace wallet {
35
struct MigrationData;
36
class ScriptPubKeyMan;
37
38
// Wallet storage things that ScriptPubKeyMans need in order to be able to store things to the wallet database.
39
// It provides access to things that are part of the entire wallet and not specific to a ScriptPubKeyMan such as
40
// wallet flags, wallet version, encryption keys, encryption status, and the database itself. This allows a
41
// ScriptPubKeyMan to have callbacks into CWallet without causing a circular dependency.
42
// WalletStorage should be the same for all ScriptPubKeyMans of a wallet.
43
class WalletStorage
44
{
45
public:
46
9.91k
    virtual ~WalletStorage() = default;
47
    virtual std::string LogName() const = 0;
48
    virtual WalletDatabase& GetDatabase() const = 0;
49
    virtual bool IsWalletFlagSet(uint64_t) const = 0;
50
    virtual void UnsetBlankWalletFlag(WalletBatch&) = 0;
51
    //! Pass the encryption key to cb().
52
    virtual bool WithEncryptionKey(std::function<bool (const CKeyingMaterial&)> cb) const = 0;
53
    virtual bool HasEncryptionKeys() const = 0;
54
    virtual bool IsLocked() const = 0;
55
    //! Callback function for after TopUp completes containing any scripts that were added by a SPKMan
56
    virtual void TopUpCallback(const std::set<CScript>&, ScriptPubKeyMan*) = 0;
57
};
58
59
//! Constant representing an unknown spkm creation time
60
static constexpr int64_t UNKNOWN_TIME = std::numeric_limits<int64_t>::max();
61
62
//! Default for -keypool
63
static const unsigned int DEFAULT_KEYPOOL_SIZE = 1000;
64
65
std::vector<CKeyID> GetAffectedKeys(const CScript& spk, const SigningProvider& provider);
66
67
struct WalletDestination
68
{
69
    CTxDestination dest;
70
    std::optional<bool> internal;
71
};
72
73
/*
74
 * A class implementing ScriptPubKeyMan manages some (or all) scriptPubKeys used in a wallet.
75
 * It contains the scripts and keys related to the scriptPubKeys it manages.
76
 * A ScriptPubKeyMan will be able to give out scriptPubKeys to be used, as well as marking
77
 * when a scriptPubKey has been used. It also handles when and how to store a scriptPubKey
78
 * and its related scripts and keys, including encryption.
79
 */
80
class ScriptPubKeyMan
81
{
82
protected:
83
    WalletStorage& m_storage;
84
85
public:
86
79.3k
    explicit ScriptPubKeyMan(WalletStorage& storage) : m_storage(storage) {}
87
79.3k
    virtual ~ScriptPubKeyMan() = default;
88
0
    virtual util::Result<CTxDestination> GetNewDestination(const OutputType type) { return util::Error{Untranslated("Not supported")}; }
89
0
    virtual bool IsMine(const CScript& script) const { return false; }
90
91
    //! Check that the given decryption key is valid for this ScriptPubKeyMan, i.e. it decrypts all of the keys handled by it.
92
0
    virtual bool CheckDecryptionKey(const CKeyingMaterial& master_key) { return false; }
93
0
    virtual bool Encrypt(const CKeyingMaterial& master_key, WalletBatch* batch) { return false; }
94
95
0
    virtual util::Result<CTxDestination> GetReservedDestination(const OutputType type, bool internal, int64_t& index) { return util::Error{Untranslated("Not supported")}; }
96
4.11k
    virtual void KeepDestination(int64_t index, const OutputType& type) {}
97
0
    virtual void ReturnDestination(int64_t index, bool internal, const CTxDestination& addr) {}
98
99
    /** Fills internal address pool. Use within ScriptPubKeyMan implementations should be used sparingly and only
100
      * when something from the address pool is removed, excluding GetNewDestination and GetReservedDestination.
101
      * External wallet code is primarily responsible for topping up prior to fetching new addresses
102
      */
103
0
    virtual bool TopUp(unsigned int size = 0) { return false; }
104
105
    /** Mark unused addresses as being used
106
     * Affects all keys up to and including the one determined by provided script.
107
     *
108
     * @param script determines the last key to mark as used
109
     *
110
     * @return All of the addresses affected
111
     */
112
0
    virtual std::vector<WalletDestination> MarkUnusedAddresses(const CScript& script) { return {}; }
113
114
    /* Returns true if HD is enabled */
115
0
    virtual bool IsHDEnabled() const { return false; }
116
117
    /* Returns true if the wallet can give out new addresses. This means it has keys in the keypool or can generate new keys */
118
0
    virtual bool CanGetAddresses(bool internal = false) const { return false; }
119
120
0
    virtual bool HavePrivateKeys() const { return false; }
121
0
    virtual bool HaveCryptedKeys() const { return false; }
122
123
    //! The action to do when the DB needs rewrite
124
0
    virtual void RewriteDB() {}
125
126
0
    virtual unsigned int GetKeyPoolSize() const { return 0; }
127
128
0
    virtual int64_t GetTimeFirstKey() const { return 0; }
129
130
0
    virtual std::unique_ptr<CKeyMetadata> GetMetadata(const CTxDestination& dest) const { return nullptr; }
131
132
0
    virtual std::unique_ptr<SigningProvider> GetSolvingProvider(const CScript& script) const { return nullptr; }
133
134
    /** Whether this ScriptPubKeyMan can provide a SigningProvider (via GetSolvingProvider) that, combined with
135
      * sigdata, can produce solving data.
136
      */
137
0
    virtual bool CanProvide(const CScript& script, SignatureData& sigdata) { return false; }
138
139
    /** Creates new signatures and adds them to the transaction. Returns whether all inputs were signed */
140
0
    virtual bool SignTransaction(CMutableTransaction& tx, const std::map<COutPoint, Coin>& coins, int sighash, std::map<int, bilingual_str>& input_errors) const { return false; }
141
    /** Sign a message with the given script */
142
0
    virtual SigningResult SignMessage(const std::string& message, const PKHash& pkhash, std::string& str_sig) const { return SigningResult::SIGNING_FAILED; };
143
    /** Adds script and derivation path information to a PSBT, and optionally signs it. */
144
0
    virtual std::optional<common::PSBTError> FillPSBT(PartiallySignedTransaction& psbt, const PrecomputedTransactionData& txdata, std::optional<int> sighash_type = std::nullopt, bool sign = true, bool bip32derivs = false, int* n_signed = nullptr, bool finalize = true) const { return common::PSBTError::UNSUPPORTED; }
145
146
0
    virtual uint256 GetID() const { return uint256(); }
147
148
    /** Returns a set of all the scriptPubKeys that this ScriptPubKeyMan watches */
149
0
    virtual std::unordered_set<CScript, SaltedSipHasher> GetScriptPubKeys() const { return {}; };
150
151
    /** Prepends the wallet name in logging output to ease debugging in multi-wallet use cases */
152
    template <typename... Params>
153
    void WalletLogPrintf(util::ConstevalFormatString<sizeof...(Params)> wallet_fmt, const Params&... params) const
154
0
    {
155
0
        LogInfo("[%s] %s", m_storage.LogName(), tfm::format(wallet_fmt, params...));
Line
Count
Source
368
0
#define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__)
Line
Count
Source
362
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(std::source_location::current(), category, level, should_ratelimit, __VA_ARGS__)
        LogInfo("[%s] %s", m_storage.LogName(), tfm::format(wallet_fmt, params...));
Line
Count
Source
368
0
#define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__)
Line
Count
Source
362
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(std::source_location::current(), category, level, should_ratelimit, __VA_ARGS__)
        LogInfo("[%s] %s", m_storage.LogName(), tfm::format(wallet_fmt, params...));
Line
Count
Source
368
0
#define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__)
Line
Count
Source
362
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(std::source_location::current(), category, level, should_ratelimit, __VA_ARGS__)
156
0
    };
Unexecuted instantiation: void wallet::ScriptPubKeyMan::WalletLogPrintf<char [12], unsigned int, unsigned int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char>>>(util::ConstevalFormatString<sizeof...(char [12], unsigned int, unsigned int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char>>)>, char const (&) [12], unsigned int const&, unsigned int const&, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char>> const&) const
Unexecuted instantiation: void wallet::ScriptPubKeyMan::WalletLogPrintf<char [20], int>(util::ConstevalFormatString<sizeof...(char [20], int)>, char const (&) [20], int const&) const
Unexecuted instantiation: void wallet::ScriptPubKeyMan::WalletLogPrintf<char [20]>(util::ConstevalFormatString<sizeof...(char [20])>, char const (&) [20]) const
157
158
    /** Keypool has new keys */
159
    boost::signals2::signal<void ()> NotifyCanGetAddressesChanged;
160
161
    /** Birth time changed */
162
    boost::signals2::signal<void (const ScriptPubKeyMan* spkm, int64_t new_birth_time)> NotifyFirstKeyTimeChanged;
163
};
164
165
/** OutputTypes supported by the LegacyScriptPubKeyMan */
166
static const std::unordered_set<OutputType> LEGACY_OUTPUT_TYPES {
167
    OutputType::LEGACY,
168
    OutputType::P2SH_SEGWIT,
169
    OutputType::BECH32,
170
};
171
172
// Manages the data for a LegacyScriptPubKeyMan.
173
// This is the minimum necessary to load a legacy wallet so that it can be migrated.
174
class LegacyDataSPKM : public ScriptPubKeyMan, public FillableSigningProvider
175
{
176
private:
177
    using WatchOnlySet = std::set<CScript>;
178
    using WatchKeyMap = std::map<CKeyID, CPubKey>;
179
    using CryptedKeyMap = std::map<CKeyID, std::pair<CPubKey, std::vector<unsigned char>>>;
180
181
    CryptedKeyMap mapCryptedKeys GUARDED_BY(cs_KeyStore);
182
    WatchOnlySet setWatchOnly GUARDED_BY(cs_KeyStore);
183
    WatchKeyMap mapWatchKeys GUARDED_BY(cs_KeyStore);
184
185
    /* the HD chain data model (external chain counters) */
186
    CHDChain m_hd_chain;
187
    std::unordered_map<CKeyID, CHDChain, SaltedSipHasher> m_inactive_hd_chains;
188
189
    //! keeps track of whether Unlock has run a thorough check before
190
    bool fDecryptionThoroughlyChecked = true;
191
192
    bool AddWatchOnlyInMem(const CScript &dest);
193
    virtual bool AddKeyPubKeyInner(const CKey& key, const CPubKey &pubkey);
194
    bool AddCryptedKeyInner(const CPubKey &vchPubKey, const std::vector<unsigned char> &vchCryptedSecret);
195
196
    // Helper function to retrieve a conservative superset of all output scripts that may be relevant to this LegacyDataSPKM.
197
    // It may include scripts that are invalid or not actually watched by this LegacyDataSPKM.
198
    // Used only in migration.
199
    std::unordered_set<CScript, SaltedSipHasher> GetCandidateScriptPubKeys() const;
200
201
    bool IsMine(const CScript& script) const override;
202
    bool CanProvide(const CScript& script, SignatureData& sigdata) override;
203
public:
204
    using ScriptPubKeyMan::ScriptPubKeyMan;
205
206
    // Map from Key ID to key metadata.
207
    std::map<CKeyID, CKeyMetadata> mapKeyMetadata GUARDED_BY(cs_KeyStore);
208
209
    // Map from Script ID to key metadata (for watch-only keys).
210
    std::map<CScriptID, CKeyMetadata> m_script_metadata GUARDED_BY(cs_KeyStore);
211
212
    // ScriptPubKeyMan overrides
213
    bool CheckDecryptionKey(const CKeyingMaterial& master_key) override;
214
    std::unordered_set<CScript, SaltedSipHasher> GetScriptPubKeys() const override;
215
    std::unique_ptr<SigningProvider> GetSolvingProvider(const CScript& script) const override;
216
0
    uint256 GetID() const override { return uint256::ONE; }
217
218
    // FillableSigningProvider overrides
219
    bool HaveKey(const CKeyID &address) const override;
220
    bool GetKey(const CKeyID &address, CKey& keyOut) const override;
221
    bool GetPubKey(const CKeyID &address, CPubKey& vchPubKeyOut) const override;
222
    bool GetKeyOrigin(const CKeyID& keyid, KeyOriginInfo& info) const override;
223
224
    //! Load metadata (used by LoadWallet)
225
    virtual void LoadKeyMetadata(const CKeyID& keyID, const CKeyMetadata &metadata);
226
    virtual void LoadScriptMetadata(const CScriptID& script_id, const CKeyMetadata &metadata);
227
228
    //! Adds a watch-only address to the store, without saving it to disk (used by LoadWallet)
229
    bool LoadWatchOnly(const CScript &dest);
230
    //! Returns whether the watch-only script is in the wallet
231
    bool HaveWatchOnly(const CScript &dest) const;
232
    //! Adds a key to the store, without saving it to disk (used by LoadWallet)
233
    bool LoadKey(const CKey& key, const CPubKey &pubkey);
234
    //! Adds an encrypted key to the store, without saving it to disk (used by LoadWallet)
235
    bool LoadCryptedKey(const CPubKey &vchPubKey, const std::vector<unsigned char> &vchCryptedSecret, bool checksum_valid);
236
    //! Adds a CScript to the store
237
    bool LoadCScript(const CScript& redeemScript);
238
    //! Load a HD chain model (used by LoadWallet)
239
    void LoadHDChain(const CHDChain& chain);
240
    void AddInactiveHDChain(const CHDChain& chain);
241
0
    const CHDChain& GetHDChain() const { return m_hd_chain; }
242
243
    //! Fetches a pubkey from mapWatchKeys if it exists there
244
    bool GetWatchPubKey(const CKeyID &address, CPubKey &pubkey_out) const;
245
246
    /**
247
     * Retrieves scripts that were imported by bugs into the legacy spkm and are
248
     * simply invalid, such as a sh(sh(pkh())) script, or not watched.
249
     */
250
    std::unordered_set<CScript, SaltedSipHasher> GetNotMineScriptPubKeys() const;
251
252
    /** Get the DescriptorScriptPubKeyMans (with private keys) that have the same scriptPubKeys as this LegacyScriptPubKeyMan.
253
     * Does not modify this ScriptPubKeyMan. */
254
    std::optional<MigrationData> MigrateToDescriptor();
255
    /** Delete all the records of this LegacyScriptPubKeyMan from disk*/
256
    bool DeleteRecordsWithDB(WalletBatch& batch);
257
};
258
259
/** Wraps a LegacyScriptPubKeyMan so that it can be returned in a new unique_ptr. Does not provide privkeys */
260
class LegacySigningProvider : public SigningProvider
261
{
262
private:
263
    const LegacyDataSPKM& m_spk_man;
264
public:
265
0
    explicit LegacySigningProvider(const LegacyDataSPKM& spk_man) : m_spk_man(spk_man) {}
266
267
0
    bool GetCScript(const CScriptID &scriptid, CScript& script) const override { return m_spk_man.GetCScript(scriptid, script); }
268
0
    bool HaveCScript(const CScriptID &scriptid) const override { return m_spk_man.HaveCScript(scriptid); }
269
0
    bool GetPubKey(const CKeyID &address, CPubKey& pubkey) const override { return m_spk_man.GetPubKey(address, pubkey); }
270
0
    bool GetKey(const CKeyID &address, CKey& key) const override { return false; }
271
0
    bool HaveKey(const CKeyID &address) const override { return false; }
272
0
    bool GetKeyOrigin(const CKeyID& keyid, KeyOriginInfo& info) const override { return m_spk_man.GetKeyOrigin(keyid, info); }
273
};
274
275
class DescriptorScriptPubKeyMan : public ScriptPubKeyMan
276
{
277
    friend class LegacyDataSPKM;
278
private:
279
    using ScriptPubKeyMap = std::map<CScript, int32_t>; // Map of scripts to descriptor range index
280
    using PubKeyMap = std::map<CPubKey, int32_t>; // Map of pubkeys involved in scripts to descriptor range index
281
    using CryptedKeyMap = std::map<CKeyID, std::pair<CPubKey, std::vector<unsigned char>>>;
282
    using KeyMap = std::map<CKeyID, CKey>;
283
284
    ScriptPubKeyMap m_map_script_pub_keys GUARDED_BY(cs_desc_man);
285
    PubKeyMap m_map_pubkeys GUARDED_BY(cs_desc_man);
286
    int32_t m_max_cached_index = -1;
287
288
    KeyMap m_map_keys GUARDED_BY(cs_desc_man);
289
    CryptedKeyMap m_map_crypted_keys GUARDED_BY(cs_desc_man);
290
291
    //! keeps track of whether Unlock has run a thorough check before
292
    bool m_decryption_thoroughly_checked = false;
293
294
    //! Number of pre-generated keys/scripts (part of the look-ahead process, used to detect payments)
295
    int64_t m_keypool_size GUARDED_BY(cs_desc_man){DEFAULT_KEYPOOL_SIZE};
296
297
    /** Map of a session id to MuSig2 secnonce
298
     *
299
     * Stores MuSig2 secnonces while the MuSig2 signing session is still ongoing.
300
     * Note that these secnonces must not be reused. In order to avoid being tricked into
301
     * reusing a nonce, this map is held only in memory and must not be written to disk.
302
     * The side effect is that signing sessions cannot persist across restarts, but this
303
     * must be done in order to prevent nonce reuse.
304
     *
305
     * The session id is an arbitrary value set by the signer in order for the signing logic
306
     * to find ongoing signing sessions. It is the SHA256 of aggregate xonly key, + participant pubkey + sighash.
307
     */
308
    mutable std::map<uint256, MuSig2SecNonce> m_musig2_secnonces;
309
310
    bool AddDescriptorKeyWithDB(WalletBatch& batch, const CKey& key, const CPubKey &pubkey) EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
311
312
    KeyMap GetKeys() const EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
313
314
    // Cached FlatSigningProviders to avoid regenerating them each time they are needed.
315
    mutable std::map<int32_t, FlatSigningProvider> m_map_signing_providers;
316
    // Fetch the SigningProvider for the given script and optionally include private keys
317
    std::unique_ptr<FlatSigningProvider> GetSigningProvider(const CScript& script, bool include_private = false) const;
318
    // Fetch the SigningProvider for a given index and optionally include private keys. Called by the above functions.
319
    std::unique_ptr<FlatSigningProvider> GetSigningProvider(int32_t index, bool include_private = false) const EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
320
321
protected:
322
    WalletDescriptor m_wallet_descriptor GUARDED_BY(cs_desc_man);
323
324
    //! Same as 'TopUp' but designed for use within a batch transaction context
325
    bool TopUpWithDB(WalletBatch& batch, unsigned int size = 0);
326
327
public:
328
    DescriptorScriptPubKeyMan(WalletStorage& storage, WalletDescriptor& descriptor, int64_t keypool_size)
329
79.3k
        :   ScriptPubKeyMan(storage),
330
79.3k
            m_keypool_size(keypool_size),
331
79.3k
            m_wallet_descriptor(descriptor)
332
79.3k
        {}
333
    DescriptorScriptPubKeyMan(WalletStorage& storage, int64_t keypool_size)
334
0
        :   ScriptPubKeyMan(storage),
335
0
            m_keypool_size(keypool_size)
336
0
        {}
337
338
    mutable RecursiveMutex cs_desc_man;
339
340
    util::Result<CTxDestination> GetNewDestination(const OutputType type) override;
341
    bool IsMine(const CScript& script) const override;
342
343
    bool CheckDecryptionKey(const CKeyingMaterial& master_key) override;
344
    bool Encrypt(const CKeyingMaterial& master_key, WalletBatch* batch) override;
345
346
    util::Result<CTxDestination> GetReservedDestination(const OutputType type, bool internal, int64_t& index) override;
347
    void ReturnDestination(int64_t index, bool internal, const CTxDestination& addr) override;
348
349
    // Tops up the descriptor cache and m_map_script_pub_keys. The cache is stored in the wallet file
350
    // and is used to expand the descriptor in GetNewDestination. DescriptorScriptPubKeyMan relies
351
    // more on ephemeral data than LegacyScriptPubKeyMan. For wallets using unhardened derivation
352
    // (with or without private keys), the "keypool" is a single xpub.
353
    bool TopUp(unsigned int size = 0) override;
354
355
    std::vector<WalletDestination> MarkUnusedAddresses(const CScript& script) override;
356
357
    bool IsHDEnabled() const override;
358
359
    //! Setup descriptors based on the given CExtkey
360
    bool SetupDescriptorGeneration(WalletBatch& batch, const CExtKey& master_key, OutputType addr_type, bool internal);
361
362
    bool HavePrivateKeys() const override;
363
    bool HasPrivKey(const CKeyID& keyid) const EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
364
    //! Retrieve the particular key if it is available. Returns nullopt if the key is not in the wallet, or if the wallet is locked.
365
    std::optional<CKey> GetKey(const CKeyID& keyid) const EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
366
    bool HaveCryptedKeys() const override;
367
368
    unsigned int GetKeyPoolSize() const override;
369
370
    int64_t GetTimeFirstKey() const override;
371
372
    std::unique_ptr<CKeyMetadata> GetMetadata(const CTxDestination& dest) const override;
373
374
    bool CanGetAddresses(bool internal = false) const override;
375
376
    std::unique_ptr<SigningProvider> GetSolvingProvider(const CScript& script) const override;
377
378
    bool CanProvide(const CScript& script, SignatureData& sigdata) override;
379
380
    // Fetch the SigningProvider for the given pubkey and always include private keys. This should only be called by signing code.
381
    std::unique_ptr<FlatSigningProvider> GetSigningProvider(const CPubKey& pubkey) const;
382
383
    bool SignTransaction(CMutableTransaction& tx, const std::map<COutPoint, Coin>& coins, int sighash, std::map<int, bilingual_str>& input_errors) const override;
384
    SigningResult SignMessage(const std::string& message, const PKHash& pkhash, std::string& str_sig) const override;
385
    std::optional<common::PSBTError> FillPSBT(PartiallySignedTransaction& psbt, const PrecomputedTransactionData& txdata, std::optional<int> sighash_type = std::nullopt, bool sign = true, bool bip32derivs = false, int* n_signed = nullptr, bool finalize = true) const override;
386
387
    uint256 GetID() const override;
388
389
    void SetCache(const DescriptorCache& cache);
390
391
    bool AddKey(const CKeyID& key_id, const CKey& key);
392
    bool AddCryptedKey(const CKeyID& key_id, const CPubKey& pubkey, const std::vector<unsigned char>& crypted_key);
393
394
    bool HasWalletDescriptor(const WalletDescriptor& desc) const;
395
    util::Result<void> UpdateWalletDescriptor(WalletDescriptor& descriptor);
396
    bool CanUpdateToWalletDescriptor(const WalletDescriptor& descriptor, std::string& error);
397
    void AddDescriptorKey(const CKey& key, const CPubKey &pubkey);
398
    void WriteDescriptor();
399
400
    WalletDescriptor GetWalletDescriptor() const EXCLUSIVE_LOCKS_REQUIRED(cs_desc_man);
401
    std::unordered_set<CScript, SaltedSipHasher> GetScriptPubKeys() const override;
402
    std::unordered_set<CScript, SaltedSipHasher> GetScriptPubKeys(int32_t minimum_index) const;
403
    int32_t GetEndRange() const;
404
405
    [[nodiscard]] bool GetDescriptorString(std::string& out, const bool priv) const;
406
407
    void UpgradeDescriptorCache();
408
};
409
410
/** struct containing information needed for migrating legacy wallets to descriptor wallets */
411
struct MigrationData
412
{
413
    CExtKey master_key;
414
    std::vector<std::pair<std::string, int64_t>> watch_descs;
415
    std::vector<std::pair<std::string, int64_t>> solvable_descs;
416
    std::vector<std::unique_ptr<DescriptorScriptPubKeyMan>> desc_spkms;
417
    std::shared_ptr<CWallet> watchonly_wallet{nullptr};
418
    std::shared_ptr<CWallet> solvable_wallet{nullptr};
419
};
420
421
} // namespace wallet
422
423
#endif // BITCOIN_WALLET_SCRIPTPUBKEYMAN_H