Line data Source code
1 : // Copyright (c) 2018-2022 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 : #include <interfaces/wallet.h>
6 :
7 : #include <common/args.h>
8 : #include <consensus/amount.h>
9 : #include <interfaces/chain.h>
10 : #include <interfaces/handler.h>
11 : #include <policy/fees.h>
12 : #include <primitives/transaction.h>
13 : #include <rpc/server.h>
14 : #include <support/allocators/secure.h>
15 : #include <sync.h>
16 : #include <uint256.h>
17 2 : #include <util/check.h>
18 2 : #include <util/translation.h>
19 : #include <util/ui_change_type.h>
20 : #include <wallet/coincontrol.h>
21 : #include <wallet/context.h>
22 : #include <wallet/feebumper.h>
23 : #include <wallet/fees.h>
24 : #include <wallet/types.h>
25 : #include <wallet/load.h>
26 : #include <wallet/receive.h>
27 2 : #include <wallet/rpc/wallet.h>
28 : #include <wallet/spend.h>
29 : #include <wallet/wallet.h>
30 :
31 : #include <memory>
32 : #include <string>
33 : #include <utility>
34 : #include <vector>
35 :
36 : using interfaces::Chain;
37 : using interfaces::FoundBlock;
38 : using interfaces::Handler;
39 : using interfaces::MakeSignalHandler;
40 : using interfaces::Wallet;
41 : using interfaces::WalletAddress;
42 : using interfaces::WalletBalances;
43 : using interfaces::WalletLoader;
44 : using interfaces::WalletMigrationResult;
45 : using interfaces::WalletOrderForm;
46 : using interfaces::WalletTx;
47 : using interfaces::WalletTxOut;
48 : using interfaces::WalletTxStatus;
49 : using interfaces::WalletValueMap;
50 :
51 : namespace wallet {
52 : // All members of the classes in this namespace are intentionally public, as the
53 0 : // classes themselves are private.
54 : namespace {
55 : //! Construct wallet tx struct.
56 0 : WalletTx MakeWalletTx(CWallet& wallet, const CWalletTx& wtx)
57 : {
58 0 : LOCK(wallet.cs_wallet);
59 0 : WalletTx result;
60 0 : result.tx = wtx.tx;
61 0 : result.txin_is_mine.reserve(wtx.tx->vin.size());
62 0 : for (const auto& txin : wtx.tx->vin) {
63 0 : result.txin_is_mine.emplace_back(InputIsMine(wallet, txin));
64 : }
65 0 : result.txout_is_mine.reserve(wtx.tx->vout.size());
66 0 : result.txout_address.reserve(wtx.tx->vout.size());
67 0 : result.txout_address_is_mine.reserve(wtx.tx->vout.size());
68 0 : for (const auto& txout : wtx.tx->vout) {
69 0 : result.txout_is_mine.emplace_back(wallet.IsMine(txout));
70 0 : result.txout_address.emplace_back();
71 0 : result.txout_address_is_mine.emplace_back(ExtractDestination(txout.scriptPubKey, result.txout_address.back()) ?
72 0 : wallet.IsMine(result.txout_address.back()) :
73 : ISMINE_NO);
74 2 : }
75 0 : result.credit = CachedTxGetCredit(wallet, wtx, ISMINE_ALL);
76 0 : result.debit = CachedTxGetDebit(wallet, wtx, ISMINE_ALL);
77 0 : result.change = CachedTxGetChange(wallet, wtx);
78 0 : result.time = wtx.GetTxTime();
79 0 : result.value_map = wtx.mapValue;
80 0 : result.is_coinbase = wtx.IsCoinBase();
81 0 : return result;
82 0 : }
83 :
84 : //! Construct wallet tx status struct.
85 0 : WalletTxStatus MakeWalletTxStatus(const CWallet& wallet, const CWalletTx& wtx)
86 : EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
87 : {
88 0 : AssertLockHeld(wallet.cs_wallet);
89 :
90 : WalletTxStatus result;
91 2 : result.block_height =
92 0 : wtx.state<TxStateConfirmed>() ? wtx.state<TxStateConfirmed>()->confirmed_block_height :
93 0 : wtx.state<TxStateConflicted>() ? wtx.state<TxStateConflicted>()->conflicting_block_height :
94 0 : std::numeric_limits<int>::max();
95 0 : result.blocks_to_maturity = wallet.GetTxBlocksToMaturity(wtx);
96 0 : result.depth_in_main_chain = wallet.GetTxDepthInMainChain(wtx);
97 0 : result.time_received = wtx.nTimeReceived;
98 0 : result.lock_time = wtx.tx->nLockTime;
99 2 : result.is_trusted = CachedTxIsTrusted(wallet, wtx);
100 0 : result.is_abandoned = wtx.isAbandoned();
101 0 : result.is_coinbase = wtx.IsCoinBase();
102 0 : result.is_in_main_chain = wallet.IsTxInMainChain(wtx);
103 0 : return result;
104 : }
105 :
106 : //! Construct wallet TxOut struct.
107 0 : WalletTxOut MakeWalletTxOut(const CWallet& wallet,
108 : const CWalletTx& wtx,
109 : int n,
110 : int depth) EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
111 : {
112 0 : WalletTxOut result;
113 0 : result.txout = wtx.tx->vout[n];
114 0 : result.time = wtx.GetTxTime();
115 0 : result.depth_in_main_chain = depth;
116 0 : result.is_spent = wallet.IsSpent(COutPoint(wtx.GetHash(), n));
117 0 : return result;
118 0 : }
119 :
120 0 : WalletTxOut MakeWalletTxOut(const CWallet& wallet,
121 : const COutput& output) EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
122 : {
123 0 : WalletTxOut result;
124 0 : result.txout = output.txout;
125 0 : result.time = output.time;
126 0 : result.depth_in_main_chain = output.depth;
127 0 : result.is_spent = wallet.IsSpent(output.outpoint);
128 0 : return result;
129 0 : }
130 :
131 : class WalletImpl : public Wallet
132 : {
133 : public:
134 0 : explicit WalletImpl(WalletContext& context, const std::shared_ptr<CWallet>& wallet) : m_context(context), m_wallet(wallet) {}
135 :
136 0 : bool encryptWallet(const SecureString& wallet_passphrase) override
137 : {
138 0 : return m_wallet->EncryptWallet(wallet_passphrase);
139 : }
140 0 : bool isCrypted() override { return m_wallet->IsCrypted(); }
141 0 : bool lock() override { return m_wallet->Lock(); }
142 0 : bool unlock(const SecureString& wallet_passphrase) override { return m_wallet->Unlock(wallet_passphrase); }
143 0 : bool isLocked() override { return m_wallet->IsLocked(); }
144 0 : bool changeWalletPassphrase(const SecureString& old_wallet_passphrase,
145 : const SecureString& new_wallet_passphrase) override
146 : {
147 0 : return m_wallet->ChangeWalletPassphrase(old_wallet_passphrase, new_wallet_passphrase);
148 : }
149 0 : void abortRescan() override { m_wallet->AbortRescan(); }
150 0 : bool backupWallet(const std::string& filename) override { return m_wallet->BackupWallet(filename); }
151 0 : std::string getWalletName() override { return m_wallet->GetName(); }
152 0 : util::Result<CTxDestination> getNewDestination(const OutputType type, const std::string& label) override
153 : {
154 0 : LOCK(m_wallet->cs_wallet);
155 0 : return m_wallet->GetNewDestination(type, label);
156 0 : }
157 0 : bool getPubKey(const CScript& script, const CKeyID& address, CPubKey& pub_key) override
158 : {
159 0 : std::unique_ptr<SigningProvider> provider = m_wallet->GetSolvingProvider(script);
160 0 : if (provider) {
161 0 : return provider->GetPubKey(address, pub_key);
162 : }
163 2 : return false;
164 2 : }
165 2 : SigningResult signMessage(const std::string& message, const PKHash& pkhash, std::string& str_sig) override
166 2 : {
167 2 : return m_wallet->SignMessage(message, pkhash, str_sig);
168 2 : }
169 2 : bool isSpendable(const CTxDestination& dest) override
170 2 : {
171 0 : LOCK(m_wallet->cs_wallet);
172 0 : return m_wallet->IsMine(dest) & ISMINE_SPENDABLE;
173 0 : }
174 0 : bool haveWatchOnly() override
175 : {
176 0 : auto spk_man = m_wallet->GetLegacyScriptPubKeyMan();
177 0 : if (spk_man) {
178 0 : return spk_man->HaveWatchOnly();
179 : }
180 0 : return false;
181 0 : };
182 0 : bool setAddressBook(const CTxDestination& dest, const std::string& name, const std::optional<AddressPurpose>& purpose) override
183 : {
184 0 : return m_wallet->SetAddressBook(dest, name, purpose);
185 : }
186 0 : bool delAddressBook(const CTxDestination& dest) override
187 : {
188 0 : return m_wallet->DelAddressBook(dest);
189 : }
190 0 : bool getAddress(const CTxDestination& dest,
191 : std::string* name,
192 : isminetype* is_mine,
193 : AddressPurpose* purpose) override
194 : {
195 0 : LOCK(m_wallet->cs_wallet);
196 0 : const auto& entry = m_wallet->FindAddressBookEntry(dest, /*allow_change=*/false);
197 0 : if (!entry) return false; // addr not found
198 0 : if (name) {
199 0 : *name = entry->GetLabel();
200 0 : }
201 0 : std::optional<isminetype> dest_is_mine;
202 0 : if (is_mine || purpose) {
203 0 : dest_is_mine = m_wallet->IsMine(dest);
204 0 : }
205 0 : if (is_mine) {
206 0 : *is_mine = *dest_is_mine;
207 0 : }
208 0 : if (purpose) {
209 : // In very old wallets, address purpose may not be recorded so we derive it from IsMine
210 0 : *purpose = entry->purpose.value_or(*dest_is_mine ? AddressPurpose::RECEIVE : AddressPurpose::SEND);
211 0 : }
212 0 : return true;
213 0 : }
214 0 : std::vector<WalletAddress> getAddresses() const override
215 : {
216 0 : LOCK(m_wallet->cs_wallet);
217 0 : std::vector<WalletAddress> result;
218 0 : m_wallet->ForEachAddrBookEntry([&](const CTxDestination& dest, const std::string& label, bool is_change, const std::optional<AddressPurpose>& purpose) EXCLUSIVE_LOCKS_REQUIRED(m_wallet->cs_wallet) {
219 0 : if (is_change) return;
220 0 : isminetype is_mine = m_wallet->IsMine(dest);
221 : // In very old wallets, address purpose may not be recorded so we derive it from IsMine
222 0 : result.emplace_back(dest, is_mine, purpose.value_or(is_mine ? AddressPurpose::RECEIVE : AddressPurpose::SEND), label);
223 0 : });
224 0 : return result;
225 0 : }
226 0 : std::vector<std::string> getAddressReceiveRequests() override {
227 0 : LOCK(m_wallet->cs_wallet);
228 0 : return m_wallet->GetAddressReceiveRequests();
229 0 : }
230 0 : bool setAddressReceiveRequest(const CTxDestination& dest, const std::string& id, const std::string& value) override {
231 : // Note: The setAddressReceiveRequest interface used by the GUI to store
232 : // receive requests is a little awkward and could be improved in the
233 : // future:
234 : //
235 : // - The same method is used to save requests and erase them, but
236 : // having separate methods could be clearer and prevent bugs.
237 : //
238 : // - Request ids are passed as strings even though they are generated as
239 : // integers.
240 : //
241 : // - Multiple requests can be stored for the same address, but it might
242 : // be better to only allow one request or only keep the current one.
243 0 : LOCK(m_wallet->cs_wallet);
244 0 : WalletBatch batch{m_wallet->GetDatabase()};
245 0 : return value.empty() ? m_wallet->EraseAddressReceiveRequest(batch, dest, id)
246 0 : : m_wallet->SetAddressReceiveRequest(batch, dest, id, value);
247 0 : }
248 0 : bool displayAddress(const CTxDestination& dest) override
249 : {
250 0 : LOCK(m_wallet->cs_wallet);
251 0 : return m_wallet->DisplayAddress(dest);
252 0 : }
253 0 : bool lockCoin(const COutPoint& output, const bool write_to_db) override
254 : {
255 0 : LOCK(m_wallet->cs_wallet);
256 0 : std::unique_ptr<WalletBatch> batch = write_to_db ? std::make_unique<WalletBatch>(m_wallet->GetDatabase()) : nullptr;
257 0 : return m_wallet->LockCoin(output, batch.get());
258 0 : }
259 0 : bool unlockCoin(const COutPoint& output) override
260 : {
261 0 : LOCK(m_wallet->cs_wallet);
262 0 : std::unique_ptr<WalletBatch> batch = std::make_unique<WalletBatch>(m_wallet->GetDatabase());
263 0 : return m_wallet->UnlockCoin(output, batch.get());
264 0 : }
265 0 : bool isLockedCoin(const COutPoint& output) override
266 : {
267 0 : LOCK(m_wallet->cs_wallet);
268 0 : return m_wallet->IsLockedCoin(output);
269 0 : }
270 2 : void listLockedCoins(std::vector<COutPoint>& outputs) override
271 : {
272 0 : LOCK(m_wallet->cs_wallet);
273 0 : return m_wallet->ListLockedCoins(outputs);
274 0 : }
275 0 : util::Result<CTransactionRef> createTransaction(const std::vector<CRecipient>& recipients,
276 : const CCoinControl& coin_control,
277 : bool sign,
278 : int& change_pos,
279 : CAmount& fee) override
280 : {
281 0 : LOCK(m_wallet->cs_wallet);
282 0 : auto res = CreateTransaction(*m_wallet, recipients, change_pos,
283 0 : coin_control, sign);
284 0 : if (!res) return util::Error{util::ErrorString(res)};
285 0 : const auto& txr = *res;
286 0 : fee = txr.fee;
287 0 : change_pos = txr.change_pos;
288 :
289 0 : return txr.tx;
290 0 : }
291 0 : void commitTransaction(CTransactionRef tx,
292 : WalletValueMap value_map,
293 : WalletOrderForm order_form) override
294 : {
295 0 : LOCK(m_wallet->cs_wallet);
296 0 : m_wallet->CommitTransaction(std::move(tx), std::move(value_map), std::move(order_form));
297 0 : }
298 0 : bool transactionCanBeAbandoned(const uint256& txid) override { return m_wallet->TransactionCanBeAbandoned(txid); }
299 0 : bool abandonTransaction(const uint256& txid) override
300 : {
301 0 : LOCK(m_wallet->cs_wallet);
302 0 : return m_wallet->AbandonTransaction(txid);
303 0 : }
304 0 : bool transactionCanBeBumped(const uint256& txid) override
305 : {
306 0 : return feebumper::TransactionCanBeBumped(*m_wallet.get(), txid);
307 : }
308 0 : bool createBumpTransaction(const uint256& txid,
309 : const CCoinControl& coin_control,
310 : std::vector<bilingual_str>& errors,
311 : CAmount& old_fee,
312 : CAmount& new_fee,
313 : CMutableTransaction& mtx) override
314 : {
315 0 : std::vector<CTxOut> outputs; // just an empty list of new recipients for now
316 0 : return feebumper::CreateRateBumpTransaction(*m_wallet.get(), txid, coin_control, errors, old_fee, new_fee, mtx, /* require_mine= */ true, outputs) == feebumper::Result::OK;
317 0 : }
318 0 : bool signBumpTransaction(CMutableTransaction& mtx) override { return feebumper::SignTransaction(*m_wallet.get(), mtx); }
319 0 : bool commitBumpTransaction(const uint256& txid,
320 : CMutableTransaction&& mtx,
321 : std::vector<bilingual_str>& errors,
322 : uint256& bumped_txid) override
323 : {
324 0 : return feebumper::CommitTransaction(*m_wallet.get(), txid, std::move(mtx), errors, bumped_txid) ==
325 : feebumper::Result::OK;
326 : }
327 0 : CTransactionRef getTx(const uint256& txid) override
328 : {
329 0 : LOCK(m_wallet->cs_wallet);
330 0 : auto mi = m_wallet->mapWallet.find(txid);
331 0 : if (mi != m_wallet->mapWallet.end()) {
332 0 : return mi->second.tx;
333 : }
334 0 : return {};
335 0 : }
336 0 : WalletTx getWalletTx(const uint256& txid) override
337 : {
338 0 : LOCK(m_wallet->cs_wallet);
339 0 : auto mi = m_wallet->mapWallet.find(txid);
340 0 : if (mi != m_wallet->mapWallet.end()) {
341 0 : return MakeWalletTx(*m_wallet, mi->second);
342 : }
343 0 : return {};
344 0 : }
345 0 : std::set<WalletTx> getWalletTxs() override
346 : {
347 0 : LOCK(m_wallet->cs_wallet);
348 0 : std::set<WalletTx> result;
349 0 : for (const auto& entry : m_wallet->mapWallet) {
350 0 : result.emplace(MakeWalletTx(*m_wallet, entry.second));
351 : }
352 0 : return result;
353 0 : }
354 0 : bool tryGetTxStatus(const uint256& txid,
355 : interfaces::WalletTxStatus& tx_status,
356 : int& num_blocks,
357 : int64_t& block_time) override
358 : {
359 0 : TRY_LOCK(m_wallet->cs_wallet, locked_wallet);
360 0 : if (!locked_wallet) {
361 0 : return false;
362 : }
363 0 : auto mi = m_wallet->mapWallet.find(txid);
364 0 : if (mi == m_wallet->mapWallet.end()) {
365 0 : return false;
366 : }
367 0 : num_blocks = m_wallet->GetLastBlockHeight();
368 0 : block_time = -1;
369 0 : CHECK_NONFATAL(m_wallet->chain().findBlock(m_wallet->GetLastBlockHash(), FoundBlock().time(block_time)));
370 0 : tx_status = MakeWalletTxStatus(*m_wallet, mi->second);
371 0 : return true;
372 0 : }
373 0 : WalletTx getWalletTxDetails(const uint256& txid,
374 : WalletTxStatus& tx_status,
375 : WalletOrderForm& order_form,
376 : bool& in_mempool,
377 : int& num_blocks) override
378 : {
379 0 : LOCK(m_wallet->cs_wallet);
380 0 : auto mi = m_wallet->mapWallet.find(txid);
381 0 : if (mi != m_wallet->mapWallet.end()) {
382 0 : num_blocks = m_wallet->GetLastBlockHeight();
383 0 : in_mempool = mi->second.InMempool();
384 0 : order_form = mi->second.vOrderForm;
385 0 : tx_status = MakeWalletTxStatus(*m_wallet, mi->second);
386 0 : return MakeWalletTx(*m_wallet, mi->second);
387 : }
388 0 : return {};
389 0 : }
390 0 : TransactionError fillPSBT(int sighash_type,
391 : bool sign,
392 : bool bip32derivs,
393 : size_t* n_signed,
394 : PartiallySignedTransaction& psbtx,
395 : bool& complete) override
396 : {
397 0 : return m_wallet->FillPSBT(psbtx, complete, sighash_type, sign, bip32derivs, n_signed);
398 : }
399 0 : WalletBalances getBalances() override
400 : {
401 0 : const auto bal = GetBalance(*m_wallet);
402 0 : WalletBalances result;
403 0 : result.balance = bal.m_mine_trusted;
404 0 : result.unconfirmed_balance = bal.m_mine_untrusted_pending;
405 0 : result.immature_balance = bal.m_mine_immature;
406 0 : result.have_watch_only = haveWatchOnly();
407 0 : if (result.have_watch_only) {
408 0 : result.watch_only_balance = bal.m_watchonly_trusted;
409 0 : result.unconfirmed_watch_only_balance = bal.m_watchonly_untrusted_pending;
410 0 : result.immature_watch_only_balance = bal.m_watchonly_immature;
411 0 : }
412 0 : return result;
413 : }
414 0 : bool tryGetBalances(WalletBalances& balances, uint256& block_hash) override
415 : {
416 0 : TRY_LOCK(m_wallet->cs_wallet, locked_wallet);
417 0 : if (!locked_wallet) {
418 0 : return false;
419 : }
420 0 : block_hash = m_wallet->GetLastBlockHash();
421 0 : balances = getBalances();
422 0 : return true;
423 0 : }
424 0 : CAmount getBalance() override { return GetBalance(*m_wallet).m_mine_trusted; }
425 0 : CAmount getAvailableBalance(const CCoinControl& coin_control) override
426 : {
427 0 : LOCK(m_wallet->cs_wallet);
428 0 : CAmount total_amount = 0;
429 : // Fetch selected coins total amount
430 0 : if (coin_control.HasSelected()) {
431 0 : FastRandomContext rng{};
432 0 : CoinSelectionParams params(rng);
433 : // Note: for now, swallow any error.
434 0 : if (auto res = FetchSelectedInputs(*m_wallet, coin_control, params)) {
435 0 : total_amount += res->total_amount;
436 0 : }
437 0 : }
438 :
439 : // And fetch the wallet available coins
440 0 : if (coin_control.m_allow_other_inputs) {
441 0 : total_amount += AvailableCoins(*m_wallet, &coin_control).GetTotalAmount();
442 0 : }
443 :
444 0 : return total_amount;
445 0 : }
446 0 : isminetype txinIsMine(const CTxIn& txin) override
447 : {
448 0 : LOCK(m_wallet->cs_wallet);
449 0 : return InputIsMine(*m_wallet, txin);
450 0 : }
451 0 : isminetype txoutIsMine(const CTxOut& txout) override
452 : {
453 0 : LOCK(m_wallet->cs_wallet);
454 0 : return m_wallet->IsMine(txout);
455 0 : }
456 0 : CAmount getDebit(const CTxIn& txin, isminefilter filter) override
457 : {
458 0 : LOCK(m_wallet->cs_wallet);
459 0 : return m_wallet->GetDebit(txin, filter);
460 0 : }
461 0 : CAmount getCredit(const CTxOut& txout, isminefilter filter) override
462 : {
463 0 : LOCK(m_wallet->cs_wallet);
464 0 : return OutputGetCredit(*m_wallet, txout, filter);
465 0 : }
466 0 : CoinsList listCoins() override
467 : {
468 0 : LOCK(m_wallet->cs_wallet);
469 0 : CoinsList result;
470 0 : for (const auto& entry : ListCoins(*m_wallet)) {
471 0 : auto& group = result[entry.first];
472 0 : for (const auto& coin : entry.second) {
473 0 : group.emplace_back(coin.outpoint,
474 0 : MakeWalletTxOut(*m_wallet, coin));
475 : }
476 : }
477 0 : return result;
478 0 : }
479 0 : std::vector<WalletTxOut> getCoins(const std::vector<COutPoint>& outputs) override
480 : {
481 0 : LOCK(m_wallet->cs_wallet);
482 0 : std::vector<WalletTxOut> result;
483 0 : result.reserve(outputs.size());
484 0 : for (const auto& output : outputs) {
485 0 : result.emplace_back();
486 0 : auto it = m_wallet->mapWallet.find(output.hash);
487 0 : if (it != m_wallet->mapWallet.end()) {
488 0 : int depth = m_wallet->GetTxDepthInMainChain(it->second);
489 0 : if (depth >= 0) {
490 0 : result.back() = MakeWalletTxOut(*m_wallet, it->second, output.n, depth);
491 0 : }
492 0 : }
493 : }
494 0 : return result;
495 0 : }
496 0 : CAmount getRequiredFee(unsigned int tx_bytes) override { return GetRequiredFee(*m_wallet, tx_bytes); }
497 0 : CAmount getMinimumFee(unsigned int tx_bytes,
498 : const CCoinControl& coin_control,
499 : int* returned_target,
500 : FeeReason* reason) override
501 : {
502 0 : FeeCalculation fee_calc;
503 : CAmount result;
504 0 : result = GetMinimumFee(*m_wallet, tx_bytes, coin_control, &fee_calc);
505 0 : if (returned_target) *returned_target = fee_calc.returnedTarget;
506 0 : if (reason) *reason = fee_calc.reason;
507 0 : return result;
508 : }
509 0 : unsigned int getConfirmTarget() override { return m_wallet->m_confirm_target; }
510 0 : bool hdEnabled() override { return m_wallet->IsHDEnabled(); }
511 0 : bool canGetAddresses() override { return m_wallet->CanGetAddresses(); }
512 0 : bool hasExternalSigner() override { return m_wallet->IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER); }
513 0 : bool privateKeysDisabled() override { return m_wallet->IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS); }
514 0 : bool taprootEnabled() override {
515 0 : if (m_wallet->IsLegacy()) return false;
516 0 : auto spk_man = m_wallet->GetScriptPubKeyMan(OutputType::BECH32M, /*internal=*/false);
517 0 : return spk_man != nullptr;
518 0 : }
519 0 : OutputType getDefaultAddressType() override { return m_wallet->m_default_address_type; }
520 0 : CAmount getDefaultMaxTxFee() override { return m_wallet->m_default_max_tx_fee; }
521 0 : void remove() override
522 : {
523 0 : RemoveWallet(m_context, m_wallet, /*load_on_start=*/false);
524 0 : }
525 0 : bool isLegacy() override { return m_wallet->IsLegacy(); }
526 0 : std::unique_ptr<Handler> handleUnload(UnloadFn fn) override
527 : {
528 0 : return MakeSignalHandler(m_wallet->NotifyUnload.connect(fn));
529 0 : }
530 0 : std::unique_ptr<Handler> handleShowProgress(ShowProgressFn fn) override
531 : {
532 0 : return MakeSignalHandler(m_wallet->ShowProgress.connect(fn));
533 0 : }
534 0 : std::unique_ptr<Handler> handleStatusChanged(StatusChangedFn fn) override
535 : {
536 0 : return MakeSignalHandler(m_wallet->NotifyStatusChanged.connect([fn](CWallet*) { fn(); }));
537 0 : }
538 0 : std::unique_ptr<Handler> handleAddressBookChanged(AddressBookChangedFn fn) override
539 : {
540 0 : return MakeSignalHandler(m_wallet->NotifyAddressBookChanged.connect(
541 0 : [fn](const CTxDestination& address, const std::string& label, bool is_mine,
542 0 : AddressPurpose purpose, ChangeType status) { fn(address, label, is_mine, purpose, status); }));
543 0 : }
544 0 : std::unique_ptr<Handler> handleTransactionChanged(TransactionChangedFn fn) override
545 : {
546 0 : return MakeSignalHandler(m_wallet->NotifyTransactionChanged.connect(
547 0 : [fn](const uint256& txid, ChangeType status) { fn(txid, status); }));
548 0 : }
549 0 : std::unique_ptr<Handler> handleWatchOnlyChanged(WatchOnlyChangedFn fn) override
550 : {
551 0 : return MakeSignalHandler(m_wallet->NotifyWatchonlyChanged.connect(fn));
552 0 : }
553 0 : std::unique_ptr<Handler> handleCanGetAddressesChanged(CanGetAddressesChangedFn fn) override
554 : {
555 0 : return MakeSignalHandler(m_wallet->NotifyCanGetAddressesChanged.connect(fn));
556 0 : }
557 0 : CWallet* wallet() override { return m_wallet.get(); }
558 :
559 : WalletContext& m_context;
560 : std::shared_ptr<CWallet> m_wallet;
561 : };
562 :
563 : class WalletLoaderImpl : public WalletLoader
564 : {
565 : public:
566 0 : WalletLoaderImpl(Chain& chain, ArgsManager& args)
567 0 : {
568 0 : m_context.chain = &chain;
569 0 : m_context.args = &args;
570 0 : }
571 0 : ~WalletLoaderImpl() override { UnloadWallets(m_context); }
572 :
573 : //! ChainClient methods
574 0 : void registerRpcs() override
575 : {
576 0 : for (const CRPCCommand& command : GetWalletRPCCommands()) {
577 0 : m_rpc_commands.emplace_back(command.category, command.name, [this, &command](const JSONRPCRequest& request, UniValue& result, bool last_handler) {
578 0 : JSONRPCRequest wallet_request = request;
579 0 : wallet_request.context = &m_context;
580 0 : return command.actor(wallet_request, result, last_handler);
581 0 : }, command.argNames, command.unique_id);
582 0 : m_rpc_handlers.emplace_back(m_context.chain->handleRpc(m_rpc_commands.back()));
583 : }
584 0 : }
585 0 : bool verify() override { return VerifyWallets(m_context); }
586 0 : bool load() override { return LoadWallets(m_context); }
587 0 : void start(CScheduler& scheduler) override { return StartWallets(m_context, scheduler); }
588 0 : void flush() override { return FlushWallets(m_context); }
589 0 : void stop() override { return StopWallets(m_context); }
590 0 : void setMockTime(int64_t time) override { return SetMockTime(time); }
591 :
592 : //! WalletLoader methods
593 0 : util::Result<std::unique_ptr<Wallet>> createWallet(const std::string& name, const SecureString& passphrase, uint64_t wallet_creation_flags, std::vector<bilingual_str>& warnings) override
594 : {
595 0 : DatabaseOptions options;
596 : DatabaseStatus status;
597 0 : ReadDatabaseArgs(*m_context.args, options);
598 0 : options.require_create = true;
599 0 : options.create_flags = wallet_creation_flags;
600 0 : options.create_passphrase = passphrase;
601 0 : bilingual_str error;
602 0 : std::unique_ptr<Wallet> wallet{MakeWallet(m_context, CreateWallet(m_context, name, /*load_on_start=*/true, options, status, error, warnings))};
603 0 : if (wallet) {
604 0 : return {std::move(wallet)};
605 : } else {
606 0 : return util::Error{error};
607 : }
608 0 : }
609 0 : util::Result<std::unique_ptr<Wallet>> loadWallet(const std::string& name, std::vector<bilingual_str>& warnings) override
610 : {
611 0 : DatabaseOptions options;
612 : DatabaseStatus status;
613 0 : ReadDatabaseArgs(*m_context.args, options);
614 0 : options.require_existing = true;
615 0 : bilingual_str error;
616 0 : std::unique_ptr<Wallet> wallet{MakeWallet(m_context, LoadWallet(m_context, name, /*load_on_start=*/true, options, status, error, warnings))};
617 0 : if (wallet) {
618 0 : return {std::move(wallet)};
619 : } else {
620 0 : return util::Error{error};
621 : }
622 0 : }
623 0 : util::Result<std::unique_ptr<Wallet>> restoreWallet(const fs::path& backup_file, const std::string& wallet_name, std::vector<bilingual_str>& warnings) override
624 : {
625 : DatabaseStatus status;
626 0 : bilingual_str error;
627 0 : std::unique_ptr<Wallet> wallet{MakeWallet(m_context, RestoreWallet(m_context, backup_file, wallet_name, /*load_on_start=*/true, status, error, warnings))};
628 0 : if (wallet) {
629 0 : return {std::move(wallet)};
630 : } else {
631 0 : return util::Error{error};
632 : }
633 0 : }
634 0 : util::Result<WalletMigrationResult> migrateWallet(const std::string& name, const SecureString& passphrase) override
635 : {
636 0 : auto res = wallet::MigrateLegacyToDescriptor(name, passphrase, m_context);
637 0 : if (!res) return util::Error{util::ErrorString(res)};
638 0 : WalletMigrationResult out{
639 0 : .wallet = MakeWallet(m_context, res->wallet),
640 0 : .watchonly_wallet_name = res->watchonly_wallet ? std::make_optional(res->watchonly_wallet->GetName()) : std::nullopt,
641 0 : .solvables_wallet_name = res->solvables_wallet ? std::make_optional(res->solvables_wallet->GetName()) : std::nullopt,
642 0 : .backup_path = res->backup_path,
643 : };
644 0 : return {std::move(out)}; // std::move to work around clang bug
645 0 : }
646 0 : std::string getWalletDir() override
647 : {
648 0 : return fs::PathToString(GetWalletDir());
649 0 : }
650 0 : std::vector<std::string> listWalletDir() override
651 : {
652 0 : std::vector<std::string> paths;
653 0 : for (auto& path : ListDatabases(GetWalletDir())) {
654 0 : paths.push_back(fs::PathToString(path));
655 : }
656 0 : return paths;
657 0 : }
658 0 : std::vector<std::unique_ptr<Wallet>> getWallets() override
659 : {
660 0 : std::vector<std::unique_ptr<Wallet>> wallets;
661 0 : for (const auto& wallet : GetWallets(m_context)) {
662 0 : wallets.emplace_back(MakeWallet(m_context, wallet));
663 : }
664 0 : return wallets;
665 0 : }
666 0 : std::unique_ptr<Handler> handleLoadWallet(LoadWalletFn fn) override
667 : {
668 0 : return HandleLoadWallet(m_context, std::move(fn));
669 0 : }
670 0 : WalletContext* context() override { return &m_context; }
671 :
672 : WalletContext m_context;
673 : const std::vector<std::string> m_wallet_filenames;
674 : std::vector<std::unique_ptr<Handler>> m_rpc_handlers;
675 : std::list<CRPCCommand> m_rpc_commands;
676 : };
677 : } // namespace
678 : } // namespace wallet
679 :
680 : namespace interfaces {
681 0 : std::unique_ptr<Wallet> MakeWallet(wallet::WalletContext& context, const std::shared_ptr<wallet::CWallet>& wallet) { return wallet ? std::make_unique<wallet::WalletImpl>(context, wallet) : nullptr; }
682 :
683 0 : std::unique_ptr<WalletLoader> MakeWalletLoader(Chain& chain, ArgsManager& args)
684 : {
685 0 : return std::make_unique<wallet::WalletLoaderImpl>(chain, args);
686 : }
687 : } // namespace interfaces
|