2016-07-11 04:56:00 +02:00
|
|
|
import StringIO
|
|
|
|
|
2016-07-11 00:47:36 +02:00
|
|
|
import mock
|
2016-10-13 19:35:55 +02:00
|
|
|
from twisted.internet import defer
|
2016-07-11 04:56:00 +02:00
|
|
|
from twisted.test import proto_helpers
|
2016-07-11 00:47:36 +02:00
|
|
|
from twisted.trial import unittest
|
|
|
|
|
2016-07-11 04:56:00 +02:00
|
|
|
from lbrynet.core import Peer
|
2016-07-11 00:47:36 +02:00
|
|
|
from lbrynet.core.server import BlobRequestHandler
|
2016-10-03 08:44:58 +02:00
|
|
|
from lbrynet.core.PaymentRateManager import NegotiatedPaymentRateManager, BasePaymentRateManager
|
2016-10-13 19:35:55 +02:00
|
|
|
from tests.mocks import DummyBlobAvailabilityTracker
|
2016-07-11 00:47:36 +02:00
|
|
|
|
|
|
|
|
|
|
|
class TestBlobRequestHandlerQueries(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.blob_manager = mock.Mock()
|
2016-10-03 08:44:58 +02:00
|
|
|
self.payment_rate_manager = NegotiatedPaymentRateManager(BasePaymentRateManager(0.001), DummyBlobAvailabilityTracker())
|
|
|
|
self.handler = BlobRequestHandler.BlobRequestHandler(self.blob_manager, None, self.payment_rate_manager)
|
2016-07-11 00:47:36 +02:00
|
|
|
|
|
|
|
def test_empty_response_when_empty_query(self):
|
2016-10-03 08:44:58 +02:00
|
|
|
self.assertEqual({}, self.successResultOf(self.handler.handle_queries({})))
|
2016-07-11 04:56:00 +02:00
|
|
|
|
2016-07-11 00:47:36 +02:00
|
|
|
def test_error_set_when_rate_is_missing(self):
|
|
|
|
query = {'requested_blob': 'blob'}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {'incoming_blob': {'error': 'RATE_UNSET'}}
|
|
|
|
self.assertEqual(response, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_error_set_when_rate_too_low(self):
|
|
|
|
query = {
|
2016-10-03 08:44:58 +02:00
|
|
|
'blob_data_payment_rate': '-1.0',
|
2016-07-11 00:47:36 +02:00
|
|
|
'requested_blob': 'blob'
|
|
|
|
}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {
|
|
|
|
'blob_data_payment_rate': 'RATE_TOO_LOW',
|
|
|
|
'incoming_blob': {'error': 'RATE_UNSET'}
|
|
|
|
}
|
|
|
|
self.assertEqual(response, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_response_when_rate_too_low(self):
|
|
|
|
query = {
|
2016-10-03 08:44:58 +02:00
|
|
|
'blob_data_payment_rate': '-1.0',
|
2016-07-11 00:47:36 +02:00
|
|
|
}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {
|
|
|
|
'blob_data_payment_rate': 'RATE_TOO_LOW',
|
|
|
|
}
|
|
|
|
self.assertEqual(response, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_blob_unavailable_when_blob_not_validated(self):
|
|
|
|
blob = mock.Mock()
|
|
|
|
blob.is_validated.return_value = False
|
|
|
|
self.blob_manager.get_blob.return_value = defer.succeed(blob)
|
|
|
|
query = {
|
2016-10-03 08:44:58 +02:00
|
|
|
'blob_data_payment_rate': 1.0,
|
2016-07-11 00:47:36 +02:00
|
|
|
'requested_blob': 'blob'
|
|
|
|
}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {
|
|
|
|
'blob_data_payment_rate': 'RATE_ACCEPTED',
|
|
|
|
'incoming_blob': {'error': 'BLOB_UNAVAILABLE'}
|
|
|
|
}
|
|
|
|
self.assertEqual(response, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_blob_unavailable_when_blob_cannot_be_opened(self):
|
|
|
|
blob = mock.Mock()
|
|
|
|
blob.is_validated.return_value = True
|
|
|
|
blob.open_for_reading.return_value = None
|
|
|
|
self.blob_manager.get_blob.return_value = defer.succeed(blob)
|
|
|
|
query = {
|
2016-10-03 08:44:58 +02:00
|
|
|
'blob_data_payment_rate': 0.0,
|
2016-07-11 00:47:36 +02:00
|
|
|
'requested_blob': 'blob'
|
|
|
|
}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {
|
|
|
|
'blob_data_payment_rate': 'RATE_ACCEPTED',
|
|
|
|
'incoming_blob': {'error': 'BLOB_UNAVAILABLE'}
|
|
|
|
}
|
|
|
|
self.assertEqual(response, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_blob_details_are_set_when_all_conditions_are_met(self):
|
|
|
|
blob = mock.Mock()
|
|
|
|
blob.is_validated.return_value = True
|
|
|
|
blob.open_for_reading.return_value = True
|
|
|
|
blob.blob_hash = 'DEADBEEF'
|
|
|
|
blob.length = 42
|
2016-10-03 08:44:58 +02:00
|
|
|
peer = mock.Mock()
|
|
|
|
peer.host = "1.2.3.4"
|
|
|
|
self.handler.peer = peer
|
2016-07-11 00:47:36 +02:00
|
|
|
self.blob_manager.get_blob.return_value = defer.succeed(blob)
|
|
|
|
query = {
|
2016-10-03 08:44:58 +02:00
|
|
|
'blob_data_payment_rate': 1.0,
|
2016-07-11 00:47:36 +02:00
|
|
|
'requested_blob': 'blob'
|
|
|
|
}
|
|
|
|
deferred = self.handler.handle_queries(query)
|
|
|
|
response = {
|
|
|
|
'blob_data_payment_rate': 'RATE_ACCEPTED',
|
|
|
|
'incoming_blob': {
|
|
|
|
'blob_hash': 'DEADBEEF',
|
|
|
|
'length': 42
|
|
|
|
}
|
|
|
|
}
|
2016-10-03 08:44:58 +02:00
|
|
|
result = self.successResultOf(deferred)
|
|
|
|
self.assertEqual(response, result)
|
2016-07-11 00:47:36 +02:00
|
|
|
|
2016-07-11 04:56:00 +02:00
|
|
|
|
|
|
|
class TestBlobRequestHandlerSender(unittest.TestCase):
|
|
|
|
def test_nothing_happens_if_not_currently_uploading(self):
|
|
|
|
handler = BlobRequestHandler.BlobRequestHandler(None, None, None)
|
|
|
|
handler.currently_uploading = None
|
|
|
|
deferred = handler.send_blob_if_requested(None)
|
|
|
|
self.assertEqual(True, self.successResultOf(deferred))
|
|
|
|
|
|
|
|
def test_file_is_sent_to_consumer(self):
|
|
|
|
# TODO: also check that the expected payment values are set
|
|
|
|
consumer = proto_helpers.StringTransport()
|
|
|
|
test_file = StringIO.StringIO('test')
|
|
|
|
handler = BlobRequestHandler.BlobRequestHandler(None, None, None)
|
|
|
|
handler.peer = mock.create_autospec(Peer.Peer)
|
|
|
|
handler.currently_uploading = mock.Mock()
|
|
|
|
handler.read_handle = test_file
|
|
|
|
handler.send_blob_if_requested(consumer)
|
|
|
|
while consumer.producer:
|
|
|
|
consumer.producer.resumeProducing()
|
|
|
|
self.assertEqual(consumer.value(), 'test')
|