mirror of
https://github.com/bitcoin/bitcoin.git
synced 2026-04-12 00:27:53 +02:00
Merge bitcoin/bitcoin#32471: wallet/rpc: fix listdescriptors RPC fails to return descriptors with private key information when wallet contains descriptors missing any key
9c7e4771b1test: Test listdescs with priv works even with missing priv keys (Novo)ed945a6854walletrpc: reject listdes with priv key on w-only wallets (Novo)9e5e9824f1descriptor: ToPrivateString() pass if at least 1 priv key exists (Novo)5c4db25b61descriptor: refactor ToPrivateString for providers (Novo)2dc74e3f4ewallet/migration: use HavePrivateKeys in place of ToPrivateString (Novo)e842eb90bbdescriptors: add HavePrivateKeys() (Novo) Pull request description: _TLDR: Currently, `listdescriptors [private=true]` will fail for a non-watch-only wallet if any descriptor has a missing private key(e.g `tr()`, `multi()`, etc.). This PR changes that while making sure `listdescriptors [private=true]` still fails if there no private keys. Closes #32078_ In non-watch-only wallets, it's possible to import descriptors as long as at least one private key is included. It's important that users can still view these descriptors when they need to create a backup—even if some private keys are missing ([#32078 (comment)](https://github.com/bitcoin/bitcoin/issues/32078#issuecomment-2781428475)). This change makes it possible to do so. This change also helps prevent `listdescriptors true` from failing completely, because one descriptor is missing some private keys. ### Notes - The new behaviour is applied to all descriptors including miniscript descriptors - `listdescriptors true` still fails for watch-only wallets to preserve existing behaviour https://github.com/bitcoin/bitcoin/pull/24361#discussion_r920801352 - Wallet migration logic previously used `Descriptor::ToPrivateString()` to determine which descriptor was watchonly. This means that modifying the `ToPrivateString()` behaviour caused descriptors that were previously recognized as "watchonly" to be "non-watchonly". **In order to keep the scope of this PR limited to the RPC behaviour, this PR uses a different method to determine `watchonly` descriptors for the purpose of wallet migration.** A follow-up PR can be opened to update migration logic to exclude descriptors with some private keys from the `watchonly` migration wallet. ### Relevant PRs https://github.com/bitcoin/bitcoin/pull/24361 https://github.com/bitcoin/bitcoin/pull/32186 ### Testing Functional tests were added to test the new behaviour EDIT **`listdescriptors [private=true]` will still fail when there are no private keys because non-watchonly wallets must have private keys and calling `listdescriptors [private=true]` for watchonly wallet returns an error** ACKs for top commit: Sjors: ACK9c7e4771b1achow101: ACK9c7e4771b1w0xlt: reACK9c7e4771b1with minor nits rkrux: re-ACK9c7e4771b1Tree-SHA512: f9b3b2c3e5425a26e158882e39e82e15b7cb13ffbfb6a5fa2868c79526e9b178fcc3cd88d3e2e286f64819d041f687353780bbcf5a355c63a136fb8179698b60
This commit is contained in:
@@ -843,6 +843,12 @@ public:
|
||||
|
||||
template<typename CTx>
|
||||
std::optional<std::string> ToString(const CTx& ctx) const {
|
||||
bool dummy{false};
|
||||
return ToString(ctx, dummy);
|
||||
}
|
||||
|
||||
template<typename CTx>
|
||||
std::optional<std::string> ToString(const CTx& ctx, bool& has_priv_key) const {
|
||||
// To construct the std::string representation for a Miniscript object, we use
|
||||
// the TreeEvalMaybe algorithm. The State is a boolean: whether the parent node is a
|
||||
// wrapper. If so, non-wrapper expressions must be prefixed with a ":".
|
||||
@@ -855,10 +861,16 @@ public:
|
||||
(node.fragment == Fragment::OR_I && node.subs[0]->fragment == Fragment::JUST_0) ||
|
||||
(node.fragment == Fragment::OR_I && node.subs[1]->fragment == Fragment::JUST_0));
|
||||
};
|
||||
auto toString = [&ctx, &has_priv_key](Key key) -> std::optional<std::string> {
|
||||
bool fragment_has_priv_key{false};
|
||||
auto key_str{ctx.ToString(key, fragment_has_priv_key)};
|
||||
if (key_str) has_priv_key = has_priv_key || fragment_has_priv_key;
|
||||
return key_str;
|
||||
};
|
||||
// The upward function computes for a node, given whether its parent is a wrapper,
|
||||
// and the string representations of its child nodes, the string representation of the node.
|
||||
const bool is_tapscript{IsTapscript(m_script_ctx)};
|
||||
auto upfn = [&ctx, is_tapscript](bool wrapped, const Node& node, std::span<std::string> subs) -> std::optional<std::string> {
|
||||
auto upfn = [is_tapscript, &toString](bool wrapped, const Node& node, std::span<std::string> subs) -> std::optional<std::string> {
|
||||
std::string ret = wrapped ? ":" : "";
|
||||
|
||||
switch (node.fragment) {
|
||||
@@ -867,13 +879,13 @@ public:
|
||||
case Fragment::WRAP_C:
|
||||
if (node.subs[0]->fragment == Fragment::PK_K) {
|
||||
// pk(K) is syntactic sugar for c:pk_k(K)
|
||||
auto key_str = ctx.ToString(node.subs[0]->keys[0]);
|
||||
auto key_str = toString(node.subs[0]->keys[0]);
|
||||
if (!key_str) return {};
|
||||
return std::move(ret) + "pk(" + std::move(*key_str) + ")";
|
||||
}
|
||||
if (node.subs[0]->fragment == Fragment::PK_H) {
|
||||
// pkh(K) is syntactic sugar for c:pk_h(K)
|
||||
auto key_str = ctx.ToString(node.subs[0]->keys[0]);
|
||||
auto key_str = toString(node.subs[0]->keys[0]);
|
||||
if (!key_str) return {};
|
||||
return std::move(ret) + "pkh(" + std::move(*key_str) + ")";
|
||||
}
|
||||
@@ -894,12 +906,12 @@ public:
|
||||
}
|
||||
switch (node.fragment) {
|
||||
case Fragment::PK_K: {
|
||||
auto key_str = ctx.ToString(node.keys[0]);
|
||||
auto key_str = toString(node.keys[0]);
|
||||
if (!key_str) return {};
|
||||
return std::move(ret) + "pk_k(" + std::move(*key_str) + ")";
|
||||
}
|
||||
case Fragment::PK_H: {
|
||||
auto key_str = ctx.ToString(node.keys[0]);
|
||||
auto key_str = toString(node.keys[0]);
|
||||
if (!key_str) return {};
|
||||
return std::move(ret) + "pk_h(" + std::move(*key_str) + ")";
|
||||
}
|
||||
@@ -925,7 +937,7 @@ public:
|
||||
CHECK_NONFATAL(!is_tapscript);
|
||||
auto str = std::move(ret) + "multi(" + util::ToString(node.k);
|
||||
for (const auto& key : node.keys) {
|
||||
auto key_str = ctx.ToString(key);
|
||||
auto key_str = toString(key);
|
||||
if (!key_str) return {};
|
||||
str += "," + std::move(*key_str);
|
||||
}
|
||||
@@ -935,7 +947,7 @@ public:
|
||||
CHECK_NONFATAL(is_tapscript);
|
||||
auto str = std::move(ret) + "multi_a(" + util::ToString(node.k);
|
||||
for (const auto& key : node.keys) {
|
||||
auto key_str = ctx.ToString(key);
|
||||
auto key_str = toString(key);
|
||||
if (!key_str) return {};
|
||||
str += "," + std::move(*key_str);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user