2019-01-25 15:05:22 -05:00
|
|
|
import os
|
|
|
|
import asyncio
|
|
|
|
import tempfile
|
|
|
|
import shutil
|
2019-12-31 15:30:13 -05:00
|
|
|
from lbry.testcase import AsyncioTestCase
|
2019-06-20 21:02:58 -04:00
|
|
|
from lbry.conf import Config
|
|
|
|
from lbry.extras.daemon.storage import SQLiteStorage
|
|
|
|
from lbry.blob.blob_manager import BlobManager
|
|
|
|
from lbry.stream.stream_manager import StreamManager
|
|
|
|
from lbry.stream.reflector.server import ReflectorServer
|
2019-01-25 15:05:22 -05:00
|
|
|
|
|
|
|
|
2020-07-15 16:39:59 -04:00
|
|
|
class TestReflector(AsyncioTestCase):
|
2019-01-25 15:05:22 -05:00
|
|
|
async def asyncSetUp(self):
|
|
|
|
self.loop = asyncio.get_event_loop()
|
|
|
|
self.key = b'deadbeef' * 4
|
|
|
|
self.cleartext = os.urandom(20000000)
|
|
|
|
|
|
|
|
tmp_dir = tempfile.mkdtemp()
|
|
|
|
self.addCleanup(lambda: shutil.rmtree(tmp_dir))
|
2019-04-15 16:14:19 -04:00
|
|
|
self.conf = Config()
|
|
|
|
self.storage = SQLiteStorage(self.conf, os.path.join(tmp_dir, "lbrynet.sqlite"))
|
2019-01-25 15:05:22 -05:00
|
|
|
await self.storage.open()
|
2019-04-15 16:14:19 -04:00
|
|
|
self.blob_manager = BlobManager(self.loop, tmp_dir, self.storage, self.conf)
|
2020-07-15 16:39:59 -04:00
|
|
|
self.addCleanup(self.blob_manager.stop)
|
2019-01-30 15:23:17 -05:00
|
|
|
self.stream_manager = StreamManager(self.loop, Config(), self.blob_manager, None, self.storage, None)
|
2019-01-25 15:05:22 -05:00
|
|
|
|
|
|
|
server_tmp_dir = tempfile.mkdtemp()
|
|
|
|
self.addCleanup(lambda: shutil.rmtree(server_tmp_dir))
|
2019-04-15 16:14:19 -04:00
|
|
|
self.server_conf = Config()
|
|
|
|
self.server_storage = SQLiteStorage(self.server_conf, os.path.join(server_tmp_dir, "lbrynet.sqlite"))
|
2019-01-25 15:05:22 -05:00
|
|
|
await self.server_storage.open()
|
2019-04-15 16:14:19 -04:00
|
|
|
self.server_blob_manager = BlobManager(self.loop, server_tmp_dir, self.server_storage, self.server_conf)
|
2020-07-15 16:39:59 -04:00
|
|
|
self.addCleanup(self.server_blob_manager.stop)
|
2019-01-25 15:05:22 -05:00
|
|
|
|
|
|
|
download_dir = tempfile.mkdtemp()
|
|
|
|
self.addCleanup(lambda: shutil.rmtree(download_dir))
|
|
|
|
|
|
|
|
# create the stream
|
|
|
|
file_path = os.path.join(tmp_dir, "test_file")
|
|
|
|
with open(file_path, 'wb') as f:
|
|
|
|
f.write(self.cleartext)
|
2020-08-11 13:04:05 -04:00
|
|
|
self.stream_manager.config.reflect_streams = False
|
2020-01-29 13:49:14 -03:00
|
|
|
self.stream = await self.stream_manager.create(file_path)
|
2019-01-25 15:05:22 -05:00
|
|
|
|
2021-05-20 18:03:53 -03:00
|
|
|
async def _test_reflect_stream(self, response_chunk_size=50, partial_needs=False):
|
|
|
|
reflector = ReflectorServer(self.server_blob_manager, response_chunk_size=response_chunk_size,
|
|
|
|
partial_needs=partial_needs)
|
2019-01-25 15:05:22 -05:00
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
2021-05-20 18:03:53 -03:00
|
|
|
if partial_needs:
|
|
|
|
server_blob = self.server_blob_manager.get_blob(self.stream.sd_hash)
|
|
|
|
client_blob = self.blob_manager.get_blob(self.stream.sd_hash)
|
|
|
|
with client_blob.reader_context() as handle:
|
|
|
|
server_blob.set_length(client_blob.get_length())
|
|
|
|
writer = server_blob.get_blob_writer('nobody', 0)
|
|
|
|
writer.write(handle.read())
|
|
|
|
self.server_blob_manager.blob_completed(server_blob)
|
2019-01-25 15:05:22 -05:00
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
2020-04-10 10:56:45 -04:00
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
2019-01-25 15:05:22 -05:00
|
|
|
sent = await self.stream.upload_to_reflector('127.0.0.1', 5566)
|
2020-04-10 10:56:45 -04:00
|
|
|
self.assertEqual(100, self.stream.reflector_progress)
|
2021-05-20 18:49:20 -03:00
|
|
|
if partial_needs:
|
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|
|
|
|
send_more = await self.stream.upload_to_reflector('127.0.0.1', 5566)
|
2021-05-20 18:53:35 -03:00
|
|
|
self.assertGreater(len(send_more), 0)
|
2021-05-20 18:49:20 -03:00
|
|
|
sent.extend(send_more)
|
2021-05-20 18:53:35 -03:00
|
|
|
sent.append(self.stream.sd_hash)
|
2019-01-25 15:05:22 -05:00
|
|
|
self.assertSetEqual(
|
|
|
|
set(sent),
|
|
|
|
set(map(lambda b: b.blob_hash,
|
|
|
|
self.stream.descriptor.blobs[:-1] + [self.blob_manager.get_blob(self.stream.sd_hash)]))
|
|
|
|
)
|
2021-05-20 18:53:35 -03:00
|
|
|
send_more = await self.stream.upload_to_reflector('127.0.0.1', 5566)
|
|
|
|
self.assertEqual(len(send_more), 0)
|
2020-07-15 16:39:59 -04:00
|
|
|
self.assertTrue(self.stream.is_fully_reflected)
|
2019-01-25 15:05:22 -05:00
|
|
|
server_sd_blob = self.server_blob_manager.get_blob(self.stream.sd_hash)
|
|
|
|
self.assertTrue(server_sd_blob.get_is_verified())
|
|
|
|
self.assertEqual(server_sd_blob.length, server_sd_blob.length)
|
|
|
|
for blob in self.stream.descriptor.blobs[:-1]:
|
|
|
|
server_blob = self.server_blob_manager.get_blob(blob.blob_hash)
|
|
|
|
self.assertTrue(server_blob.get_is_verified())
|
|
|
|
self.assertEqual(server_blob.length, blob.length)
|
|
|
|
|
|
|
|
sent = await self.stream.upload_to_reflector('127.0.0.1', 5566)
|
|
|
|
self.assertListEqual(sent, [])
|
2019-05-06 16:49:03 -03:00
|
|
|
|
2019-06-07 13:23:20 -04:00
|
|
|
async def test_reflect_stream(self):
|
|
|
|
return await asyncio.wait_for(self._test_reflect_stream(response_chunk_size=50), 3, loop=self.loop)
|
|
|
|
|
2021-05-20 18:03:53 -03:00
|
|
|
async def test_reflect_stream_but_reflector_changes_its_mind(self):
|
|
|
|
return await asyncio.wait_for(self._test_reflect_stream(partial_needs=True), 3, loop=self.loop)
|
|
|
|
|
2019-06-07 13:23:20 -04:00
|
|
|
async def test_reflect_stream_small_response_chunks(self):
|
|
|
|
return await asyncio.wait_for(self._test_reflect_stream(response_chunk_size=30), 3, loop=self.loop)
|
|
|
|
|
2019-05-06 16:49:03 -03:00
|
|
|
async def test_announces(self):
|
|
|
|
to_announce = await self.storage.get_blobs_to_announce()
|
|
|
|
self.assertIn(self.stream.sd_hash, to_announce, "sd blob not set to announce")
|
2022-02-08 22:10:42 -03:00
|
|
|
self.assertNotIn(self.stream.descriptor.blobs[0].blob_hash, to_announce, "head blob set to announce")
|
2020-07-15 16:39:59 -04:00
|
|
|
|
|
|
|
async def test_result_from_disconnect_mid_sd_transfer(self):
|
|
|
|
stop = asyncio.Event()
|
|
|
|
incoming = asyncio.Event()
|
|
|
|
reflector = ReflectorServer(
|
|
|
|
self.server_blob_manager, response_chunk_size=50, stop_event=stop, incoming_event=incoming
|
|
|
|
)
|
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
|
|
|
reflect_task = asyncio.create_task(self.stream.upload_to_reflector('127.0.0.1', 5566))
|
|
|
|
await incoming.wait()
|
|
|
|
stop.set()
|
|
|
|
# this used to raise (and then propagate) a CancelledError
|
2021-01-08 11:54:02 -05:00
|
|
|
self.assertListEqual(await reflect_task, [])
|
2020-07-15 16:39:59 -04:00
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|
2020-08-11 13:04:05 -04:00
|
|
|
self.assertFalse(self.server_blob_manager.get_blob(self.stream.sd_hash).get_is_verified())
|
2020-07-15 16:39:59 -04:00
|
|
|
|
|
|
|
async def test_result_from_disconnect_after_sd_transfer(self):
|
|
|
|
stop = asyncio.Event()
|
|
|
|
incoming = asyncio.Event()
|
|
|
|
not_incoming = asyncio.Event()
|
|
|
|
reflector = ReflectorServer(
|
|
|
|
self.server_blob_manager, response_chunk_size=50, stop_event=stop, incoming_event=incoming,
|
|
|
|
not_incoming_event=not_incoming
|
|
|
|
)
|
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
|
|
|
reflect_task = asyncio.create_task(self.stream.upload_to_reflector('127.0.0.1', 5566))
|
|
|
|
await incoming.wait()
|
|
|
|
await not_incoming.wait()
|
|
|
|
stop.set()
|
2021-01-08 11:54:02 -05:00
|
|
|
sent = await reflect_task
|
|
|
|
self.assertListEqual([self.stream.sd_hash], sent)
|
2020-07-15 16:39:59 -04:00
|
|
|
self.assertTrue(self.server_blob_manager.get_blob(self.stream.sd_hash).get_is_verified())
|
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|
|
|
|
|
|
|
|
async def test_result_from_disconnect_after_data_transfer(self):
|
|
|
|
stop = asyncio.Event()
|
|
|
|
incoming = asyncio.Event()
|
|
|
|
not_incoming = asyncio.Event()
|
|
|
|
reflector = ReflectorServer(
|
|
|
|
self.server_blob_manager, response_chunk_size=50, stop_event=stop, incoming_event=incoming,
|
|
|
|
not_incoming_event=not_incoming
|
|
|
|
)
|
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
|
|
|
reflect_task = asyncio.create_task(self.stream.upload_to_reflector('127.0.0.1', 5566))
|
|
|
|
await incoming.wait()
|
|
|
|
await not_incoming.wait()
|
|
|
|
await incoming.wait()
|
|
|
|
await not_incoming.wait()
|
|
|
|
stop.set()
|
2021-01-08 11:54:02 -05:00
|
|
|
sent = await reflect_task
|
|
|
|
self.assertListEqual([self.stream.sd_hash, self.stream.descriptor.blobs[0].blob_hash], sent)
|
2020-07-15 16:39:59 -04:00
|
|
|
self.assertTrue(self.server_blob_manager.get_blob(self.stream.sd_hash).get_is_verified())
|
|
|
|
self.assertTrue(self.server_blob_manager.get_blob(self.stream.descriptor.blobs[0].blob_hash).get_is_verified())
|
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|
|
|
|
|
|
|
|
async def test_result_from_disconnect_mid_data_transfer(self):
|
|
|
|
stop = asyncio.Event()
|
|
|
|
incoming = asyncio.Event()
|
|
|
|
not_incoming = asyncio.Event()
|
|
|
|
reflector = ReflectorServer(
|
|
|
|
self.server_blob_manager, response_chunk_size=50, stop_event=stop, incoming_event=incoming,
|
|
|
|
not_incoming_event=not_incoming
|
|
|
|
)
|
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
|
|
|
reflect_task = asyncio.create_task(self.stream.upload_to_reflector('127.0.0.1', 5566))
|
|
|
|
await incoming.wait()
|
|
|
|
await not_incoming.wait()
|
|
|
|
await incoming.wait()
|
|
|
|
stop.set()
|
2021-01-08 11:54:02 -05:00
|
|
|
self.assertListEqual(await reflect_task, [self.stream.sd_hash])
|
2020-07-15 16:39:59 -04:00
|
|
|
self.assertTrue(self.server_blob_manager.get_blob(self.stream.sd_hash).get_is_verified())
|
|
|
|
self.assertFalse(
|
|
|
|
self.server_blob_manager.get_blob(self.stream.descriptor.blobs[0].blob_hash).get_is_verified()
|
|
|
|
)
|
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|
2020-07-16 12:08:10 -04:00
|
|
|
|
|
|
|
async def test_delete_file_during_reflector_upload(self):
|
|
|
|
stop = asyncio.Event()
|
|
|
|
incoming = asyncio.Event()
|
|
|
|
not_incoming = asyncio.Event()
|
|
|
|
reflector = ReflectorServer(
|
|
|
|
self.server_blob_manager, response_chunk_size=50, stop_event=stop, incoming_event=incoming,
|
|
|
|
not_incoming_event=not_incoming
|
|
|
|
)
|
|
|
|
reflector.start_server(5566, '127.0.0.1')
|
|
|
|
await reflector.started_listening.wait()
|
|
|
|
self.addCleanup(reflector.stop_server)
|
|
|
|
self.assertEqual(0, self.stream.reflector_progress)
|
|
|
|
reflect_task = asyncio.create_task(self.stream.upload_to_reflector('127.0.0.1', 5566))
|
|
|
|
await incoming.wait()
|
|
|
|
await not_incoming.wait()
|
|
|
|
await incoming.wait()
|
|
|
|
await self.stream_manager.delete(self.stream, delete_file=True)
|
|
|
|
# this used to raise OSError when it can't read the deleted blob for the upload
|
2021-01-08 11:54:02 -05:00
|
|
|
sent = await reflect_task
|
|
|
|
self.assertListEqual([self.stream.sd_hash], sent)
|
2020-07-16 12:08:10 -04:00
|
|
|
self.assertTrue(self.server_blob_manager.get_blob(self.stream.sd_hash).get_is_verified())
|
|
|
|
self.assertFalse(
|
|
|
|
self.server_blob_manager.get_blob(self.stream.descriptor.blobs[0].blob_hash).get_is_verified()
|
|
|
|
)
|
|
|
|
self.assertFalse(self.stream.is_fully_reflected)
|