mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-03-09 15:37:00 -04:00

0bbe26a1af
wallet: filter for keys only before record deser in salvage (Andrew Chow)544e12a4e8
walletdb: Add KeyFilterFn to ReadKeyValue (Andrew Chow) Pull request description: When salvaging a wallet, the only things that matter are the private keys. It is not necessary to attempt to deserialize any other records, especially if those records are corrupted too. This PR adds a `KeyFilterFn` function callback to `ReadKeyValue` that salvage uses to filter for only the records that it wants. Of course doing it this way also lets us do other filters in the future from other places should we so desire. ACKs for top commit: ryanofsky: Code review ACK0bbe26a1af
. Looks great! This should make the recovery code more robust. Normally it'd be good to have a test case for the problem this fixes, but Marco already wrote one in #19078, so I think we're covered laanwj: Code review ACK0bbe26a1af
Tree-SHA512: 8e3ee283a22a79273915711c4fb751f3c9b02ce94e6bf08dc468f1cfdf9fac35c693bbfd2435ce43c3a06c601b9b0a67e209621f6814bedfe3bc7a7ccc37bb01
300 lines
10 KiB
C++
300 lines
10 KiB
C++
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#ifndef BITCOIN_WALLET_WALLETDB_H
|
|
#define BITCOIN_WALLET_WALLETDB_H
|
|
|
|
#include <amount.h>
|
|
#include <script/sign.h>
|
|
#include <wallet/bdb.h>
|
|
#include <wallet/db.h>
|
|
#include <wallet/walletutil.h>
|
|
#include <key.h>
|
|
|
|
#include <stdint.h>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
/**
|
|
* Overview of wallet database classes:
|
|
*
|
|
* - WalletBatch is an abstract modifier object for the wallet database, and encapsulates a database
|
|
* batch update as well as methods to act on the database. It should be agnostic to the database implementation.
|
|
*
|
|
* The following classes are implementation specific:
|
|
* - BerkeleyEnvironment is an environment in which the database exists.
|
|
* - BerkeleyDatabase represents a wallet database.
|
|
* - BerkeleyBatch is a low-level database batch update.
|
|
*/
|
|
|
|
static const bool DEFAULT_FLUSHWALLET = true;
|
|
|
|
struct CBlockLocator;
|
|
class CKeyPool;
|
|
class CMasterKey;
|
|
class CScript;
|
|
class CWallet;
|
|
class CWalletTx;
|
|
class uint160;
|
|
class uint256;
|
|
|
|
/** Error statuses for the wallet database */
|
|
enum class DBErrors
|
|
{
|
|
LOAD_OK,
|
|
CORRUPT,
|
|
NONCRITICAL_ERROR,
|
|
TOO_NEW,
|
|
LOAD_FAIL,
|
|
NEED_REWRITE
|
|
};
|
|
|
|
namespace DBKeys {
|
|
extern const std::string ACENTRY;
|
|
extern const std::string ACTIVEEXTERNALSPK;
|
|
extern const std::string ACTIVEINTERNALSPK;
|
|
extern const std::string BESTBLOCK;
|
|
extern const std::string BESTBLOCK_NOMERKLE;
|
|
extern const std::string CRYPTED_KEY;
|
|
extern const std::string CSCRIPT;
|
|
extern const std::string DEFAULTKEY;
|
|
extern const std::string DESTDATA;
|
|
extern const std::string FLAGS;
|
|
extern const std::string HDCHAIN;
|
|
extern const std::string KEY;
|
|
extern const std::string KEYMETA;
|
|
extern const std::string MASTER_KEY;
|
|
extern const std::string MINVERSION;
|
|
extern const std::string NAME;
|
|
extern const std::string OLD_KEY;
|
|
extern const std::string ORDERPOSNEXT;
|
|
extern const std::string POOL;
|
|
extern const std::string PURPOSE;
|
|
extern const std::string SETTINGS;
|
|
extern const std::string TX;
|
|
extern const std::string VERSION;
|
|
extern const std::string WALLETDESCRIPTOR;
|
|
extern const std::string WALLETDESCRIPTORCKEY;
|
|
extern const std::string WALLETDESCRIPTORKEY;
|
|
extern const std::string WATCHMETA;
|
|
extern const std::string WATCHS;
|
|
} // namespace DBKeys
|
|
|
|
/* simple HD chain data model */
|
|
class CHDChain
|
|
{
|
|
public:
|
|
uint32_t nExternalChainCounter;
|
|
uint32_t nInternalChainCounter;
|
|
CKeyID seed_id; //!< seed hash160
|
|
|
|
static const int VERSION_HD_BASE = 1;
|
|
static const int VERSION_HD_CHAIN_SPLIT = 2;
|
|
static const int CURRENT_VERSION = VERSION_HD_CHAIN_SPLIT;
|
|
int nVersion;
|
|
|
|
CHDChain() { SetNull(); }
|
|
|
|
SERIALIZE_METHODS(CHDChain, obj)
|
|
{
|
|
READWRITE(obj.nVersion, obj.nExternalChainCounter, obj.seed_id);
|
|
if (obj.nVersion >= VERSION_HD_CHAIN_SPLIT) {
|
|
READWRITE(obj.nInternalChainCounter);
|
|
}
|
|
}
|
|
|
|
void SetNull()
|
|
{
|
|
nVersion = CHDChain::CURRENT_VERSION;
|
|
nExternalChainCounter = 0;
|
|
nInternalChainCounter = 0;
|
|
seed_id.SetNull();
|
|
}
|
|
|
|
bool operator==(const CHDChain& chain) const
|
|
{
|
|
return seed_id == chain.seed_id;
|
|
}
|
|
};
|
|
|
|
class CKeyMetadata
|
|
{
|
|
public:
|
|
static const int VERSION_BASIC=1;
|
|
static const int VERSION_WITH_HDDATA=10;
|
|
static const int VERSION_WITH_KEY_ORIGIN = 12;
|
|
static const int CURRENT_VERSION=VERSION_WITH_KEY_ORIGIN;
|
|
int nVersion;
|
|
int64_t nCreateTime; // 0 means unknown
|
|
std::string hdKeypath; //optional HD/bip32 keypath. Still used to determine whether a key is a seed. Also kept for backwards compatibility
|
|
CKeyID hd_seed_id; //id of the HD seed used to derive this key
|
|
KeyOriginInfo key_origin; // Key origin info with path and fingerprint
|
|
bool has_key_origin = false; //!< Whether the key_origin is useful
|
|
|
|
CKeyMetadata()
|
|
{
|
|
SetNull();
|
|
}
|
|
explicit CKeyMetadata(int64_t nCreateTime_)
|
|
{
|
|
SetNull();
|
|
nCreateTime = nCreateTime_;
|
|
}
|
|
|
|
SERIALIZE_METHODS(CKeyMetadata, obj)
|
|
{
|
|
READWRITE(obj.nVersion, obj.nCreateTime);
|
|
if (obj.nVersion >= VERSION_WITH_HDDATA) {
|
|
READWRITE(obj.hdKeypath, obj.hd_seed_id);
|
|
}
|
|
if (obj.nVersion >= VERSION_WITH_KEY_ORIGIN)
|
|
{
|
|
READWRITE(obj.key_origin);
|
|
READWRITE(obj.has_key_origin);
|
|
}
|
|
}
|
|
|
|
void SetNull()
|
|
{
|
|
nVersion = CKeyMetadata::CURRENT_VERSION;
|
|
nCreateTime = 0;
|
|
hdKeypath.clear();
|
|
hd_seed_id.SetNull();
|
|
key_origin.clear();
|
|
has_key_origin = false;
|
|
}
|
|
};
|
|
|
|
/** Access to the wallet database.
|
|
* Opens the database and provides read and write access to it. Each read and write is its own transaction.
|
|
* Multiple operation transactions can be started using TxnBegin() and committed using TxnCommit()
|
|
* Otherwise the transaction will be committed when the object goes out of scope.
|
|
* Optionally (on by default) it will flush to disk on close.
|
|
* Every 1000 writes will automatically trigger a flush to disk.
|
|
*/
|
|
class WalletBatch
|
|
{
|
|
private:
|
|
template <typename K, typename T>
|
|
bool WriteIC(const K& key, const T& value, bool fOverwrite = true)
|
|
{
|
|
if (!m_batch->Write(key, value, fOverwrite)) {
|
|
return false;
|
|
}
|
|
m_database.IncrementUpdateCounter();
|
|
if (m_database.nUpdateCounter % 1000 == 0) {
|
|
m_batch->Flush();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename K>
|
|
bool EraseIC(const K& key)
|
|
{
|
|
if (!m_batch->Erase(key)) {
|
|
return false;
|
|
}
|
|
m_database.IncrementUpdateCounter();
|
|
if (m_database.nUpdateCounter % 1000 == 0) {
|
|
m_batch->Flush();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
public:
|
|
explicit WalletBatch(WalletDatabase& database, const char* pszMode = "r+", bool _fFlushOnClose = true) :
|
|
m_batch(database.MakeBatch(pszMode, _fFlushOnClose)),
|
|
m_database(database)
|
|
{
|
|
}
|
|
WalletBatch(const WalletBatch&) = delete;
|
|
WalletBatch& operator=(const WalletBatch&) = delete;
|
|
|
|
bool WriteName(const std::string& strAddress, const std::string& strName);
|
|
bool EraseName(const std::string& strAddress);
|
|
|
|
bool WritePurpose(const std::string& strAddress, const std::string& purpose);
|
|
bool ErasePurpose(const std::string& strAddress);
|
|
|
|
bool WriteTx(const CWalletTx& wtx);
|
|
bool EraseTx(uint256 hash);
|
|
|
|
bool WriteKeyMetadata(const CKeyMetadata& meta, const CPubKey& pubkey, const bool overwrite);
|
|
bool WriteKey(const CPubKey& vchPubKey, const CPrivKey& vchPrivKey, const CKeyMetadata &keyMeta);
|
|
bool WriteCryptedKey(const CPubKey& vchPubKey, const std::vector<unsigned char>& vchCryptedSecret, const CKeyMetadata &keyMeta);
|
|
bool WriteMasterKey(unsigned int nID, const CMasterKey& kMasterKey);
|
|
|
|
bool WriteCScript(const uint160& hash, const CScript& redeemScript);
|
|
|
|
bool WriteWatchOnly(const CScript &script, const CKeyMetadata &keymeta);
|
|
bool EraseWatchOnly(const CScript &script);
|
|
|
|
bool WriteBestBlock(const CBlockLocator& locator);
|
|
bool ReadBestBlock(CBlockLocator& locator);
|
|
|
|
bool WriteOrderPosNext(int64_t nOrderPosNext);
|
|
|
|
bool ReadPool(int64_t nPool, CKeyPool& keypool);
|
|
bool WritePool(int64_t nPool, const CKeyPool& keypool);
|
|
bool ErasePool(int64_t nPool);
|
|
|
|
bool WriteMinVersion(int nVersion);
|
|
|
|
bool WriteDescriptorKey(const uint256& desc_id, const CPubKey& pubkey, const CPrivKey& privkey);
|
|
bool WriteCryptedDescriptorKey(const uint256& desc_id, const CPubKey& pubkey, const std::vector<unsigned char>& secret);
|
|
bool WriteDescriptor(const uint256& desc_id, const WalletDescriptor& descriptor);
|
|
bool WriteDescriptorDerivedCache(const CExtPubKey& xpub, const uint256& desc_id, uint32_t key_exp_index, uint32_t der_index);
|
|
bool WriteDescriptorParentCache(const CExtPubKey& xpub, const uint256& desc_id, uint32_t key_exp_index);
|
|
|
|
/// Write destination data key,value tuple to database
|
|
bool WriteDestData(const std::string &address, const std::string &key, const std::string &value);
|
|
/// Erase destination data tuple from wallet database
|
|
bool EraseDestData(const std::string &address, const std::string &key);
|
|
|
|
bool WriteActiveScriptPubKeyMan(uint8_t type, const uint256& id, bool internal);
|
|
|
|
DBErrors LoadWallet(CWallet* pwallet);
|
|
DBErrors FindWalletTx(std::vector<uint256>& vTxHash, std::list<CWalletTx>& vWtx);
|
|
DBErrors ZapSelectTx(std::vector<uint256>& vHashIn, std::vector<uint256>& vHashOut);
|
|
/* Function to determine if a certain KV/key-type is a key (cryptographical key) type */
|
|
static bool IsKeyType(const std::string& strType);
|
|
|
|
//! write the hdchain model (external chain child index counter)
|
|
bool WriteHDChain(const CHDChain& chain);
|
|
|
|
bool WriteWalletFlags(const uint64_t flags);
|
|
//! Begin a new transaction
|
|
bool TxnBegin();
|
|
//! Commit current transaction
|
|
bool TxnCommit();
|
|
//! Abort current transaction
|
|
bool TxnAbort();
|
|
private:
|
|
std::unique_ptr<DatabaseBatch> m_batch;
|
|
WalletDatabase& m_database;
|
|
};
|
|
|
|
//! Compacts BDB state so that wallet.dat is self-contained (if there are changes)
|
|
void MaybeCompactWalletDB();
|
|
|
|
//! Callback for filtering key types to deserialize in ReadKeyValue
|
|
using KeyFilterFn = std::function<bool(const std::string&)>;
|
|
|
|
//! Unserialize a given Key-Value pair and load it into the wallet
|
|
bool ReadKeyValue(CWallet* pwallet, CDataStream& ssKey, CDataStream& ssValue, std::string& strType, std::string& strErr, const KeyFilterFn& filter_fn = nullptr);
|
|
|
|
/** Return whether a wallet database is currently loaded. */
|
|
bool IsWalletLoaded(const fs::path& wallet_path);
|
|
|
|
/** Return object for accessing database at specified path. */
|
|
std::unique_ptr<WalletDatabase> CreateWalletDatabase(const fs::path& path);
|
|
|
|
/** Return object for accessing dummy database with no read/write capabilities. */
|
|
std::unique_ptr<WalletDatabase> CreateDummyWalletDatabase();
|
|
|
|
/** Return object for accessing temporary in-memory database. */
|
|
std::unique_ptr<WalletDatabase> CreateMockWalletDatabase();
|
|
|
|
#endif // BITCOIN_WALLET_WALLETDB_H
|