Improve tests, update suggested file name when necessary

This commit is contained in:
Miroslav Kovar 2019-10-11 17:54:12 +02:00 committed by Lex Berezhny
parent 15376c19be
commit 869be3e361
4 changed files with 93 additions and 55 deletions

View file

@ -2810,7 +2810,7 @@ class Daemon(metaclass=JSONRPCServerType):
@requires(WALLET_COMPONENT, STREAM_MANAGER_COMPONENT, BLOB_COMPONENT, DATABASE_COMPONENT) @requires(WALLET_COMPONENT, STREAM_MANAGER_COMPONENT, BLOB_COMPONENT, DATABASE_COMPONENT)
async def jsonrpc_stream_update( async def jsonrpc_stream_update(
self, claim_id, bid=None, file_path=None, self, claim_id, bid=None, file_path=None, file_name=None,
channel_id=None, channel_name=None, channel_account_id=None, clear_channel=False, channel_id=None, channel_name=None, channel_account_id=None, clear_channel=False,
account_id=None, wallet_id=None, claim_address=None, funding_account_ids=None, account_id=None, wallet_id=None, claim_address=None, funding_account_ids=None,
preview=False, blocking=False, replace=False, **kwargs): preview=False, blocking=False, replace=False, **kwargs):
@ -2979,7 +2979,7 @@ class Daemon(metaclass=JSONRPCServerType):
claim.stream.update(file_path=file_path, **kwargs) claim.stream.update(file_path=file_path, **kwargs)
else: else:
claim = Claim.from_bytes(old_txo.claim.to_bytes()) claim = Claim.from_bytes(old_txo.claim.to_bytes())
claim.stream.update(file_path=file_path, **kwargs) claim.stream.update(file_path=file_path, file_name=file_name, **kwargs)
tx = await Transaction.claim_update( tx = await Transaction.claim_update(
old_txo, claim, amount, claim_address, funding_accounts, funding_accounts[0], channel old_txo, claim, amount, claim_address, funding_accounts, funding_accounts[0], channel
) )
@ -2988,10 +2988,13 @@ class Daemon(metaclass=JSONRPCServerType):
stream_hash = None stream_hash = None
if not preview: if not preview:
old_stream_hash = await self.storage.get_stream_hash_for_sd_hash(old_txo.claim.stream.source.sd_hash) old_stream_hash = await self.storage.get_stream_hash_for_sd_hash(old_txo.claim.stream.source.sd_hash)
if file_path is not None: old_stream = self.stream_manager.get_stream_by_stream_hash(old_stream_hash)
if old_stream_hash: if not file_path and file_name and old_stream:
stream_to_delete = self.stream_manager.get_stream_by_stream_hash(old_stream_hash) old_path, _ = os.path.split(old_stream.full_path)
await self.stream_manager.delete_stream(stream_to_delete, delete_file=False) file_path = os.path.join(old_path, file_name)
if file_path:
if old_stream:
await self.stream_manager.delete_stream(old_stream, delete_file=False)
file_stream = await self.stream_manager.create_stream(file_path) file_stream = await self.stream_manager.create_stream(file_path)
new_txo.claim.stream.source.sd_hash = file_stream.sd_hash new_txo.claim.stream.source.sd_hash = file_stream.sd_hash
new_txo.script.generate() new_txo.script.generate()

View file

@ -15,6 +15,17 @@ from lbry.error import InvalidStreamDescriptorError
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
_exprs = [
'[<>:"/\\\|\?\*]+', # Illegal characters
'[\\x00-\\x1F]+', # All characters in range 0-31
'[ \t]*(\.)+[ \t]*$', # Dots at the end
'(^[ \t]+|[ \t]+$)', # Leading and trailing whitespace
'^CON$', '^PRN$', '^AUX$', # Illegal names
'^NUL$', '^COM[1-9]$', '^LPT[1-9]$'
]
RES = re.compile('(' + '|'.join((expr for expr in _exprs)) + ')')
def format_sd_info(stream_name: str, key: str, suggested_file_name: str, stream_hash: str, def format_sd_info(stream_name: str, key: str, suggested_file_name: str, stream_hash: str,
blobs: typing.List[typing.Dict]) -> typing.Dict: blobs: typing.List[typing.Dict]) -> typing.Dict:
@ -48,16 +59,6 @@ def file_reader(file_path: str):
def sanitize_file_name(dirty_name: str): def sanitize_file_name(dirty_name: str):
exprs = [
'[<>:"/\\\|\?\*]+', # Illegal characters
'[\\x00-\\x1F]+', # All characters in range 0-31
'[ \t]*(\.)+[ \t]*$', # Dots at the end
'(^[ \t]+|[ \t]+$)', # Leading and trailing whitespace
'^CON$', '^PRN$', '^AUX$', # Illegal names
'^NUL$', '^COM[1-9]$', '^LPT[1-9]$']
RES = re.compile('(' + '|'.join((expr for expr in exprs)) + ')')
file_name, ext = os.path.splitext(dirty_name) file_name, ext = os.path.splitext(dirty_name)
file_name = re.sub(RES, '', file_name) file_name = re.sub(RES, '', file_name)
ext = re.sub(RES, '', ext) ext = re.sub(RES, '', ext)

