2019-03-23 05:07:22 +01:00
|
|
|
|
import hashlib
|
2019-03-22 23:44:17 +01:00
|
|
|
|
from binascii import unhexlify
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-25 14:59:32 +01:00
|
|
|
|
import base64
|
2019-03-23 05:07:22 +01:00
|
|
|
|
import ecdsa
|
|
|
|
|
|
|
|
|
|
from lbrynet.wallet.transaction import Transaction, Output
|
2019-03-24 21:55:04 +01:00
|
|
|
|
from torba.client.errors import InsufficientFundsError
|
2019-03-22 20:38:14 +01:00
|
|
|
|
from lbrynet.schema.compat import OldClaimMessage
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-02-12 03:35:07 +01:00
|
|
|
|
from integration.testcase import CommandTestCase
|
2019-03-23 05:07:22 +01:00
|
|
|
|
from torba.client.hash import sha256, Base58
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
class ChannelCommands(CommandTestCase):
|
|
|
|
|
|
|
|
|
|
async def test_create_channel_names(self):
|
|
|
|
|
# claim new name
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.channel_create('@foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '8.991893')
|
|
|
|
|
|
|
|
|
|
# fail to claim duplicate
|
|
|
|
|
with self.assertRaisesRegex(Exception, "You already have a channel under the name '@foo'."):
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.channel_create('@foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# fail to claim invalid name
|
2019-03-26 03:06:36 +01:00
|
|
|
|
with self.assertRaisesRegex(Exception, "Channel names must start with '@' symbol."):
|
|
|
|
|
await self.channel_create('foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# nothing's changed after failed attempts
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '8.991893')
|
|
|
|
|
|
|
|
|
|
# succeed overriding duplicate restriction
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.channel_create('@foo', allow_duplicate_name=True)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 2)
|
|
|
|
|
await self.assertBalance(self.account, '7.983786')
|
|
|
|
|
|
|
|
|
|
async def test_channel_bids(self):
|
|
|
|
|
# enough funds
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.channel_create('@foo', '5.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
claim_id = tx['outputs'][0]['claim_id']
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '4.991893')
|
|
|
|
|
|
|
|
|
|
# bid preserved on update
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.channel_update(claim_id)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(tx['outputs'][0]['amount'], '5.0')
|
|
|
|
|
|
|
|
|
|
# bid changed on update
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.channel_update(claim_id, bid='4.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(tx['outputs'][0]['amount'], '4.0')
|
|
|
|
|
|
|
|
|
|
await self.assertBalance(self.account, '5.991447')
|
|
|
|
|
|
|
|
|
|
# not enough funds
|
|
|
|
|
with self.assertRaisesRegex(
|
|
|
|
|
InsufficientFundsError, "Not enough funds to cover this transaction."):
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.channel_create('@foo2', '9.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '5.991447')
|
|
|
|
|
|
|
|
|
|
# spend exactly amount available, no change
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.channel_create('@foo3', '5.981266')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
await self.assertBalance(self.account, '0.0')
|
|
|
|
|
self.assertEqual(len(tx['outputs']), 1) # no change
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 2)
|
|
|
|
|
|
|
|
|
|
async def test_setting_channel_fields(self):
|
|
|
|
|
values = {
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'tags': ["cool", "awesome"],
|
2019-03-24 21:55:04 +01:00
|
|
|
|
'title': "Cool Channel",
|
|
|
|
|
'description': "Best channel on LBRY.",
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'thumbnail_url': "https://co.ol/thumbnail.png",
|
|
|
|
|
'language': "en",
|
2019-03-24 21:55:04 +01:00
|
|
|
|
'contact_email': "human@email.com",
|
|
|
|
|
'homepage_url': "https://co.ol",
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'cover_url': "https://co.ol/cover.png",
|
2019-03-24 21:55:04 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# create new channel with all fields set
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_create('@bigchannel', **values))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
txo['value']['channel'],
|
|
|
|
|
{'public_key': txo['value']['channel']['public_key'], **values}
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# create channel with nothing set
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_create('@lightchannel'))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
txo['value']['channel'],
|
|
|
|
|
{'public_key': txo['value']['channel']['public_key']}
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# create channel with just some tags
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_create('@updatedchannel', tags='blah'))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
claim_id = txo['claim_id']
|
|
|
|
|
public_key = txo['value']['channel']['public_key']
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
txo['value']['channel'],
|
|
|
|
|
{'public_key': public_key, 'tags': ['blah']}
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# update channel setting all fields
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_update(claim_id, **values))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
values['public_key'] = public_key
|
|
|
|
|
values['tags'].insert(0, 'blah') # existing tag
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
txo['value']['channel'],
|
|
|
|
|
values
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# clearing and settings tags
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_update(claim_id, tags='single', clear_tags=True))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
values['tags'] = ['single']
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
txo['value']['channel'],
|
|
|
|
|
values
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# reset signing key
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_update(claim_id, new_signing_key=True))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
self.assertNotEqual(
|
|
|
|
|
txo['value']['channel']['public_key'],
|
|
|
|
|
values['public_key']
|
|
|
|
|
)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# send channel to someone else
|
|
|
|
|
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'])
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# before sending
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 3)
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list(account_id=account2_id)), 0)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
other_address = await account2.receiving.get_or_create_usable_address()
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.channel_update(claim_id, claim_address=other_address))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# after sending
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list()), 2)
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_channel_list(account_id=account2_id)), 1)
|
|
|
|
|
|
|
|
|
|
# shoud not have private key
|
|
|
|
|
txo = (await account2.get_channels())[0]
|
|
|
|
|
self.assertIsNone(txo.private_key)
|
|
|
|
|
|
|
|
|
|
# send the private key too
|
|
|
|
|
txoid = f"{tx['outputs'][0]['txid']}:{tx['outputs'][0]['nout']}"
|
|
|
|
|
account2.channel_keys[txoid] = self.account.channel_keys[txoid]
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# now should have private key
|
|
|
|
|
txo = (await account2.get_channels())[0]
|
|
|
|
|
self.assertIsNotNone(txo.private_key)
|
|
|
|
|
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
class StreamCommands(CommandTestCase):
|
2019-03-25 00:45:54 +01:00
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
async def test_create_stream_names(self):
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# claim new name
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create('foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '8.993893')
|
|
|
|
|
|
|
|
|
|
# fail to claim duplicate
|
2019-03-26 03:06:36 +01:00
|
|
|
|
with self.assertRaisesRegex(
|
|
|
|
|
Exception, "You already have a stream claim published under the name 'foo'."):
|
|
|
|
|
await self.stream_create('foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# fail claim starting with @
|
2019-03-26 03:06:36 +01:00
|
|
|
|
with self.assertRaisesRegex(
|
|
|
|
|
Exception, "Stream names cannot start with '@' symbol."):
|
|
|
|
|
await self.stream_create('@foo')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '8.993893')
|
|
|
|
|
|
|
|
|
|
# succeed overriding duplicate restriction
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create('foo', allow_duplicate_name=True)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 2)
|
|
|
|
|
await self.assertBalance(self.account, '7.987786')
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
async def test_stream_bids(self):
|
2019-03-24 21:55:04 +01:00
|
|
|
|
# enough funds
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_create('foo', '2.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
claim_id = tx['outputs'][0]['claim_id']
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '7.993893')
|
|
|
|
|
|
|
|
|
|
# bid preserved on update
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_update(claim_id)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(tx['outputs'][0]['amount'], '2.0')
|
|
|
|
|
|
|
|
|
|
# bid changed on update
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_update(claim_id, bid='3.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(tx['outputs'][0]['amount'], '3.0')
|
|
|
|
|
|
|
|
|
|
await self.assertBalance(self.account, '6.993384')
|
|
|
|
|
|
|
|
|
|
# not enough funds
|
|
|
|
|
with self.assertRaisesRegex(
|
|
|
|
|
InsufficientFundsError, "Not enough funds to cover this transaction."):
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create('foo2', '9.0')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 1)
|
|
|
|
|
await self.assertBalance(self.account, '6.993384')
|
|
|
|
|
|
|
|
|
|
# spend exactly amount available, no change
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_create('foo3', '6.98527700')
|
2019-03-24 21:55:04 +01:00
|
|
|
|
await self.assertBalance(self.account, '0.0')
|
|
|
|
|
self.assertEqual(len(tx['outputs']), 1) # no change
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 2)
|
|
|
|
|
|
|
|
|
|
async def test_publishing_checks_all_accounts_for_channel(self):
|
2019-02-11 23:45:52 +01:00
|
|
|
|
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'])
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.out(self.channel_create('@spam', '1.0'))
|
2019-02-11 23:45:52 +01:00
|
|
|
|
self.assertEqual('8.989893', await self.daemon.jsonrpc_account_balance())
|
|
|
|
|
|
2019-03-25 04:12:57 +01:00
|
|
|
|
result = await self.out(self.daemon.jsonrpc_account_send(
|
2019-02-11 23:45:52 +01:00
|
|
|
|
'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))
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
baz_tx = await self.out(self.channel_create('@baz', '1.0', account_id=account2_id))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
baz_id = baz_tx['outputs'][0]['claim_id']
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
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
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create('hovercraft1', channel_id=baz_id)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
# uses only the specific accounts which contains the channel
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create('hovercraft2', channel_id=baz_id, channel_account_id=[account2_id])
|
2019-02-11 23:45:52 +01:00
|
|
|
|
# fails when specifying account which does not contain channel
|
2019-03-24 21:55:04 +01:00
|
|
|
|
with self.assertRaisesRegex(ValueError, "Couldn't find channel with channel_id"):
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create(
|
2019-03-24 21:55:04 +01:00
|
|
|
|
'hovercraft3', channel_id=baz_id, channel_account_id=[account1_id]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
async def test_setting_claim_fields(self):
|
|
|
|
|
values = {
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'tags': ["cool", "awesome"],
|
2019-03-25 14:59:32 +01:00
|
|
|
|
'title': "Cool Content",
|
|
|
|
|
'description': "Best content on LBRY.",
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'thumbnail_url': "https://co.ol/thumbnail.png",
|
2019-03-25 14:59:32 +01:00
|
|
|
|
'language': "en",
|
2019-03-25 17:30:30 +01:00
|
|
|
|
|
|
|
|
|
'author': "Jules Verne",
|
2019-03-25 14:59:32 +01:00
|
|
|
|
'license': 'Public Domain',
|
2019-03-25 17:30:30 +01:00
|
|
|
|
'license_url': "https://co.ol/license",
|
|
|
|
|
'release_time': 123456,
|
|
|
|
|
|
2019-03-25 14:59:32 +01:00
|
|
|
|
'fee_currency': 'usd',
|
|
|
|
|
'fee_amount': '2.99',
|
|
|
|
|
'fee_address': 'mmCsWAiXMUVecFQ3fVzUwvpT9XFMXno2Ca',
|
2019-03-25 17:30:30 +01:00
|
|
|
|
|
2019-03-25 14:59:32 +01:00
|
|
|
|
'video_width': 800,
|
|
|
|
|
'video_height': 600
|
2019-03-24 21:55:04 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# create new channel with all fields set
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_create('big', **values))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
2019-03-25 14:59:32 +01:00
|
|
|
|
stream = txo['value']['stream']
|
|
|
|
|
fixed_values = values.copy()
|
|
|
|
|
fixed_values['hash'] = stream['hash']
|
|
|
|
|
fixed_values['file'] = stream['file']
|
|
|
|
|
fixed_values['media_type'] = 'application/octet-stream'
|
|
|
|
|
fixed_values['release_time'] = str(values['release_time'])
|
|
|
|
|
fixed_values['fee'] = {
|
|
|
|
|
'address': base64.b64encode(Base58.decode(fixed_values.pop('fee_address'))).decode(),
|
|
|
|
|
'amount': fixed_values.pop('fee_amount').replace('.', ''),
|
|
|
|
|
'currency': fixed_values.pop('fee_currency').upper()
|
|
|
|
|
}
|
|
|
|
|
fixed_values['video'] = {
|
|
|
|
|
'height': fixed_values.pop('video_height'),
|
|
|
|
|
'width': fixed_values.pop('video_width')
|
|
|
|
|
}
|
|
|
|
|
self.assertEqual(stream, fixed_values)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# create channel with nothing set
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_create('light'))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
self.assertEqual(
|
2019-03-25 14:59:32 +01:00
|
|
|
|
txo['value']['stream'], {
|
|
|
|
|
'file': {'size': '3'},
|
|
|
|
|
'media_type': 'application/octet-stream',
|
|
|
|
|
'hash': txo['value']['stream']['hash']
|
|
|
|
|
}
|
2019-03-24 21:55:04 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# create channel with just some tags
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_create('updated', tags='blah'))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
|
|
|
|
claim_id = txo['claim_id']
|
2019-03-25 14:59:32 +01:00
|
|
|
|
fixed_values['hash'] = txo['value']['stream']['hash']
|
2019-03-24 21:55:04 +01:00
|
|
|
|
self.assertEqual(
|
2019-03-25 14:59:32 +01:00
|
|
|
|
txo['value']['stream'], {
|
|
|
|
|
'file': {'size': '3'},
|
|
|
|
|
'media_type': 'application/octet-stream',
|
|
|
|
|
'hash': fixed_values['hash'],
|
|
|
|
|
'tags': ['blah']
|
|
|
|
|
}
|
2019-03-24 21:55:04 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# update channel setting all fields
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_update(claim_id, **values))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
2019-03-25 14:59:32 +01:00
|
|
|
|
fixed_values['tags'].insert(0, 'blah') # existing tag
|
|
|
|
|
self.assertEqual(txo['value']['stream'], fixed_values)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# clearing and settings tags
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_update(claim_id, tags='single', clear_tags=True))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
txo = tx['outputs'][0]
|
2019-03-25 14:59:32 +01:00
|
|
|
|
fixed_values['tags'] = ['single']
|
|
|
|
|
self.assertEqual(txo['value']['stream'], fixed_values)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
2019-03-25 14:59:32 +01:00
|
|
|
|
# send claim to someone else
|
2019-03-24 21:55:04 +01:00
|
|
|
|
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'])
|
|
|
|
|
|
|
|
|
|
# before sending
|
2019-03-25 14:59:32 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 3)
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list(account_id=account2_id)), 0)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
other_address = await account2.receiving.get_or_create_usable_address()
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.out(self.stream_update(claim_id, claim_address=other_address))
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
# after sending
|
2019-03-25 14:59:32 +01:00
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list()), 2)
|
|
|
|
|
self.assertEqual(len(await self.daemon.jsonrpc_claim_list(account_id=account2_id)), 1)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
|
|
|
|
|
async def test_create_update_and_abandon_claim(self):
|
2019-02-11 23:45:52 +01:00
|
|
|
|
await self.assertBalance(self.account, '10.0')
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_create(bid='2.5') # creates new claim
|
2019-03-24 21:55:04 +01:00
|
|
|
|
claim_id = tx['outputs'][0]['claim_id']
|
|
|
|
|
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_id)
|
|
|
|
|
self.assertEqual(txs[0]['value'], '0.0')
|
|
|
|
|
self.assertEqual(txs[0]['fee'], '-0.020107')
|
|
|
|
|
await self.assertBalance(self.account, '7.479893')
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_update(claim_id, bid='1.0') # updates previous claim
|
2019-03-24 21:55:04 +01:00
|
|
|
|
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_id)
|
|
|
|
|
self.assertEqual(txs[0]['value'], '0.0')
|
|
|
|
|
self.assertEqual(txs[0]['fee'], '-0.000184')
|
|
|
|
|
await self.assertBalance(self.account, '8.979709')
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.claim_abandon(claim_id)
|
2019-03-24 21:55:04 +01:00
|
|
|
|
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_id)
|
|
|
|
|
self.assertEqual(txs[0]['value'], '0.0')
|
|
|
|
|
self.assertEqual(txs[0]['fee'], '-0.000107')
|
|
|
|
|
await self.assertBalance(self.account, '9.979602')
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
async def test_abandoning_claim_at_loss(self):
|
|
|
|
|
await self.assertBalance(self.account, '10.0')
|
2019-03-26 03:06:36 +01:00
|
|
|
|
tx = await self.stream_create(bid='0.0001')
|
2019-02-11 23:45:52 +01:00
|
|
|
|
await self.assertBalance(self.account, '9.979793')
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.claim_abandon(tx['outputs'][0]['claim_id'])
|
2019-02-11 23:45:52 +01:00
|
|
|
|
await self.assertBalance(self.account, '9.97968399')
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
async def test_claim_search(self):
|
|
|
|
|
# search for channel claim
|
|
|
|
|
channel = await self.channel_create('@abc', '1.0')
|
|
|
|
|
channel_id, txid = channel['outputs'][0]['claim_id'], channel['txid']
|
|
|
|
|
value = channel['outputs'][0]['value']
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
claims = await self.claim_search('@abc')
|
|
|
|
|
self.assertEqual(claims[0]['value'], value)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
claims = await self.claim_search(txid=txid, nout=0)
|
|
|
|
|
self.assertEqual(claims[0]['value'], value)
|
|
|
|
|
|
|
|
|
|
claims = await self.claim_search(claim_id=channel_id)
|
|
|
|
|
self.assertEqual(claims[0]['value'], value)
|
|
|
|
|
|
|
|
|
|
await self.claim_abandon(txid=txid, nout=0)
|
|
|
|
|
self.assertEqual(len(await self.claim_search(txid=txid, nout=0)), 0)
|
|
|
|
|
|
|
|
|
|
# search stream claims
|
|
|
|
|
channel = await self.channel_create('@abc', '1.0')
|
|
|
|
|
channel_id, txid = channel['outputs'][0]['claim_id'], channel['txid']
|
|
|
|
|
|
|
|
|
|
signed = await self.stream_create('on-channel-claim', '0.0001', channel_id=channel_id)
|
|
|
|
|
unsigned = await self.stream_create('unsigned', '0.0001')
|
|
|
|
|
|
|
|
|
|
claims = await self.claim_search('on-channel-claim')
|
|
|
|
|
self.assertEqual(claims[0]['value'], signed['outputs'][0]['value'])
|
|
|
|
|
|
|
|
|
|
claims = await self.claim_search('unsigned')
|
|
|
|
|
self.assertEqual(claims[0]['value'], unsigned['outputs'][0]['value'])
|
|
|
|
|
|
|
|
|
|
# list streams in a channel
|
|
|
|
|
await self.stream_create('on-channel-claim-2', '0.0001', channel_id=channel_id)
|
|
|
|
|
await self.stream_create('on-channel-claim-3', '0.0001', channel_id=channel_id)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
claims = await self.claim_search(channel_id=channel_id)
|
|
|
|
|
self.assertEqual(len(claims), 3)
|
|
|
|
|
|
|
|
|
|
await self.claim_abandon(claim_id=claims[0]['claim_id'])
|
|
|
|
|
await self.claim_abandon(claim_id=claims[1]['claim_id'])
|
|
|
|
|
await self.claim_abandon(claim_id=claims[2]['claim_id'])
|
|
|
|
|
|
|
|
|
|
claims = await self.claim_search(channel_id=channel_id)
|
|
|
|
|
self.assertEqual(len(claims), 0)
|
|
|
|
|
|
|
|
|
|
tx = await self.daemon.jsonrpc_account_fund(None, None, '0.001', outputs=100, broadcast=True)
|
|
|
|
|
await self.confirm_tx(tx.id)
|
|
|
|
|
|
|
|
|
|
# 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_tx = await self.stream_create(f'c{j}-{k}', '0.000001', channel_id=channel_id, confirm=False)
|
|
|
|
|
same_height_claims.append(claim_tx['outputs'][0]['claim_id'])
|
|
|
|
|
await self.on_transaction_dict(claim_tx)
|
|
|
|
|
claim_tx = await self.stream_create(f'c{j}-4', '0.000001', channel_id=channel_id, confirm=True)
|
|
|
|
|
same_height_claims.append(claim_tx['outputs'][0]['claim_id'])
|
|
|
|
|
same_height_claims.sort(key=lambda x: int(x, 16))
|
|
|
|
|
claims = same_height_claims + claims
|
|
|
|
|
|
|
|
|
|
page = await self.claim_search(page_size=20, channel_id=channel_id)
|
|
|
|
|
page_claim_ids = [item['claim_id'] for item in page]
|
|
|
|
|
self.assertEqual(page_claim_ids, claims)
|
|
|
|
|
|
|
|
|
|
page = await self.claim_search(page_size=6, channel_id=channel_id)
|
|
|
|
|
page_claim_ids = [item['claim_id'] for item in page]
|
|
|
|
|
self.assertEqual(page_claim_ids, claims[:6])
|
|
|
|
|
|
|
|
|
|
out_of_bounds = await self.claim_search(page=2, page_size=20, channel_id=channel_id)
|
|
|
|
|
self.assertEqual(out_of_bounds, [])
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
async def test_abandoned_channel_with_signed_claims(self):
|
2019-03-26 03:06:36 +01:00
|
|
|
|
channel = (await self.channel_create('@abc', '1.0'))['outputs'][0]
|
|
|
|
|
orphan_claim = await self.stream_create('on-channel-claim', '0.0001', channel_id=channel['claim_id'])
|
|
|
|
|
await self.claim_abandon(txid=channel['txid'], nout=0)
|
|
|
|
|
channel = (await self.channel_create('@abc', '1.0'))['outputs'][0]
|
2019-03-24 23:53:58 +01:00
|
|
|
|
orphan_claim_id = orphan_claim['outputs'][0]['claim_id']
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
# Original channel doesnt exists anymore, so the signature is invalid. For invalid signatures, resolution is
|
|
|
|
|
# only possible outside a channel
|
|
|
|
|
response = await self.resolve('lbry://@abc/on-channel-claim')
|
|
|
|
|
self.assertNotIn('claim', response['lbry://@abc/on-channel-claim'])
|
|
|
|
|
response = await self.resolve('lbry://on-channel-claim')
|
|
|
|
|
self.assertFalse(response['lbry://on-channel-claim']['claim']['signature_is_valid'])
|
2019-03-24 23:53:58 +01:00
|
|
|
|
direct_uri = 'lbry://on-channel-claim#' + orphan_claim_id
|
2019-02-11 23:45:52 +01:00
|
|
|
|
response = await self.resolve(direct_uri)
|
|
|
|
|
self.assertFalse(response[direct_uri]['claim']['signature_is_valid'])
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.claim_abandon(claim_id=orphan_claim_id)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
|
|
|
|
uri = 'lbry://@abc/on-channel-claim'
|
|
|
|
|
# now, claim something on this channel (it will update the invalid claim, but we save and forcefully restore)
|
2019-03-26 03:06:36 +01:00
|
|
|
|
valid_claim = await self.stream_create('on-channel-claim', '0.00000001', channel_id=channel['claim_id'])
|
2019-02-11 23:45:52 +01:00
|
|
|
|
# resolves normally
|
|
|
|
|
response = await self.resolve(uri)
|
|
|
|
|
self.assertTrue(response[uri]['claim']['signature_is_valid'])
|
|
|
|
|
|
2019-03-22 23:44:17 +01:00
|
|
|
|
# ooops! claimed a valid conflict! (this happens on the wild, mostly by accident or race condition)
|
2019-03-26 03:06:36 +01:00
|
|
|
|
await self.stream_create(
|
2019-03-24 23:53:58 +01:00
|
|
|
|
'on-channel-claim', '0.00000001', channel_id=channel['claim_id'], allow_duplicate_name=True
|
2019-03-22 23:44:17 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# it still resolves! but to the older claim
|
|
|
|
|
response = await self.resolve(uri)
|
|
|
|
|
self.assertTrue(response[uri]['claim']['signature_is_valid'])
|
2019-03-24 23:53:58 +01:00
|
|
|
|
self.assertEqual(response[uri]['claim']['txid'], valid_claim['txid'])
|
2019-03-26 03:06:36 +01:00
|
|
|
|
claims = (await self.daemon.jsonrpc_claim_search('on-channel-claim'))['items']
|
2019-03-22 23:44:17 +01:00
|
|
|
|
self.assertEqual(2, len(claims))
|
|
|
|
|
signer_ids = set([claim['value'].signing_channel_id for claim in claims])
|
|
|
|
|
self.assertEqual({channel['claim_id']}, signer_ids)
|
2019-02-11 23:45:52 +01:00
|
|
|
|
|
2019-02-14 20:54:23 +01:00
|
|
|
|
async def test_normalization_resolution(self):
|
|
|
|
|
|
|
|
|
|
# this test assumes that the lbrycrd forks normalization at height == 250 on regtest
|
|
|
|
|
|
2019-03-26 03:06:36 +01:00
|
|
|
|
c1 = await self.stream_create('ΣίσυφοςfiÆ', '0.1')
|
|
|
|
|
c2 = await self.stream_create('ΣΊΣΥΦΟσFIæ', '0.2')
|
2019-02-14 20:54:23 +01:00
|
|
|
|
|
|
|
|
|
r1 = await self.daemon.jsonrpc_resolve(urls='lbry://ΣίσυφοςfiÆ')
|
|
|
|
|
r2 = await self.daemon.jsonrpc_resolve(urls='lbry://ΣΊΣΥΦΟσFIæ')
|
|
|
|
|
|
|
|
|
|
r1c = list(r1.values())[0]['claim']['claim_id']
|
|
|
|
|
r2c = list(r2.values())[0]['claim']['claim_id']
|
2019-03-25 04:23:06 +01:00
|
|
|
|
self.assertEqual(c1['outputs'][0]['claim_id'], r1c)
|
|
|
|
|
self.assertEqual(c2['outputs'][0]['claim_id'], r2c)
|
2019-02-14 20:54:23 +01:00
|
|
|
|
self.assertNotEqual(r1c, r2c)
|
|
|
|
|
|
|
|
|
|
await self.generate(50)
|
|
|
|
|
head = await self.daemon.jsonrpc_block_show()
|
|
|
|
|
self.assertTrue(head['height'] > 250)
|
|
|
|
|
|
|
|
|
|
r3 = await self.daemon.jsonrpc_resolve(urls='lbry://ΣίσυφοςfiÆ')
|
|
|
|
|
r4 = await self.daemon.jsonrpc_resolve(urls='lbry://ΣΊΣΥΦΟσFIæ')
|
|
|
|
|
|
|
|
|
|
r3c = list(r3.values())[0]['claim']['claim_id']
|
|
|
|
|
r4c = list(r4.values())[0]['claim']['claim_id']
|
|
|
|
|
r3n = list(r3.values())[0]['claim']['name']
|
|
|
|
|
r4n = list(r4.values())[0]['claim']['name']
|
|
|
|
|
|
2019-03-25 04:23:06 +01:00
|
|
|
|
self.assertEqual(c2['outputs'][0]['claim_id'], r3c)
|
|
|
|
|
self.assertEqual(c2['outputs'][0]['claim_id'], r4c)
|
2019-02-14 20:54:23 +01:00
|
|
|
|
self.assertEqual(r3c, r4c)
|
|
|
|
|
self.assertEqual(r3n, r4n)
|
2019-03-23 05:07:22 +01:00
|
|
|
|
|
|
|
|
|
async def test_resolve_old_claim(self):
|
2019-03-25 03:27:34 +01:00
|
|
|
|
channel = await self.daemon.jsonrpc_channel_create('@olds', '1.0')
|
|
|
|
|
await self.confirm_tx(channel.id)
|
|
|
|
|
address = channel.outputs[0].get_address(self.account.ledger)
|
|
|
|
|
claim = generate_signed_legacy('example', address, channel.outputs[0])
|
|
|
|
|
tx = await Transaction.claim_create('example', claim.SerializeToString(), 1, address, [self.account], self.account)
|
2019-03-23 05:07:22 +01:00
|
|
|
|
await tx.sign([self.account])
|
|
|
|
|
await self.broadcast(tx)
|
2019-03-23 05:32:07 +01:00
|
|
|
|
await self.confirm_tx(tx.id)
|
2019-03-23 05:07:22 +01:00
|
|
|
|
|
|
|
|
|
response = await self.daemon.jsonrpc_resolve(urls='@olds/example')
|
|
|
|
|
self.assertTrue(response['@olds/example']['claim']['signature_is_valid'])
|
|
|
|
|
|
|
|
|
|
claim.publisherSignature.signature = bytes(reversed(claim.publisherSignature.signature))
|
2019-03-25 03:27:34 +01:00
|
|
|
|
tx = await Transaction.claim_create(
|
2019-03-23 05:07:22 +01:00
|
|
|
|
'bad_example', claim.SerializeToString(), 1, address, [self.account], self.account
|
|
|
|
|
)
|
|
|
|
|
await tx.sign([self.account])
|
|
|
|
|
await self.broadcast(tx)
|
2019-03-23 05:32:07 +01:00
|
|
|
|
await self.confirm_tx(tx.id)
|
2019-03-23 05:07:22 +01:00
|
|
|
|
|
|
|
|
|
response = await self.daemon.jsonrpc_resolve(urls='bad_example')
|
|
|
|
|
self.assertFalse(response['bad_example']['claim']['signature_is_valid'], response)
|
|
|
|
|
response = await self.daemon.jsonrpc_resolve(urls='@olds/bad_example')
|
|
|
|
|
self.assertEqual('URI lbry://@olds/bad_example cannot be resolved', response['@olds/bad_example']['error'])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def generate_signed_legacy(name: str, address: bytes, output: Output):
|
|
|
|
|
decoded_address = Base58.decode(address)
|
|
|
|
|
claim = OldClaimMessage()
|
|
|
|
|
claim.ParseFromString(unhexlify(
|
|
|
|
|
'080110011aee04080112a604080410011a2b4865726520617265203520526561736f6e73204920e29da4e'
|
|
|
|
|
'fb88f204e657874636c6f7564207c20544c4722920346696e64206f7574206d6f72652061626f7574204e'
|
|
|
|
|
'657874636c6f75643a2068747470733a2f2f6e657874636c6f75642e636f6d2f0a0a596f752063616e206'
|
|
|
|
|
'6696e64206d65206f6e20746865736520736f6369616c733a0a202a20466f72756d733a2068747470733a'
|
|
|
|
|
'2f2f666f72756d2e6865617679656c656d656e742e696f2f0a202a20506f64636173743a2068747470733'
|
|
|
|
|
'a2f2f6f6666746f706963616c2e6e65740a202a2050617472656f6e3a2068747470733a2f2f7061747265'
|
|
|
|
|
'6f6e2e636f6d2f7468656c696e757867616d65720a202a204d657263683a2068747470733a2f2f7465657'
|
|
|
|
|
'37072696e672e636f6d2f73746f7265732f6f6666696369616c2d6c696e75782d67616d65720a202a2054'
|
|
|
|
|
'77697463683a2068747470733a2f2f7477697463682e74762f786f6e64616b0a202a20547769747465723'
|
|
|
|
|
'a2068747470733a2f2f747769747465722e636f6d2f7468656c696e757867616d65720a0a2e2e2e0a6874'
|
|
|
|
|
'7470733a2f2f7777772e796f75747562652e636f6d2f77617463683f763d4672546442434f535f66632a0'
|
|
|
|
|
'f546865204c696e75782047616d6572321c436f7079726967687465642028636f6e746163742061757468'
|
|
|
|
|
'6f722938004a2968747470733a2f2f6265726b2e6e696e6a612f7468756d626e61696c732f46725464424'
|
|
|
|
|
'34f535f666352005a001a41080110011a30040e8ac6e89c061f982528c23ad33829fd7146435bf7a4cc22'
|
|
|
|
|
'f0bff70c4fe0b91fd36da9a375e3e1c171db825bf5d1f32209766964656f2f6d70342a5c080110031a406'
|
|
|
|
|
'2b2dd4c45e364030fbfad1a6fefff695ebf20ea33a5381b947753e2a0ca359989a5cc7d15e5392a0d354c'
|
|
|
|
|
'0b68498382b2701b22c03beb8dcb91089031b871e72214feb61536c007cdf4faeeaab4876cb397feaf6b51'
|
|
|
|
|
))
|
|
|
|
|
claim.ClearField("publisherSignature")
|
|
|
|
|
digest = sha256(b''.join([
|
|
|
|
|
decoded_address,
|
|
|
|
|
claim.SerializeToString(),
|
|
|
|
|
output.claim_hash
|
|
|
|
|
]))
|
|
|
|
|
private_key = ecdsa.SigningKey.from_pem(output.private_key, hashfunc=hashlib.sha256)
|
|
|
|
|
signature = private_key.sign_digest_deterministic(digest, hashfunc=hashlib.sha256)
|
|
|
|
|
claim.publisherSignature.version = 1
|
|
|
|
|
claim.publisherSignature.signatureType = 1
|
|
|
|
|
claim.publisherSignature.signature = signature
|
|
|
|
|
claim.publisherSignature.certificateId = output.claim_hash
|
2019-03-26 03:06:36 +01:00
|
|
|
|
return claim
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class SupportCommands(CommandTestCase):
|
|
|
|
|
|
|
|
|
|
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']
|
|
|
|
|
account2 = self.daemon.get_account_or_error(account2_id)
|
|
|
|
|
|
|
|
|
|
# send account2 5 LBC out of the 10 LBC in account1
|
|
|
|
|
result = await self.out(self.daemon.jsonrpc_account_send(
|
|
|
|
|
'5.0', await self.daemon.jsonrpc_address_unused(account2_id)
|
|
|
|
|
))
|
|
|
|
|
await self.on_transaction_dict(result)
|
|
|
|
|
|
|
|
|
|
# account1 and account2 balances:
|
|
|
|
|
await self.assertBalance(self.account, '4.999876')
|
|
|
|
|
await self.assertBalance(account2, '5.0')
|
|
|
|
|
|
|
|
|
|
# create the claim we'll be tipping and supporting
|
|
|
|
|
tx = await self.stream_create()
|
|
|
|
|
claim_id = tx['outputs'][0]['claim_id']
|
|
|
|
|
|
|
|
|
|
# account1 and account2 balances:
|
|
|
|
|
await self.assertBalance(self.account, '3.979769')
|
|
|
|
|
await self.assertBalance(account2, '5.0')
|
|
|
|
|
|
|
|
|
|
# send a tip to the claim using account2
|
|
|
|
|
tip = await self.out(
|
|
|
|
|
self.daemon.jsonrpc_support_create(claim_id, '1.0', True, 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
|
|
|
|
|
await self.assertBalance(self.account, '3.979769')
|
|
|
|
|
await self.assertBalance(account2, '3.9998585')
|
|
|
|
|
|
|
|
|
|
# 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_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_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_support_create(claim_id, '2.0', False, account2_id)
|
|
|
|
|
)
|
|
|
|
|
await self.on_transaction_dict(support)
|
|
|
|
|
await self.generate(1)
|
|
|
|
|
await self.on_transaction_dict(support)
|
|
|
|
|
|
|
|
|
|
# account2 balance went down ~2
|
|
|
|
|
await self.assertBalance(self.account, '3.979769')
|
|
|
|
|
await self.assertBalance(account2, '1.999717')
|
|
|
|
|
|
|
|
|
|
# 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_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')
|
|
|
|
|
|
|
|
|
|
|