/root/bitcoin/src/wallet/walletdb.h
Line | Count | Source |
1 | | // Copyright (c) 2009-2010 Satoshi Nakamoto |
2 | | // Copyright (c) 2009-present The Bitcoin Core developers |
3 | | // Distributed under the MIT software license, see the accompanying |
4 | | // file COPYING or http://www.opensource.org/licenses/mit-license.php. |
5 | | |
6 | | #ifndef BITCOIN_WALLET_WALLETDB_H |
7 | | #define BITCOIN_WALLET_WALLETDB_H |
8 | | |
9 | | #include <key.h> |
10 | | #include <primitives/transaction_identifier.h> |
11 | | #include <script/sign.h> |
12 | | #include <wallet/db.h> |
13 | | #include <wallet/walletutil.h> |
14 | | |
15 | | #include <cstdint> |
16 | | #include <string> |
17 | | #include <vector> |
18 | | |
19 | | class CScript; |
20 | | class uint160; |
21 | | class uint256; |
22 | | struct CBlockLocator; |
23 | | |
24 | | namespace wallet { |
25 | | class CMasterKey; |
26 | | class CWallet; |
27 | | class CWalletTx; |
28 | | struct WalletContext; |
29 | | |
30 | | /** |
31 | | * Overview of wallet database classes: |
32 | | * |
33 | | * - WalletBatch is an abstract modifier object for the wallet database, and encapsulates a database |
34 | | * batch update as well as methods to act on the database. It should be agnostic to the database implementation. |
35 | | */ |
36 | | |
37 | | /** Error statuses for the wallet database. |
38 | | * Values are in order of severity. When multiple errors occur, the most severe (highest value) will be returned. |
39 | | */ |
40 | | enum class DBErrors : int |
41 | | { |
42 | | LOAD_OK = 0, |
43 | | NEED_RESCAN = 1, |
44 | | NEED_REWRITE = 2, |
45 | | EXTERNAL_SIGNER_SUPPORT_REQUIRED = 3, |
46 | | NONCRITICAL_ERROR = 4, |
47 | | TOO_NEW = 5, |
48 | | UNKNOWN_DESCRIPTOR = 6, |
49 | | LOAD_FAIL = 7, |
50 | | UNEXPECTED_LEGACY_ENTRY = 8, |
51 | | LEGACY_WALLET = 9, |
52 | | CORRUPT = 10, |
53 | | }; |
54 | | |
55 | | namespace DBKeys { |
56 | | extern const std::string ACENTRY; |
57 | | extern const std::string ACTIVEEXTERNALSPK; |
58 | | extern const std::string ACTIVEINTERNALSPK; |
59 | | extern const std::string BESTBLOCK; |
60 | | extern const std::string BESTBLOCK_NOMERKLE; |
61 | | extern const std::string CRYPTED_KEY; |
62 | | extern const std::string CSCRIPT; |
63 | | extern const std::string DEFAULTKEY; |
64 | | extern const std::string DESTDATA; |
65 | | extern const std::string FLAGS; |
66 | | extern const std::string HDCHAIN; |
67 | | extern const std::string KEY; |
68 | | extern const std::string KEYMETA; |
69 | | extern const std::string LOCKED_UTXO; |
70 | | extern const std::string MASTER_KEY; |
71 | | extern const std::string MINVERSION; |
72 | | extern const std::string NAME; |
73 | | extern const std::string OLD_KEY; |
74 | | extern const std::string ORDERPOSNEXT; |
75 | | extern const std::string POOL; |
76 | | extern const std::string PURPOSE; |
77 | | extern const std::string SETTINGS; |
78 | | extern const std::string TX; |
79 | | extern const std::string VERSION; |
80 | | extern const std::string WALLETDESCRIPTOR; |
81 | | extern const std::string WALLETDESCRIPTORCKEY; |
82 | | extern const std::string WALLETDESCRIPTORKEY; |
83 | | extern const std::string WATCHMETA; |
84 | | extern const std::string WATCHS; |
85 | | |
86 | | // Keys in this set pertain only to the legacy wallet (LegacyScriptPubKeyMan) and are removed during migration from legacy to descriptors. |
87 | | extern const std::unordered_set<std::string> LEGACY_TYPES; |
88 | | } // namespace DBKeys |
89 | | |
90 | | /* simple HD chain data model */ |
91 | | class CHDChain |
92 | | { |
93 | | public: |
94 | | uint32_t nExternalChainCounter; |
95 | | uint32_t nInternalChainCounter; |
96 | | CKeyID seed_id; //!< seed hash160 |
97 | | int64_t m_next_external_index{0}; // Next index in the keypool to be used. Memory only. |
98 | | int64_t m_next_internal_index{0}; // Next index in the keypool to be used. Memory only. |
99 | | |
100 | | static const int VERSION_HD_BASE = 1; |
101 | | static const int VERSION_HD_CHAIN_SPLIT = 2; |
102 | | static const int CURRENT_VERSION = VERSION_HD_CHAIN_SPLIT; |
103 | | int nVersion; |
104 | | |
105 | 0 | CHDChain() { SetNull(); } |
106 | | |
107 | | SERIALIZE_METHODS(CHDChain, obj) |
108 | 0 | { |
109 | 0 | READWRITE(obj.nVersion, obj.nExternalChainCounter, obj.seed_id); |
110 | 0 | if (obj.nVersion >= VERSION_HD_CHAIN_SPLIT) { |
111 | 0 | READWRITE(obj.nInternalChainCounter); |
112 | 0 | } |
113 | 0 | } |
114 | | |
115 | | void SetNull() |
116 | 0 | { |
117 | 0 | nVersion = CHDChain::CURRENT_VERSION; |
118 | 0 | nExternalChainCounter = 0; |
119 | 0 | nInternalChainCounter = 0; |
120 | 0 | seed_id.SetNull(); |
121 | 0 | } |
122 | | |
123 | | bool operator==(const CHDChain& chain) const |
124 | 0 | { |
125 | 0 | return seed_id == chain.seed_id; |
126 | 0 | } |
127 | | }; |
128 | | |
129 | | class CKeyMetadata |
130 | | { |
131 | | public: |
132 | | static const int VERSION_BASIC=1; |
133 | | static const int VERSION_WITH_HDDATA=10; |
134 | | static const int VERSION_WITH_KEY_ORIGIN = 12; |
135 | | static const int CURRENT_VERSION=VERSION_WITH_KEY_ORIGIN; |
136 | | int nVersion; |
137 | | int64_t nCreateTime; // 0 means unknown |
138 | | std::string hdKeypath; //optional HD/bip32 keypath. Still used to determine whether a key is a seed. Also kept for backwards compatibility |
139 | | CKeyID hd_seed_id; //id of the HD seed used to derive this key |
140 | | KeyOriginInfo key_origin; // Key origin info with path and fingerprint |
141 | | bool has_key_origin = false; //!< Whether the key_origin is useful |
142 | | |
143 | | CKeyMetadata() |
144 | 0 | { |
145 | 0 | SetNull(); |
146 | 0 | } |
147 | | explicit CKeyMetadata(int64_t nCreateTime_) |
148 | 0 | { |
149 | 0 | SetNull(); |
150 | 0 | nCreateTime = nCreateTime_; |
151 | 0 | } |
152 | | |
153 | | SERIALIZE_METHODS(CKeyMetadata, obj) |
154 | 0 | { |
155 | 0 | READWRITE(obj.nVersion, obj.nCreateTime); |
156 | 0 | if (obj.nVersion >= VERSION_WITH_HDDATA) { |
157 | 0 | READWRITE(obj.hdKeypath, obj.hd_seed_id); |
158 | 0 | } |
159 | 0 | if (obj.nVersion >= VERSION_WITH_KEY_ORIGIN) |
160 | 0 | { |
161 | 0 | READWRITE(obj.key_origin); |
162 | 0 | READWRITE(obj.has_key_origin); |
163 | 0 | } |
164 | 0 | } Unexecuted instantiation: _ZN6wallet12CKeyMetadata16SerializationOpsI10DataStreamS0_17ActionUnserializeEEvRT0_RT_T1_ Unexecuted instantiation: _ZN6wallet12CKeyMetadata16SerializationOpsI10DataStreamKS0_15ActionSerializeEEvRT0_RT_T1_ |
165 | | |
166 | | void SetNull() |
167 | 0 | { |
168 | 0 | nVersion = CKeyMetadata::CURRENT_VERSION; |
169 | 0 | nCreateTime = 0; |
170 | 0 | hdKeypath.clear(); |
171 | 0 | hd_seed_id.SetNull(); |
172 | 0 | key_origin.clear(); |
173 | 0 | has_key_origin = false; |
174 | 0 | } |
175 | | }; |
176 | | |
177 | | struct DbTxnListener |
178 | | { |
179 | | std::function<void()> on_commit, on_abort; |
180 | | }; |
181 | | |
182 | | /** Access to the wallet database. |
183 | | * Opens the database and provides read and write access to it. Each read and write is its own transaction. |
184 | | * Multiple operation transactions can be started using TxnBegin() and committed using TxnCommit() |
185 | | * Otherwise the transaction will be committed when the object goes out of scope. |
186 | | * Optionally (on by default) it will flush to disk on close. |
187 | | * Every 1000 writes will automatically trigger a flush to disk. |
188 | | */ |
189 | | class WalletBatch |
190 | | { |
191 | | private: |
192 | | template <typename K, typename T> |
193 | | bool WriteIC(const K& key, const T& value, bool fOverwrite = true) |
194 | 0 | { |
195 | 0 | if (!m_batch->Write(key, value, fOverwrite)) { |
196 | 0 | return false; |
197 | 0 | } |
198 | 0 | return true; |
199 | 0 | } Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES8_ES8_EEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE22transaction_identifierILb0EEENS_9CWalletTxEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CPubKeyENS_12CKeyMetadataEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CPubKeyES2_ISt6vectorIh16secure_allocatorIhEE7uint256EEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CPubKeyES2_ISt6vectorIhSaIhEE7uint256EEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEjENS_10CMasterKeyEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CScriptENS_12CKeyMetadataEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CScriptEhEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE13CBlockLocatorEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEElEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEhE7uint256EEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_I7uint2567CPubKeyEES2_ISt6vectorIh16secure_allocatorIhEES9_EEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_I7uint2567CPubKeyEESt6vectorIhSaIhEEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7uint256ENS_16WalletDescriptorEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairIS2_INSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7uint256ES2_IjjEESt6vectorIhSaIhEEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairIS2_INSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7uint256EjESt6vectorIhSaIhEEEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_I22transaction_identifierILb0EEjEEhEEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_IS8_S8_EES8_EEbRKT_RKT0_b Unexecuted instantiation: _ZN6wallet11WalletBatch7WriteICINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEmEEbRKT_RKT0_b |
200 | | |
201 | | template <typename K> |
202 | | bool EraseIC(const K& key) |
203 | 0 | { |
204 | 0 | if (!m_batch->Erase(key)) { |
205 | 0 | return false; |
206 | 0 | } |
207 | 0 | return true; |
208 | 0 | } Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES8_EEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7uint256EEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CPubKeyEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEjEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE7CScriptEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEhEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_I7uint2567CPubKeyEEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_I22transaction_identifierILb0EEjEEEEbRKT_ Unexecuted instantiation: _ZN6wallet11WalletBatch7EraseICISt4pairINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEES2_IS8_S8_EEEEbRKT_ |
209 | | |
210 | | public: |
211 | | explicit WalletBatch(WalletDatabase &database) : |
212 | 0 | m_batch(database.MakeBatch()) |
213 | 0 | { |
214 | 0 | } |
215 | | WalletBatch(const WalletBatch&) = delete; |
216 | | WalletBatch& operator=(const WalletBatch&) = delete; |
217 | | |
218 | | bool WriteName(const std::string& strAddress, const std::string& strName); |
219 | | bool EraseName(const std::string& strAddress); |
220 | | |
221 | | bool WritePurpose(const std::string& strAddress, const std::string& purpose); |
222 | | bool ErasePurpose(const std::string& strAddress); |
223 | | |
224 | | bool WriteTx(const CWalletTx& wtx); |
225 | | bool EraseTx(Txid hash); |
226 | | |
227 | | bool WriteKeyMetadata(const CKeyMetadata& meta, const CPubKey& pubkey, const bool overwrite); |
228 | | bool WriteKey(const CPubKey& vchPubKey, const CPrivKey& vchPrivKey, const CKeyMetadata &keyMeta); |
229 | | bool WriteCryptedKey(const CPubKey& vchPubKey, const std::vector<unsigned char>& vchCryptedSecret, const CKeyMetadata &keyMeta); |
230 | | bool WriteMasterKey(unsigned int nID, const CMasterKey& kMasterKey); |
231 | | bool EraseMasterKey(unsigned int id); |
232 | | |
233 | | bool WriteWatchOnly(const CScript &script, const CKeyMetadata &keymeta); |
234 | | bool EraseWatchOnly(const CScript &script); |
235 | | |
236 | | bool WriteBestBlock(const CBlockLocator& locator); |
237 | | bool ReadBestBlock(CBlockLocator& locator); |
238 | | |
239 | | // Returns true if wallet stores encryption keys |
240 | | bool IsEncrypted(); |
241 | | |
242 | | bool WriteOrderPosNext(int64_t nOrderPosNext); |
243 | | |
244 | | bool WriteDescriptorKey(const uint256& desc_id, const CPubKey& pubkey, const CPrivKey& privkey); |
245 | | bool WriteCryptedDescriptorKey(const uint256& desc_id, const CPubKey& pubkey, const std::vector<unsigned char>& secret); |
246 | | bool WriteDescriptor(const uint256& desc_id, const WalletDescriptor& descriptor); |
247 | | bool WriteDescriptorDerivedCache(const CExtPubKey& xpub, const uint256& desc_id, uint32_t key_exp_index, uint32_t der_index); |
248 | | bool WriteDescriptorParentCache(const CExtPubKey& xpub, const uint256& desc_id, uint32_t key_exp_index); |
249 | | bool WriteDescriptorLastHardenedCache(const CExtPubKey& xpub, const uint256& desc_id, uint32_t key_exp_index); |
250 | | bool WriteDescriptorCacheItems(const uint256& desc_id, const DescriptorCache& cache); |
251 | | |
252 | | bool WriteLockedUTXO(const COutPoint& output); |
253 | | bool EraseLockedUTXO(const COutPoint& output); |
254 | | |
255 | | bool WriteAddressPreviouslySpent(const CTxDestination& dest, bool previously_spent); |
256 | | bool WriteAddressReceiveRequest(const CTxDestination& dest, const std::string& id, const std::string& receive_request); |
257 | | bool EraseAddressReceiveRequest(const CTxDestination& dest, const std::string& id); |
258 | | bool EraseAddressData(const CTxDestination& dest); |
259 | | |
260 | | bool WriteActiveScriptPubKeyMan(uint8_t type, const uint256& id, bool internal); |
261 | | bool EraseActiveScriptPubKeyMan(uint8_t type, bool internal); |
262 | | |
263 | | DBErrors LoadWallet(CWallet* pwallet); |
264 | | |
265 | | //! Delete records of the given types |
266 | | bool EraseRecords(const std::unordered_set<std::string>& types); |
267 | | |
268 | | bool WriteWalletFlags(const uint64_t flags); |
269 | | //! Begin a new transaction |
270 | | bool TxnBegin(); |
271 | | //! Commit current transaction |
272 | | bool TxnCommit(); |
273 | | //! Abort current transaction |
274 | | bool TxnAbort(); |
275 | 0 | bool HasActiveTxn() { return m_batch->HasActiveTxn(); } |
276 | | |
277 | | //! Registers db txn callback functions |
278 | | void RegisterTxnListener(const DbTxnListener& l); |
279 | | |
280 | | private: |
281 | | std::unique_ptr<DatabaseBatch> m_batch; |
282 | | |
283 | | // External functions listening to the current db txn outcome. |
284 | | // Listeners are cleared at the end of the transaction. |
285 | | std::vector<DbTxnListener> m_txn_listeners; |
286 | | }; |
287 | | |
288 | | /** |
289 | | * Executes the provided function 'func' within a database transaction context. |
290 | | * |
291 | | * This function ensures that all db modifications performed within 'func()' are |
292 | | * atomically committed to the db at the end of the process. And, in case of a |
293 | | * failure during execution, all performed changes are rolled back. |
294 | | * |
295 | | * @param database The db connection instance to perform the transaction on. |
296 | | * @param process_desc A description of the process being executed, used for logging purposes in the event of a failure. |
297 | | * @param func The function to be executed within the db txn context. It returns a boolean indicating whether to commit or roll back the txn. |
298 | | * @return true if the db txn executed successfully, false otherwise. |
299 | | */ |
300 | | bool RunWithinTxn(WalletDatabase& database, std::string_view process_desc, const std::function<bool(WalletBatch&)>& func); |
301 | | |
302 | | bool LoadKey(CWallet* pwallet, DataStream& ssKey, DataStream& ssValue, std::string& strErr); |
303 | | bool LoadCryptedKey(CWallet* pwallet, DataStream& ssKey, DataStream& ssValue, std::string& strErr); |
304 | | bool LoadEncryptionKey(CWallet* pwallet, DataStream& ssKey, DataStream& ssValue, std::string& strErr); |
305 | | bool LoadHDChain(CWallet* pwallet, DataStream& ssValue, std::string& strErr); |
306 | | |
307 | | //! Returns true if there are any DBKeys::LEGACY_TYPES record in the wallet db |
308 | | bool HasLegacyRecords(CWallet& wallet); |
309 | | bool HasLegacyRecords(CWallet& wallet, DatabaseBatch& batch); |
310 | | } // namespace wallet |
311 | | |
312 | | #endif // BITCOIN_WALLET_WALLETDB_H |