View file

@ -197,9 +197,11 @@ class CommandTestCase(IntegrationTestCase):
""" Synchronous version of `out` method. """ """ Synchronous version of `out` method. """
return json.loads(jsonrpc_dumps_pretty(value, ledger=self.ledger))['result'] return json.loads(jsonrpc_dumps_pretty(value, ledger=self.ledger))['result']
async def stream_create(self, name='hovercraft', bid='1.0', data=b'hi!', confirm=True, def create_tempfile(self, data=None, prefix=None, suffix=None, dir=None):
prefix=None, suffix=None, **kwargs): dir = dir or self.daemon.conf.download_dir
file = tempfile.NamedTemporaryFile(prefix=prefix, suffix=suffix) file = tempfile.NamedTemporaryFile(prefix=prefix, suffix=suffix, dir=dir)
file.write(data)
file.flush()
def cleanup(): def cleanup():
try: try:
@ -208,10 +210,13 @@ class CommandTestCase(IntegrationTestCase):
pass pass
self.addCleanup(cleanup) self.addCleanup(cleanup)
file.write(data) return file.name
file.flush()
async def stream_create(self, name='hovercraft', bid='1.0', data=b'hi!', confirm=True,
prefix=None, suffix=None, **kwargs):
file_path = self.create_tempfile(data=data, prefix=prefix, suffix=suffix)
claim = await self.out( claim = await self.out(
self.daemon.jsonrpc_stream_create(name, bid, file_path=file.name, **kwargs) self.daemon.jsonrpc_stream_create(name, bid, file_path=file_path, **kwargs)
) )
self.assertEqual(claim['outputs'][0]['name'], name) self.assertEqual(claim['outputs'][0]['name'], name)
if confirm: if confirm:
@ -222,19 +227,9 @@ class CommandTestCase(IntegrationTestCase):
async def stream_update(self, claim_id, data=None, confirm=True, **kwargs): async def stream_update(self, claim_id, data=None, confirm=True, **kwargs):
if data: if data:
file = tempfile.NamedTemporaryFile() file_path = self.create_tempfile(data)
file.write(data)
file.flush()
def cleanup():
try:
file.close()
except FileNotFoundError:
pass
self.addCleanup(cleanup)
claim = await self.out( claim = await self.out(
self.daemon.jsonrpc_stream_update(claim_id, file_path=file.name, **kwargs) self.daemon.jsonrpc_stream_update(claim_id, file_path=file_path, **kwargs)
) )
else: else:
claim = await self.out(self.daemon.jsonrpc_stream_update(claim_id, **kwargs)) claim = await self.out(self.daemon.jsonrpc_stream_update(claim_id, **kwargs))

View file

