mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-25 10:43:19 -03:00
Merge #17283: rpc: improve getaddressinfo test coverage, help, code docs
33f5fc32e5
test: add rpc getaddressinfo labels test coverage (Jon Atack)0f3539ac6d
test: add listlabels test in wallet_labels.py (Jon Atack)1388de8390
rpc: add getaddressinfo code documentation (Jon Atack)2ee0cb3330
rpc: update getaddressinfo RPCExamples to bech32 (Jon Atack)8d1ed0c263
rpc: clarify label vs labels in getaddressinfo RPCHelpman (Jon Atack)5a0ed85070
rpc: improve getaddressinfo RPCHelpman content (Jon Atack)70cda342cd
rpc: improve getaddressinfo RPCHelpman formatting (Jon Atack) Pull request description: This PR is a continuation of the work in https://github.com/bitcoin/bitcoin/pull/12892. Main motivations: - There is currently no test coverage for the getaddressinfo `labels` response. Coverage here is a prerequisite before deprecating the `label` response or adding multiple labels per address. - `bitcoin-cli help getaddressinfo` returns a few content errors, difficult-to-read formatting, and no explanation why it returns both `label` and `labels` and how they relate, which can be confusing for application developers. Changes by order of commits: - [x] improve/fix getaddressinfo RPCHelpman layout formatting - [x] improve/fix getaddressinfo RPCHelpman content - [x] clarify the `label` and `labels` fields in getaddressinfo RPCHelpman - [x] update getaddressinfo RPCExamples addresses to bech32 - [x] add getaddressinfo code docs - [x] add a `listlabels` test assertion in wallet_labels.py - [x] add missing getaddressinfo `labels` test coverage and improve the existing `label` tests Here are gists of the CLI help output: [`bitcoin-cli help getaddressinfo` before this PR](https://gist.github.com/jonatack/022af5221a85c069780359a22643c810) [`bitcoin-cli help getaddressinfo` after this PR](https://gist.github.com/jonatack/4ee5f6abc62a3d99269570206a5f90ba) It seems we ought to begin a deprecation process for the getaddressinfo `label` field? If yes, I have a follow-up ready. _--> EDIT: Deprecation follow-ups #17578 and #17585 now build on this PR._ ACKs for top commit: fjahr: Re-ACK33f5fc32e5
jnewbery: ACK33f5fc32e5
. Tree-SHA512: a001aa863090ec2566a31059477945b1c303ebeb430b33472f8b150e420fa5742fc33bca9d95571746395b607f43f6078dd5b53e238ac1f3fc648b51c8f79a07
This commit is contained in:
commit
d8a66626d6
7 changed files with 118 additions and 60 deletions
|
@ -951,7 +951,7 @@ static UniValue addmultisigaddress(const JSONRPCRequest& request)
|
||||||
}
|
}
|
||||||
|
|
||||||
RPCHelpMan{"addmultisigaddress",
|
RPCHelpMan{"addmultisigaddress",
|
||||||
"\nAdd a nrequired-to-sign multisignature address to the wallet. Requires a new wallet backup.\n"
|
"\nAdd an nrequired-to-sign multisignature address to the wallet. Requires a new wallet backup.\n"
|
||||||
"Each key is a Bitcoin address or hex-encoded public key.\n"
|
"Each key is a Bitcoin address or hex-encoded public key.\n"
|
||||||
"This functionality is only intended for use with non-watchonly addresses.\n"
|
"This functionality is only intended for use with non-watchonly addresses.\n"
|
||||||
"See `importaddress` for watchonly p2sh address support.\n"
|
"See `importaddress` for watchonly p2sh address support.\n"
|
||||||
|
@ -3709,52 +3709,57 @@ UniValue getaddressinfo(const JSONRPCRequest& request)
|
||||||
}
|
}
|
||||||
|
|
||||||
RPCHelpMan{"getaddressinfo",
|
RPCHelpMan{"getaddressinfo",
|
||||||
"\nReturn information about the given bitcoin address. Some information requires the address\n"
|
"\nReturn information about the given bitcoin address.\n"
|
||||||
"to be in the wallet.\n",
|
"Some of the information will only be present if the address is in the active wallet.\n",
|
||||||
{
|
{
|
||||||
{"address", RPCArg::Type::STR, RPCArg::Optional::NO, "The bitcoin address to get the information of."},
|
{"address", RPCArg::Type::STR, RPCArg::Optional::NO, "The bitcoin address for which to get information."},
|
||||||
},
|
},
|
||||||
RPCResult{
|
RPCResult{
|
||||||
"{\n"
|
"{\n"
|
||||||
" \"address\" : \"address\", (string) The bitcoin address validated\n"
|
" \"address\" : \"address\", (string) The bitcoin address validated.\n"
|
||||||
" \"scriptPubKey\" : \"hex\", (string) The hex-encoded scriptPubKey generated by the address\n"
|
" \"scriptPubKey\" : \"hex\", (string) The hex-encoded scriptPubKey generated by the address.\n"
|
||||||
" \"ismine\" : true|false, (boolean) If the address is yours or not\n"
|
" \"ismine\" : true|false, (boolean) If the address is yours.\n"
|
||||||
" \"iswatchonly\" : true|false, (boolean) If the address is watchonly\n"
|
" \"iswatchonly\" : true|false, (boolean) If the address is watchonly.\n"
|
||||||
" \"solvable\" : true|false, (boolean) Whether we know how to spend coins sent to this address, ignoring the possible lack of private keys\n"
|
" \"solvable\" : true|false, (boolean) If we know how to spend coins sent to this address, ignoring the possible lack of private keys.\n"
|
||||||
" \"desc\" : \"desc\", (string, optional) A descriptor for spending coins sent to this address (only when solvable)\n"
|
" \"desc\" : \"desc\", (string, optional) A descriptor for spending coins sent to this address (only when solvable).\n"
|
||||||
" \"isscript\" : true|false, (boolean) If the key is a script\n"
|
" \"isscript\" : true|false, (boolean) If the key is a script.\n"
|
||||||
" \"ischange\" : true|false, (boolean) If the address was used for change output\n"
|
" \"ischange\" : true|false, (boolean) If the address was used for change output.\n"
|
||||||
" \"iswitness\" : true|false, (boolean) If the address is a witness address\n"
|
" \"iswitness\" : true|false, (boolean) If the address is a witness address.\n"
|
||||||
" \"witness_version\" : version (numeric, optional) The version number of the witness program\n"
|
" \"witness_version\" : version (numeric, optional) The version number of the witness program.\n"
|
||||||
" \"witness_program\" : \"hex\" (string, optional) The hex value of the witness program\n"
|
" \"witness_program\" : \"hex\" (string, optional) The hex value of the witness program.\n"
|
||||||
" \"script\" : \"type\" (string, optional) The output script type. Only if \"isscript\" is true and the redeemscript is known. Possible types: nonstandard, pubkey, pubkeyhash, scripthash, multisig, nulldata, witness_v0_keyhash, witness_v0_scripthash, witness_unknown\n"
|
" \"script\" : \"type\" (string, optional) The output script type. Only if isscript is true and the redeemscript is known. Possible\n"
|
||||||
" \"hex\" : \"hex\", (string, optional) The redeemscript for the p2sh address\n"
|
" types: nonstandard, pubkey, pubkeyhash, scripthash, multisig, nulldata, witness_v0_keyhash,\n"
|
||||||
" \"pubkeys\" (string, optional) Array of pubkeys associated with the known redeemscript (only if \"script\" is \"multisig\")\n"
|
" witness_v0_scripthash, witness_unknown.\n"
|
||||||
|
" \"hex\" : \"hex\", (string, optional) The redeemscript for the p2sh address.\n"
|
||||||
|
" \"pubkeys\" (array, optional) Array of pubkeys associated with the known redeemscript (only if script is multisig).\n"
|
||||||
" [\n"
|
" [\n"
|
||||||
" \"pubkey\"\n"
|
" \"pubkey\" (string)\n"
|
||||||
" ,...\n"
|
" ,...\n"
|
||||||
" ]\n"
|
" ]\n"
|
||||||
" \"sigsrequired\" : xxxxx (numeric, optional) Number of signatures required to spend multisig output (only if \"script\" is \"multisig\")\n"
|
" \"sigsrequired\" : xxxxx (numeric, optional) The number of signatures required to spend multisig output (only if script is multisig).\n"
|
||||||
" \"pubkey\" : \"publickeyhex\", (string, optional) The hex value of the raw public key, for single-key addresses (possibly embedded in P2SH or P2WSH)\n"
|
" \"pubkey\" : \"publickeyhex\", (string, optional) The hex value of the raw public key for single-key addresses (possibly embedded in P2SH or P2WSH).\n"
|
||||||
" \"embedded\" : {...}, (object, optional) Information about the address embedded in P2SH or P2WSH, if relevant and known. It includes all getaddressinfo output fields for the embedded address, excluding metadata (\"timestamp\", \"hdkeypath\", \"hdseedid\") and relation to the wallet (\"ismine\", \"iswatchonly\").\n"
|
" \"embedded\" : {...}, (object, optional) Information about the address embedded in P2SH or P2WSH, if relevant and known. Includes all\n"
|
||||||
" \"iscompressed\" : true|false, (boolean, optional) If the pubkey is compressed\n"
|
" getaddressinfo output fields for the embedded address, excluding metadata (timestamp, hdkeypath,\n"
|
||||||
" \"label\" : \"label\" (string) The label associated with the address, \"\" is the default label\n"
|
" hdseedid) and relation to the wallet (ismine, iswatchonly).\n"
|
||||||
" \"timestamp\" : timestamp, (number, optional) The creation time of the key if available in seconds since epoch (Jan 1 1970 GMT)\n"
|
" \"iscompressed\" : true|false, (boolean, optional) If the pubkey is compressed.\n"
|
||||||
" \"hdkeypath\" : \"keypath\" (string, optional) The HD keypath if the key is HD and available\n"
|
" \"label\" : \"label\" (string) The label associated with the address. Defaults to \"\". Equivalent to the name field in the labels array.\n"
|
||||||
" \"hdseedid\" : \"<hash160>\" (string, optional) The Hash160 of the HD seed\n"
|
" \"timestamp\" : timestamp, (number, optional) The creation time of the key if available, expressed in seconds since Epoch Time (Jan 1 1970 GMT).\n"
|
||||||
" \"hdmasterfingerprint\" : \"<hash160>\" (string, optional) The fingperint of the master key.\n"
|
" \"hdkeypath\" : \"keypath\" (string, optional) The HD keypath, if the key is HD and available.\n"
|
||||||
" \"labels\" (object) Array of labels associated with the address.\n"
|
" \"hdseedid\" : \"<hash160>\" (string, optional) The Hash160 of the HD seed.\n"
|
||||||
|
" \"hdmasterfingerprint\" : \"<hash160>\" (string, optional) The fingerprint of the master key.\n"
|
||||||
|
" \"labels\" (object) An array of labels associated with the address. Currently limited to one label but returned\n"
|
||||||
|
" as an array to keep the API stable if multiple labels are enabled in the future.\n"
|
||||||
" [\n"
|
" [\n"
|
||||||
" { (json object of label data)\n"
|
" { (json object of label data)\n"
|
||||||
" \"name\": \"labelname\" (string) The label\n"
|
" \"name\": \"label name\" (string) The label name. Defaults to \"\". Equivalent to the label field above.\n"
|
||||||
" \"purpose\": \"string\" (string) Purpose of address (\"send\" for sending address, \"receive\" for receiving address)\n"
|
" \"purpose\": \"purpose\" (string) The purpose of the associated address (send or receive).\n"
|
||||||
" },...\n"
|
" },...\n"
|
||||||
" ]\n"
|
" ]\n"
|
||||||
"}\n"
|
"}\n"
|
||||||
},
|
},
|
||||||
RPCExamples{
|
RPCExamples{
|
||||||
HelpExampleCli("getaddressinfo", "\"1PSSGeFHDnKNxiEyFrD1wcEaHr9hrQDDWc\"")
|
HelpExampleCli("getaddressinfo", "\"bc1q09vm5lfy0j5reeulh4x5752q25uqqvz34hufdl\"") +
|
||||||
+ HelpExampleRpc("getaddressinfo", "\"1PSSGeFHDnKNxiEyFrD1wcEaHr9hrQDDWc\"")
|
HelpExampleRpc("getaddressinfo", "\"bc1q09vm5lfy0j5reeulh4x5752q25uqqvz34hufdl\"")
|
||||||
},
|
},
|
||||||
}.Check(request);
|
}.Check(request);
|
||||||
|
|
||||||
|
@ -3773,23 +3778,39 @@ UniValue getaddressinfo(const JSONRPCRequest& request)
|
||||||
|
|
||||||
CScript scriptPubKey = GetScriptForDestination(dest);
|
CScript scriptPubKey = GetScriptForDestination(dest);
|
||||||
ret.pushKV("scriptPubKey", HexStr(scriptPubKey.begin(), scriptPubKey.end()));
|
ret.pushKV("scriptPubKey", HexStr(scriptPubKey.begin(), scriptPubKey.end()));
|
||||||
|
|
||||||
const SigningProvider* provider = pwallet->GetSigningProvider(scriptPubKey);
|
const SigningProvider* provider = pwallet->GetSigningProvider(scriptPubKey);
|
||||||
|
|
||||||
isminetype mine = pwallet->IsMine(dest);
|
isminetype mine = pwallet->IsMine(dest);
|
||||||
ret.pushKV("ismine", bool(mine & ISMINE_SPENDABLE));
|
ret.pushKV("ismine", bool(mine & ISMINE_SPENDABLE));
|
||||||
|
|
||||||
bool solvable = provider && IsSolvable(*provider, scriptPubKey);
|
bool solvable = provider && IsSolvable(*provider, scriptPubKey);
|
||||||
ret.pushKV("solvable", solvable);
|
ret.pushKV("solvable", solvable);
|
||||||
|
|
||||||
if (solvable) {
|
if (solvable) {
|
||||||
ret.pushKV("desc", InferDescriptor(scriptPubKey, *provider)->ToString());
|
ret.pushKV("desc", InferDescriptor(scriptPubKey, *provider)->ToString());
|
||||||
}
|
}
|
||||||
|
|
||||||
ret.pushKV("iswatchonly", bool(mine & ISMINE_WATCH_ONLY));
|
ret.pushKV("iswatchonly", bool(mine & ISMINE_WATCH_ONLY));
|
||||||
|
|
||||||
|
// Return DescribeWalletAddress fields.
|
||||||
|
// Always returned: isscript, ischange, iswitness.
|
||||||
|
// Optional: witness_version, witness_program, script, hex, pubkeys (array),
|
||||||
|
// sigsrequired, pubkey, embedded, iscompressed.
|
||||||
UniValue detail = DescribeWalletAddress(pwallet, dest);
|
UniValue detail = DescribeWalletAddress(pwallet, dest);
|
||||||
ret.pushKVs(detail);
|
ret.pushKVs(detail);
|
||||||
|
|
||||||
|
// Return label field if existing. Currently only one label can be
|
||||||
|
// associated with an address, so the label should be equivalent to the
|
||||||
|
// value of the name key/value pair in the labels hash array below.
|
||||||
if (pwallet->mapAddressBook.count(dest)) {
|
if (pwallet->mapAddressBook.count(dest)) {
|
||||||
ret.pushKV("label", pwallet->mapAddressBook[dest].name);
|
ret.pushKV("label", pwallet->mapAddressBook[dest].name);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret.pushKV("ischange", pwallet->IsChange(scriptPubKey));
|
ret.pushKV("ischange", pwallet->IsChange(scriptPubKey));
|
||||||
|
|
||||||
|
// Fetch KeyMetadata, if present, for the timestamp, hdkeypath, hdseedid,
|
||||||
|
// and hdmasterfingerprint fields.
|
||||||
ScriptPubKeyMan* spk_man = pwallet->GetScriptPubKeyMan(scriptPubKey);
|
ScriptPubKeyMan* spk_man = pwallet->GetScriptPubKeyMan(scriptPubKey);
|
||||||
if (spk_man) {
|
if (spk_man) {
|
||||||
if (const CKeyMetadata* meta = spk_man->GetMetadata(dest)) {
|
if (const CKeyMetadata* meta = spk_man->GetMetadata(dest)) {
|
||||||
|
@ -3802,9 +3823,11 @@ UniValue getaddressinfo(const JSONRPCRequest& request)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Currently only one label can be associated with an address, return an array
|
// Return a labels array containing a hash of key/value pairs for the label
|
||||||
// so the API remains stable if we allow multiple labels to be associated with
|
// name and address purpose. The name value is equivalent to the label field
|
||||||
// an address.
|
// above. Currently only one label can be associated with an address, but we
|
||||||
|
// return an array so the API remains stable if we allow multiple labels to
|
||||||
|
// be associated with an address in the future.
|
||||||
UniValue labels(UniValue::VARR);
|
UniValue labels(UniValue::VARR);
|
||||||
std::map<CTxDestination, CAddressBookData>::iterator mi = pwallet->mapAddressBook.find(dest);
|
std::map<CTxDestination, CAddressBookData>::iterator mi = pwallet->mapAddressBook.find(dest);
|
||||||
if (mi != pwallet->mapAddressBook.end()) {
|
if (mi != pwallet->mapAddressBook.end()) {
|
||||||
|
|
|
@ -88,6 +88,11 @@ def get_multisig(node):
|
||||||
p2sh_p2wsh_script=CScript([OP_HASH160, witness_script, OP_EQUAL]).hex(),
|
p2sh_p2wsh_script=CScript([OP_HASH160, witness_script, OP_EQUAL]).hex(),
|
||||||
p2sh_p2wsh_addr=script_to_p2sh_p2wsh(script_code))
|
p2sh_p2wsh_addr=script_to_p2sh_p2wsh(script_code))
|
||||||
|
|
||||||
|
def labels_value(name="", purpose="receive"):
|
||||||
|
"""Generate a getaddressinfo labels array from a name and purpose.
|
||||||
|
Often used as the value of a labels kwarg for calling test_address below."""
|
||||||
|
return [{"name": name, "purpose": purpose}]
|
||||||
|
|
||||||
def test_address(node, address, **kwargs):
|
def test_address(node, address, **kwargs):
|
||||||
"""Get address info for `address` and test whether the returned values are as expected."""
|
"""Get address info for `address` and test whether the returned values are as expected."""
|
||||||
addr_info = node.getaddressinfo(address)
|
addr_info = node.getaddressinfo(address)
|
||||||
|
|
|
@ -15,6 +15,10 @@ from test_framework.util import (
|
||||||
connect_nodes,
|
connect_nodes,
|
||||||
wait_until,
|
wait_until,
|
||||||
)
|
)
|
||||||
|
from test_framework.wallet_util import (
|
||||||
|
labels_value,
|
||||||
|
test_address,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class WalletTest(BitcoinTestFramework):
|
class WalletTest(BitcoinTestFramework):
|
||||||
|
@ -390,7 +394,7 @@ class WalletTest(BitcoinTestFramework):
|
||||||
for label in [u'рыба', u'𝅘𝅥𝅯']:
|
for label in [u'рыба', u'𝅘𝅥𝅯']:
|
||||||
addr = self.nodes[0].getnewaddress()
|
addr = self.nodes[0].getnewaddress()
|
||||||
self.nodes[0].setlabel(addr, label)
|
self.nodes[0].setlabel(addr, label)
|
||||||
assert_equal(self.nodes[0].getaddressinfo(addr)['label'], label)
|
test_address(self.nodes[0], addr, label=label, labels=labels_value(name=label))
|
||||||
assert label in self.nodes[0].listlabels()
|
assert label in self.nodes[0].listlabels()
|
||||||
self.nodes[0].rpc.ensure_ascii = True # restore to default
|
self.nodes[0].rpc.ensure_ascii = True # restore to default
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,10 @@ with and without a label.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from test_framework.test_framework import BitcoinTestFramework
|
from test_framework.test_framework import BitcoinTestFramework
|
||||||
from test_framework.wallet_util import test_address
|
from test_framework.wallet_util import (
|
||||||
|
labels_value,
|
||||||
|
test_address,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class ImportWithLabel(BitcoinTestFramework):
|
class ImportWithLabel(BitcoinTestFramework):
|
||||||
|
@ -36,7 +39,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
address,
|
address,
|
||||||
iswatchonly=True,
|
iswatchonly=True,
|
||||||
ismine=False,
|
ismine=False,
|
||||||
label=label)
|
label=label,
|
||||||
|
labels=labels_value(name=label))
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
"Import the watch-only address's private key without a "
|
"Import the watch-only address's private key without a "
|
||||||
|
@ -47,7 +51,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
|
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
address,
|
address,
|
||||||
label=label)
|
label=label,
|
||||||
|
labels=labels_value(name=label))
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
"Test importaddress without label and importprivkey with label."
|
"Test importaddress without label and importprivkey with label."
|
||||||
|
@ -59,7 +64,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
address2,
|
address2,
|
||||||
iswatchonly=True,
|
iswatchonly=True,
|
||||||
ismine=False,
|
ismine=False,
|
||||||
label="")
|
label="",
|
||||||
|
labels=labels_value())
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
"Import the watch-only address's private key with a "
|
"Import the watch-only address's private key with a "
|
||||||
|
@ -71,7 +77,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
|
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
address2,
|
address2,
|
||||||
label=label2)
|
label=label2,
|
||||||
|
labels=labels_value(name=label2))
|
||||||
|
|
||||||
self.log.info("Test importaddress with label and importprivkey with label.")
|
self.log.info("Test importaddress with label and importprivkey with label.")
|
||||||
self.log.info("Import a watch-only address with a label.")
|
self.log.info("Import a watch-only address with a label.")
|
||||||
|
@ -82,7 +89,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
address3,
|
address3,
|
||||||
iswatchonly=True,
|
iswatchonly=True,
|
||||||
ismine=False,
|
ismine=False,
|
||||||
label=label3_addr)
|
label=label3_addr,
|
||||||
|
labels=labels_value(name=label3_addr))
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
"Import the watch-only address's private key with a "
|
"Import the watch-only address's private key with a "
|
||||||
|
@ -94,7 +102,8 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
|
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
address3,
|
address3,
|
||||||
label=label3_priv)
|
label=label3_priv,
|
||||||
|
labels=labels_value(name=label3_priv))
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
"Test importprivkey won't label new dests with the same "
|
"Test importprivkey won't label new dests with the same "
|
||||||
|
@ -109,6 +118,7 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
iswatchonly=True,
|
iswatchonly=True,
|
||||||
ismine=False,
|
ismine=False,
|
||||||
label=label4_addr,
|
label=label4_addr,
|
||||||
|
labels=labels_value(name=label4_addr),
|
||||||
embedded=None)
|
embedded=None)
|
||||||
|
|
||||||
self.log.info(
|
self.log.info(
|
||||||
|
@ -123,10 +133,13 @@ class ImportWithLabel(BitcoinTestFramework):
|
||||||
|
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
embedded_addr,
|
embedded_addr,
|
||||||
label="")
|
label="",
|
||||||
|
labels=labels_value())
|
||||||
|
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
address4,
|
address4,
|
||||||
label=label4_addr)
|
label=label4_addr,
|
||||||
|
labels=labels_value(name=label4_addr))
|
||||||
|
|
||||||
self.stop_nodes()
|
self.stop_nodes()
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,7 @@ from test_framework.util import (
|
||||||
from test_framework.wallet_util import (
|
from test_framework.wallet_util import (
|
||||||
get_key,
|
get_key,
|
||||||
get_multisig,
|
get_multisig,
|
||||||
|
labels_value,
|
||||||
test_address,
|
test_address,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -121,7 +122,7 @@ class ImportMultiTest(BitcoinTestFramework):
|
||||||
self.test_importmulti({"scriptPubKey": key.p2pkh_script,
|
self.test_importmulti({"scriptPubKey": key.p2pkh_script,
|
||||||
"timestamp": "now",
|
"timestamp": "now",
|
||||||
"internal": True,
|
"internal": True,
|
||||||
"label": "Example label"},
|
"label": "Unsuccessful labelling for internal addresses"},
|
||||||
success=False,
|
success=False,
|
||||||
error_code=-8,
|
error_code=-8,
|
||||||
error_message='Internal addresses should not have a label')
|
error_message='Internal addresses should not have a label')
|
||||||
|
@ -550,7 +551,7 @@ class ImportMultiTest(BitcoinTestFramework):
|
||||||
self.log.info("Should not import a p2sh-p2wpkh address from descriptor without checksum and private key")
|
self.log.info("Should not import a p2sh-p2wpkh address from descriptor without checksum and private key")
|
||||||
self.test_importmulti({"desc": "sh(wpkh(" + key.pubkey + "))",
|
self.test_importmulti({"desc": "sh(wpkh(" + key.pubkey + "))",
|
||||||
"timestamp": "now",
|
"timestamp": "now",
|
||||||
"label": "Descriptor import test",
|
"label": "Unsuccessful P2SH-P2WPKH descriptor import",
|
||||||
"keys": [key.privkey]},
|
"keys": [key.privkey]},
|
||||||
success=False,
|
success=False,
|
||||||
error_code=-5,
|
error_code=-5,
|
||||||
|
@ -558,17 +559,19 @@ class ImportMultiTest(BitcoinTestFramework):
|
||||||
|
|
||||||
# Test importing of a P2SH-P2WPKH address via descriptor + private key
|
# Test importing of a P2SH-P2WPKH address via descriptor + private key
|
||||||
key = get_key(self.nodes[0])
|
key = get_key(self.nodes[0])
|
||||||
|
p2sh_p2wpkh_label = "Successful P2SH-P2WPKH descriptor import"
|
||||||
self.log.info("Should import a p2sh-p2wpkh address from descriptor and private key")
|
self.log.info("Should import a p2sh-p2wpkh address from descriptor and private key")
|
||||||
self.test_importmulti({"desc": descsum_create("sh(wpkh(" + key.pubkey + "))"),
|
self.test_importmulti({"desc": descsum_create("sh(wpkh(" + key.pubkey + "))"),
|
||||||
"timestamp": "now",
|
"timestamp": "now",
|
||||||
"label": "Descriptor import test",
|
"label": p2sh_p2wpkh_label,
|
||||||
"keys": [key.privkey]},
|
"keys": [key.privkey]},
|
||||||
success=True)
|
success=True)
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
key.p2sh_p2wpkh_addr,
|
key.p2sh_p2wpkh_addr,
|
||||||
solvable=True,
|
solvable=True,
|
||||||
ismine=True,
|
ismine=True,
|
||||||
label="Descriptor import test")
|
label=p2sh_p2wpkh_label,
|
||||||
|
labels=labels_value(name=p2sh_p2wpkh_label))
|
||||||
|
|
||||||
# Test ranged descriptor fails if range is not specified
|
# Test ranged descriptor fails if range is not specified
|
||||||
xpriv = "tprv8ZgxMBicQKsPeuVhWwi6wuMQGfPKi9Li5GtX35jVNknACgqe3CY4g5xgkfDDJcmtF7o1QnxWDRYw4H5P26PXq7sbcUkEqeR4fg3Kxp2tigg"
|
xpriv = "tprv8ZgxMBicQKsPeuVhWwi6wuMQGfPKi9Li5GtX35jVNknACgqe3CY4g5xgkfDDJcmtF7o1QnxWDRYw4H5P26PXq7sbcUkEqeR4fg3Kxp2tigg"
|
||||||
|
@ -628,17 +631,19 @@ class ImportMultiTest(BitcoinTestFramework):
|
||||||
|
|
||||||
# Test importing of a P2PKH address via descriptor
|
# Test importing of a P2PKH address via descriptor
|
||||||
key = get_key(self.nodes[0])
|
key = get_key(self.nodes[0])
|
||||||
|
p2pkh_label = "P2PKH descriptor import"
|
||||||
self.log.info("Should import a p2pkh address from descriptor")
|
self.log.info("Should import a p2pkh address from descriptor")
|
||||||
self.test_importmulti({"desc": descsum_create("pkh(" + key.pubkey + ")"),
|
self.test_importmulti({"desc": descsum_create("pkh(" + key.pubkey + ")"),
|
||||||
"timestamp": "now",
|
"timestamp": "now",
|
||||||
"label": "Descriptor import test"},
|
"label": p2pkh_label},
|
||||||
True,
|
True,
|
||||||
warnings=["Some private keys are missing, outputs will be considered watchonly. If this is intentional, specify the watchonly flag."])
|
warnings=["Some private keys are missing, outputs will be considered watchonly. If this is intentional, specify the watchonly flag."])
|
||||||
test_address(self.nodes[1],
|
test_address(self.nodes[1],
|
||||||
key.p2pkh_addr,
|
key.p2pkh_addr,
|
||||||
solvable=True,
|
solvable=True,
|
||||||
ismine=False,
|
ismine=False,
|
||||||
label="Descriptor import test")
|
label=p2pkh_label,
|
||||||
|
labels=labels_value(name=p2pkh_label))
|
||||||
|
|
||||||
# Test import fails if both desc and scriptPubKey are provided
|
# Test import fails if both desc and scriptPubKey are provided
|
||||||
key = get_key(self.nodes[0])
|
key = get_key(self.nodes[0])
|
||||||
|
|
|
@ -13,6 +13,10 @@ from collections import defaultdict
|
||||||
|
|
||||||
from test_framework.test_framework import BitcoinTestFramework
|
from test_framework.test_framework import BitcoinTestFramework
|
||||||
from test_framework.util import assert_equal, assert_raises_rpc_error
|
from test_framework.util import assert_equal, assert_raises_rpc_error
|
||||||
|
from test_framework.wallet_util import (
|
||||||
|
labels_value,
|
||||||
|
test_address,
|
||||||
|
)
|
||||||
|
|
||||||
class WalletLabelsTest(BitcoinTestFramework):
|
class WalletLabelsTest(BitcoinTestFramework):
|
||||||
def set_test_params(self):
|
def set_test_params(self):
|
||||||
|
@ -152,14 +156,14 @@ class Label:
|
||||||
def verify(self, node):
|
def verify(self, node):
|
||||||
if self.receive_address is not None:
|
if self.receive_address is not None:
|
||||||
assert self.receive_address in self.addresses
|
assert self.receive_address in self.addresses
|
||||||
|
|
||||||
for address in self.addresses:
|
for address in self.addresses:
|
||||||
assert_equal(
|
test_address(
|
||||||
node.getaddressinfo(address)['labels'][0],
|
node,
|
||||||
{"name": self.name,
|
address,
|
||||||
"purpose": self.purpose[address]})
|
label=self.name,
|
||||||
assert_equal(node.getaddressinfo(address)['label'], self.name)
|
labels=labels_value(name=self.name, purpose=self.purpose[address])
|
||||||
|
)
|
||||||
|
assert self.name in node.listlabels()
|
||||||
assert_equal(
|
assert_equal(
|
||||||
node.getaddressesbylabel(self.name),
|
node.getaddressesbylabel(self.name),
|
||||||
{address: {"purpose": self.purpose[address]} for address in self.addresses})
|
{address: {"purpose": self.purpose[address]} for address in self.addresses})
|
||||||
|
|
|
@ -11,6 +11,10 @@ from test_framework.util import (
|
||||||
assert_equal,
|
assert_equal,
|
||||||
assert_raises_rpc_error,
|
assert_raises_rpc_error,
|
||||||
)
|
)
|
||||||
|
from test_framework.wallet_util import (
|
||||||
|
labels_value,
|
||||||
|
test_address,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class ReceivedByTest(BitcoinTestFramework):
|
class ReceivedByTest(BitcoinTestFramework):
|
||||||
|
@ -127,7 +131,7 @@ class ReceivedByTest(BitcoinTestFramework):
|
||||||
# set pre-state
|
# set pre-state
|
||||||
label = ''
|
label = ''
|
||||||
address = self.nodes[1].getnewaddress()
|
address = self.nodes[1].getnewaddress()
|
||||||
assert_equal(self.nodes[1].getaddressinfo(address)['label'], label)
|
test_address(self.nodes[1], address, label=label, labels=labels_value(name=label))
|
||||||
received_by_label_json = [r for r in self.nodes[1].listreceivedbylabel() if r["label"] == label][0]
|
received_by_label_json = [r for r in self.nodes[1].listreceivedbylabel() if r["label"] == label][0]
|
||||||
balance_by_label = self.nodes[1].getreceivedbylabel(label)
|
balance_by_label = self.nodes[1].getreceivedbylabel(label)
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue