2019-02-12 18:50:43 -03:00
|
|
|
import asyncio
|
2019-12-31 14:52:57 -05:00
|
|
|
|
2019-06-03 16:47:08 -03:00
|
|
|
from unittest.mock import Mock
|
2020-04-11 17:27:41 -04:00
|
|
|
from binascii import unhexlify
|
2018-10-17 14:32:45 -03:00
|
|
|
|
2020-04-11 17:27:41 -04:00
|
|
|
import lbry
|
2020-01-03 02:40:37 -05:00
|
|
|
from lbry.wallet.network import Network
|
2019-12-31 14:52:57 -05:00
|
|
|
from lbry.wallet.orchstr8.node import SPVNode
|
|
|
|
from lbry.wallet.rpc import RPCSession
|
2019-12-31 15:30:13 -05:00
|
|
|
from lbry.testcase import IntegrationTestCase, AsyncioTestCase
|
2018-10-11 00:07:38 -03:00
|
|
|
|
|
|
|
|
2019-06-26 03:41:35 -03:00
|
|
|
class NetworkTests(IntegrationTestCase):
|
|
|
|
|
|
|
|
async def test_remote_height_updated_automagically(self):
|
|
|
|
initial_height = self.ledger.network.remote_height
|
|
|
|
await self.blockchain.generate(1)
|
|
|
|
await self.ledger.network.on_header.first
|
|
|
|
self.assertEqual(self.ledger.network.remote_height, initial_height + 1)
|
|
|
|
|
2019-09-17 01:29:00 -03:00
|
|
|
async def test_server_features(self):
|
2019-10-08 18:01:38 +03:00
|
|
|
self.assertDictEqual({
|
2019-09-17 16:30:27 -03:00
|
|
|
'genesis_hash': self.conductor.spv_node.coin_class.GENESIS_HASH,
|
|
|
|
'hash_function': 'sha256',
|
|
|
|
'hosts': {},
|
2020-01-06 11:33:07 -05:00
|
|
|
'protocol_max': '0.99.0',
|
2020-02-03 13:39:36 -05:00
|
|
|
'protocol_min': '0.54.0',
|
2019-09-17 16:30:27 -03:00
|
|
|
'pruning': None,
|
|
|
|
'description': '',
|
|
|
|
'payment_address': '',
|
2020-02-11 22:19:13 -03:00
|
|
|
'donation_address': '',
|
2020-01-06 01:53:51 -03:00
|
|
|
'daily_fee': '0',
|
2020-03-03 20:26:07 -05:00
|
|
|
'server_version': lbry.__version__,
|
|
|
|
'trending_algorithm': 'zscore',
|
|
|
|
}, await self.ledger.network.get_server_features())
|
2019-09-17 01:29:00 -03:00
|
|
|
await self.conductor.spv_node.stop()
|
2020-02-11 22:19:13 -03:00
|
|
|
payment_address, donation_address = await self.account.get_addresses(limit=2)
|
2020-01-12 00:48:46 -03:00
|
|
|
await self.conductor.spv_node.start(
|
|
|
|
self.conductor.blockchain_node,
|
|
|
|
extraconf={
|
|
|
|
'DESCRIPTION': 'Fastest server in the west.',
|
2020-02-11 22:19:13 -03:00
|
|
|
'PAYMENT_ADDRESS': payment_address,
|
|
|
|
'DONATION_ADDRESS': donation_address,
|
2020-01-12 00:48:46 -03:00
|
|
|
'DAILY_FEE': '42'
|
|
|
|
}
|
|
|
|
)
|
2019-09-17 01:29:00 -03:00
|
|
|
await self.ledger.network.on_connected.first
|
2019-10-08 18:01:38 +03:00
|
|
|
self.assertDictEqual({
|
2019-09-17 16:30:27 -03:00
|
|
|
'genesis_hash': self.conductor.spv_node.coin_class.GENESIS_HASH,
|
|
|
|
'hash_function': 'sha256',
|
|
|
|
'hosts': {},
|
2020-01-06 11:33:07 -05:00
|
|
|
'protocol_max': '0.99.0',
|
2020-02-03 13:39:36 -05:00
|
|
|
'protocol_min': '0.54.0',
|
2019-09-17 16:30:27 -03:00
|
|
|
'pruning': None,
|
|
|
|
'description': 'Fastest server in the west.',
|
2020-02-11 22:19:13 -03:00
|
|
|
'payment_address': payment_address,
|
|
|
|
'donation_address': donation_address,
|
2020-01-06 01:53:51 -03:00
|
|
|
'daily_fee': '42',
|
2020-03-03 20:26:07 -05:00
|
|
|
'server_version': lbry.__version__,
|
|
|
|
'trending_algorithm': 'zscore',
|
|
|
|
}, await self.ledger.network.get_server_features())
|
2019-09-17 01:29:00 -03:00
|
|
|
|
2019-06-26 03:41:35 -03:00
|
|
|
|
2020-03-30 17:02:08 -04:00
|
|
|
class ReconnectTests(IntegrationTestCase):
|
2018-10-11 00:07:38 -03:00
|
|
|
|
2019-08-29 01:12:49 -03:00
|
|
|
async def test_multiple_servers(self):
|
|
|
|
# we have a secondary node that connects later, so
|
|
|
|
node2 = SPVNode(self.conductor.spv_module, node_number=2)
|
|
|
|
self.ledger.network.config['default_servers'].append((node2.hostname, node2.port))
|
|
|
|
await asyncio.wait_for(self.ledger.stop(), timeout=1)
|
|
|
|
await asyncio.wait_for(self.ledger.start(), timeout=1)
|
|
|
|
self.ledger.network.session_pool.new_connection_event.clear()
|
|
|
|
await node2.start(self.blockchain)
|
|
|
|
# this is only to speed up the test as retrying would take 4+ seconds
|
|
|
|
for session in self.ledger.network.session_pool.sessions:
|
|
|
|
session.trigger_urgent_reconnect.set()
|
|
|
|
await asyncio.wait_for(self.ledger.network.session_pool.new_connection_event.wait(), timeout=1)
|
2019-08-30 19:53:51 -03:00
|
|
|
self.assertEqual(2, len(list(self.ledger.network.session_pool.available_sessions)))
|
2019-08-29 01:12:49 -03:00
|
|
|
self.assertTrue(self.ledger.network.is_connected)
|
|
|
|
switch_event = self.ledger.network.on_connected.first
|
|
|
|
await node2.stop(True)
|
|
|
|
# secondary down, but primary is ok, do not switch! (switches trigger new on_connected events)
|
|
|
|
with self.assertRaises(asyncio.TimeoutError):
|
|
|
|
await asyncio.wait_for(switch_event, timeout=1)
|
|
|
|
|
2019-10-08 21:45:47 -03:00
|
|
|
async def test_direct_sync(self):
|
|
|
|
await self.ledger.stop()
|
2019-10-08 22:17:09 -03:00
|
|
|
initial_height = self.ledger.local_height_including_downloaded_height
|
2019-10-08 21:45:47 -03:00
|
|
|
await self.blockchain.generate(100)
|
2020-01-13 22:48:07 -03:00
|
|
|
while self.conductor.spv_node.server.session_mgr.notified_height < initial_height + 99: # off by 1
|
2019-10-08 21:45:47 -03:00
|
|
|
await asyncio.sleep(0.1)
|
2019-10-08 22:17:09 -03:00
|
|
|
self.assertEqual(initial_height, self.ledger.local_height_including_downloaded_height)
|
2020-03-21 04:32:03 -03:00
|
|
|
await self.ledger.headers.open()
|
|
|
|
await self.ledger.network.start()
|
|
|
|
await self.ledger.network.on_connected.first
|
|
|
|
await self.ledger.initial_headers_sync()
|
|
|
|
self.assertEqual(initial_height + 100, self.ledger.local_height_including_downloaded_height)
|
2019-10-08 21:45:47 -03:00
|
|
|
|
2018-10-11 00:07:38 -03:00
|
|
|
async def test_connection_drop_still_receives_events_after_reconnected(self):
|
2018-10-15 00:45:21 -04:00
|
|
|
address1 = await self.account.receiving.get_or_create_usable_address()
|
2019-08-06 02:17:39 -03:00
|
|
|
# disconnect and send a new tx, should reconnect and get it
|
2018-10-17 14:32:45 -03:00
|
|
|
self.ledger.network.client.connection_lost(Exception())
|
2019-08-06 02:17:39 -03:00
|
|
|
self.assertFalse(self.ledger.network.is_connected)
|
2020-04-11 17:27:41 -04:00
|
|
|
tx_hash = unhexlify((await self.blockchain.send_to_address(address1, 1.1337)))[::-1]
|
|
|
|
await asyncio.wait_for(self.on_transaction_hash(tx_hash), 2.0) # mempool
|
2018-10-11 00:07:38 -03:00
|
|
|
await self.blockchain.generate(1)
|
2020-04-11 17:27:41 -04:00
|
|
|
await self.on_transaction_hash(tx_hash) # confirmed
|
2019-08-12 18:16:53 -03:00
|
|
|
self.assertLess(self.ledger.network.client.response_time, 1) # response time properly set lower, we are fine
|
2018-10-11 00:07:38 -03:00
|
|
|
|
2018-11-18 23:17:39 -05:00
|
|
|
await self.assertBalance(self.account, '1.1337')
|
2018-10-11 00:07:38 -03:00
|
|
|
# is it real? are we rich!? let me see this tx...
|
2020-04-11 17:27:41 -04:00
|
|
|
d = self.ledger.network.get_transaction(tx_hash)
|
2018-10-11 00:07:38 -03:00
|
|
|
# what's that smoke on my ethernet cable? oh no!
|
2019-08-29 01:12:49 -03:00
|
|
|
master_client = self.ledger.network.client
|
2018-10-17 14:32:45 -03:00
|
|
|
self.ledger.network.client.connection_lost(Exception())
|
2019-08-12 18:16:53 -03:00
|
|
|
with self.assertRaises(asyncio.TimeoutError):
|
2019-08-31 06:37:17 -03:00
|
|
|
await d
|
2019-08-29 01:12:49 -03:00
|
|
|
self.assertIsNone(master_client.response_time) # response time unknown as it failed
|
2018-10-11 00:07:38 -03:00
|
|
|
# rich but offline? no way, no water, let's retry
|
|
|
|
with self.assertRaisesRegex(ConnectionError, 'connection is not available'):
|
2020-04-11 17:27:41 -04:00
|
|
|
await self.ledger.network.get_transaction(tx_hash)
|
2018-10-11 00:07:38 -03:00
|
|
|
# * goes to pick some water outside... * time passes by and another donation comes in
|
2020-04-11 17:27:41 -04:00
|
|
|
tx_hash = unhexlify((await self.blockchain.send_to_address(address1, 42)))[::-1]
|
2018-10-11 00:07:38 -03:00
|
|
|
await self.blockchain.generate(1)
|
2019-10-02 21:04:30 +03:00
|
|
|
# (this is just so the test doesn't hang forever if it doesn't reconnect)
|
2019-08-12 13:32:20 -03:00
|
|
|
if not self.ledger.network.is_connected:
|
|
|
|
await asyncio.wait_for(self.ledger.network.on_connected.first, timeout=1.0)
|
2018-10-11 00:07:38 -03:00
|
|
|
# omg, the burned cable still works! torba is fire proof!
|
2020-04-11 17:27:41 -04:00
|
|
|
await self.ledger.network.get_transaction(tx_hash)
|
2019-02-12 18:50:43 -03:00
|
|
|
|
|
|
|
async def test_timeout_then_reconnect(self):
|
2019-08-06 02:17:39 -03:00
|
|
|
# tests that it connects back after some failed attempts
|
2019-06-04 02:34:58 -03:00
|
|
|
await self.conductor.spv_node.stop()
|
|
|
|
self.assertFalse(self.ledger.network.is_connected)
|
2019-08-06 02:17:39 -03:00
|
|
|
await asyncio.sleep(0.2) # let it retry and fail once
|
2019-06-04 02:34:58 -03:00
|
|
|
await self.conductor.spv_node.start(self.conductor.blockchain_node)
|
|
|
|
await self.ledger.network.on_connected.first
|
2019-06-03 05:21:57 -03:00
|
|
|
self.assertTrue(self.ledger.network.is_connected)
|
|
|
|
|
2019-09-24 03:23:09 -03:00
|
|
|
async def test_online_but_still_unavailable(self):
|
|
|
|
# Edge case. See issue #2445 for context
|
|
|
|
self.assertIsNotNone(self.ledger.network.session_pool.fastest_session)
|
|
|
|
for session in self.ledger.network.session_pool.sessions:
|
|
|
|
session.response_time = None
|
|
|
|
self.assertIsNone(self.ledger.network.session_pool.fastest_session)
|
|
|
|
|
2019-06-03 16:47:08 -03:00
|
|
|
|
|
|
|
class ServerPickingTestCase(AsyncioTestCase):
|
2019-07-19 20:03:18 -03:00
|
|
|
|
|
|
|
async def _make_fake_server(self, latency=1.0, port=1):
|
2019-06-03 16:03:20 -03:00
|
|
|
# local fake server with artificial latency
|
2019-07-19 20:03:18 -03:00
|
|
|
class FakeSession(RPCSession):
|
|
|
|
async def handle_request(self, request):
|
|
|
|
await asyncio.sleep(latency)
|
2020-01-15 12:57:37 -05:00
|
|
|
if request.method == 'server.version':
|
|
|
|
return tuple(request.args)
|
|
|
|
return {'height': 1}
|
2019-07-19 20:03:18 -03:00
|
|
|
server = await self.loop.create_server(lambda: FakeSession(), host='127.0.0.1', port=port)
|
|
|
|
self.addCleanup(server.close)
|
|
|
|
return '127.0.0.1', port
|
2019-07-12 19:54:04 -03:00
|
|
|
|
2019-07-19 20:03:18 -03:00
|
|
|
async def _make_bad_server(self, port=42420):
|
|
|
|
async def echo(reader, writer):
|
|
|
|
while True:
|
|
|
|
writer.write(await reader.read())
|
|
|
|
server = await asyncio.start_server(echo, host='127.0.0.1', port=port)
|
2019-06-03 16:03:20 -03:00
|
|
|
self.addCleanup(server.close)
|
2019-07-12 19:54:04 -03:00
|
|
|
return '127.0.0.1', port
|
2019-06-03 16:03:20 -03:00
|
|
|
|
|
|
|
async def test_pick_fastest(self):
|
2019-06-03 16:47:08 -03:00
|
|
|
ledger = Mock(config={
|
|
|
|
'default_servers': [
|
2019-07-19 20:03:18 -03:00
|
|
|
# fast but unhealthy, should be discarded
|
|
|
|
await self._make_bad_server(),
|
|
|
|
('localhost', 1),
|
|
|
|
('example.that.doesnt.resolve', 9000),
|
2019-08-06 02:17:39 -03:00
|
|
|
await self._make_fake_server(latency=1.0, port=1340),
|
|
|
|
await self._make_fake_server(latency=0.1, port=1337),
|
|
|
|
await self._make_fake_server(latency=0.4, port=1339),
|
2019-06-03 16:47:08 -03:00
|
|
|
],
|
2019-07-19 20:03:18 -03:00
|
|
|
'connect_timeout': 3
|
2019-06-03 16:47:08 -03:00
|
|
|
})
|
2019-06-03 16:03:20 -03:00
|
|
|
|
2020-01-03 02:40:37 -05:00
|
|
|
network = Network(ledger)
|
2019-06-03 16:10:41 -03:00
|
|
|
self.addCleanup(network.stop)
|
2019-06-03 16:03:20 -03:00
|
|
|
asyncio.ensure_future(network.start())
|
2019-08-06 02:17:39 -03:00
|
|
|
await asyncio.wait_for(network.on_connected.first, timeout=1)
|
2019-06-03 16:03:20 -03:00
|
|
|
self.assertTrue(network.is_connected)
|
2019-10-08 18:01:38 +03:00
|
|
|
self.assertTupleEqual(network.client.server, ('127.0.0.1', 1337))
|
2019-08-06 02:17:39 -03:00
|
|
|
self.assertTrue(all([not session.is_closing() for session in network.session_pool.available_sessions]))
|
|
|
|
# ensure we are connected to all of them after a while
|
|
|
|
await asyncio.sleep(1)
|
2019-08-30 19:53:51 -03:00
|
|
|
self.assertEqual(len(list(network.session_pool.available_sessions)), 3)
|