@ -45,40 +45,79 @@ class FileCommands(CommandTestCase):
{stream.sd_hash, stream.descriptor.blobs[0].blob_hash} {stream.sd_hash, stream.descriptor.blobs[0].blob_hash}
) )
async def _purge_file(self, claim_name, full_path):
self.assertTrue(
await self.daemon.jsonrpc_file_delete(claim_name=claim_name, delete_from_download_dir=True)
)
self.assertEqual(len(self.daemon.jsonrpc_file_list()), 0)
self.assertFalse(os.path.isfile(full_path))
async def test_publish_with_illegal_chars(self): async def test_publish_with_illegal_chars(self):
def check_prefix_suffix(name, prefix, suffix):
self.assertTrue(name.startswith(prefix))
self.assertTrue(name.endswith(suffix))
# Stream a file with file name containing invalid chars # Stream a file with file name containing invalid chars
prefix = '?an|t:z< m<' claim_name = 'lolwindows'
suffix = '.ext.' prefix, suffix = 'derp?', '.ext.'
san_prefix = 'antz m' san_prefix, san_suffix = 'derp', '.ext'
san_suffix = '.ext' tx = await self.stream_create(claim_name, '0.01', prefix=prefix, suffix=suffix)
tx = await self.stream_create('foo', '0.01', prefix=prefix, suffix=suffix)
stream = self.daemon.jsonrpc_file_list()[0] stream = self.daemon.jsonrpc_file_list()[0]
claim_id = self.get_claim_id(tx)
# Assert that file list and source contains the local unsanitized name, but suggested name is sanitized # Assert that file list and source contains the local unsanitized name, but suggested name is sanitized
full_path = (await self.daemon.jsonrpc_get('lbry://' + claim_name)).full_path
stream_file_name = os.path.basename(full_path)
source_file_name = tx['outputs'][0]['value']['source']['name'] source_file_name = tx['outputs'][0]['value']['source']['name']
file_list_name = stream.file_name file_list_name = stream.file_name
suggested_file_name = stream.descriptor.suggested_file_name suggested_file_name = stream.descriptor.suggested_file_name
self.assertTrue(source_file_name.startswith(prefix))
self.assertTrue(source_file_name.endswith(suffix))
self.assertEqual(file_list_name, source_file_name)
self.assertTrue(suggested_file_name.startswith(san_prefix))
self.assertTrue(suggested_file_name.endswith(san_suffix))
# Delete the file, re-download and assert that the file name is sanitized
self.assertTrue(await self.daemon.jsonrpc_file_delete(claim_name='foo'))
self.assertEqual(len(self.daemon.jsonrpc_file_list()), 0)
full_path = (await self.daemon.jsonrpc_get('lbry://foo', save_file=True)).full_path
file_name = os.path.basename(full_path)
self.assertTrue(os.path.isfile(full_path)) self.assertTrue(os.path.isfile(full_path))
self.assertTrue(file_name.startswith(san_prefix)) check_prefix_suffix(stream_file_name, prefix, suffix)
self.assertTrue(file_name.endswith(san_suffix)) self.assertEqual(stream_file_name, source_file_name)
self.assertEqual(stream_file_name, file_list_name)
check_prefix_suffix(suggested_file_name, san_prefix, san_suffix)
await self._purge_file(claim_name, full_path)
# Re-download deleted file and assert that the file name is sanitized
full_path = (await self.daemon.jsonrpc_get('lbry://' + claim_name, save_file=True)).full_path
stream_file_name = os.path.basename(full_path)
stream = self.daemon.jsonrpc_file_list()[0]
file_list_name = stream.file_name
suggested_file_name = stream.descriptor.suggested_file_name
self.assertTrue(os.path.isfile(full_path))
check_prefix_suffix(stream_file_name, san_prefix, san_suffix)
self.assertEqual(stream_file_name, file_list_name)
self.assertEqual(stream_file_name, suggested_file_name)
await self._purge_file(claim_name, full_path)
# Assert that the downloaded file name is not sanitized when user provides custom file name # Assert that the downloaded file name is not sanitized when user provides custom file name
self.assertTrue(await self.daemon.jsonrpc_file_delete(claim_name='foo')) custom_name = 'cust*m_name'
file_name = 'my <u?|*m.name' full_path = (await self.daemon.jsonrpc_get(
full_path = (await self.daemon.jsonrpc_get('lbry://foo', file_name=file_name, save_file=True)).full_path 'lbry://' + claim_name, file_name=custom_name, save_file=True)).full_path
file_name_on_disk = os.path.basename(full_path)
self.assertTrue(os.path.isfile(full_path)) self.assertTrue(os.path.isfile(full_path))
self.assertEqual(file_name, os.path.basename(full_path)) self.assertEqual(custom_name, file_name_on_disk)
# Update the stream and assert the file name is not sanitized, but the suggested file name is
prefix, suffix = 'derpyderp?', '.ext.'
san_prefix, san_suffix = 'derpyderp', '.ext'
new_file_name = os.path.basename(self.create_tempfile(data=b'amazing content', prefix=prefix, suffix=suffix))
tx = await self.stream_update(claim_id, file_name=new_file_name)
full_path = (await self.daemon.jsonrpc_get('lbry://' + claim_name, save_file=True)).full_path
updated_stream = self.daemon.jsonrpc_file_list()[0]
stream_file_name = os.path.basename(full_path)
source_file_name = tx['outputs'][0]['value']['source']['name']
file_list_name = updated_stream.file_name
suggested_file_name = updated_stream.descriptor.suggested_file_name
self.assertTrue(os.path.isfile(full_path))
check_prefix_suffix(stream_file_name, prefix, suffix)
self.assertEqual(stream_file_name, source_file_name)
self.assertEqual(stream_file_name, file_list_name)
check_prefix_suffix(suggested_file_name, san_prefix, san_suffix)
async def test_file_list_fields(self): async def test_file_list_fields(self):
await self.stream_create('foo', '0.01') await self.stream_create('foo', '0.01')