lbry-sdk/tests/integration/commands/test_claim_commands.py

1941 lines
93 KiB
Python
Raw Normal View History

import os.path
2019-03-27 16:02:17 -04:00
import tempfile
import logging
import asyncio
2020-11-16 12:31:44 -05:00
from unittest import skip
from binascii import unhexlify
from urllib.request import urlopen
2019-02-11 17:45:52 -05:00
2019-12-31 15:30:13 -05:00
from lbry.error import InsufficientFundsError
from lbry.testcase import CommandTestCase
2020-05-18 08:29:15 -04:00
from lbry.blockchain import dewies_to_lbc as lbc
from lbry.service.api import DEFAULT_PAGE_SIZE
2019-02-11 17:45:52 -05:00
log = logging.getLogger(__name__)
class ClaimTestCase(CommandTestCase):
2019-11-13 22:14:20 -05:00
files_directory = os.path.join(os.path.dirname(__file__), 'files')
video_file_url = 'http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/ForBiggerEscapes.mp4'
video_file_name = os.path.join(files_directory, 'ForBiggerEscapes.mp4')
image_data = unhexlify(
b'89504e470d0a1a0a0000000d49484452000000050000000708020000004fc'
b'510b9000000097048597300000b1300000b1301009a9c1800000015494441'
b'5408d763fcffff3f031260624005d4e603004c45030b5286e9ea000000004'
b'9454e44ae426082'
)
def setUp(self):
if not os.path.exists(self.video_file_name):
if not os.path.exists(self.files_directory):
os.mkdir(self.files_directory)
log.info(f'downloading test video from {self.video_file_name}')
with urlopen(self.video_file_url) as response, \
open(self.video_file_name, 'wb') as video_file:
video_file.write(response.read())
class ClaimSearchCommand(ClaimTestCase):
2019-05-05 22:25:43 -04:00
async def create_channel(self):
self.channel = await self.channel_create('@abc', '1.0')
self.channel_id = self.get_claim_id(self.channel)
2019-05-05 22:25:43 -04:00
async def create_lots_of_streams(self):
2020-05-18 08:29:15 -04:00
await self.account_fund(None, None, '0.001', outputs=100, broadcast=True)
# 4 claims per block, 3 blocks. Sorted by height (descending) then claim name (ascending).
2019-05-05 22:25:43 -04:00
self.streams = []
for j in range(4):
2019-05-05 22:25:43 -04:00
same_height_claims = []
for k in range(5):
2019-05-05 22:25:43 -04:00
claim_tx = await self.stream_create(
f'c{j}-{k}', '0.000001', channel_id=self.channel_id, confirm=False)
same_height_claims.append(claim_tx['outputs'][0]['name'])
await self.on_transaction_dict(claim_tx)
claim_tx = await self.stream_create(
f'c{j}-6', '0.000001', channel_id=self.channel_id, confirm=True)
2019-05-05 22:25:43 -04:00
same_height_claims.append(claim_tx['outputs'][0]['name'])
self.streams = same_height_claims + self.streams
async def assertFindsClaim(self, claim, **kwargs):
await self.assertFindsClaims([claim], **kwargs)
async def assertFindsClaims(self, claims, **kwargs):
2019-07-20 13:30:41 -04:00
kwargs.setdefault('order_by', ['height', '^name'])
2019-05-05 22:25:43 -04:00
results = await self.claim_search(**kwargs)
self.assertEqual(len(claims), len(results))
for claim, result in zip(claims, results):
self.assertEqual(
(claim['txid'], self.get_claim_id(claim)),
2019-05-05 22:25:43 -04:00
(result['txid'], result['claim_id'])
)
2020-11-16 12:31:44 -05:00
@skip
async def test_disconnect_on_memory_error(self):
claim_ids = [
'0000000000000000000000000000000000000000',
] * 23828
self.assertListEqual([], await self.claim_search(claim_ids=claim_ids))
# this should do nothing... if the resolve (which is retried) results in the server disconnecting,
# it kerplodes
2020-05-18 08:29:15 -04:00
await asyncio.wait_for(self.api.resolve([
f'0000000000000000000000000000000000000000{i}' for i in range(30000)
]), 30)
# 23829 claim ids makes the request just large enough
claim_ids = [
'0000000000000000000000000000000000000000',
] * 23829
with self.assertRaises(ConnectionResetError):
await self.claim_search(claim_ids=claim_ids)
2019-05-05 22:25:43 -04:00
async def test_basic_claim_search(self):
await self.create_channel()
2019-06-03 16:37:21 -04:00
channel_txo = self.channel['outputs'][0]
2019-05-05 22:25:43 -04:00
channel2 = await self.channel_create('@abc', '0.1', allow_duplicate_name=True)
2019-06-03 16:37:21 -04:00
channel_txo2 = channel2['outputs'][0]
channel_id2 = channel_txo2['claim_id']
2019-05-05 22:25:43 -04:00
# finding a channel
await self.assertFindsClaims([channel2, self.channel], name='@abc')
await self.assertFindsClaim(self.channel, name='@abc', is_controlling=True)
await self.assertFindsClaim(self.channel, claim_id=self.channel_id)
await self.assertFindsClaim(self.channel, txid=self.channel['txid'], nout=0)
await self.assertFindsClaim(channel2, claim_id=channel_id2)
await self.assertFindsClaim(channel2, txid=channel2['txid'], nout=0)
2019-06-03 16:37:21 -04:00
await self.assertFindsClaim(
channel2, public_key_id=channel_txo2['value']['public_key_id'])
await self.assertFindsClaim(
self.channel, public_key_id=channel_txo['value']['public_key_id'])
2019-05-05 22:25:43 -04:00
signed = await self.stream_create('on-channel-claim', '0.001', channel_id=self.channel_id)
signed2 = await self.stream_create('on-channel-claim', '0.0001', channel_id=channel_id2,
allow_duplicate_name=True)
unsigned = await self.stream_create('unsigned', '0.0001')
# finding claims with and without a channel
await self.assertFindsClaims([signed2, signed], name='on-channel-claim')
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([signed2, signed], channel_id=[self.channel_id, channel_id2])
await self.assertFindsClaim(signed, name='on-channel-claim', channel_id=[self.channel_id])
await self.assertFindsClaim(signed2, name='on-channel-claim', channel_id=[channel_id2])
2019-05-05 22:25:43 -04:00
await self.assertFindsClaim(unsigned, name='unsigned')
await self.assertFindsClaim(unsigned, txid=unsigned['txid'], nout=0)
await self.assertFindsClaim(unsigned, claim_id=self.get_claim_id(unsigned))
2019-05-05 22:25:43 -04:00
two = await self.stream_create('on-channel-claim-2', '0.0001', channel_id=self.channel_id)
three = await self.stream_create('on-channel-claim-3', '0.0001', channel_id=self.channel_id)
# three streams in channel, zero streams in abandoned channel
claims = [three, two, signed]
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims(claims, channel_id=[self.channel_id])
2019-05-05 22:25:43 -04:00
await self.assertFindsClaims(claims, channel=f"@abc#{self.channel_id}")
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([three, two, signed2, signed], channel_id=[channel_id2, self.channel_id])
2019-05-05 22:25:43 -04:00
await self.channel_abandon(claim_id=self.channel_id)
await self.assertFindsClaims([], channel=f"@abc#{self.channel_id}", valid_channel_signature=True)
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([signed2], channel_id=[channel_id2], valid_channel_signature=True)
# pass `invalid_channel_signature=False` to catch a bug in argument processing
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([signed2], channel_id=[channel_id2],
valid_channel_signature=True, invalid_channel_signature=False)
2020-11-16 12:31:44 -05:00
# in old SDK abandoned channels caused content to have invalid signature,
# in new SDK this is not the case
# TODO: create situation where streams legitimately have invalid signature, harder in new SDK
# await self.assertFindsClaims([], channel_id=[self.channel_id], valid_channel_signature=True)
# invalid signature still returns channel_id
2020-11-16 12:31:44 -05:00
#invalid_claims = await self.claim_search(invalid_channel_signature=True, has_channel_signature=True)
#self.assertEqual(3, len(invalid_claims))
#self.assertTrue(all([not c['is_channel_signature_valid'] for c in invalid_claims]))
#self.assertEqual({'channel_id': self.channel_id}, invalid_claims[0]['signing_channel'])
self.assertEqual(
0, len(await self.claim_search(invalid_channel_signature=True, has_channel_signature=True))
)
valid_claims = await self.claim_search(valid_channel_signature=True, has_channel_signature=True)
2020-11-16 12:31:44 -05:00
self.assertEqual(4, len(valid_claims))
self.assertTrue(all([c['is_channel_signature_valid'] for c in valid_claims]))
2020-11-16 12:31:44 -05:00
self.assertEqual('@abc', valid_claims[1]['signing_channel']['name'])
2019-05-05 22:25:43 -04:00
# abandoned stream won't show up for streams in channel search
await self.stream_abandon(txid=signed2['txid'], nout=0)
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([], channel_id=[channel_id2])
2019-05-05 22:25:43 -04:00
async def test_pagination(self):
await self.create_channel()
await self.create_lots_of_streams()
2019-05-05 22:25:43 -04:00
# with and without totals
2020-11-16 12:31:44 -05:00
results = await self.api.claim_search(include_total=True)
self.assertEqual(results['total_pages'], 2)
self.assertEqual(results['total_items'], 25)
2020-05-18 08:29:15 -04:00
results = await self.api.claim_search()
self.assertNotIn('total_pages', results)
self.assertNotIn('total_items', results)
2019-05-05 22:25:43 -04:00
# defaults
page = await self.claim_search(channel='@abc', order_by=['height', '^name'])
page_claim_ids = [item['name'] for item in page]
2019-10-18 02:01:40 +03:00
self.assertEqual(page_claim_ids, self.streams[:DEFAULT_PAGE_SIZE])
2019-05-05 22:25:43 -04:00
# page with default page_size
2019-10-18 02:01:40 +03:00
page = await self.claim_search(page=2, channel='@abc', order_by=['height', '^name'])
page_claim_ids = [item['name'] for item in page]
self.assertEqual(page_claim_ids, self.streams[DEFAULT_PAGE_SIZE:(DEFAULT_PAGE_SIZE*2)])
# page_size larger than dataset
page = await self.claim_search(page_size=50, channel='@abc', order_by=['height', '^name'])
page_claim_ids = [item['name'] for item in page]
self.assertEqual(page_claim_ids, self.streams)
2019-10-18 02:01:40 +03:00
# page_size less than dataset
page = await self.claim_search(page_size=6, channel='@abc', order_by=['height', '^name'])
page_claim_ids = [item['name'] for item in page]
self.assertEqual(page_claim_ids, self.streams[:6])
2019-10-18 02:01:40 +03:00
# page and page_size
page = await self.claim_search(page=2, page_size=6, channel='@abc', order_by=['height', '^name'])
page_claim_ids = [item['name'] for item in page]
self.assertEqual(page_claim_ids, self.streams[6:12])
out_of_bounds = await self.claim_search(page=4, page_size=20, channel='@abc')
self.assertEqual(out_of_bounds, [])
2019-05-05 22:25:43 -04:00
async def test_tag_search(self):
2020-11-16 12:31:44 -05:00
claim1 = await self.stream_create('claim1', tag=['aBc'])
claim2 = await self.stream_create('claim2', tag=['#abc', 'def'])
claim3 = await self.stream_create('claim3', tag=['abc', 'ghi', 'jkl'])
claim4 = await self.stream_create('claim4', tag=['abc\t', 'ghi', 'mno'])
claim5 = await self.stream_create('claim5', tag=['pqr'])
2019-05-05 22:25:43 -04:00
# any_tags
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([claim5, claim4, claim3, claim2, claim1], any_tag=['\tabc', 'pqr'])
await self.assertFindsClaims([claim4, claim3, claim2, claim1], any_tag=['abc'])
await self.assertFindsClaims([claim4, claim3, claim2, claim1], any_tag=['abc', 'ghi'])
await self.assertFindsClaims([claim4, claim3], any_tag=['ghi'])
await self.assertFindsClaims([claim4, claim3], any_tag=['ghi', 'xyz'])
await self.assertFindsClaims([], any_tag=['xyz'])
2019-05-05 22:25:43 -04:00
# all_tags
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([], all_tag=['abc', 'pqr'])
await self.assertFindsClaims([claim4, claim3, claim2, claim1], all_tag=['ABC'])
await self.assertFindsClaims([claim4, claim3], all_tag=['abc', 'ghi'])
await self.assertFindsClaims([claim4, claim3], all_tag=['ghi'])
await self.assertFindsClaims([], all_tag=['ghi', 'xyz'])
await self.assertFindsClaims([], all_tag=['xyz'])
2019-05-05 22:25:43 -04:00
# not_tags
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([], not_tag=['abc', 'pqr'])
await self.assertFindsClaims([claim5], not_tag=['abC'])
await self.assertFindsClaims([claim5], not_tag=['abc', 'ghi'])
await self.assertFindsClaims([claim5, claim2, claim1], not_tag=['ghi'])
await self.assertFindsClaims([claim5, claim2, claim1], not_tag=['ghi', 'xyz'])
await self.assertFindsClaims([claim5, claim4, claim3, claim2, claim1], not_tag=['xyz'])
# combinations
2020-11-16 12:31:44 -05:00
await self.assertFindsClaims([claim3], all_tag=['abc', 'ghi'], not_tag=['mno'])
await self.assertFindsClaims([claim3], all_tag=['abc', 'ghi'], any_tag=['jkl'], not_tag=['mno'])
await self.assertFindsClaims([claim4, claim3, claim2], all_tags=['abc'], any_tag=['def', 'ghi'])
2019-05-17 23:54:03 -04:00
async def test_order_by(self):
2020-11-17 14:30:33 -05:00
height = await self.api.block_tip()
2019-05-17 23:54:03 -04:00
claims = [await self.stream_create(f'claim{i}') for i in range(5)]
await self.assertFindsClaims(claims, order_by=["^height"])
await self.assertFindsClaims(list(reversed(claims)), order_by=["height"])
await self.assertFindsClaims([claims[0]], height=height+1)
await self.assertFindsClaims([claims[4]], height=height+5)
await self.assertFindsClaims(claims[:1], height=f'<{height+2}', order_by=["^height"])
await self.assertFindsClaims(claims[:2], height=f'<={height+2}', order_by=["^height"])
await self.assertFindsClaims(claims[2:], height=f'>{height+2}', order_by=["^height"])
await self.assertFindsClaims(claims[1:], height=f'>={height+2}', order_by=["^height"])
await self.assertFindsClaims(claims, order_by=["^name"])
async def test_search_by_fee(self):
claim1 = await self.stream_create('claim1', fee_amount='1.0', fee_currency='lbc')
claim2 = await self.stream_create('claim2', fee_amount='0.9', fee_currency='lbc')
claim3 = await self.stream_create('claim3', fee_amount='0.5', fee_currency='lbc')
claim4 = await self.stream_create('claim4', fee_amount='0.1', fee_currency='lbc')
claim5 = await self.stream_create('claim5', fee_amount='1.0', fee_currency='usd')
await self.assertFindsClaims([claim5, claim4, claim3, claim2, claim1], fee_amount='>0')
await self.assertFindsClaims([claim4, claim3, claim2, claim1], fee_currency='lbc')
await self.assertFindsClaims([claim3, claim2, claim1], fee_amount='>0.1', fee_currency='lbc')
await self.assertFindsClaims([claim4, claim3, claim2], fee_amount='<1.0', fee_currency='lbc')
await self.assertFindsClaims([claim3], fee_amount='0.5', fee_currency='lbc')
await self.assertFindsClaims([claim5], fee_currency='usd')
await self.assertFindsClaims([], fee_currency='foo')
async def test_search_by_channel(self):
match = self.assertFindsClaims
chan1_id = self.get_claim_id(await self.channel_create('@chan1'))
chan2_id = self.get_claim_id(await self.channel_create('@chan2'))
chan3_id = self.get_claim_id(await self.channel_create('@chan3'))
chan4 = await self.channel_create('@chan4', '0.1')
claim1 = await self.stream_create('claim1')
claim2 = await self.stream_create('claim2', channel_id=chan1_id)
claim3 = await self.stream_create('claim3', channel_id=chan1_id)
claim4 = await self.stream_create('claim4', channel_id=chan2_id)
claim5 = await self.stream_create('claim5', channel_id=chan2_id)
claim6 = await self.stream_create('claim6', channel_id=chan3_id)
2020-11-16 15:12:34 -05:00
await self.channel_abandon(claim_id=chan3_id)
# {has/valid/invalid}_channel_signature
await match([claim6, claim5, claim4, claim3, claim2], has_channel_signature=True)
await match([claim5, claim4, claim3, claim2, claim1], valid_channel_signature=True, claim_type='stream')
await match([claim6, claim1], invalid_channel_signature=True, claim_type='stream')
await match([claim5, claim4, claim3, claim2], has_channel_signature=True, valid_channel_signature=True)
await match([claim6], has_channel_signature=True, invalid_channel_signature=True)
# not_channel_ids
await match([claim6, claim5, claim4, claim3, claim2, claim1], not_channel_ids=['abc123'], claim_type='stream')
await match([claim5, claim4, claim3, claim2, claim1], not_channel_ids=[chan3_id], claim_type='stream')
await match([claim6, claim5, claim4, claim1], not_channel_ids=[chan1_id], claim_type='stream')
await match([claim6, claim3, claim2, claim1], not_channel_ids=[chan2_id], claim_type='stream')
await match([claim6, claim1], not_channel_ids=[chan1_id, chan2_id], claim_type='stream')
await match([claim6, claim1, chan4], not_channel_ids=[chan1_id, chan2_id])
# not_channel_ids + valid_channel_signature
await match([claim5, claim4, claim3, claim2, claim1],
not_channel_ids=['abc123'], valid_channel_signature=True, claim_type='stream')
await match([claim5, claim4, claim1],
not_channel_ids=[chan1_id], valid_channel_signature=True, claim_type='stream')
await match([claim3, claim2, claim1],
not_channel_ids=[chan2_id], valid_channel_signature=True, claim_type='stream')
await match([claim1], not_channel_ids=[chan1_id, chan2_id], valid_channel_signature=True, claim_type='stream')
# not_channel_ids + has_channel_signature
await match([claim6, claim5, claim4, claim3, claim2], not_channel_ids=['abc123'], has_channel_signature=True)
await match([claim6, claim5, claim4], not_channel_ids=[chan1_id], has_channel_signature=True)
await match([claim6, claim3, claim2], not_channel_ids=[chan2_id], has_channel_signature=True)
await match([claim6], not_channel_ids=[chan1_id, chan2_id], has_channel_signature=True)
# not_channel_ids + has_channel_signature + valid_channel_signature
await match([claim5, claim4, claim3, claim2],
not_channel_ids=['abc123'], has_channel_signature=True, valid_channel_signature=True)
await match([claim5, claim4],
not_channel_ids=[chan1_id], has_channel_signature=True, valid_channel_signature=True)
await match([claim3, claim2],
not_channel_ids=[chan2_id], has_channel_signature=True, valid_channel_signature=True)
await match([], not_channel_ids=[chan1_id, chan2_id], has_channel_signature=True, valid_channel_signature=True)
async def test_claim_type_and_media_type_search(self):
# create an invalid/unknown claim
address = await self.account.receiving.get_or_create_usable_address()
2020-05-18 08:29:15 -04:00
tx = await self.wallet.claims.create(
'unknown', b'{"sources":{"lbry_sd_hash":""}}', 1, address, [self.account], self.account)
2020-11-16 15:12:34 -05:00
await self.service.broadcast(tx)
await self.generate(1)
await self.service.wait(tx)
octet = await self.stream_create()
video = await self.stream_create('chrome', file_path=self.video_file_name)
image = await self.stream_create('blank-image', data=self.image_data, suffix='.png')
repost = await self.stream_repost(self.get_claim_id(image))
channel = await self.channel_create()
unknown = self.sout(tx)
# claim_type
await self.assertFindsClaims([image, video, octet, unknown], claim_type='stream')
await self.assertFindsClaims([channel], claim_type='channel')
await self.assertFindsClaims([repost], claim_type='repost')
# stream_type
2020-11-17 14:30:33 -05:00
await self.assertFindsClaims([octet, unknown], stream_type=['binary'])
await self.assertFindsClaims([video], stream_type=['video'])
await self.assertFindsClaims([image], stream_type=['image'])
await self.assertFindsClaims([image, video], stream_type=['video', 'image'])
# media_type
2020-11-17 14:30:33 -05:00
await self.assertFindsClaims([octet, unknown], media_type=['application/octet-stream'])
await self.assertFindsClaims([video], media_type=['video/mp4'])
await self.assertFindsClaims([image], media_type=['image/png'])
await self.assertFindsClaims([image, video], media_type=['video/mp4', 'image/png'])
2020-01-29 15:41:40 -05:00
# duration
2020-01-29 16:08:20 -05:00
await self.assertFindsClaim(video, duration='>14')
await self.assertFindsClaim(video, duration='<16')
2020-01-29 15:41:40 -05:00
await self.assertFindsClaim(video, duration=15)
2020-01-29 16:08:20 -05:00
await self.assertFindsClaims([], duration='>100')
await self.assertFindsClaims([], duration='<14')
2020-01-29 15:41:40 -05:00
2021-01-07 22:28:07 -05:00
@skip
2019-11-14 14:31:49 -05:00
async def test_search_by_text(self):
chan1_id = self.get_claim_id(await self.channel_create('@SatoshiNakamoto'))
chan2_id = self.get_claim_id(await self.channel_create('@Bitcoin'))
chan3_id = self.get_claim_id(await self.channel_create('@IAmSatoshi'))
claim1 = await self.stream_create(
"the-real-satoshi", title="The Real Satoshi Nakamoto",
description="Documentary about the real Satoshi Nakamoto, creator of bitcoin.",
tags=['satoshi nakamoto', 'bitcoin', 'documentary']
)
claim2 = await self.stream_create(
"about-me", channel_id=chan1_id, title="Satoshi Nakamoto Autobiography",
description="I am Satoshi Nakamoto and this is my autobiography.",
tags=['satoshi nakamoto', 'bitcoin', 'documentary', 'autobiography']
)
claim3 = await self.stream_create(
"history-of-bitcoin", channel_id=chan2_id, title="History of Bitcoin",
description="History of bitcoin and its creator Satoshi Nakamoto.",
tags=['satoshi nakamoto', 'bitcoin', 'documentary', 'history']
)
claim4 = await self.stream_create(
"satoshi-conspiracies", channel_id=chan3_id, title="Satoshi Nakamoto Conspiracies",
description="Documentary detailing various conspiracies surrounding Satoshi Nakamoto.",
tags=['conspiracies', 'bitcoin', 'satoshi nakamoto']
)
await self.assertFindsClaims([], text='cheese')
await self.assertFindsClaims([claim2], text='autobiography')
2019-11-14 14:31:49 -05:00
await self.assertFindsClaims([claim3], text='history')
await self.assertFindsClaims([claim4], text='conspiracy')
await self.assertFindsClaims([], text='conspiracy AND history')
await self.assertFindsClaims([claim4, claim3], text='conspiracy OR history')
2019-11-14 14:31:49 -05:00
await self.assertFindsClaims([claim1, claim4, claim2, claim3], text='documentary')
await self.assertFindsClaims([claim4, claim1, claim2, claim3], text='satoshi')
claim2 = await self.stream_update(
self.get_claim_id(claim2), clear_tags=True, tags=['cloud'],
title="Satoshi Nakamoto Nography",
description="I am Satoshi Nakamoto and this is my nography.",
)
await self.assertFindsClaims([], text='autobiography')
await self.assertFindsClaims([claim2], text='cloud')
await self.stream_abandon(self.get_claim_id(claim2))
await self.assertFindsClaims([], text='cloud')
2019-05-05 22:25:43 -04:00
2020-03-07 00:34:47 -05:00
class TransactionCommands(ClaimTestCase):
async def test_transaction_list(self):
channel_id = self.get_claim_id(await self.channel_create())
await self.channel_update(channel_id, bid='0.5')
await self.channel_abandon(claim_id=channel_id)
stream_id = self.get_claim_id(await self.stream_create())
await self.stream_update(stream_id, bid='0.5')
await self.stream_abandon(claim_id=stream_id)
r = await self.transaction_list()
self.assertEqual(7, len(r))
self.assertEqual(stream_id, r[0]['abandon_info'][0]['claim_id'])
self.assertEqual(stream_id, r[1]['update_info'][0]['claim_id'])
self.assertEqual(stream_id, r[2]['claim_info'][0]['claim_id'])
self.assertEqual(channel_id, r[3]['abandon_info'][0]['claim_id'])
self.assertEqual(channel_id, r[4]['update_info'][0]['claim_id'])
self.assertEqual(channel_id, r[5]['claim_info'][0]['claim_id'])
class TransactionOutputCommands(ClaimTestCase):
2020-03-20 19:07:16 -04:00
async def test_txo_list_and_sum_filtering(self):
2020-03-07 00:34:47 -05:00
channel_id = self.get_claim_id(await self.channel_create())
2020-04-01 20:44:34 -04:00
self.assertEqual('1.0', lbc(await self.txo_sum(type='channel', is_not_spent=True)))
2020-03-07 00:34:47 -05:00
await self.channel_update(channel_id, bid='0.5')
2020-04-01 20:44:34 -04:00
self.assertEqual('0.5', lbc(await self.txo_sum(type='channel', is_not_spent=True)))
2020-03-20 19:07:16 -04:00
self.assertEqual('1.5', lbc(await self.txo_sum(type='channel')))
stream_id = self.get_claim_id(await self.stream_create(bid='1.3'))
2020-04-01 20:44:34 -04:00
self.assertEqual('1.3', lbc(await self.txo_sum(type='stream', is_not_spent=True)))
2020-03-20 19:07:16 -04:00
await self.stream_update(stream_id, bid='0.7')
2020-04-01 20:44:34 -04:00
self.assertEqual('0.7', lbc(await self.txo_sum(type='stream', is_not_spent=True)))
2020-03-20 19:07:16 -04:00
self.assertEqual('2.0', lbc(await self.txo_sum(type='stream')))
2020-04-01 20:44:34 -04:00
self.assertEqual('1.2', lbc(await self.txo_sum(type=['stream', 'channel'], is_not_spent=True)))
2020-03-20 19:07:16 -04:00
self.assertEqual('3.5', lbc(await self.txo_sum(type=['stream', 'channel'])))
2020-03-07 00:34:47 -05:00
# type filtering
r = await self.txo_list(type='channel')
self.assertEqual(2, len(r))
self.assertEqual('channel', r[0]['value_type'])
self.assertFalse(r[0]['is_spent'])
self.assertEqual('channel', r[1]['value_type'])
self.assertTrue(r[1]['is_spent'])
r = await self.txo_list(type='stream')
self.assertEqual(2, len(r))
self.assertEqual('stream', r[0]['value_type'])
self.assertFalse(r[0]['is_spent'])
self.assertEqual('stream', r[1]['value_type'])
self.assertTrue(r[1]['is_spent'])
r = await self.txo_list(type=['stream', 'channel'])
self.assertEqual(4, len(r))
self.assertEqual({'stream', 'channel'}, {c['value_type'] for c in r})
# claim_id filtering
r = await self.txo_list(claim_id=stream_id)
self.assertEqual(2, len(r))
self.assertEqual({stream_id}, {c['claim_id'] for c in r})
r = await self.txo_list(claim_id=[stream_id, channel_id])
self.assertEqual(4, len(r))
self.assertEqual({stream_id, channel_id}, {c['claim_id'] for c in r})
stream_name, _, channel_name, _ = (c['name'] for c in r)
r = await self.txo_list(claim_id=['beef'])
self.assertEqual(0, len(r))
# claim_name filtering
r = await self.txo_list(name=stream_name)
self.assertEqual(2, len(r))
self.assertEqual({stream_id}, {c['claim_id'] for c in r})
r = await self.txo_list(name=[stream_name, channel_name])
self.assertEqual(4, len(r))
self.assertEqual({stream_id, channel_id}, {c['claim_id'] for c in r})
r = await self.txo_list(name=['beef'])
self.assertEqual(0, len(r))
r = await self.txo_list()
self.assertEqual(9, len(r))
await self.stream_abandon(claim_id=stream_id)
r = await self.txo_list()
self.assertEqual(10, len(r))
r = await self.txo_list(claim_id=stream_id)
self.assertEqual(2, len(r))
self.assertTrue(r[0]['is_spent'])
self.assertTrue(r[1]['is_spent'])
async def test_txo_list_my_input_output_filtering(self):
2020-05-18 08:29:15 -04:00
wallet2 = await self.api.wallet_create('wallet2', create_account=True)
address2 = await self.api.address_unused(wallet_id=wallet2.id)
await self.channel_create('@kept-channel')
await self.channel_create('@sent-channel', claim_address=address2)
# all txos on second wallet
received_channel, = await self.txo_list(wallet_id=wallet2.id, is_my_input_or_output=True)
self.assertEqual('1.0', received_channel['amount'])
self.assertFalse(received_channel['is_my_input'])
self.assertTrue(received_channel['is_my_output'])
self.assertFalse(received_channel['is_internal_transfer'])
# all txos on default wallet
r = await self.txo_list(is_my_input_or_output=True)
self.assertEqual(
['1.0', '7.947786', '1.0', '8.973893', '10.0'],
[t['amount'] for t in r]
)
2020-03-08 23:11:03 -04:00
sent_channel, change2, kept_channel, change1, initial_funds = r
self.assertTrue(sent_channel['is_my_input'])
self.assertFalse(sent_channel['is_my_output'])
self.assertFalse(sent_channel['is_internal_transfer'])
self.assertTrue(change2['is_my_input'])
self.assertTrue(change2['is_my_output'])
self.assertTrue(change2['is_internal_transfer'])
self.assertTrue(kept_channel['is_my_input'])
self.assertTrue(kept_channel['is_my_output'])
self.assertFalse(kept_channel['is_internal_transfer'])
self.assertTrue(change1['is_my_input'])
self.assertTrue(change1['is_my_output'])
self.assertTrue(change1['is_internal_transfer'])
self.assertFalse(initial_funds['is_my_input'])
self.assertTrue(initial_funds['is_my_output'])
self.assertFalse(initial_funds['is_internal_transfer'])
# my stuff and stuff i sent excluding "change"
r = await self.txo_list(is_my_input_or_output=True, exclude_internal_transfers=True)
self.assertEqual([sent_channel, kept_channel, initial_funds], r)
# my unspent stuff and stuff i sent excluding "change"
2020-04-01 20:44:34 -04:00
r = await self.txo_list(is_my_input_or_output=True, is_not_spent=True, exclude_internal_transfers=True)
self.assertEqual([sent_channel, kept_channel], r)
# only "change"
r = await self.txo_list(is_my_input=True, is_my_output=True, type="other")
self.assertEqual([change2, change1], r)
# only unspent "change"
2020-04-01 20:44:34 -04:00
r = await self.txo_list(is_my_input=True, is_my_output=True, type="other", is_not_spent=True)
self.assertEqual([change2], r)
# only spent "change"
r = await self.txo_list(is_my_input=True, is_my_output=True, type="other", is_spent=True)
self.assertEqual([change1], r)
# all my unspent stuff
2020-04-01 20:44:34 -04:00
r = await self.txo_list(is_my_output=True, is_not_spent=True)
self.assertEqual([change2, kept_channel], r)
# stuff i sent
r = await self.txo_list(is_not_my_output=True)
self.assertEqual([sent_channel], r)
2020-03-08 23:11:03 -04:00
async def test_txo_plot(self):
day_blocks = int((24 * 60 * 60) / self.ledger.headers.timestamp_average_offset)
stream_id = self.get_claim_id(await self.stream_create())
await self.support_create(stream_id, '0.3')
await self.support_create(stream_id, '0.2')
await self.generate(day_blocks)
await self.support_create(stream_id, '0.4')
await self.support_create(stream_id, '0.5')
await self.generate(day_blocks)
await self.support_create(stream_id, '0.6')
plot = await self.txo_plot(type='support')
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-25', 'total': '0.6'},
], plot)
plot = await self.txo_plot(type='support', days_back=1)
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-24', 'total': '0.9'},
{'day': '2016-06-25', 'total': '0.6'},
], plot)
plot = await self.txo_plot(type='support', days_back=2)
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-23', 'total': '0.5'},
{'day': '2016-06-24', 'total': '0.9'},
{'day': '2016-06-25', 'total': '0.6'},
], plot)
plot = await self.txo_plot(type='support', start_day='2016-06-23')
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-23', 'total': '0.5'},
{'day': '2016-06-24', 'total': '0.9'},
{'day': '2016-06-25', 'total': '0.6'},
], plot)
plot = await self.txo_plot(type='support', start_day='2016-06-24')
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-24', 'total': '0.9'},
{'day': '2016-06-25', 'total': '0.6'},
], plot)
plot = await self.txo_plot(type='support', start_day='2016-06-23', end_day='2016-06-24')
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-23', 'total': '0.5'},
{'day': '2016-06-24', 'total': '0.9'},
], plot)
plot = await self.txo_plot(type='support', start_day='2016-06-23', days_after=1)
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-23', 'total': '0.5'},
{'day': '2016-06-24', 'total': '0.9'},
], plot)
plot = await self.txo_plot(type='support', start_day='2016-06-23', days_after=2)
self.assertEqual([
2020-03-26 01:13:09 -04:00
{'day': '2016-06-23', 'total': '0.5'},
{'day': '2016-06-24', 'total': '0.9'},
{'day': '2016-06-25', 'total': '0.6'},
], plot)
async def test_txo_spend(self):
stream_id = self.get_claim_id(await self.stream_create())
for _ in range(10):
await self.support_create(stream_id, '0.1')
await self.assertBalance(self.account, '7.978478')
2020-04-01 20:44:34 -04:00
self.assertEqual('1.0', lbc(await self.txo_sum(type='support', is_not_spent=True)))
txs = await self.txo_spend(type='support', batch_size=3, include_full_tx=True)
self.assertEqual(4, len(txs))
self.assertEqual(3, len(txs[0]['inputs']))
self.assertEqual(3, len(txs[1]['inputs']))
self.assertEqual(3, len(txs[2]['inputs']))
self.assertEqual(1, len(txs[3]['inputs']))
2020-04-01 20:44:34 -04:00
self.assertEqual('0.0', lbc(await self.txo_sum(type='support', is_not_spent=True)))
await self.assertBalance(self.account, '8.977606')
await self.support_create(stream_id, '0.1')
2020-05-18 08:29:15 -04:00
txs = await self.txo_spend(type='support', batch_size=3)
self.assertEqual(1, len(txs))
self.assertEqual({'txid'}, set(txs[0]))
2020-03-07 00:34:47 -05:00
class ClaimCommands(ClaimTestCase):
2020-03-07 00:34:47 -05:00
async def test_claim_list_filtering(self):
channel_id = self.get_claim_id(await self.channel_create())
stream_id = self.get_claim_id(await self.stream_create())
2020-03-07 00:34:47 -05:00
# type filtering
r = await self.claim_list(claim_type='channel')
self.assertEqual(1, len(r))
self.assertEqual('channel', r[0]['value_type'])
r = await self.claim_list(claim_type='stream')
self.assertEqual(1, len(r))
self.assertEqual('stream', r[0]['value_type'])
r = await self.claim_list(claim_type=['stream', 'channel'])
self.assertEqual(2, len(r))
self.assertEqual({'stream', 'channel'}, {c['value_type'] for c in r})
2020-03-07 00:34:47 -05:00
# claim_id filtering
r = await self.claim_list(claim_id=stream_id)
self.assertEqual(1, len(r))
self.assertEqual({stream_id}, {c['claim_id'] for c in r})
r = await self.claim_list(claim_id=[stream_id, channel_id])
self.assertEqual(2, len(r))
self.assertEqual({stream_id, channel_id}, {c['claim_id'] for c in r})
stream_name, channel_name = (c['name'] for c in r)
r = await self.claim_list(claim_id=['beef'])
self.assertEqual(0, len(r))
# claim_name filtering
r = await self.claim_list(name=stream_name)
self.assertEqual(1, len(r))
self.assertEqual({stream_id}, {c['claim_id'] for c in r})
r = await self.claim_list(name=[stream_name, channel_name])
self.assertEqual(2, len(r))
self.assertEqual({stream_id, channel_id}, {c['claim_id'] for c in r})
r = await self.claim_list(name=['beef'])
self.assertEqual(0, len(r))
async def test_claim_stream_channel_list_with_resolve(self):
self.assertListEqual([], await self.claim_list(resolve=True))
await self.channel_create()
await self.stream_create()
r = await self.claim_list()
self.assertNotIn('short_url', r[0])
self.assertNotIn('short_url', r[1])
self.assertNotIn('short_url', (await self.stream_list())[0])
self.assertNotIn('short_url', (await self.channel_list())[0])
r = await self.claim_list(resolve=True)
self.assertIn('short_url', r[0])
self.assertIn('short_url', r[1])
self.assertIn('short_url', (await self.stream_list(resolve=True))[0])
self.assertIn('short_url', (await self.channel_list(resolve=True))[0])
# unconfirmed channel won't resolve
2020-05-18 08:29:15 -04:00
channel_tx = await self.api.channel_create('@foo', '1.0')
await self.service.wait(channel_tx)
r = await self.claim_list(resolve=True)
self.assertEqual('NOT_FOUND', r[0]['meta']['error']['name'])
self.assertTrue(r[1]['meta']['is_controlling'])
r = await self.channel_list(resolve=True)
self.assertEqual('NOT_FOUND', r[0]['meta']['error']['name'])
self.assertTrue(r[1]['meta']['is_controlling'])
# confirm it
await self.generate(1)
await self.ledger.wait(channel_tx, self.blockchain.block_expected)
# all channel claims resolve
r = await self.claim_list(resolve=True)
self.assertTrue(r[0]['meta']['is_controlling'])
self.assertTrue(r[1]['meta']['is_controlling'])
r = await self.channel_list(resolve=True)
self.assertTrue(r[0]['meta']['is_controlling'])
self.assertTrue(r[1]['meta']['is_controlling'])
# unconfirmed stream won't resolve
2020-05-18 08:29:15 -04:00
stream_tx = await self.api.stream_create(
'foo', '1.0', file_path=self.create_upload_file(data=b'hi')
)
2020-05-18 08:29:15 -04:00
await self.service.wait(stream_tx)
r = await self.claim_list(resolve=True)
self.assertEqual('NOT_FOUND', r[0]['meta']['error']['name'])
self.assertTrue(r[1]['meta']['is_controlling'])
r = await self.stream_list(resolve=True)
self.assertEqual('NOT_FOUND', r[0]['meta']['error']['name'])
self.assertTrue(r[1]['meta']['is_controlling'])
# confirm it
await self.generate(1)
await self.ledger.wait(stream_tx, self.blockchain.block_expected)
# all claims resolve
r = await self.claim_list(resolve=True)
self.assertTrue(r[0]['meta']['is_controlling'])
self.assertTrue(r[1]['meta']['is_controlling'])
self.assertTrue(r[2]['meta']['is_controlling'])
self.assertTrue(r[3]['meta']['is_controlling'])
r = await self.stream_list(resolve=True)
self.assertTrue(r[0]['meta']['is_controlling'])
self.assertTrue(r[1]['meta']['is_controlling'])
r = await self.channel_list(resolve=True)
self.assertTrue(r[0]['meta']['is_controlling'])
self.assertTrue(r[1]['meta']['is_controlling'])
# check that metadata is transfered
self.assertTrue(r[0]['is_my_output'])
async def assertClaimList(self, claim_ids, **kwargs):
self.assertEqual(claim_ids, [c['claim_id'] for c in await self.claim_list(**kwargs)])
async def test_list_streams_in_channel_and_order_by(self):
channel1_id = self.get_claim_id(await self.channel_create('@chan-one'))
channel2_id = self.get_claim_id(await self.channel_create('@chan-two'))
stream1_id = self.get_claim_id(await self.stream_create('stream-a', bid='0.3', channel_id=channel1_id))
stream2_id = self.get_claim_id(await self.stream_create('stream-b', bid='0.9', channel_id=channel1_id))
stream3_id = self.get_claim_id(await self.stream_create('stream-c', bid='0.6', channel_id=channel2_id))
await self.assertClaimList([stream2_id, stream1_id], channel_id=channel1_id)
await self.assertClaimList([stream3_id], channel_id=channel2_id)
await self.assertClaimList([stream3_id, stream2_id, stream1_id], channel_id=[channel1_id, channel2_id])
await self.assertClaimList([stream1_id, stream2_id, stream3_id], claim_type='stream', order_by='name')
await self.assertClaimList([stream1_id, stream3_id, stream2_id], claim_type='stream', order_by='amount')
await self.assertClaimList([stream3_id, stream2_id, stream1_id], claim_type='stream', order_by='height')
async def test_claim_list_with_tips(self):
2020-05-18 08:29:15 -04:00
wallet2 = await self.api.wallet_create('wallet2', create_account=True)
address2 = await self.api.address_unused(wallet_id=wallet2.id)
await self.wallet_send('5.0', address2)
stream1_id = self.get_claim_id(await self.stream_create('one'))
stream2_id = self.get_claim_id(await self.stream_create('two'))
claims = await self.claim_list()
self.assertNotIn('received_tips', claims[0])
self.assertNotIn('received_tips', claims[1])
claims = await self.claim_list(include_received_tips=True)
self.assertEqual('0.0', claims[0]['received_tips'])
self.assertEqual('0.0', claims[1]['received_tips'])
await self.support_create(stream1_id, '0.7', tip=True)
await self.support_create(stream1_id, '0.3', tip=True, wallet_id=wallet2.id)
await self.support_create(stream1_id, '0.2', tip=True, wallet_id=wallet2.id)
await self.support_create(stream2_id, '0.4', tip=True, wallet_id=wallet2.id)
await self.support_create(stream2_id, '0.5', tip=True, wallet_id=wallet2.id)
await self.support_create(stream2_id, '0.1', tip=True, wallet_id=wallet2.id)
claims = await self.claim_list(include_received_tips=True)
self.assertEqual('1.0', claims[0]['received_tips'])
self.assertEqual('1.2', claims[1]['received_tips'])
2021-01-08 00:11:12 -05:00
await self.support_abandon(claim_id=stream1_id)
claims = await self.claim_list(include_received_tips=True)
self.assertEqual('1.0', claims[0]['received_tips'])
self.assertEqual('0.0', claims[1]['received_tips'])
2021-01-08 00:11:12 -05:00
await self.support_abandon(claim_id=stream2_id)
claims = await self.claim_list(include_received_tips=True)
self.assertEqual('0.0', claims[0]['received_tips'])
self.assertEqual('0.0', claims[1]['received_tips'])
2019-03-24 16:55:04 -04:00
class ChannelCommands(CommandTestCase):
async def test_create_channel_names(self):
# claim new name
2019-03-25 22:06:36 -04:00
await self.channel_create('@foo')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '8.991893')
# fail to claim duplicate
2020-05-18 08:29:15 -04:00
with self.assertRaisesRegex(Exception, "You already have a claim published under the name '@foo'."):
2019-03-25 22:06:36 -04:00
await self.channel_create('@foo')
2019-03-24 16:55:04 -04:00
# fail to claim invalid name
2019-03-25 22:06:36 -04:00
with self.assertRaisesRegex(Exception, "Channel names must start with '@' symbol."):
await self.channel_create('foo')
2019-03-24 16:55:04 -04:00
# nothing's changed after failed attempts
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '8.991893')
# succeed overriding duplicate restriction
2019-03-25 22:06:36 -04:00
await self.channel_create('@foo', allow_duplicate_name=True)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 2)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '7.983786')
async def test_channel_bids(self):
# enough funds
2019-03-25 22:06:36 -04:00
tx = await self.channel_create('@foo', '5.0')
claim_id = self.get_claim_id(tx)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '4.991893')
# bid preserved on update
2019-03-25 22:06:36 -04:00
tx = await self.channel_update(claim_id)
2019-03-24 16:55:04 -04:00
self.assertEqual(tx['outputs'][0]['amount'], '5.0')
# bid changed on update
2019-03-25 22:06:36 -04:00
tx = await self.channel_update(claim_id, bid='4.0')
2019-03-24 16:55:04 -04: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-25 22:06:36 -04:00
await self.channel_create('@foo2', '9.0')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '5.991447')
# spend exactly amount available, no change
2019-03-25 22:06:36 -04:00
tx = await self.channel_create('@foo3', '5.981266')
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '0.0')
self.assertEqual(len(tx['outputs']), 1) # no change
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 2)
2019-03-24 16:55:04 -04:00
async def test_setting_channel_fields(self):
values = {
'title': "Cool Channel",
'description': "Best channel on LBRY.",
2019-03-25 12:30:30 -04:00
'thumbnail_url': "https://co.ol/thumbnail.png",
2020-05-18 08:29:15 -04:00
'tag': ["cool", "awesome"],
'language': ["en-US"],
'location': ['US::Manchester'],
2019-04-20 01:12:43 -04:00
'email': "human@email.com",
'website_url': "https://co.ol",
2019-03-25 12:30:30 -04:00
'cover_url': "https://co.ol/cover.png",
'featured': ['cafe']
2019-03-24 16:55:04 -04:00
}
fixed_values = values.copy()
2019-04-20 01:12:43 -04:00
fixed_values['thumbnail'] = {'url': fixed_values.pop('thumbnail_url')}
fixed_values['cover'] = {'url': fixed_values.pop('cover_url')}
2020-05-18 08:29:15 -04:00
fixed_values['tags'] = fixed_values.pop('tag')
fixed_values['languages'] = fixed_values.pop('language')
fixed_values['locations'] = [{'country': 'US', 'city': 'Manchester'}]
fixed_values.pop('location')
2019-03-24 16:55:04 -04:00
# create new channel with all fields set
2020-05-18 08:29:15 -04:00
tx = await self.channel_create('@bigchannel', **values)
channel = tx['outputs'][0]['value']
self.assertEqual(channel, {
'public_key': channel['public_key'],
'public_key_id': channel['public_key_id'],
**fixed_values
})
2019-02-11 17:45:52 -05:00
2019-03-24 16:55:04 -04:00
# create channel with nothing set
2020-05-18 08:29:15 -04:00
tx = await self.channel_create('@lightchannel')
channel = tx['outputs'][0]['value']
self.assertEqual(
channel, {'public_key': channel['public_key'], 'public_key_id': channel['public_key_id']})
2019-02-11 17:45:52 -05:00
# create channel with just a featured claim
2020-05-18 08:29:15 -04:00
tx = await self.channel_create('@featurechannel', featured='beef')
2019-03-24 16:55:04 -04:00
txo = tx['outputs'][0]
claim_id, channel = txo['claim_id'], txo['value']
fixed_values['public_key'] = channel['public_key']
fixed_values['public_key_id'] = channel['public_key_id']
self.assertEqual(channel, {
'public_key': fixed_values['public_key'],
'public_key_id': fixed_values['public_key_id'],
'featured': ['beef']
})
2019-02-11 17:45:52 -05:00
2019-05-20 01:51:54 +05:30
# update channel "@featurechannel" setting all fields
2020-05-18 08:29:15 -04:00
tx = await self.channel_update(claim_id, **values)
channel = tx['outputs'][0]['value']
fixed_values['featured'].insert(0, 'beef') # existing featured claim
self.assertEqual(channel, fixed_values)
2019-02-11 17:45:52 -05:00
# clearing and settings featured content
2020-05-18 08:29:15 -04:00
tx = await self.channel_update(claim_id, featured='beefcafe', clear_featured=True)
channel = tx['outputs'][0]['value']
fixed_values['featured'] = ['beefcafe']
self.assertEqual(channel, fixed_values)
2019-02-11 17:45:52 -05:00
2019-03-24 16:55:04 -04:00
# reset signing key
2020-05-18 08:29:15 -04:00
tx = await self.channel_update(claim_id, new_signing_key=True)
channel = tx['outputs'][0]['value']
self.assertNotEqual(channel['public_key'], fixed_values['public_key'])
2019-02-11 17:45:52 -05:00
# replace mode (clears everything except public_key)
2020-05-18 08:29:15 -04:00
tx = await self.channel_update(claim_id, replace=True, title='foo', email='new@email.com')
self.assertEqual(tx['outputs'][0]['value'], {
'public_key': channel['public_key'],
'public_key_id': channel['public_key_id'],
'title': 'foo', 'email': 'new@email.com'}
)
# move channel to another account
2020-05-18 08:29:15 -04:00
account2 = await self.api.account_create('second account')
2019-02-11 17:45:52 -05:00
# before moving
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 3)
self.assertEqual(len(await self.channel_list(account_id=account2.id)), 0)
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
await self.channel_update(claim_id, account_id=account2.id)
2019-03-24 16:55:04 -04:00
# after moving
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.channel_list()), 3)
self.assertEqual(len(await self.channel_list(account_id=self.account.id)), 2)
self.assertEqual(len(await self.channel_list(account_id=account2.id)), 1)
2019-03-24 16:55:04 -04:00
2020-11-16 15:12:34 -05:00
@skip
async def test_channel_export_import_before_sending_channel(self):
# export
2019-05-22 16:03:57 +05:30
tx = await self.channel_create('@foo', '1.0')
claim_id = self.get_claim_id(tx)
2020-11-16 15:12:34 -05:00
channels = await self.wallet.channels.list()
2020-05-18 08:29:15 -04:00
channel_private_key = channels[0].private_key
exported_data = await self.out(self.api.channel_export(claim_id))
# import
2020-05-18 08:29:15 -04:00
api2 = (await self.add_daemon()).api
self.assertEqual(len((await self.out(api2.channel_list()))['items']), 0)
await api2.channel_import(exported_data)
channels = (await self.out(api2.channel_list()))['items']
2019-05-29 17:40:22 -04:00
self.assertEqual(1, len(channels))
self.assertEqual(channel_private_key.to_string(), channels[0].private_key.to_string())
2019-05-22 16:03:57 +05:30
# second wallet can't update until channel is sent to it
with self.assertRaisesRegex(AssertionError, 'Cannot find private key for signing output.'):
2020-05-18 08:29:15 -04:00
await api2.channel_update(claim_id, bid='0.5')
# now send the channel as well
2020-05-18 08:29:15 -04:00
await self.channel_update(claim_id, claim_address=await api2.address_unused())
# second wallet should be able to update now
2020-05-18 08:29:15 -04:00
await api2.channel_update(claim_id, bid='0.5')
async def test_move_channel_between_accounts(self):
account2 = await self.api.account_create('second account')
# initial case
channel = await self.channel_create('@spam', '1.0', account_id=account2.id)
self.assertFalse(self.account.channel_keys)
self.assertTrue(account2.channel_keys)
self.assertEqual(await self.wallet.get_account_for_address(self.get_address(channel)), account2)
# updating channel leaves existing address
await self.channel_update(self.get_claim_id(channel), bid='1.5')
updated = await self.channel_list()
self.assertEqual(updated[0]['amount'], '1.5')
self.assertEqual(updated[0]['address'], self.get_address(channel))
self.assertFalse(self.account.channel_keys)
self.assertTrue(account2.channel_keys)
# moving channel gives it new address and adds channel key to other account
await self.channel_update(self.get_claim_id(channel), bid='2.0', account_id=self.account.id)
updated = await self.channel_list()
self.assertEqual(updated[0]['amount'], '2.0')
self.assertEqual(await self.wallet.get_account_for_address(updated[0]['address']), self.account)
self.assertTrue(self.account.channel_keys)
self.assertTrue(account2.channel_keys)
2019-09-25 12:28:02 +02:00
async def test_tag_normalization(self):
2020-05-18 08:29:15 -04:00
tx1 = await self.channel_create('@abc', '1.0', tag=['aBc', ' ABC ', 'xYZ ', 'xyz'])
2019-09-25 12:28:02 +02:00
claim_id = self.get_claim_id(tx1)
self.assertCountEqual(tx1['outputs'][0]['value']['tags'], ['abc', 'xyz'])
2020-05-18 08:29:15 -04:00
tx2 = await self.channel_update(claim_id, tag=[' pqr', 'PQr '])
2019-09-25 12:28:02 +02:00
self.assertCountEqual(tx2['outputs'][0]['value']['tags'], ['abc', 'xyz', 'pqr'])
2020-05-18 08:29:15 -04:00
tx3 = await self.channel_update(claim_id, tag=' pqr')
2019-09-25 12:28:02 +02:00
self.assertCountEqual(tx3['outputs'][0]['value']['tags'], ['abc', 'xyz', 'pqr'])
2020-05-18 08:29:15 -04:00
tx4 = await self.channel_update(claim_id, tag=[' pqr', 'PQr '], clear_tags=True)
2019-09-25 12:28:02 +02:00
self.assertEqual(tx4['outputs'][0]['value']['tags'], ['pqr'])
2019-03-24 16:55:04 -04:00
2019-10-24 16:48:00 -04:00
class StreamCommands(ClaimTestCase):
2019-03-25 22:06:36 -04:00
async def test_create_stream_names(self):
2019-03-24 16:55:04 -04:00
# claim new name
2019-03-25 22:06:36 -04:00
await self.stream_create('foo')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '8.993893')
# fail to claim duplicate
2019-03-25 22:06:36 -04:00
with self.assertRaisesRegex(
2020-11-16 15:26:50 -05:00
Exception, "You already have a claim published under the name 'foo'."):
2019-03-25 22:06:36 -04:00
await self.stream_create('foo')
2019-03-24 16:55:04 -04:00
# fail claim starting with @
2019-03-25 22:06:36 -04:00
with self.assertRaisesRegex(
Exception, "Stream names cannot start with '@' symbol."):
await self.stream_create('@foo')
2019-03-24 16:55:04 -04:00
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '8.993893')
# succeed overriding duplicate restriction
2019-03-25 22:06:36 -04:00
await self.stream_create('foo', allow_duplicate_name=True)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 2)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '7.987786')
2019-03-25 22:06:36 -04:00
async def test_stream_bids(self):
2019-03-24 16:55:04 -04:00
# enough funds
2019-03-25 22:06:36 -04:00
tx = await self.stream_create('foo', '2.0')
claim_id = self.get_claim_id(tx)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 1)
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '7.993893')
# bid preserved on update
2019-03-25 22:06:36 -04:00
tx = await self.stream_update(claim_id)
2019-03-24 16:55:04 -04:00
self.assertEqual(tx['outputs'][0]['amount'], '2.0')
# bid changed on update
2019-03-25 22:06:36 -04:00
tx = await self.stream_update(claim_id, bid='3.0')
2019-03-24 16:55:04 -04:00
self.assertEqual(tx['outputs'][0]['amount'], '3.0')
await self.assertBalance(self.account, '6.993319')
2019-03-24 16:55:04 -04:00
# not enough funds
with self.assertRaisesRegex(
InsufficientFundsError, "Not enough funds to cover this transaction."):
2019-03-25 22:06:36 -04:00
await self.stream_create('foo2', '9.0')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 1)
await self.assertBalance(self.account, '6.993319')
2019-03-24 16:55:04 -04:00
# spend exactly amount available, no change
tx = await self.stream_create('foo3', '6.98523')
2019-03-24 16:55:04 -04:00
await self.assertBalance(self.account, '0.0')
self.assertEqual(len(tx['outputs']), 1) # no change
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 2)
2019-03-24 16:55:04 -04:00
async def test_stream_update_and_abandon_across_accounts(self):
2020-05-18 08:29:15 -04:00
account2 = await self.api.account_create('second account')
stream = await self.stream_create('spam', '1.0', account_id=account2.id)
# stream not in account1
with self.assertRaisesRegex(Exception, "Can't find the stream"):
await self.stream_update(self.get_claim_id(stream), bid='2.0', account_id=self.account.id)
# stream is in account2
await self.stream_update(self.get_claim_id(stream), bid='2.0', account_id=account2.id)
2020-05-18 08:29:15 -04:00
self.assertEqual((await self.stream_list())[0]['amount'], '2.0')
# check all accounts for stream
await self.stream_update(self.get_claim_id(stream), bid='3.0')
2020-05-18 08:29:15 -04:00
self.assertEqual((await self.stream_list())[0]['amount'], '3.0')
await self.stream_abandon(self.get_claim_id(stream))
2019-03-24 16:55:04 -04:00
async def test_publishing_checks_all_accounts_for_channel(self):
2020-05-18 08:29:15 -04:00
account1 = self.account
account2 = await self.api.account_create('second account')
2019-02-11 17:45:52 -05:00
2019-03-25 22:06:36 -04:00
await self.out(self.channel_create('@spam', '1.0'))
2020-05-18 08:29:15 -04:00
await self.assertBalance(self.account, '8.989893')
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
await self.account_send('5.0', await self.api.address_unused(account2.id))
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
await self.assertBalance(self.account, '3.989769')
await self.assertBalance(account2, '5.0')
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
baz_tx = await self.channel_create('@baz', '1.0', account_id=account2.id)
baz_id = self.get_claim_id(baz_tx)
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
channels = await self.channel_list(account1.id)
self.assertEqual(len(channels), 1)
self.assertEqual(channels[0]['name'], '@spam')
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
channels = await self.channel_list(account2.id)
self.assertEqual(len(channels), 1)
self.assertEqual(channels[0]['name'], '@baz')
2019-02-11 17:45:52 -05:00
2020-05-18 08:29:15 -04:00
channels = await self.channel_list()
self.assertEqual(len(channels), 2)
self.assertEqual(channels[0]['name'], '@baz')
self.assertEqual(channels[1]['name'], '@spam')
2019-02-11 17:45:52 -05:00
# defaults to using all accounts to lookup channel
await self.stream_create('hovercraft1', '0.1', channel_id=baz_id)
2019-04-29 00:38:58 -04:00
self.assertEqual((await self.claim_search(name='hovercraft1'))[0]['signing_channel']['name'], '@baz')
# lookup by channel_name in all accounts
await self.stream_create('hovercraft2', '0.1', channel_name='@baz')
2019-04-29 00:38:58 -04:00
self.assertEqual((await self.claim_search(name='hovercraft2'))[0]['signing_channel']['name'], '@baz')
# uses only the specific accounts which contains the channel
2020-05-18 08:29:15 -04:00
await self.stream_create('hovercraft3', '0.1', channel_id=baz_id, channel_account_id=[account2.id])
2019-04-29 00:38:58 -04:00
self.assertEqual((await self.claim_search(name='hovercraft3'))[0]['signing_channel']['name'], '@baz')
# lookup by channel_name in specific account
2020-05-18 08:29:15 -04:00
await self.stream_create('hovercraft4', '0.1', channel_name='@baz', channel_account_id=[account2.id])
2019-04-29 00:38:58 -04:00
self.assertEqual((await self.claim_search(name='hovercraft4'))[0]['signing_channel']['name'], '@baz')
2019-02-11 17:45:52 -05:00
# fails when specifying account which does not contain channel
2019-03-24 16:55:04 -04:00
with self.assertRaisesRegex(ValueError, "Couldn't find channel with channel_id"):
2019-03-25 22:06:36 -04:00
await self.stream_create(
2020-05-18 08:29:15 -04:00
'hovercraft5', '0.1', channel_id=baz_id, channel_account_id=[account1.id]
)
# fail with channel_name
with self.assertRaisesRegex(ValueError, "Couldn't find channel with channel_name '@baz'"):
await self.stream_create(
2020-05-18 08:29:15 -04:00
'hovercraft5', '0.1', channel_name='@baz', channel_account_id=[account1.id]
2019-03-24 16:55:04 -04:00
)
# signing with channel works even if channel and certificate are in different accounts
await self.channel_update(
2020-05-18 08:29:15 -04:00
baz_id, account_id=account2.id,
claim_address=await self.api.address_unused(account1.id)
)
await self.stream_create(
'hovercraft5', '0.1', channel_id=baz_id
)
async def test_preview_works_with_signed_streams(self):
await self.channel_create('@spam', '1.0')
signed = await self.stream_create('bar', '1.0', channel_name='@spam', preview=True, confirm=False)
self.assertTrue(signed['outputs'][0]['is_channel_signature_valid'])
2019-10-14 05:17:37 -03:00
async def test_repost(self):
sql = self.conductor.spv_node.server.bp.sql
sql.execute(sql.TAG_INDEXES)
await self.channel_create('@goodies', '1.0')
2020-05-18 08:29:15 -04:00
tx = await self.stream_create('newstuff', '1.1', channel_name='@goodies', tag=['foo', 'gaming'])
claim_id = self.get_claim_id(tx)
self.assertEqual((await self.claim_search(name='newstuff'))[0]['meta']['reposted'], 0)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.txo_list(reposted_claim_id=claim_id)), 0)
self.assertEqual(len(await self.txo_list(type='repost')), 0)
tx = await self.stream_repost(claim_id, 'newstuff-again', '1.1')
repost_id = self.get_claim_id(tx)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list(claim_type='repost')), 1)
self.assertEqual((await self.claim_search(name='newstuff'))[0]['meta']['reposted'], 1)
self.assertEqual((await self.claim_search(reposted_claim_id=claim_id))[0]['claim_id'], repost_id)
2020-03-20 20:22:57 -04:00
self.assertEqual((await self.txo_list(reposted_claim_id=claim_id))[0]['claim_id'], repost_id)
self.assertEqual((await self.txo_list(type='repost'))[0]['claim_id'], repost_id)
# tags are inherited (non-common / indexed tags)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_search(any_tags=['foo'], claim_type=['stream', 'repost'])), 2)
self.assertEqual(len(await self.claim_search(all_tags=['foo'], claim_type=['stream', 'repost'])), 2)
self.assertEqual(len(await self.claim_search(not_tags=['foo'], claim_type=['stream', 'repost'])), 0)
# "common" / indexed tags work too
self.assertIn('gaming', sql.TAG_INDEXES) # if this breaks, next test doesn't make sense
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_search(any_tags=['gaming'], claim_type=['stream', 'repost'])), 2)
self.assertEqual(len(await self.claim_search(all_tags=['gaming'], claim_type=['stream', 'repost'])), 2)
self.assertEqual(len(await self.claim_search(not_tags=['gaming'], claim_type=['stream', 'repost'])), 0)
await self.channel_create('@reposting-goodies', '1.0')
2019-10-14 05:17:37 -03:00
await self.stream_repost(claim_id, 'repost-on-channel', '1.1', channel_name='@reposting-goodies')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list(claim_type='repost')), 2)
self.assertEqual(len(await self.claim_search(reposted_claim_id=claim_id)), 2)
self.assertEqual((await self.claim_search(name='newstuff'))[0]['meta']['reposted'], 2)
search_results = await self.claim_search(reposted='>=2')
self.assertEqual(len(search_results), 1)
self.assertEqual(search_results[0]['name'], 'newstuff')
search_results = await self.claim_search(name='repost-on-channel')
2019-10-14 05:17:37 -03:00
self.assertEqual(len(search_results), 1)
search = search_results[0]
self.assertEqual(search['name'], 'repost-on-channel')
self.assertEqual(search['signing_channel']['name'], '@reposting-goodies')
self.assertEqual(search['reposted_claim']['name'], 'newstuff')
self.assertEqual(search['reposted_claim']['meta']['reposted'], 2)
self.assertEqual(search['reposted_claim']['signing_channel']['name'], '@goodies')
resolved = await self.out(
2020-05-18 08:29:15 -04:00
self.api.resolve(['@reposting-goodies/repost-on-channel', 'newstuff-again'])
)
self.assertEqual(resolved['@reposting-goodies/repost-on-channel'], search)
self.assertEqual(resolved['newstuff-again']['reposted_claim']['name'], 'newstuff')
2019-10-14 05:17:37 -03:00
2019-11-05 04:02:05 -03:00
async def test_filtering_channels_for_removing_content(self):
await self.channel_create('@some_channel', '0.1')
2020-05-18 08:29:15 -04:00
await self.stream_create('good_content', '0.1', channel_name='@some_channel', tag=['good'])
2020-01-18 20:58:30 -05:00
bad_content_id = self.get_claim_id(
2020-05-18 08:29:15 -04:00
await self.stream_create('bad_content', '0.1', channel_name='@some_channel', tag=['bad'])
2020-01-18 20:58:30 -05:00
)
filtering_channel_id = self.get_claim_id(
await self.channel_create('@filtering', '0.1')
2020-01-18 20:58:30 -05:00
)
self.conductor.spv_node.server.db.sql.filtering_channel_hashes.add(
unhexlify(filtering_channel_id)[::-1]
2019-11-05 04:02:05 -03:00
)
self.assertEqual(0, len(self.conductor.spv_node.server.db.sql.filtered_streams))
await self.stream_repost(bad_content_id, 'filter1', '0.1', channel_name='@filtering')
self.assertEqual(1, len(self.conductor.spv_node.server.db.sql.filtered_streams))
2020-01-18 20:58:30 -05:00
# search for filtered content directly
2020-05-18 08:29:15 -04:00
result = await self.out(self.api.claim_search(name='bad_content'))
blocked = result['blocked']
2020-01-18 20:58:30 -05:00
self.assertEqual([], result['items'])
self.assertEqual(1, blocked['total'])
self.assertEqual(1, len(blocked['channels']))
self.assertEqual(1, blocked['channels'][0]['blocked'])
self.assertTrue(blocked['channels'][0]['channel']['short_url'].startswith('lbry://@filtering#'))
2020-01-18 20:58:30 -05:00
# search inside channel containing filtered content
2020-05-18 08:29:15 -04:00
result = await self.out(self.api.claim_search(channel='@some_channel'))
filtered = result['blocked']
2020-01-18 20:58:30 -05:00
self.assertEqual(1, len(result['items']))
self.assertEqual(1, filtered['total'])
self.assertEqual(1, len(filtered['channels']))
self.assertEqual(1, filtered['channels'][0]['blocked'])
self.assertTrue(filtered['channels'][0]['channel']['short_url'].startswith('lbry://@filtering#'))
2020-01-18 20:58:30 -05:00
# content was filtered by not_tag before censoring
2020-05-18 08:29:15 -04:00
result = await self.out(self.api.claim_search(channel='@some_channel', not_tags=["good", "bad"]))
2020-01-18 20:58:30 -05:00
self.assertEqual(0, len(result['items']))
self.assertEqual({"channels": [], "total": 0}, result['blocked'])
2019-11-05 04:02:05 -03:00
# filtered content can still be resolved
result = await self.resolve('lbry://@some_channel/bad_content')
self.assertEqual(bad_content_id, result['claim_id'])
blocking_channel_id = self.get_claim_id(
await self.channel_create('@blocking', '0.1')
)
self.conductor.spv_node.server.db.sql.blocking_channel_hashes.add(
unhexlify(blocking_channel_id)[::-1]
)
self.assertEqual(0, len(self.conductor.spv_node.server.db.sql.blocked_streams))
await self.stream_repost(bad_content_id, 'block1', '0.1', channel_name='@blocking')
self.assertEqual(1, len(self.conductor.spv_node.server.db.sql.blocked_streams))
# blocked content is not resolveable
error = (await self.resolve('lbry://@some_channel/bad_content'))['error']
self.assertEqual(error['name'], 'BLOCKED')
self.assertTrue(error['text'].startswith("Resolve of 'lbry://@some_channel/bad_content' was censored"))
self.assertTrue(error['censor']['short_url'].startswith('lbry://@blocking#'))
# a filtered/blocked channel impacts all content inside it
bad_channel_id = self.get_claim_id(
2020-05-18 08:29:15 -04:00
await self.channel_create('@bad_channel', '0.1', tag=['bad-stuff'])
)
worse_content_id = self.get_claim_id(
2020-05-18 08:29:15 -04:00
await self.stream_create('worse_content', '0.1', channel_name='@bad_channel', tag=['bad-stuff'])
)
# check search before filtering channel
2020-05-18 08:29:15 -04:00
result = await self.out(self.api.claim_search(any_tags=['bad-stuff'], order_by=['height']))
self.assertEqual(2, result['total_items'])
self.assertEqual('worse_content', result['items'][0]['name'])
self.assertEqual('@bad_channel', result['items'][1]['name'])
# filter channel out
self.assertEqual(0, len(self.conductor.spv_node.server.db.sql.filtered_channels))
await self.stream_repost(bad_channel_id, 'filter2', '0.1', channel_name='@filtering')
self.assertEqual(1, len(self.conductor.spv_node.server.db.sql.filtered_channels))
# same claim search as previous now returns 0 results
2020-05-18 08:29:15 -04:00
result = await self.out(self.api.claim_search(any_tags=['bad-stuff'], order_by=['height']))
filtered = result['blocked']
self.assertEqual(0, len(result['items']))
self.assertEqual(3, filtered['total'])
self.assertEqual(1, len(filtered['channels']))
self.assertEqual(3, filtered['channels'][0]['blocked'])
self.assertTrue(filtered['channels'][0]['channel']['short_url'].startswith('lbry://@filtering#'))
# filtered channel should still resolve
result = await self.resolve('lbry://@bad_channel')
self.assertEqual(bad_channel_id, result['claim_id'])
result = await self.resolve('lbry://@bad_channel/worse_content')
self.assertEqual(worse_content_id, result['claim_id'])
# block channel
self.assertEqual(0, len(self.conductor.spv_node.server.db.sql.blocked_channels))
await self.stream_repost(bad_channel_id, 'block2', '0.1', channel_name='@blocking')
self.assertEqual(1, len(self.conductor.spv_node.server.db.sql.blocked_channels))
# channel, claim in channel or claim individually no longer resolve
self.assertEqual((await self.resolve('lbry://@bad_channel'))['error']['name'], 'BLOCKED')
self.assertEqual((await self.resolve('lbry://worse_content'))['error']['name'], 'BLOCKED')
self.assertEqual((await self.resolve('lbry://@bad_channel/worse_content'))['error']['name'], 'BLOCKED')
async def test_publish_updates_file_list(self):
tx = await self.stream_create(title='created')
txo = tx['outputs'][0]
claim_id, expected = txo['claim_id'], txo['value']
files = await self.file_list()
self.assertEqual(1, len(files))
self.assertEqual(tx['txid'], files[0]['txid'])
self.assertEqual(expected, files[0]['metadata'])
# update with metadata-only changes
tx = await self.stream_update(claim_id, title='update 1')
files = await self.file_list()
expected['title'] = 'update 1'
self.assertEqual(1, len(files))
self.assertEqual(tx['txid'], files[0]['txid'])
self.assertEqual(expected, files[0]['metadata'])
# update with new data
tx = await self.stream_update(claim_id, title='update 2', data=b'updated data')
expected = tx['outputs'][0]['value']
files = await self.file_list()
self.assertEqual(1, len(files))
self.assertEqual(tx['txid'], files[0]['txid'])
self.assertEqual(expected, files[0]['metadata'])
async def test_setting_stream_fields(self):
2019-03-24 16:55:04 -04:00
values = {
2019-03-25 09:59:32 -04:00
'title': "Cool Content",
'description': "Best content on LBRY.",
2019-03-25 12:30:30 -04:00
'thumbnail_url': "https://co.ol/thumbnail.png",
2020-05-18 08:29:15 -04:00
'tag': ["cool", "awesome"],
'language': ["en"],
'location': ['US:NH:Manchester:03101:42.990605:-71.460989'],
2019-03-25 12:30:30 -04:00
'author': "Jules Verne",
2019-03-25 09:59:32 -04:00
'license': 'Public Domain',
2019-03-25 12:30:30 -04:00
'license_url': "https://co.ol/license",
'release_time': 123456,
2019-03-25 09:59:32 -04:00
'fee_currency': 'usd',
'fee_amount': '2.99',
'fee_address': 'mmCsWAiXMUVecFQ3fVzUwvpT9XFMXno2Ca',
2019-03-24 16:55:04 -04:00
}
fixed_values = values.copy()
fixed_values['locations'] = [{
'country': 'US',
'state': 'NH',
'city': 'Manchester',
'code': '03101',
'latitude': '42.990605',
'longitude': '-71.460989'
}]
2019-04-20 01:12:43 -04:00
fixed_values['thumbnail'] = {'url': fixed_values.pop('thumbnail_url')}
2019-03-25 09:59:32 -04:00
fixed_values['release_time'] = str(values['release_time'])
2019-06-04 14:39:05 -04:00
fixed_values['stream_type'] = 'binary'
2019-04-20 01:12:43 -04:00
fixed_values['source'] = {
'hash': '56bf5dbae43f77a63d075b0f2ae9c7c3e3098db93779c7f9840da0f4db9c2f8c8454f4edd1373e2b64ee2e68350d916e',
2019-04-20 01:12:43 -04:00
'media_type': 'application/octet-stream',
'size': '3'
}
2019-03-25 09:59:32 -04:00
fixed_values['fee'] = {
'address': fixed_values.pop('fee_address'),
'amount': fixed_values.pop('fee_amount'),
2019-03-25 09:59:32 -04:00
'currency': fixed_values.pop('fee_currency').upper()
}
2019-04-20 01:12:43 -04:00
# create new stream with all fields set
2019-04-20 01:12:43 -04:00
tx = await self.out(self.stream_create('big', **values))
stream = tx['outputs'][0]['value']
fixed_values['source']['name'] = stream['source']['name']
2019-04-20 01:12:43 -04:00
fixed_values['source']['sd_hash'] = stream['source']['sd_hash']
2019-03-25 09:59:32 -04:00
self.assertEqual(stream, fixed_values)
2019-03-24 16:55:04 -04:00
# create stream with nothing set
2019-03-25 22:06:36 -04:00
tx = await self.out(self.stream_create('light'))
stream = tx['outputs'][0]['value']
2019-03-24 16:55:04 -04:00
self.assertEqual(
stream, {
2019-06-04 14:39:05 -04:00
'stream_type': 'binary',
2019-04-20 01:12:43 -04:00
'source': {
'size': '3',
'media_type': 'application/octet-stream',
'name': stream['source']['name'],
'hash': '56bf5dbae43f77a63d075b0f2ae9c7c3e3098db93779c7f9840da0f4db9c2f8c8454f4edd1373e2b64ee2e68350d916e',
'sd_hash': stream['source']['sd_hash']
2019-04-20 01:12:43 -04:00
},
2019-03-25 09:59:32 -04:00
}
2019-03-24 16:55:04 -04:00
)
# create stream with just some tags, langs and locations
2020-05-18 08:29:15 -04:00
tx = await self.out(self.stream_create('updated', tag='blah', language='uk', location='UA::Kyiv'))
2019-03-24 16:55:04 -04:00
txo = tx['outputs'][0]
claim_id, stream = txo['claim_id'], txo['value']
fixed_values['source']['name'] = stream['source']['name']
fixed_values['source']['sd_hash'] = stream['source']['sd_hash']
2019-03-24 16:55:04 -04:00
self.assertEqual(
stream, {
2019-06-04 14:39:05 -04:00
'stream_type': 'binary',
2019-04-20 01:12:43 -04:00
'source': {
'size': '3',
'media_type': 'application/octet-stream',
'name': fixed_values['source']['name'],
'hash': '56bf5dbae43f77a63d075b0f2ae9c7c3e3098db93779c7f9840da0f4db9c2f8c8454f4edd1373e2b64ee2e68350d916e',
2019-04-20 01:12:43 -04:00
'sd_hash': fixed_values['source']['sd_hash'],
},
'tags': ['blah'],
'languages': ['uk'],
'locations': [{'country': 'UA', 'city': 'Kyiv'}]
2019-03-25 09:59:32 -04:00
}
2019-03-24 16:55:04 -04:00
)
# update stream setting all fields, 'source' doesn't change
2019-03-25 22:06:36 -04:00
tx = await self.out(self.stream_update(claim_id, **values))
stream = tx['outputs'][0]['value']
2019-03-25 09:59:32 -04:00
fixed_values['tags'].insert(0, 'blah') # existing tag
fixed_values['languages'].insert(0, 'uk') # existing language
fixed_values['locations'].insert(0, {'country': 'UA', 'city': 'Kyiv'}) # existing location
self.assertEqual(stream, fixed_values)
2019-03-24 16:55:04 -04:00
# clearing and settings tags, languages and locations
tx = await self.out(self.stream_update(
claim_id, tags='single', clear_tags=True,
languages='pt', clear_languages=True,
locations='BR', clear_locations=True,
))
2019-03-24 16:55:04 -04:00
txo = tx['outputs'][0]
2019-03-25 09:59:32 -04:00
fixed_values['tags'] = ['single']
fixed_values['languages'] = ['pt']
fixed_values['locations'] = [{'country': 'BR'}]
2019-04-20 01:12:43 -04:00
self.assertEqual(txo['value'], fixed_values)
2019-03-24 16:55:04 -04:00
# modifying hash/size/name
fixed_values['source']['name'] = 'changed_name'
fixed_values['source']['hash'] = 'cafebeef'
fixed_values['source']['size'] = '42'
tx = await self.out(self.stream_update(
claim_id, file_name='changed_name', file_hash='cafebeef', file_size=42
))
self.assertEqual(tx['outputs'][0]['value'], fixed_values)
2019-05-08 23:35:25 -04:00
# stream_update re-signs with the same channel
channel_id = self.get_claim_id(await self.channel_create('@chan'))
2019-05-08 23:35:25 -04:00
tx = await self.stream_update(claim_id, channel_id=channel_id)
self.assertEqual(tx['outputs'][0]['signing_channel']['name'], '@chan')
tx = await self.stream_update(claim_id, title='channel re-signs')
self.assertEqual(tx['outputs'][0]['value']['title'], 'channel re-signs')
self.assertEqual(tx['outputs'][0]['signing_channel']['name'], '@chan')
2019-03-25 09:59:32 -04:00
# send claim to someone else
2020-05-18 08:29:15 -04:00
account2 = await self.api.account_create('second account')
2019-03-24 16:55:04 -04:00
# before sending
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 4)
self.assertEqual(len(await self.claim_list(account_id=self.account.id)), 4)
self.assertEqual(len(await self.claim_list(account_id=account2.id)), 0)
2019-03-24 16:55:04 -04:00
other_address = await account2.receiving.get_or_create_usable_address()
2020-05-18 08:29:15 -04:00
await self.stream_update(claim_id, claim_address=other_address)
2019-03-24 16:55:04 -04:00
# after sending
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.claim_list()), 4)
self.assertEqual(len(await self.claim_list(account_id=self.account.id)), 3)
self.assertEqual(len(await self.claim_list(account_id=account2.id)), 1)
2019-03-24 16:55:04 -04:00
2019-06-19 05:33:47 -04:00
async def test_setting_fee_fields(self):
tx = await self.out(self.stream_create('paid-stream'))
txo = tx['outputs'][0]
claim_id, stream = txo['claim_id'], txo['value']
fee_address = 'mmCsWAiXMUVecFQ3fVzUwvpT9XFMXno2Ca'
self.assertNotIn('fee', stream)
# --replace=false
# validation
with self.assertRaisesRegex(Exception, 'please specify a fee currency'):
await self.stream_update(claim_id, fee_amount='0.1')
with self.assertRaisesRegex(Exception, 'unknown currency provided: foo'):
await self.stream_update(claim_id, fee_amount='0.1', fee_currency="foo")
with self.assertRaisesRegex(Exception, 'please specify a fee amount'):
await self.stream_update(claim_id, fee_currency='usd')
with self.assertRaisesRegex(Exception, 'please specify a fee amount'):
await self.stream_update(claim_id, fee_address=fee_address)
# set just amount and currency with default address
tx = await self.stream_update(
claim_id, fee_amount='0.99', fee_currency='lbc'
)
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.99', 'currency': 'LBC', 'address': txo['address']}
)
# set all fee fields
tx = await self.stream_update(
claim_id, fee_amount='0.1', fee_currency='usd', fee_address=fee_address
)
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.1', 'currency': 'USD', 'address': fee_address}
)
# change just address
tx = await self.stream_update(claim_id, fee_address=txo['address'])
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.1', 'currency': 'USD', 'address': txo['address']}
)
# change just amount (does not reset fee_address)
tx = await self.stream_update(claim_id, fee_amount='0.2')
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.2', 'currency': 'USD', 'address': txo['address']}
)
# changing currency without an amount is never allowed, even if previous amount exists
with self.assertRaises(Exception, msg='In order to set a fee currency, please specify a fee amount'):
await self.stream_update(claim_id, fee_currency='usd')
# clearing fee
tx = await self.out(self.stream_update(claim_id, clear_fee=True))
self.assertNotIn('fee', tx['outputs'][0]['value'])
# --replace=true
# set just amount and currency with default address
tx = await self.stream_update(
claim_id, fee_amount='0.99', fee_currency='lbc', replace=True
)
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.99', 'currency': 'LBC', 'address': txo['address']}
)
# set all fee fields
tx = await self.stream_update(
claim_id, fee_amount='0.1', fee_currency='usd', fee_address=fee_address, replace=True
)
self.assertEqual(
tx['outputs'][0]['value']['fee'],
{'amount': '0.1', 'currency': 'USD', 'address': fee_address}
)
# validation
with self.assertRaisesRegex(Exception, 'please specify a fee currency'):
await self.stream_update(claim_id, fee_amount='0.1', replace=True)
with self.assertRaisesRegex(Exception, 'unknown currency provided: foo'):
await self.stream_update(claim_id, fee_amount='0.1', fee_currency="foo", replace=True)
with self.assertRaisesRegex(Exception, 'please specify a fee amount'):
await self.stream_update(claim_id, fee_currency='usd', replace=True)
with self.assertRaisesRegex(Exception, 'please specify a fee amount'):
await self.stream_update(claim_id, fee_address=fee_address, replace=True)
async def test_automatic_type_and_metadata_detection_for_image(self):
txo = (await self.stream_create('blank-image', data=self.image_data, suffix='.png'))['outputs'][0]
self.assertEqual(
txo['value'], {
'source': {
'size': '99',
'name': txo['value']['source']['name'],
'media_type': 'image/png',
'hash': '6c7df435d412c603390f593ef658c199817c7830ba3f16b7eadd8f99fa50e85dbd0d2b3dc61eadc33fe096e3872d1545',
'sd_hash': txo['value']['source']['sd_hash'],
},
'stream_type': 'image',
'image': {
'width': 5,
'height': 7
}
}
)
async def test_automatic_type_and_metadata_detection_for_video(self):
txo = (await self.stream_create('chrome', file_path=self.video_file_name))['outputs'][0]
self.assertEqual(
txo['value'], {
'source': {
'size': '2299653',
'name': 'ForBiggerEscapes.mp4',
'media_type': 'video/mp4',
'hash': '5f6811c83c1616df06f10bf5309ca61edb5ff949a9c1212ce784602d837bfdfc1c3db1e0580ef7bd1dadde41d8acf315',
'sd_hash': txo['value']['source']['sd_hash'],
},
'stream_type': 'video',
'video': {
'width': 1280,
'height': 720,
'duration': 15
}
}
)
async def test_overriding_automatic_metadata_detection(self):
2020-05-18 08:29:15 -04:00
tx = await self.stream_create(
'chrome', '1.0', file_path=self.video_file_name, width=99, height=88, duration=9
)
txo = tx['outputs'][0]
self.assertEqual(
txo['value'], {
'source': {
'size': '2299653',
'name': 'ForBiggerEscapes.mp4',
'media_type': 'video/mp4',
'hash': '5f6811c83c1616df06f10bf5309ca61edb5ff949a9c1212ce784602d837bfdfc1c3db1e0580ef7bd1dadde41d8acf315',
'sd_hash': txo['value']['source']['sd_hash'],
},
'stream_type': 'video',
'video': {
'width': 99,
'height': 88,
'duration': 9
}
}
)
2019-10-04 11:29:35 +02:00
async def test_update_file_type(self):
video_txo = (await self.stream_create('chrome', file_path=self.video_file_name))['outputs'][0]
self.assertSetEqual(set(video_txo['value']), {'source', 'stream_type', 'video'})
2019-10-04 11:29:35 +02:00
self.assertEqual(video_txo['value']['stream_type'], 'video')
self.assertEqual(video_txo['value']['source']['media_type'], 'video/mp4')
self.assertEqual(
video_txo['value']['video'], {
'duration': 15,
'height': 720,
'width': 1280
}
)
claim_id = video_txo['claim_id']
2019-10-04 11:29:35 +02:00
binary_txo = (await self.stream_update(claim_id, data=b'hi!'))['outputs'][0]
self.assertEqual(binary_txo['value']['stream_type'], 'binary')
self.assertEqual(binary_txo['value']['source']['media_type'], 'application/octet-stream')
self.assertSetEqual(set(binary_txo['value']), {'source', 'stream_type'})
image_txo = (await self.stream_update(claim_id, data=self.image_data, suffix='.png'))['outputs'][0]
self.assertSetEqual(set(image_txo['value']), {'source', 'stream_type', 'image'})
2019-10-04 11:29:35 +02:00
self.assertEqual(image_txo['value']['stream_type'], 'image')
self.assertEqual(image_txo['value']['source']['media_type'], 'image/png')
self.assertEqual(image_txo['value']['image'], {'height': 7, 'width': 5})
2019-10-04 11:29:35 +02:00
async def test_replace_mode_preserves_source_and_type(self):
expected = {
'tags': ['blah'],
'languages': ['uk'],
'locations': [{'country': 'UA', 'city': 'Kyiv'}],
'source': {
'size': '2299653',
'name': 'ForBiggerEscapes.mp4',
'media_type': 'video/mp4',
'hash': '5f6811c83c1616df06f10bf5309ca61edb5ff949a9c1212ce784602d837bfdfc1c3db1e0580ef7bd1dadde41d8acf315',
},
'stream_type': 'video',
'video': {
'width': 1280,
'height': 720,
'duration': 15
}
}
channel = await self.channel_create('@chan')
2020-05-18 08:29:15 -04:00
tx = await self.out(self.stream_create(
'chrome', '1.0', file_path=self.video_file_name,
tags='blah', languages='uk', locations='UA::Kyiv',
channel_id=self.get_claim_id(channel)
))
await self.on_transaction_dict(tx)
txo = tx['outputs'][0]
expected['source']['sd_hash'] = txo['value']['source']['sd_hash']
self.assertEqual(txo['value'], expected)
self.assertEqual(txo['signing_channel']['name'], '@chan')
2020-05-18 08:29:15 -04:00
tx = await self.stream_update(
txo['claim_id'], title='new title', replace=True
2020-05-18 08:29:15 -04:00
)
txo = tx['outputs'][0]
expected['title'] = 'new title'
del expected['tags']
del expected['languages']
del expected['locations']
self.assertEqual(txo['value'], expected)
self.assertNotIn('signing_channel', txo)
async def test_create_update_and_abandon_stream(self):
2019-02-11 17:45:52 -05:00
await self.assertBalance(self.account, '10.0')
2019-03-25 22:06:36 -04:00
tx = await self.stream_create(bid='2.5') # creates new claim
claim_id = self.get_claim_id(tx)
2020-03-07 00:34:47 -05:00
txs = await self.transaction_list()
2019-03-24 16:55:04 -04:00
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.assertFalse(txs[0]['claim_info'][0]['is_spent'])
2019-03-24 16:55:04 -04:00
self.assertEqual(txs[0]['value'], '0.0')
self.assertEqual(txs[0]['fee'], '-0.020107')
await self.assertBalance(self.account, '7.479893')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 1)
2020-05-18 08:29:15 -04:00
await self.file_delete(delete_all=True)
self.assertEqual(len(await self.file_list()), 0)
2019-03-24 16:55:04 -04:00
2019-03-25 22:06:36 -04:00
await self.stream_update(claim_id, bid='1.0') # updates previous claim
2020-03-07 00:34:47 -05:00
txs = await self.transaction_list()
2019-03-24 16:55:04 -04:00
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.assertFalse(txs[0]['update_info'][0]['is_spent'])
self.assertTrue(txs[1]['claim_info'][0]['is_spent'])
2019-03-24 16:55:04 -04:00
self.assertEqual(txs[0]['value'], '0.0')
self.assertEqual(txs[0]['fee'], '-0.0002165')
await self.assertBalance(self.account, '8.9796765')
2019-03-24 16:55:04 -04:00
await self.stream_abandon(claim_id)
2020-03-07 00:34:47 -05:00
txs = await self.transaction_list()
2019-03-24 16:55:04 -04:00
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.assertTrue(txs[1]['update_info'][0]['is_spent'])
self.assertTrue(txs[2]['claim_info'][0]['is_spent'])
2019-03-24 16:55:04 -04:00
self.assertEqual(txs[0]['value'], '0.0')
self.assertEqual(txs[0]['fee'], '-0.000107')
await self.assertBalance(self.account, '9.9795695')
2019-02-11 17:45:52 -05:00
async def test_abandoning_stream_at_loss(self):
2019-02-11 17:45:52 -05:00
await self.assertBalance(self.account, '10.0')
2019-03-25 22:06:36 -04:00
tx = await self.stream_create(bid='0.0001')
2019-02-11 17:45:52 -05:00
await self.assertBalance(self.account, '9.979793')
2020-11-16 15:26:50 -05:00
await self.stream_abandon(claim_id=self.get_claim_id(tx))
2019-02-11 17:45:52 -05:00
await self.assertBalance(self.account, '9.97968399')
2019-03-27 16:02:17 -04:00
async def test_publish(self):
# errors on missing arguments to create a stream
with self.assertRaisesRegex(Exception, "'bid' is a required argument for new publishes."):
2020-05-18 08:29:15 -04:00
await self.publish('foo')
2019-03-27 16:02:17 -04:00
with self.assertRaisesRegex(Exception, "'file_path' is a required argument for new publishes."):
2020-05-18 08:29:15 -04:00
await self.publish('foo', bid='1.0')
2019-03-27 16:02:17 -04:00
# successfully create stream
with tempfile.NamedTemporaryFile() as file:
file.write(b'hi')
file.flush()
tx1 = await self.publish('foo', bid='1.0', file_path=file.name)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 1)
2019-04-16 09:27:19 -04:00
2019-03-27 16:02:17 -04:00
# doesn't error on missing arguments when doing an update stream
tx2 = await self.publish('foo', tags='updated')
2019-04-16 09:27:19 -04:00
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 1)
self.assertEqual(self.get_claim_id(tx1), self.get_claim_id(tx2))
2019-03-27 16:02:17 -04:00
# update conflict with two claims of the same name
tx3 = await self.stream_create('foo', allow_duplicate_name=True)
with self.assertRaisesRegex(Exception, "There are 2 claims for 'foo'"):
2020-05-18 08:29:15 -04:00
await self.publish('foo')
2019-03-27 16:02:17 -04:00
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 2)
# abandon duplicate stream
await self.stream_abandon(self.get_claim_id(tx3))
2019-03-27 16:02:17 -04:00
# publish to a channel
await self.channel_create('@abc')
tx3 = await self.publish('foo', channel_name='@abc')
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 2)
2019-03-27 16:02:17 -04:00
r = await self.resolve('lbry://@abc/foo')
self.assertEqual(
r['claim_id'],
self.get_claim_id(tx3)
2019-03-27 16:02:17 -04:00
)
2019-05-08 23:35:25 -04:00
# publishing again clears channel
2019-09-25 12:28:02 +02:00
tx4 = await self.publish('foo', languages='uk-UA', tags=['Anime', 'anime '])
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.file_list()), 2)
claim = await self.resolve('lbry://foo')
self.assertEqual(claim['txid'], tx4['outputs'][0]['txid'])
2019-04-29 00:38:58 -04:00
self.assertNotIn('signing_channel', claim)
2019-04-20 01:12:43 -04:00
self.assertEqual(claim['value']['languages'], ['uk-UA'])
2019-09-25 12:28:02 +02:00
self.assertEqual(claim['value']['tags'], ['anime'])
2019-03-25 22:06:36 -04:00
class SupportCommands(CommandTestCase):
async def test_regular_supports_and_tip_supports(self):
2020-05-18 08:29:15 -04:00
wallet2 = await self.api.wallet_create('wallet2', create_account=True)
account2 = wallet2.accounts[0]
2019-03-25 22:06:36 -04:00
# send account2 5 LBC out of the 10 LBC in account1
2020-05-18 08:29:15 -04:00
await self.account_send('5.0', await self.api.address_unused(wallet_id='wallet2'))
2019-03-25 22:06:36 -04:00
# 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
claim_id = self.get_claim_id(await self.stream_create())
2019-03-25 22:06:36 -04:00
# 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
2020-05-18 08:29:15 -04:00
tip = await self.support_create(
claim_id, '1.0', tip=True, account_id=account2.id, wallet_id='wallet2',
funding_account_ids=[account2.id]
2019-03-25 22:06:36 -04:00
)
# 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
2020-03-07 00:34:47 -05:00
txs = await self.transaction_list(account_id=self.account.id)
2019-03-25 22:06:36 -04:00
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)
2019-10-02 19:58:51 +03:00
self.assertTrue(txs[0]['support_info'][0]['is_tip'])
self.assertFalse(txs[0]['support_info'][0]['is_spent'])
2019-03-25 22:06:36 -04:00
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
2020-03-07 00:34:47 -05:00
txs2 = await self.transaction_list(wallet_id='wallet2', account_id=account2.id)
2019-03-25 22:06:36 -04:00
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)
2019-10-02 19:58:51 +03:00
self.assertTrue(txs2[0]['support_info'][0]['is_tip'])
self.assertFalse(txs2[0]['support_info'][0]['is_spent'])
2019-03-25 22:06:36 -04:00
self.assertEqual(txs2[0]['value'], '-1.0')
self.assertEqual(txs2[0]['fee'], '-0.0001415')
# send a support to the claim using account2
2020-05-18 08:29:15 -04:00
support = self.support_create(
claim_id, '2.0', tip=False, account_id=account2.id, wallet_id='wallet2',
funding_account_ids=[account2.id]
2019-03-25 22:06:36 -04:00
)
# 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
2020-03-07 00:34:47 -05:00
txs2 = await self.transaction_list(wallet_id='wallet2')
2019-03-25 22:06:36 -04:00
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)
2019-10-02 19:58:51 +03:00
self.assertFalse(txs2[0]['support_info'][0]['is_tip'])
self.assertFalse(txs2[0]['support_info'][0]['is_spent'])
2019-03-25 22:06:36 -04:00
self.assertEqual(txs2[0]['value'], '0.0')
self.assertEqual(txs2[0]['fee'], '-0.0001415')
2019-11-13 17:50:35 -05:00
# abandoning the tip increases balance and shows tip as spent
await self.support_abandon(claim_id)
await self.assertBalance(self.account, '4.979662')
2020-03-07 00:34:47 -05:00
txs = await self.transaction_list(account_id=self.account.id)
self.assertEqual(len(txs[0]['abandon_info']), 1)
self.assertEqual(len(txs[1]['support_info']), 1)
self.assertTrue(txs[1]['support_info'][0]['is_tip'])
self.assertTrue(txs[1]['support_info'][0]['is_spent'])
2019-11-13 17:50:35 -05:00
class CollectionCommands(CommandTestCase):
async def test_collections(self):
claim_ids = [
self.get_claim_id(tx) for tx in [
await self.stream_create('stream-one'),
await self.stream_create('stream-two')
]
]
claim_ids.append(claim_ids[0])
claim_ids.append('beef')
tx = await self.collection_create('radjingles', claims=claim_ids, title="boring title")
claim_id = self.get_claim_id(tx)
2020-05-18 08:29:15 -04:00
collections = await self.collection_list()
self.assertEqual(collections[0]['value']['title'], 'boring title')
self.assertEqual(collections[0]['value']['claims'], claim_ids)
self.assertEqual(collections[0]['value_type'], 'collection')
self.assertEqual(len(collections), 1)
2019-11-13 17:50:35 -05:00
await self.assertBalance(self.account, '6.939679')
with self.assertRaisesRegex(Exception, "You already have a collection under the name 'radjingles'."):
await self.collection_create('radjingles', claims=claim_ids)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.collection_list()), 1)
2019-11-13 17:50:35 -05:00
await self.assertBalance(self.account, '6.939679')
2020-05-18 08:29:15 -04:00
self.assertEqual((await self.collection_list())[0]['value']['title'], 'boring title')
2019-11-13 17:50:35 -05:00
await self.collection_update(claim_id, title='fancy title')
2020-05-18 08:29:15 -04:00
collections = await self.collection_list()
self.assertEqual(collections[0]['value']['title'], 'fancy title')
self.assertEqual(collections[0]['value']['claims'], claim_ids)
self.assertNotIn('claims', collections[0])
2019-11-13 17:50:35 -05:00
2019-11-13 21:16:27 -05:00
tx = await self.collection_create('radjingles', claims=claim_ids, allow_duplicate_name=True)
claim_id2 = self.get_claim_id(tx)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.collection_list()), 2)
2019-11-13 17:50:35 -05:00
2019-11-14 13:25:33 -05:00
await self.collection_update(claim_id, clear_claims=True, claims=claim_ids[:2])
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.collection_list()), 2)
2019-11-14 13:25:33 -05:00
2019-11-13 17:50:35 -05:00
await self.collection_abandon(claim_id)
2020-05-18 08:29:15 -04:00
self.assertEqual(len(await self.collection_list()), 1)
collections = await self.collection_list(resolve_claims=2)
self.assertEqual(len(collections[0]['claims']), 2)
collections = await self.collection_list(resolve_claims=10)
self.assertEqual(len(collections[0]['claims']), 4)
self.assertEqual(collections[0]['claims'][0]['name'], 'stream-one')
self.assertEqual(collections[0]['claims'][1]['name'], 'stream-two')
self.assertEqual(collections[0]['claims'][2]['name'], 'stream-one')
self.assertIsNone(collections[0]['claims'][3])
claims = await self.claim_list()
self.assertEqual(claims[0]['name'], 'radjingles')
self.assertEqual(claims[1]['name'], 'stream-two')
self.assertEqual(claims[2]['name'], 'stream-one')
claims = await self.collection_resolve(claim_id2)
self.assertEqual(claims[0]['name'], 'stream-one')
self.assertEqual(claims[1]['name'], 'stream-two')
self.assertEqual(claims[2]['name'], 'stream-one')
self.assertEqual(await self.collection_resolve(claim_id2, page=10), [])