Changeset View
Changeset View
Standalone View
Standalone View
test/functional/wallet_labels.py
#!/usr/bin/env python3 | #!/usr/bin/env python3 | ||||
# Copyright (c) 2016-2017 The Bitcoin Core developers | # Copyright (c) 2016-2017 The Bitcoin Core developers | ||||
# Distributed under the MIT software license, see the accompanying | # Distributed under the MIT software license, see the accompanying | ||||
# file COPYING or http://www.opensource.org/licenses/mit-license.php. | # file COPYING or http://www.opensource.org/licenses/mit-license.php. | ||||
"""Test label RPCs. | """Test label RPCs. | ||||
RPCs tested are: | RPCs tested are: | ||||
- getlabeladdress | - getaccountaddress | ||||
- getaddressesbyaccount/getaddressesbylabel | - getaddressesbyaccount/getaddressesbylabel | ||||
- listaddressgroupings | - listaddressgroupings | ||||
- setlabel | - setlabel | ||||
- sendfrom (with account arguments) | - sendfrom (with account arguments) | ||||
- move (with account arguments) | - move (with account arguments) | ||||
Run the test twice - once using the accounts API and once using the labels API. | Run the test twice - once using the accounts API and once using the labels API. | ||||
The accounts API test can be removed in V0.21. | The accounts API test can be removed in V0.21. | ||||
▲ Show 20 Lines • Show All 68 Lines • ▼ Show 20 Lines | def _run_subtest(self, accounts_api, node): | ||||
amount_to_send = 1.0 | amount_to_send = 1.0 | ||||
# Create labels and make sure subsequent label API calls | # Create labels and make sure subsequent label API calls | ||||
# recognize the label/address associations. | # recognize the label/address associations. | ||||
labels = [Label(name, accounts_api) | labels = [Label(name, accounts_api) | ||||
for name in ("a", "b", "c", "d", "e")] | for name in ("a", "b", "c", "d", "e")] | ||||
for label in labels: | for label in labels: | ||||
label.add_receive_address( | if accounts_api: | ||||
node.getlabeladdress(label=label.name, force=True)) | address = node.getaccountaddress(label.name) | ||||
else: | |||||
address = node.getnewaddress(label.name) | |||||
label.add_receive_address(address) | |||||
label.verify(node) | label.verify(node) | ||||
# Check all labels are returned by listlabels. | # Check all labels are returned by listlabels. | ||||
assert_equal(node.listlabels(), [label.name for label in labels]) | assert_equal(node.listlabels(), [label.name for label in labels]) | ||||
# Send a transaction to each label, and make sure this forces | # Send a transaction to each label, and make sure this forces | ||||
# getlabeladdress to generate a new receiving address. | # getaccountaddress to generate a new receiving address. | ||||
for label in labels: | for label in labels: | ||||
if accounts_api: | |||||
node.sendtoaddress(label.receive_address, amount_to_send) | node.sendtoaddress(label.receive_address, amount_to_send) | ||||
label.add_receive_address(node.getlabeladdress(label.name)) | label.add_receive_address(node.getaccountaddress(label.name)) | ||||
else: | |||||
node.sendtoaddress(label.addresses[0], amount_to_send) | |||||
label.verify(node) | label.verify(node) | ||||
# Check the amounts received. | # Check the amounts received. | ||||
node.generate(1) | node.generate(1) | ||||
for label in labels: | for label in labels: | ||||
assert_equal( | assert_equal( | ||||
node.getreceivedbyaddress(label.addresses[0]), amount_to_send) | node.getreceivedbyaddress(label.addresses[0]), amount_to_send) | ||||
assert_equal(node.getreceivedbylabel(label.name), amount_to_send) | assert_equal(node.getreceivedbylabel(label.name), amount_to_send) | ||||
# Check that sendfrom label reduces listaccounts balances. | # Check that sendfrom label reduces listaccounts balances. | ||||
for i, label in enumerate(labels): | for i, label in enumerate(labels): | ||||
to_label = labels[(i + 1) % len(labels)] | to_label = labels[(i + 1) % len(labels)] | ||||
node.sendfrom(label.name, to_label.receive_address, amount_to_send) | if accounts_api: | ||||
node.sendfrom( | |||||
label.name, to_label.receive_address, amount_to_send) | |||||
else: | |||||
node.sendfrom( | |||||
label.name, to_label.addresses[0], amount_to_send) | |||||
node.generate(1) | node.generate(1) | ||||
for label in labels: | for label in labels: | ||||
label.add_receive_address(node.getlabeladdress(label.name)) | if accounts_api: | ||||
address = node.getaccountaddress(label.name) | |||||
else: | |||||
address = node.getnewaddress(label.name) | |||||
label.add_receive_address(address) | |||||
label.verify(node) | label.verify(node) | ||||
assert_equal(node.getreceivedbylabel(label.name), 2) | assert_equal(node.getreceivedbylabel(label.name), 2) | ||||
if accounts_api: | if accounts_api: | ||||
node.move(label.name, "", node.getbalance(label.name)) | node.move(label.name, "", node.getbalance(label.name)) | ||||
label.verify(node) | label.verify(node) | ||||
node.generate(101) | node.generate(101) | ||||
expected_account_balances = {"": 5200 + fee} | expected_account_balances = {"": 5200 + fee} | ||||
for label in labels: | for label in labels: | ||||
expected_account_balances[label.name] = 0 | expected_account_balances[label.name] = 0 | ||||
if accounts_api: | if accounts_api: | ||||
assert_equal(node.listaccounts(), expected_account_balances) | assert_equal(node.listaccounts(), expected_account_balances) | ||||
assert_equal(node.getbalance(""), 5200 + fee) | assert_equal(node.getbalance(""), 5200 + fee) | ||||
# Check that setlabel can assign a label to a new unused address. | # Check that setlabel can assign a label to a new unused address. | ||||
for label in labels: | for label in labels: | ||||
address = node.getlabeladdress(label="", force=True) | address = node.getnewaddress() | ||||
node.setlabel(address, label.name) | node.setlabel(address, label.name) | ||||
label.add_address(address) | label.add_address(address) | ||||
label.verify(node) | label.verify(node) | ||||
if accounts_api: | if accounts_api: | ||||
assert(address not in node.getaddressesbyaccount("")) | assert address not in node.getaddressesbyaccount("") | ||||
else: | else: | ||||
assert_raises_rpc_error(-11, "No addresses with label", | assert_raises_rpc_error(-11, "No addresses with label", | ||||
node.getaddressesbylabel, "") | node.getaddressesbylabel, "") | ||||
# Check that addmultisigaddress can assign labels. | # Check that addmultisigaddress can assign labels. | ||||
for label in labels: | for label in labels: | ||||
addresses = [] | addresses = [] | ||||
for x in range(10): | for x in range(10): | ||||
addresses.append(node.getnewaddress()) | addresses.append(node.getnewaddress()) | ||||
multisig_address = node.addmultisigaddress( | multisig_address = node.addmultisigaddress( | ||||
5, addresses, label.name)['address'] | 5, addresses, label.name)['address'] | ||||
label.add_address(multisig_address) | label.add_address(multisig_address) | ||||
label.purpose[multisig_address] = "send" | label.purpose[multisig_address] = "send" | ||||
label.verify(node) | label.verify(node) | ||||
node.sendfrom("", multisig_address, 50) | node.sendfrom("", multisig_address, 50) | ||||
node.generate(101) | node.generate(101) | ||||
if accounts_api: | if accounts_api: | ||||
for label in labels: | for label in labels: | ||||
assert_equal(node.getbalance(label.name), 50) | assert_equal(node.getbalance(label.name), 50) | ||||
# Check that setlabel can change the label of an address from a | # Check that setlabel can change the label of an address from a | ||||
# different label. | # different label. | ||||
change_label(node, labels[0].addresses[0], labels[0], labels[1]) | change_label(node, labels[0].addresses[0], | ||||
labels[0], labels[1], accounts_api) | |||||
# Check that setlabel can change the label of an address which | |||||
# is the receiving address of a different label. | |||||
change_label(node, labels[0].receive_address, labels[0], labels[1]) | |||||
# Check that setlabel can set the label of an address already | # Check that setlabel can set the label of an address already | ||||
# in the label. This is a no-op. | # in the label. This is a no-op. | ||||
change_label(node, labels[2].addresses[0], labels[2], labels[2]) | change_label(node, labels[2].addresses[0], | ||||
labels[2], labels[2], accounts_api) | |||||
# Check that setlabel can set the label of an address which is | if accounts_api: | ||||
# Check that setaccount can change the label of an address which | |||||
# is the receiving address of a different label. | |||||
change_label(node, labels[0].receive_address, | |||||
labels[0], labels[1], accounts_api) | |||||
# Check that setaccount can set the label of an address which is | |||||
# already the receiving address of the label. This is a no-op. | # already the receiving address of the label. This is a no-op. | ||||
change_label(node, labels[2].receive_address, labels[2], labels[2]) | change_label(node, labels[2].receive_address, | ||||
labels[2], labels[2], accounts_api) | |||||
class Label: | class Label: | ||||
def __init__(self, name, accounts_api): | def __init__(self, name, accounts_api): | ||||
# Label name | # Label name | ||||
self.name = name | self.name = name | ||||
self.accounts_api = accounts_api | self.accounts_api = accounts_api | ||||
# Current receiving address associated with this label. | # Current receiving address associated with this label. | ||||
self.receive_address = None | self.receive_address = None | ||||
# List of all addresses assigned with this label | # List of all addresses assigned with this label | ||||
self.addresses = [] | self.addresses = [] | ||||
# Map of address to address purpose | # Map of address to address purpose | ||||
self.purpose = defaultdict(lambda: "receive") | self.purpose = defaultdict(lambda: "receive") | ||||
def add_address(self, address): | def add_address(self, address): | ||||
assert_equal(address not in self.addresses, True) | assert_equal(address not in self.addresses, True) | ||||
self.addresses.append(address) | self.addresses.append(address) | ||||
def add_receive_address(self, address): | def add_receive_address(self, address): | ||||
self.add_address(address) | self.add_address(address) | ||||
if self.accounts_api: | |||||
self.receive_address = address | self.receive_address = address | ||||
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 | ||||
assert_equal(node.getlabeladdress(self.name), self.receive_address) | if self.accounts_api: | ||||
assert_equal(node.getaccountaddress( | |||||
self.name), self.receive_address) | |||||
for address in self.addresses: | for address in self.addresses: | ||||
assert_equal( | assert_equal( | ||||
node.getaddressinfo(address)['labels'][0], | node.getaddressinfo(address)['labels'][0], | ||||
{"name": self.name, | {"name": self.name, | ||||
"purpose": self.purpose[address]}) | "purpose": self.purpose[address]}) | ||||
if self.accounts_api: | if self.accounts_api: | ||||
assert_equal(node.getaccount(address), self.name) | assert_equal(node.getaccount(address), self.name) | ||||
else: | else: | ||||
assert_equal(node.getaddressinfo(address)['label'], self.name) | assert_equal(node.getaddressinfo(address)['label'], self.name) | ||||
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}) | ||||
if self.accounts_api: | if self.accounts_api: | ||||
assert_equal(set(node.getaddressesbyaccount( | assert_equal(set(node.getaddressesbyaccount( | ||||
self.name)), set(self.addresses)) | self.name)), set(self.addresses)) | ||||
def change_label(node, address, old_label, new_label): | def change_label(node, address, old_label, new_label, accounts_api): | ||||
assert_equal(address in old_label.addresses, True) | assert_equal(address in old_label.addresses, True) | ||||
if accounts_api: | |||||
node.setaccount(address, new_label.name) | |||||
else: | |||||
node.setlabel(address, new_label.name) | node.setlabel(address, new_label.name) | ||||
old_label.addresses.remove(address) | old_label.addresses.remove(address) | ||||
new_label.add_address(address) | new_label.add_address(address) | ||||
# Calling setlabel on an address which was previously the receiving | # Calling setaccount on an address which was previously the receiving | ||||
# address of a different label should reset the receiving address of | # address of a different account should reset the receiving address of | ||||
# the old label, causing getlabeladdress to return a brand new | # the old account, causing getaccountaddress to return a brand new | ||||
# address. | # address. | ||||
if accounts_api: | |||||
if old_label.name != new_label.name and address == old_label.receive_address: | if old_label.name != new_label.name and address == old_label.receive_address: | ||||
new_address = node.getlabeladdress(old_label.name) | new_address = node.getaccountaddress(old_label.name) | ||||
assert_equal(new_address not in old_label.addresses, True) | assert_equal(new_address not in old_label.addresses, True) | ||||
assert_equal(new_address not in new_label.addresses, True) | assert_equal(new_address not in new_label.addresses, True) | ||||
old_label.add_receive_address(new_address) | old_label.add_receive_address(new_address) | ||||
old_label.verify(node) | old_label.verify(node) | ||||
new_label.verify(node) | new_label.verify(node) | ||||
if __name__ == '__main__': | if __name__ == '__main__': | ||||
WalletLabelsTest().main() | WalletLabelsTest().main() |