Improve tests, update suggested file name when necessary
This commit is contained in:
parent
15376c19be
commit
869be3e361
4 changed files with 93 additions and 55 deletions
|
@ -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()
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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))
|
||||||
|
|
|
@ -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')
|
||||||
|
|
Loading…
Add table
Reference in a new issue