lbry-sdk/tests/unit/core/test_Strategy.py

152 lines
6.4 KiB
Python
Raw Normal View History

2016-10-13 14:00:46 -04:00
import itertools
2016-10-03 02:44:58 -04:00
import random
from unittest import mock
from twisted.trial import unittest
from lbrynet.p2p.PaymentRateManager import NegotiatedPaymentRateManager, BasePaymentRateManager
from lbrynet.p2p.Strategy import BasicAvailabilityWeightedStrategy
from lbrynet.p2p.Offer import Offer
2019-01-21 15:55:50 -05:00
from lbrynet.conf import Config
from tests.mocks import BlobAvailabilityTracker as DummyBlobAvailabilityTracker
MAX_NEGOTIATION_TURNS = 10
2016-10-13 13:48:18 -04:00
random.seed(12345)
2016-10-03 02:44:58 -04:00
def get_random_sample(list_to_sample):
2017-09-29 11:44:22 +01:00
result = list_to_sample[
random.randint(1, len(list_to_sample)):random.randint(1, len(list_to_sample))]
2016-10-03 02:44:58 -04:00
if not result:
return get_random_sample(list_to_sample)
return result
2016-10-03 02:44:58 -04:00
2017-09-29 11:44:22 +01:00
def calculate_negotation_turns(client_base, host_base, host_is_generous=True,
client_is_generous=True):
2016-10-13 14:00:46 -04:00
blobs = [
'b2e48bb4c88cf46b76adf0d47a72389fae0cd1f19ed27dc5'
'09138c99509a25423a4cef788d571dca7988e1dca69e6fa0',
'd7c82e6cac093b3f16107d2ae2b2c75424f1fcad2c7fbdbe'
'66e4a13c0b6bd27b67b3a29c403b82279ab0f7c1c48d6787',
'5a450b416275da4bdff604ee7b58eaedc7913c5005b7184f'
'c3bc5ef0b1add00613587f54217c91097fc039ed9eace9dd',
'f99d24cd50d4bfd77c2598bfbeeb8415bf0feef21200bdf0'
'b8fbbde7751a77b7a2c68e09c25465a2f40fba8eecb0b4e0',
'9dbda74a472a2e5861a5d18197aeba0f5de67c67e401124c'
'243d2f0f41edf01d7a26aeb0b5fc9bf47f6361e0f0968e2c',
'91dc64cf1ff42e20d627b033ad5e4c3a4a96856ed8a6e3fb'
'4cd5fa1cfba4bf72eefd325f579db92f45f4355550ace8e7',
'6d8017aba362e5c5d0046625a039513419810a0397d72831'
'8c328a5cc5d96efb589fbca0728e54fe5adbf87e9545ee07',
'6af95cd062b4a179576997ef1054c9d2120f8592eea045e9'
'667bea411d520262cd5a47b137eabb7a7871f5f8a79c92dd',
'8c70d5e2f5c3a6085006198e5192d157a125d92e73787944'
'72007a61947992768926513fc10924785bdb1761df3c37e6',
'c84aa1fd8f5009f7c4e71e444e40d95610abc1480834f835'
'eefb267287aeb10025880a3ce22580db8c6d92efb5bc0c9c'
2016-10-13 14:00:46 -04:00
]
host = mock.Mock()
host.host = "1.2.3.4"
client = mock.Mock()
client.host = "1.2.3.5"
2019-01-21 15:55:50 -05:00
conf = Config()
client_base_prm = BasePaymentRateManager(client_base, conf.min_info_rate)
2016-10-13 14:00:46 -04:00
client_prm = NegotiatedPaymentRateManager(client_base_prm,
DummyBlobAvailabilityTracker(),
2019-01-21 15:55:50 -05:00
client_is_generous)
host_base_prm = BasePaymentRateManager(host_base, conf.min_info_rate)
2016-10-13 14:00:46 -04:00
host_prm = NegotiatedPaymentRateManager(host_base_prm,
DummyBlobAvailabilityTracker(),
2019-01-21 15:55:50 -05:00
host_is_generous)
2016-10-13 14:00:46 -04:00
blobs_to_query = get_random_sample(blobs)
accepted = False
turns = 0
while not accepted:
rate = client_prm.get_rate_blob_data(host, blobs_to_query)
offer = Offer(rate)
accepted = host_prm.accept_rate_blob_data(client, blobs_to_query, offer)
turns += 1
return turns
2016-10-03 02:44:58 -04:00
class AvailabilityWeightedStrategyTests(unittest.TestCase):
2017-01-16 22:23:20 -05:00
def test_first_offer_is_zero_and_second_is_not_if_offer_not_accepted(self):
2019-01-21 15:55:50 -05:00
conf = Config()
strategy = BasicAvailabilityWeightedStrategy(
DummyBlobAvailabilityTracker(), conf.data_rate, conf.is_generous_host
)
peer = "1.1.1.1"
blobs = strategy.price_model.blob_tracker.availability.keys()
offer1 = strategy.make_offer(peer, blobs)
offer2 = strategy.make_offer(peer, blobs)
2018-07-21 16:55:43 -04:00
self.assertEqual(offer1.rate, 0.0)
self.assertNotEqual(offer2.rate, 0.0)
def test_accept_zero_and_persist_if_accepted(self):
2019-01-21 15:55:50 -05:00
conf = Config()
host_strategy = BasicAvailabilityWeightedStrategy(
DummyBlobAvailabilityTracker(), conf.data_rate, conf.is_generous_host
)
client_strategy = BasicAvailabilityWeightedStrategy(
DummyBlobAvailabilityTracker(), conf.data_rate, conf.is_generous_host
)
client = "1.1.1.1"
host = "1.1.1.2"
blobs = host_strategy.price_model.blob_tracker.availability.keys()
2016-10-03 02:44:58 -04:00
offer = client_strategy.make_offer(host, blobs)
response1 = host_strategy.respond_to_offer(offer, client, blobs)
2016-10-13 13:35:55 -04:00
client_strategy.update_accepted_offers(host, response1)
2016-10-03 02:44:58 -04:00
offer = client_strategy.make_offer(host, blobs)
response2 = host_strategy.respond_to_offer(offer, client, blobs)
2016-10-13 13:35:55 -04:00
client_strategy.update_accepted_offers(host, response2)
self.assertFalse(response1.is_too_low)
self.assertTrue(response1.is_accepted)
2018-07-21 16:55:43 -04:00
self.assertEqual(response1.rate, 0.0)
self.assertFalse(response2.is_too_low)
self.assertTrue(response2.is_accepted)
2018-07-21 16:55:43 -04:00
self.assertEqual(response2.rate, 0.0)
def test_how_many_turns_before_accept_with_similar_rate_settings(self):
2016-10-13 14:00:46 -04:00
base_rates = [0.0001 * n for n in range(1, 10)]
for host_base, client_base in itertools.product(base_rates, base_rates):
turns = calculate_negotation_turns(host_base,
client_base,
client_is_generous=False,
host_is_generous=False)
self.assertGreater(MAX_NEGOTIATION_TURNS, turns)
def test_generous_connects_in_one_turn(self):
2016-10-13 14:00:46 -04:00
base_rates = [0.0001 * n for n in range(1, 10)]
for host_base, client_base in itertools.product(base_rates, base_rates):
turns = calculate_negotation_turns(host_base, client_base)
self.assertEqual(1, turns)
def test_how_many_turns_with_generous_client(self):
2016-10-13 14:00:46 -04:00
base_rates = [0.0001 * n for n in range(1, 10)]
for host_base, client_base in itertools.product(base_rates, base_rates):
turns = calculate_negotation_turns(host_base,
client_base,
host_is_generous=False)
self.assertGreater(MAX_NEGOTIATION_TURNS, turns)
def test_how_many_turns_with_generous_host(self):
2016-10-13 14:00:46 -04:00
base_rates = [0.0001 * n for n in range(1, 10)]
for host_base, client_base in itertools.product(base_rates, base_rates):
turns = calculate_negotation_turns(host_base,
client_base,
client_is_generous=False)
self.assertGreater(MAX_NEGOTIATION_TURNS, turns)