moved all other tests out of test_chris45.py and into dedicated files
This commit is contained in:
parent
a3f59f041a
commit
5366779b1b
1 changed files with 1 additions and 582 deletions
|
@ -1,114 +1,9 @@
|
||||||
import json
|
|
||||||
import tempfile
|
import tempfile
|
||||||
import logging
|
from .testcase import CommandTestCase
|
||||||
from binascii import unhexlify
|
|
||||||
|
|
||||||
from lbrynet.extras.wallet.transaction import Transaction
|
|
||||||
from lbrynet.error import InsufficientFundsError
|
|
||||||
from lbrynet.schema.claim import ClaimDict
|
|
||||||
|
|
||||||
from torba.testcase import IntegrationTestCase
|
|
||||||
|
|
||||||
import lbrynet.schema
|
|
||||||
lbrynet.schema.BLOCKCHAIN_NAME = 'lbrycrd_regtest'
|
|
||||||
|
|
||||||
from lbrynet.conf import Config
|
|
||||||
from lbrynet.extras.daemon.Daemon import Daemon, jsonrpc_dumps_pretty
|
|
||||||
from lbrynet.extras.wallet import LbryWalletManager
|
|
||||||
from lbrynet.extras.daemon.Components import WalletComponent
|
|
||||||
from lbrynet.extras.daemon.Components import (
|
|
||||||
DHT_COMPONENT, HASH_ANNOUNCER_COMPONENT, PEER_PROTOCOL_SERVER_COMPONENT,
|
|
||||||
UPNP_COMPONENT, EXCHANGE_RATE_MANAGER_COMPONENT
|
|
||||||
)
|
|
||||||
from lbrynet.extras.daemon.ComponentManager import ComponentManager
|
|
||||||
|
|
||||||
|
|
||||||
class CommandTestCase(IntegrationTestCase):
|
|
||||||
|
|
||||||
timeout = 180
|
|
||||||
MANAGER = LbryWalletManager
|
|
||||||
VERBOSITY = logging.WARN
|
|
||||||
|
|
||||||
async def asyncSetUp(self):
|
|
||||||
await super().asyncSetUp()
|
|
||||||
|
|
||||||
logging.getLogger('lbrynet.blob_exchange').setLevel(self.VERBOSITY)
|
|
||||||
logging.getLogger('lbrynet.daemon').setLevel(self.VERBOSITY)
|
|
||||||
|
|
||||||
conf = Config()
|
|
||||||
conf.data_dir = self.wallet_node.data_path
|
|
||||||
conf.wallet_dir = self.wallet_node.data_path
|
|
||||||
conf.download_dir = self.wallet_node.data_path
|
|
||||||
conf.share_usage_data = False
|
|
||||||
conf.use_upnp = False
|
|
||||||
conf.reflect_streams = False
|
|
||||||
conf.blockchain_name = 'lbrycrd_regtest'
|
|
||||||
conf.lbryum_servers = [('localhost', 50001)]
|
|
||||||
conf.reflector_servers = []
|
|
||||||
conf.known_dht_nodes = []
|
|
||||||
|
|
||||||
await self.account.ensure_address_gap()
|
|
||||||
address = (await self.account.receiving.get_addresses(limit=1, only_usable=True))[0]
|
|
||||||
sendtxid = await self.blockchain.send_to_address(address, 10)
|
|
||||||
await self.confirm_tx(sendtxid)
|
|
||||||
await self.generate(5)
|
|
||||||
|
|
||||||
def wallet_maker(component_manager):
|
|
||||||
self.wallet_component = WalletComponent(component_manager)
|
|
||||||
self.wallet_component.wallet_manager = self.manager
|
|
||||||
self.wallet_component._running = True
|
|
||||||
return self.wallet_component
|
|
||||||
|
|
||||||
conf.components_to_skip = [
|
|
||||||
DHT_COMPONENT, UPNP_COMPONENT, HASH_ANNOUNCER_COMPONENT,
|
|
||||||
PEER_PROTOCOL_SERVER_COMPONENT, EXCHANGE_RATE_MANAGER_COMPONENT
|
|
||||||
]
|
|
||||||
self.daemon = Daemon(conf, ComponentManager(
|
|
||||||
conf, skip_components=conf.components_to_skip, wallet=wallet_maker
|
|
||||||
))
|
|
||||||
await self.daemon.initialize()
|
|
||||||
self.manager.old_db = self.daemon.storage
|
|
||||||
|
|
||||||
async def asyncTearDown(self):
|
|
||||||
await super().asyncTearDown()
|
|
||||||
self.wallet_component._running = False
|
|
||||||
await self.daemon.stop()
|
|
||||||
|
|
||||||
async def confirm_tx(self, txid):
|
|
||||||
""" Wait for tx to be in mempool, then generate a block, wait for tx to be in a block. """
|
|
||||||
await self.on_transaction_id(txid)
|
|
||||||
await self.generate(1)
|
|
||||||
await self.on_transaction_id(txid)
|
|
||||||
|
|
||||||
async def on_transaction_dict(self, tx):
|
|
||||||
await self.ledger.wait(
|
|
||||||
self.ledger.transaction_class(unhexlify(tx['hex']))
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_all_addresses(tx):
|
|
||||||
addresses = set()
|
|
||||||
for txi in tx['inputs']:
|
|
||||||
addresses.add(txi['address'])
|
|
||||||
for txo in tx['outputs']:
|
|
||||||
addresses.add(txo['address'])
|
|
||||||
return list(addresses)
|
|
||||||
|
|
||||||
async def generate(self, blocks):
|
|
||||||
""" Ask lbrycrd to generate some blocks and wait until ledger has them. """
|
|
||||||
await self.blockchain.generate(blocks)
|
|
||||||
await self.ledger.on_header.where(self.blockchain.is_expected_block)
|
|
||||||
|
|
||||||
async def out(self, awaitable):
|
|
||||||
""" Converts Daemon API call results (dictionary)
|
|
||||||
to JSON and then back to a dictionary. """
|
|
||||||
return json.loads(jsonrpc_dumps_pretty(await awaitable, ledger=self.ledger))['result']
|
|
||||||
|
|
||||||
|
|
||||||
class EpicAdventuresOfChris45(CommandTestCase):
|
class EpicAdventuresOfChris45(CommandTestCase):
|
||||||
|
|
||||||
VERBOSITY = logging.WARN
|
|
||||||
|
|
||||||
async def test_no_this_is_not_a_test_its_an_adventure(self):
|
async def test_no_this_is_not_a_test_its_an_adventure(self):
|
||||||
# Chris45 is an avid user of LBRY and this is his story. It's fact and fiction
|
# Chris45 is an avid user of LBRY and this is his story. It's fact and fiction
|
||||||
# and everything in between; it's also the setting of some record setting
|
# and everything in between; it's also the setting of some record setting
|
||||||
|
@ -327,479 +222,3 @@ class EpicAdventuresOfChris45(CommandTestCase):
|
||||||
# He them checks that the claim doesn't resolve anymore.
|
# He them checks that the claim doesn't resolve anymore.
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri=uri))
|
response = await self.out(self.daemon.jsonrpc_resolve(uri=uri))
|
||||||
self.assertNotIn('claim', response[uri])
|
self.assertNotIn('claim', response[uri])
|
||||||
|
|
||||||
|
|
||||||
class AccountManagement(CommandTestCase):
|
|
||||||
|
|
||||||
VERBOSE = False
|
|
||||||
|
|
||||||
async def test_performing_account_management_commands(self):
|
|
||||||
# check initial account
|
|
||||||
response = await self.daemon.jsonrpc_account_list()
|
|
||||||
self.assertEqual(len(response['lbc_regtest']), 1)
|
|
||||||
|
|
||||||
# change account name and gap
|
|
||||||
account_id = response['lbc_regtest'][0]['id']
|
|
||||||
self.daemon.jsonrpc_account_set(
|
|
||||||
account_id=account_id, new_name='test account',
|
|
||||||
receiving_gap=95, receiving_max_uses=96,
|
|
||||||
change_gap=97, change_max_uses=98
|
|
||||||
)
|
|
||||||
response = (await self.daemon.jsonrpc_account_list())['lbc_regtest'][0]
|
|
||||||
self.assertEqual(response['name'], 'test account')
|
|
||||||
self.assertEqual(
|
|
||||||
response['address_generator']['receiving'],
|
|
||||||
{'gap': 95, 'maximum_uses_per_address': 96}
|
|
||||||
)
|
|
||||||
self.assertEqual(
|
|
||||||
response['address_generator']['change'],
|
|
||||||
{'gap': 97, 'maximum_uses_per_address': 98}
|
|
||||||
)
|
|
||||||
|
|
||||||
# create another account
|
|
||||||
await self.daemon.jsonrpc_account_create('second account')
|
|
||||||
response = await self.daemon.jsonrpc_account_list()
|
|
||||||
self.assertEqual(len(response['lbc_regtest']), 2)
|
|
||||||
self.assertEqual(response['lbc_regtest'][1]['name'], 'second account')
|
|
||||||
account_id2 = response['lbc_regtest'][1]['id']
|
|
||||||
|
|
||||||
# make new account the default
|
|
||||||
self.daemon.jsonrpc_account_set(account_id=account_id2, default=True)
|
|
||||||
response = await self.daemon.jsonrpc_account_list(show_seed=True)
|
|
||||||
self.assertEqual(response['lbc_regtest'][0]['name'], 'second account')
|
|
||||||
|
|
||||||
account_seed = response['lbc_regtest'][1]['seed']
|
|
||||||
|
|
||||||
# remove account
|
|
||||||
self.daemon.jsonrpc_account_remove(response['lbc_regtest'][1]['id'])
|
|
||||||
response = await self.daemon.jsonrpc_account_list()
|
|
||||||
self.assertEqual(len(response['lbc_regtest']), 1)
|
|
||||||
|
|
||||||
# add account
|
|
||||||
await self.daemon.jsonrpc_account_add('recreated account', seed=account_seed)
|
|
||||||
response = await self.daemon.jsonrpc_account_list()
|
|
||||||
self.assertEqual(len(response['lbc_regtest']), 2)
|
|
||||||
self.assertEqual(response['lbc_regtest'][1]['name'], 'recreated account')
|
|
||||||
|
|
||||||
# list specific account
|
|
||||||
response = await self.daemon.jsonrpc_account_list(account_id, include_claims=True)
|
|
||||||
self.assertEqual(response['name'], 'recreated account')
|
|
||||||
|
|
||||||
|
|
||||||
class ClaimManagement(CommandTestCase):
|
|
||||||
|
|
||||||
VERBOSITY = logging.WARN
|
|
||||||
|
|
||||||
async def make_claim(self, name='hovercraft', amount='1.0', data=b'hi!', channel_name=None, confirm=True):
|
|
||||||
with tempfile.NamedTemporaryFile() as file:
|
|
||||||
file.write(data)
|
|
||||||
file.flush()
|
|
||||||
claim = await self.out(self.daemon.jsonrpc_publish(
|
|
||||||
name, amount, file_path=file.name, channel_name=channel_name
|
|
||||||
))
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
if confirm:
|
|
||||||
await self.on_transaction_dict(claim['tx'])
|
|
||||||
await self.generate(1)
|
|
||||||
await self.on_transaction_dict(claim['tx'])
|
|
||||||
return claim
|
|
||||||
|
|
||||||
async def craft_claim(self, name, amount_dewies, claim_dict, address):
|
|
||||||
# FIXME: this is here mostly because publish has defensive code for situations that happens accidentally
|
|
||||||
# However, it still happens... So, let's reproduce them.
|
|
||||||
claim = ClaimDict.load_dict(claim_dict)
|
|
||||||
address = address or (await self.account.receiving.get_addresses(limit=1, only_usable=True))[0]
|
|
||||||
tx = await Transaction.claim(name, claim, amount_dewies, address, [self.account], self.account)
|
|
||||||
await self.broadcast(tx)
|
|
||||||
await self.ledger.wait(tx)
|
|
||||||
await self.generate(1)
|
|
||||||
await self.ledger.wait(tx)
|
|
||||||
return tx
|
|
||||||
|
|
||||||
async def test_create_update_and_abandon_claim(self):
|
|
||||||
self.assertEqual('10.0', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
claim = await self.make_claim(amount='2.5') # creates new claim
|
|
||||||
txs = await self.out(self.daemon.jsonrpc_transaction_list())
|
|
||||||
self.assertEqual(len(txs[0]['claim_info']), 1)
|
|
||||||
self.assertEqual(txs[0]['confirmations'], 1)
|
|
||||||
self.assertEqual(txs[0]['claim_info'][0]['balance_delta'], '-2.5')
|
|
||||||
self.assertEqual(txs[0]['claim_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs[0]['value'], '0.0')
|
|
||||||
self.assertEqual(txs[0]['fee'], '-0.020107')
|
|
||||||
self.assertEqual('7.479893', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
await self.make_claim(amount='1.0') # updates previous claim
|
|
||||||
txs = await self.out(self.daemon.jsonrpc_transaction_list())
|
|
||||||
self.assertEqual(len(txs[0]['update_info']), 1)
|
|
||||||
self.assertEqual(txs[0]['update_info'][0]['balance_delta'], '1.5')
|
|
||||||
self.assertEqual(txs[0]['update_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs[0]['value'], '0.0')
|
|
||||||
self.assertEqual(txs[0]['fee'], '-0.0001985')
|
|
||||||
self.assertEqual('8.9796945', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
await self.out(self.daemon.jsonrpc_claim_abandon(claim['claim_id']))
|
|
||||||
txs = await self.out(self.daemon.jsonrpc_transaction_list())
|
|
||||||
self.assertEqual(len(txs[0]['abandon_info']), 1)
|
|
||||||
self.assertEqual(txs[0]['abandon_info'][0]['balance_delta'], '1.0')
|
|
||||||
self.assertEqual(txs[0]['abandon_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs[0]['value'], '0.0')
|
|
||||||
self.assertEqual(txs[0]['fee'], '-0.000107')
|
|
||||||
self.assertEqual('9.9795875', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
async def test_update_claim_holding_address(self):
|
|
||||||
other_account_id = (await self.daemon.jsonrpc_account_create('second account'))['id']
|
|
||||||
other_account = self.daemon.get_account_or_error(other_account_id)
|
|
||||||
other_address = await other_account.receiving.get_or_create_usable_address()
|
|
||||||
|
|
||||||
self.assertEqual('10.0', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
# create the initial name claim
|
|
||||||
claim = await self.make_claim()
|
|
||||||
|
|
||||||
self.assertEqual(len(await self.daemon.jsonrpc_claim_list_mine()), 1)
|
|
||||||
self.assertEqual(len(await self.daemon.jsonrpc_claim_list_mine(account_id=other_account_id)), 0)
|
|
||||||
tx = await self.daemon.jsonrpc_claim_send_to_address(
|
|
||||||
claim['claim_id'], other_address
|
|
||||||
)
|
|
||||||
await self.ledger.wait(tx)
|
|
||||||
self.assertEqual(len(await self.daemon.jsonrpc_claim_list_mine()), 0)
|
|
||||||
self.assertEqual(len(await self.daemon.jsonrpc_claim_list_mine(account_id=other_account_id)), 1)
|
|
||||||
|
|
||||||
async def test_publishing_checks_all_accounts_for_certificate(self):
|
|
||||||
account1_id, account1 = self.account.id, self.account
|
|
||||||
new_account = await self.daemon.jsonrpc_account_create('second account')
|
|
||||||
account2_id, account2 = new_account['id'], self.daemon.get_account_or_error(new_account['id'])
|
|
||||||
|
|
||||||
spam_channel = await self.out(self.daemon.jsonrpc_channel_new('@spam', '1.0'))
|
|
||||||
self.assertTrue(spam_channel['success'])
|
|
||||||
await self.confirm_tx(spam_channel['tx']['txid'])
|
|
||||||
|
|
||||||
self.assertEqual('8.989893', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
result = await self.out(self.daemon.jsonrpc_wallet_send(
|
|
||||||
'5.0', await self.daemon.jsonrpc_address_unused(account2_id)
|
|
||||||
))
|
|
||||||
await self.confirm_tx(result['txid'])
|
|
||||||
|
|
||||||
self.assertEqual('3.989769', await self.daemon.jsonrpc_account_balance())
|
|
||||||
self.assertEqual('5.0', await self.daemon.jsonrpc_account_balance(account2_id))
|
|
||||||
|
|
||||||
baz_channel = await self.out(self.daemon.jsonrpc_channel_new('@baz', '1.0', account2_id))
|
|
||||||
self.assertTrue(baz_channel['success'])
|
|
||||||
await self.confirm_tx(baz_channel['tx']['txid'])
|
|
||||||
|
|
||||||
channels = await self.out(self.daemon.jsonrpc_channel_list(account1_id))
|
|
||||||
self.assertEqual(len(channels), 1)
|
|
||||||
self.assertEqual(channels[0]['name'], '@spam')
|
|
||||||
self.assertEqual(channels, await self.out(self.daemon.jsonrpc_channel_list()))
|
|
||||||
|
|
||||||
channels = await self.out(self.daemon.jsonrpc_channel_list(account2_id))
|
|
||||||
self.assertEqual(len(channels), 1)
|
|
||||||
self.assertEqual(channels[0]['name'], '@baz')
|
|
||||||
|
|
||||||
# defaults to using all accounts to lookup channel
|
|
||||||
with tempfile.NamedTemporaryFile() as file:
|
|
||||||
file.write(b'hi!')
|
|
||||||
file.flush()
|
|
||||||
claim1 = await self.out(self.daemon.jsonrpc_publish(
|
|
||||||
'hovercraft', '1.0', file_path=file.name, channel_name='@baz'
|
|
||||||
))
|
|
||||||
self.assertTrue(claim1['success'])
|
|
||||||
await self.confirm_tx(claim1['tx']['txid'])
|
|
||||||
|
|
||||||
# uses only the specific accounts which contains the channel
|
|
||||||
with tempfile.NamedTemporaryFile() as file:
|
|
||||||
file.write(b'hi!')
|
|
||||||
file.flush()
|
|
||||||
claim1 = await self.out(self.daemon.jsonrpc_publish(
|
|
||||||
'hovercraft', '1.0', file_path=file.name,
|
|
||||||
channel_name='@baz', channel_account_id=[account2_id]
|
|
||||||
))
|
|
||||||
self.assertTrue(claim1['success'])
|
|
||||||
await self.confirm_tx(claim1['tx']['txid'])
|
|
||||||
|
|
||||||
# fails when specifying account which does not contain channel
|
|
||||||
with tempfile.NamedTemporaryFile() as file:
|
|
||||||
file.write(b'hi!')
|
|
||||||
file.flush()
|
|
||||||
with self.assertRaisesRegex(ValueError, "Couldn't find channel with name '@baz'."):
|
|
||||||
await self.out(self.daemon.jsonrpc_publish(
|
|
||||||
'hovercraft', '1.0', file_path=file.name,
|
|
||||||
channel_name='@baz', channel_account_id=[account1_id]
|
|
||||||
))
|
|
||||||
|
|
||||||
async def test_updating_claim_includes_claim_value_in_balance_check(self):
|
|
||||||
self.assertEqual('10.0', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
await self.make_claim(amount='9.0')
|
|
||||||
self.assertEqual('0.979893', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
# update the same claim
|
|
||||||
await self.make_claim(amount='9.0')
|
|
||||||
self.assertEqual('0.9796205', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
# update the claim a second time but use even more funds
|
|
||||||
await self.make_claim(amount='9.97')
|
|
||||||
self.assertEqual('0.009348', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
# fails when specifying more than available
|
|
||||||
with tempfile.NamedTemporaryFile() as file:
|
|
||||||
file.write(b'hi!')
|
|
||||||
file.flush()
|
|
||||||
with self.assertRaisesRegex(
|
|
||||||
InsufficientFundsError,
|
|
||||||
"Please lower the bid value, the maximum amount"
|
|
||||||
" you can specify for this claim is 9.979274."
|
|
||||||
):
|
|
||||||
await self.out(self.daemon.jsonrpc_publish(
|
|
||||||
'hovercraft', '9.98', file_path=file.name
|
|
||||||
))
|
|
||||||
|
|
||||||
async def test_abandoning_claim_at_loss(self):
|
|
||||||
self.assertEqual('10.0', await self.daemon.jsonrpc_account_balance())
|
|
||||||
claim = await self.make_claim(amount='0.0001')
|
|
||||||
self.assertEqual('9.979793', await self.daemon.jsonrpc_account_balance())
|
|
||||||
await self.out(self.daemon.jsonrpc_claim_abandon(claim['claim_id']))
|
|
||||||
self.assertEqual('9.97968399', await self.daemon.jsonrpc_account_balance())
|
|
||||||
|
|
||||||
async def test_claim_show(self):
|
|
||||||
channel = await self.out(self.daemon.jsonrpc_channel_new('@abc', "1.0"))
|
|
||||||
self.assertTrue(channel['success'])
|
|
||||||
await self.confirm_tx(channel['tx']['txid'])
|
|
||||||
channel_from_claim_show = await self.out(
|
|
||||||
self.daemon.jsonrpc_claim_show(txid=channel['tx']['txid'], nout=channel['output']['nout'])
|
|
||||||
)
|
|
||||||
self.assertEqual(channel_from_claim_show['value'], channel['output']['value'])
|
|
||||||
channel_from_claim_show = await self.out(
|
|
||||||
self.daemon.jsonrpc_claim_show(claim_id=channel['claim_id'])
|
|
||||||
)
|
|
||||||
self.assertEqual(channel_from_claim_show['value'], channel['output']['value'])
|
|
||||||
|
|
||||||
abandon = await self.out(self.daemon.jsonrpc_claim_abandon(txid=channel['tx']['txid'], nout=0, blocking=False))
|
|
||||||
self.assertTrue(abandon['success'])
|
|
||||||
await self.confirm_tx(abandon['tx']['txid'])
|
|
||||||
not_a_claim = await self.out(
|
|
||||||
self.daemon.jsonrpc_claim_show(txid=abandon['tx']['txid'], nout=0)
|
|
||||||
)
|
|
||||||
self.assertEqual(not_a_claim, 'claim not found')
|
|
||||||
|
|
||||||
async def test_claim_list(self):
|
|
||||||
channel = await self.out(self.daemon.jsonrpc_channel_new('@abc', "1.0"))
|
|
||||||
self.assertTrue(channel['success'])
|
|
||||||
await self.confirm_tx(channel['tx']['txid'])
|
|
||||||
claim = await self.make_claim(amount='0.0001', name='on-channel-claim', channel_name='@abc')
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
unsigned_claim = await self.make_claim(amount='0.0001', name='unsigned')
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
|
|
||||||
channel_from_claim_list = await self.out(self.daemon.jsonrpc_claim_list('@abc'))
|
|
||||||
self.assertEqual(channel_from_claim_list['claims'][0]['value'], channel['output']['value'])
|
|
||||||
signed_claim_from_claim_list = await self.out(self.daemon.jsonrpc_claim_list('on-channel-claim'))
|
|
||||||
self.assertEqual(signed_claim_from_claim_list['claims'][0]['value'], claim['output']['value'])
|
|
||||||
unsigned_claim_from_claim_list = await self.out(self.daemon.jsonrpc_claim_list('unsigned'))
|
|
||||||
self.assertEqual(unsigned_claim_from_claim_list['claims'][0]['value'], unsigned_claim['output']['value'])
|
|
||||||
|
|
||||||
abandon = await self.out(self.daemon.jsonrpc_claim_abandon(txid=channel['tx']['txid'], nout=0, blocking=False))
|
|
||||||
self.assertTrue(abandon['success'])
|
|
||||||
await self.confirm_tx(abandon['tx']['txid'])
|
|
||||||
|
|
||||||
empty = await self.out(self.daemon.jsonrpc_claim_list('@abc'))
|
|
||||||
self.assertEqual(len(empty['claims']), 0)
|
|
||||||
|
|
||||||
async def test_abandoned_channel_with_signed_claims(self):
|
|
||||||
channel = await self.out(self.daemon.jsonrpc_channel_new('@abc', "1.0"))
|
|
||||||
self.assertTrue(channel['success'])
|
|
||||||
await self.confirm_tx(channel['tx']['txid'])
|
|
||||||
claim = await self.make_claim(amount='0.0001', name='on-channel-claim', channel_name='@abc')
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
abandon = await self.out(self.daemon.jsonrpc_claim_abandon(txid=channel['tx']['txid'], nout=0, blocking=False))
|
|
||||||
self.assertTrue(abandon['success'])
|
|
||||||
channel = await self.out(self.daemon.jsonrpc_channel_new('@abc', "1.0"))
|
|
||||||
self.assertTrue(channel['success'])
|
|
||||||
await self.confirm_tx(channel['tx']['txid'])
|
|
||||||
|
|
||||||
# Original channel doesnt exists anymore, so the signature is invalid. For invalid signatures, resolution is
|
|
||||||
# only possible outside a channel
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri='lbry://@abc/on-channel-claim'))
|
|
||||||
self.assertNotIn('claim', response['lbry://@abc/on-channel-claim'])
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri='lbry://on-channel-claim'))
|
|
||||||
self.assertIn('claim', response['lbry://on-channel-claim'])
|
|
||||||
self.assertFalse(response['lbry://on-channel-claim']['claim']['signature_is_valid'])
|
|
||||||
direct_uri = 'lbry://on-channel-claim#' + claim['claim_id']
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri=direct_uri))
|
|
||||||
self.assertIn('claim', response[direct_uri])
|
|
||||||
self.assertFalse(response[direct_uri]['claim']['signature_is_valid'])
|
|
||||||
|
|
||||||
uri = 'lbry://@abc/on-channel-claim'
|
|
||||||
# now, claim something on this channel (it will update the invalid claim, but we save and forcefully restore)
|
|
||||||
original_claim = await self.make_claim(amount='0.00000001', name='on-channel-claim', channel_name='@abc')
|
|
||||||
self.assertTrue(original_claim['success'])
|
|
||||||
# resolves normally
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri=uri))
|
|
||||||
self.assertIn('claim', response[uri])
|
|
||||||
self.assertTrue(response[uri]['claim']['signature_is_valid'])
|
|
||||||
|
|
||||||
# tamper it, invalidating the signature
|
|
||||||
value = response[uri]['claim']['value'].copy()
|
|
||||||
value['stream']['metadata']['author'] = 'some troll'
|
|
||||||
address = response[uri]['claim']['address']
|
|
||||||
await self.craft_claim('on-channel-claim', 1, value, address)
|
|
||||||
# it resolves to the now only valid claim under the channel, ignoring the fake one
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri=uri))
|
|
||||||
self.assertIn('claim', response[uri])
|
|
||||||
self.assertTrue(response[uri]['claim']['signature_is_valid'])
|
|
||||||
|
|
||||||
# ooops! claimed a valid conflict! (this happens on the wild, mostly by accident or race condition)
|
|
||||||
await self.craft_claim('on-channel-claim', 1, response[uri]['claim']['value'], address)
|
|
||||||
|
|
||||||
# it still resolves! but to the older claim
|
|
||||||
response = await self.out(self.daemon.jsonrpc_resolve(uri=uri))
|
|
||||||
self.assertIn('claim', response[uri])
|
|
||||||
self.assertTrue(response[uri]['claim']['signature_is_valid'])
|
|
||||||
self.assertEqual(response[uri]['claim']['txid'], original_claim['tx']['txid'])
|
|
||||||
|
|
||||||
async def test_claim_list_by_channel(self):
|
|
||||||
self.maxDiff = None
|
|
||||||
tx = await self.daemon.jsonrpc_account_fund(None, None, '0.001', outputs=100, broadcast=True)
|
|
||||||
await self.ledger.wait(tx)
|
|
||||||
await self.generate(1)
|
|
||||||
await self.ledger.wait(tx)
|
|
||||||
channel = await self.out(self.daemon.jsonrpc_channel_new('@abc', "0.0001"))
|
|
||||||
self.assertTrue(channel['success'])
|
|
||||||
await self.confirm_tx(channel['tx']['txid'])
|
|
||||||
|
|
||||||
# 4 claims per block, 3 blocks. Sorted by height (descending) then claim_id (ascending).
|
|
||||||
claims = []
|
|
||||||
for j in range(3):
|
|
||||||
same_height_claims = []
|
|
||||||
for k in range(3):
|
|
||||||
claim = await self.make_claim(amount='0.000001', name=f'c{j}-{k}', channel_name='@abc', confirm=False)
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
same_height_claims.append(claim['claim_id'])
|
|
||||||
await self.on_transaction_dict(claim['tx'])
|
|
||||||
claim = await self.make_claim(amount='0.000001', name=f'c{j}-4', channel_name='@abc', confirm=True)
|
|
||||||
self.assertTrue(claim['success'])
|
|
||||||
same_height_claims.append(claim['claim_id'])
|
|
||||||
same_height_claims.sort(key=lambda x: int(x, 16))
|
|
||||||
claims = same_height_claims + claims
|
|
||||||
|
|
||||||
page = await self.out(self.daemon.jsonrpc_claim_list_by_channel(1, page_size=20, uri='@abc'))
|
|
||||||
page_claim_ids = [item['claim_id'] for item in page['@abc']['claims_in_channel']]
|
|
||||||
self.assertEqual(page_claim_ids, claims)
|
|
||||||
page = await self.out(self.daemon.jsonrpc_claim_list_by_channel(1, page_size=6, uri='@abc'))
|
|
||||||
page_claim_ids = [item['claim_id'] for item in page['@abc']['claims_in_channel']]
|
|
||||||
self.assertEqual(page_claim_ids, claims[:6])
|
|
||||||
out_of_bounds = await self.out(self.daemon.jsonrpc_claim_list_by_channel(2, page_size=20, uri='@abc'))
|
|
||||||
self.assertEqual(out_of_bounds['error'], 'claim 20 greater than max 12')
|
|
||||||
|
|
||||||
async def test_regular_supports_and_tip_supports(self):
|
|
||||||
# account2 will be used to send tips and supports to account1
|
|
||||||
account2_id = (await self.daemon.jsonrpc_account_create('second account'))['id']
|
|
||||||
|
|
||||||
# send account2 5 LBC out of the 10 LBC in account1
|
|
||||||
result = await self.out(self.daemon.jsonrpc_wallet_send(
|
|
||||||
'5.0', await self.daemon.jsonrpc_address_unused(account2_id)
|
|
||||||
))
|
|
||||||
await self.confirm_tx(result['txid'])
|
|
||||||
|
|
||||||
# account1 and account2 balances:
|
|
||||||
self.assertEqual('4.999876', await self.daemon.jsonrpc_account_balance())
|
|
||||||
self.assertEqual('5.0', await self.daemon.jsonrpc_account_balance(account2_id))
|
|
||||||
|
|
||||||
# create the claim we'll be tipping and supporting
|
|
||||||
claim = await self.make_claim()
|
|
||||||
|
|
||||||
# account1 and account2 balances:
|
|
||||||
self.assertEqual('3.979769', await self.daemon.jsonrpc_account_balance())
|
|
||||||
self.assertEqual('5.0', await self.daemon.jsonrpc_account_balance(account2_id))
|
|
||||||
|
|
||||||
# send a tip to the claim using account2
|
|
||||||
tip = await self.out(
|
|
||||||
self.daemon.jsonrpc_claim_tip(claim['claim_id'], '1.0', account2_id)
|
|
||||||
)
|
|
||||||
await self.on_transaction_dict(tip)
|
|
||||||
await self.generate(1)
|
|
||||||
await self.on_transaction_dict(tip)
|
|
||||||
|
|
||||||
# tips don't affect balance so account1 balance is same but account2 balance went down
|
|
||||||
self.assertEqual('3.979769', await self.daemon.jsonrpc_account_balance())
|
|
||||||
self.assertEqual('3.9998585', await self.daemon.jsonrpc_account_balance(account2_id))
|
|
||||||
|
|
||||||
# verify that the incoming tip is marked correctly as is_tip=True in account1
|
|
||||||
txs = await self.out(self.daemon.jsonrpc_transaction_list())
|
|
||||||
self.assertEqual(len(txs[0]['support_info']), 1)
|
|
||||||
self.assertEqual(txs[0]['support_info'][0]['balance_delta'], '1.0')
|
|
||||||
self.assertEqual(txs[0]['support_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs[0]['support_info'][0]['is_tip'], True)
|
|
||||||
self.assertEqual(txs[0]['value'], '1.0')
|
|
||||||
self.assertEqual(txs[0]['fee'], '0.0')
|
|
||||||
|
|
||||||
# verify that the outgoing tip is marked correctly as is_tip=True in account2
|
|
||||||
txs2 = await self.out(
|
|
||||||
self.daemon.jsonrpc_transaction_list(account2_id)
|
|
||||||
)
|
|
||||||
self.assertEqual(len(txs2[0]['support_info']), 1)
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['balance_delta'], '-1.0')
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['is_tip'], True)
|
|
||||||
self.assertEqual(txs2[0]['value'], '-1.0')
|
|
||||||
self.assertEqual(txs2[0]['fee'], '-0.0001415')
|
|
||||||
|
|
||||||
# send a support to the claim using account2
|
|
||||||
support = await self.out(
|
|
||||||
self.daemon.jsonrpc_claim_new_support('hovercraft', claim['claim_id'], '2.0', account2_id)
|
|
||||||
)
|
|
||||||
await self.on_transaction_dict(support)
|
|
||||||
await self.generate(1)
|
|
||||||
await self.on_transaction_dict(support)
|
|
||||||
|
|
||||||
# account2 balance went down ~2
|
|
||||||
self.assertEqual('3.979769', await self.daemon.jsonrpc_account_balance())
|
|
||||||
self.assertEqual('1.999717', await self.daemon.jsonrpc_account_balance(account2_id))
|
|
||||||
|
|
||||||
# verify that the outgoing support is marked correctly as is_tip=False in account2
|
|
||||||
txs2 = await self.out(self.daemon.jsonrpc_transaction_list(account2_id))
|
|
||||||
self.assertEqual(len(txs2[0]['support_info']), 1)
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['balance_delta'], '-2.0')
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['claim_id'], claim['claim_id'])
|
|
||||||
self.assertEqual(txs2[0]['support_info'][0]['is_tip'], False)
|
|
||||||
self.assertEqual(txs2[0]['value'], '0.0')
|
|
||||||
self.assertEqual(txs2[0]['fee'], '-0.0001415')
|
|
||||||
|
|
||||||
|
|
||||||
class TransactionCommandsTestCase(CommandTestCase):
|
|
||||||
|
|
||||||
async def test_transaction_show(self):
|
|
||||||
# local tx
|
|
||||||
result = await self.out(self.daemon.jsonrpc_wallet_send(
|
|
||||||
'5.0', await self.daemon.jsonrpc_address_unused(self.account.id)
|
|
||||||
))
|
|
||||||
await self.confirm_tx(result['txid'])
|
|
||||||
tx = await self.daemon.jsonrpc_transaction_show(result['txid'])
|
|
||||||
self.assertEqual(tx.id, result['txid'])
|
|
||||||
|
|
||||||
# someone's tx
|
|
||||||
change_address = await self.blockchain.get_raw_change_address()
|
|
||||||
sendtxid = await self.blockchain.send_to_address(change_address, 10)
|
|
||||||
tx = await self.daemon.jsonrpc_transaction_show(sendtxid)
|
|
||||||
self.assertEqual(tx.id, sendtxid)
|
|
||||||
self.assertEqual(tx.height, -2)
|
|
||||||
await self.generate(1)
|
|
||||||
tx = await self.daemon.jsonrpc_transaction_show(sendtxid)
|
|
||||||
self.assertEqual(tx.height, self.ledger.headers.height)
|
|
||||||
|
|
||||||
# inexistent
|
|
||||||
result = await self.daemon.jsonrpc_transaction_show('0'*64)
|
|
||||||
self.assertFalse(result['success'])
|
|
||||||
|
|
||||||
async def test_utxo_release(self):
|
|
||||||
sendtxid = await self.blockchain.send_to_address(
|
|
||||||
await self.account.receiving.get_or_create_usable_address(), 1
|
|
||||||
)
|
|
||||||
await self.confirm_tx(sendtxid)
|
|
||||||
await self.assertBalance(self.account, '11.0')
|
|
||||||
await self.ledger.reserve_outputs(await self.account.get_utxos())
|
|
||||||
await self.assertBalance(self.account, '0.0')
|
|
||||||
await self.daemon.jsonrpc_utxo_release()
|
|
||||||
await self.assertBalance(self.account, '11.0')
|
|
||||||
|
|
Loading…
Reference in a new issue