mirror of
https://github.com/bigchaindb/bigchaindb.git
synced 2024-10-13 13:34:05 +00:00

* Isolate handling of potential KeyError * Add tests for bigchaindb/util.py * Simplify the logic a bit * Raise the exception
213 lines
6.5 KiB
Python
213 lines
6.5 KiB
Python
import queue
|
|
from unittest.mock import patch, call
|
|
|
|
import pytest
|
|
|
|
from cryptoconditions import ThresholdSha256Fulfillment
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_queue(monkeypatch):
|
|
|
|
class MockQueue:
|
|
items = []
|
|
|
|
def get(self, timeout=None):
|
|
try:
|
|
return self.items.pop()
|
|
except IndexError:
|
|
if timeout:
|
|
raise queue.Empty()
|
|
raise
|
|
|
|
def put(self, item):
|
|
self.items.append(item)
|
|
|
|
mockqueue = MockQueue()
|
|
|
|
monkeypatch.setattr('queue.Queue', lambda: mockqueue)
|
|
return mockqueue
|
|
|
|
|
|
def test_transform_create(b, user_sk, user_vk):
|
|
from bigchaindb import util
|
|
tx = util.create_tx(user_vk, user_vk, None, 'CREATE')
|
|
tx = util.transform_create(tx)
|
|
tx = util.sign_tx(tx, b.me_private)
|
|
|
|
assert tx['transaction']['fulfillments'][0]['current_owners'][0] == b.me
|
|
assert tx['transaction']['conditions'][0]['new_owners'][0] == user_vk
|
|
assert util.validate_fulfillments(tx)
|
|
|
|
|
|
def test_empty_pool_is_populated_with_instances(mock_queue):
|
|
from bigchaindb import util
|
|
|
|
pool = util.pool(lambda: 'hello', 4)
|
|
|
|
assert len(mock_queue.items) == 0
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 1
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 2
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 3
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 4
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 4
|
|
|
|
|
|
def test_pool_blocks_if_no_instances_available(mock_queue):
|
|
from bigchaindb import util
|
|
|
|
pool = util.pool(lambda: 'hello', 4)
|
|
|
|
assert len(mock_queue.items) == 0
|
|
|
|
# We need to manually trigger the `__enter__` method so the context
|
|
# manager will "hang" and not return the resource to the pool
|
|
assert pool().__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
assert pool().__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
assert pool().__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
# We need to keep a reference of the last context manager so we can
|
|
# manually release the resource
|
|
last = pool()
|
|
assert last.__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
# This would block using `queue.Queue` but since we mocked it it will
|
|
# just raise a IndexError because it's trying to pop from an empty list.
|
|
with pytest.raises(IndexError):
|
|
assert pool().__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
# Release the last resource
|
|
last.__exit__(None, None, None)
|
|
assert len(mock_queue.items) == 1
|
|
|
|
assert pool().__enter__() == 'hello'
|
|
assert len(mock_queue.items) == 0
|
|
|
|
|
|
def test_pool_raises_empty_exception_when_timeout(mock_queue):
|
|
from bigchaindb import util
|
|
|
|
pool = util.pool(lambda: 'hello', 1, timeout=1)
|
|
|
|
assert len(mock_queue.items) == 0
|
|
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
assert len(mock_queue.items) == 1
|
|
|
|
# take the only resource available
|
|
assert pool().__enter__() == 'hello'
|
|
|
|
with pytest.raises(queue.Empty):
|
|
with pool() as instance:
|
|
assert instance == 'hello'
|
|
|
|
|
|
@patch('multiprocessing.Process')
|
|
def test_process_group_instantiates_and_start_processes(mock_process):
|
|
from bigchaindb.util import ProcessGroup
|
|
|
|
def noop():
|
|
pass
|
|
|
|
concurrency = 10
|
|
|
|
pg = ProcessGroup(concurrency=concurrency, group='test_group', target=noop)
|
|
pg.start()
|
|
|
|
mock_process.assert_has_calls([call(group='test_group', target=noop,
|
|
name=None, args=(), kwargs={},
|
|
daemon=None)
|
|
for i in range(concurrency)], any_order=True)
|
|
|
|
for process in pg.processes:
|
|
process.start.assert_called_with()
|
|
|
|
|
|
def test_create_tx_with_empty_inputs():
|
|
from bigchaindb.util import create_tx
|
|
tx = create_tx(None, None, [], None)
|
|
assert 'id' in tx
|
|
assert 'transaction' in tx
|
|
assert 'version' in tx
|
|
assert 'fulfillments' in tx['transaction']
|
|
assert 'conditions' in tx['transaction']
|
|
assert 'operation' in tx['transaction']
|
|
assert 'timestamp' in tx['transaction']
|
|
assert 'data' in tx['transaction']
|
|
assert len(tx['transaction']['fulfillments']) == 1
|
|
assert tx['transaction']['fulfillments'][0] == {
|
|
'current_owners': [], 'input': None, 'fulfillment': None, 'fid': 0}
|
|
|
|
|
|
def test_fulfill_threshold_signature_fulfillment_pubkey_notfound(monkeypatch):
|
|
from bigchaindb.exceptions import KeypairMismatchException
|
|
from bigchaindb.util import fulfill_threshold_signature_fulfillment
|
|
monkeypatch.setattr(
|
|
ThresholdSha256Fulfillment,
|
|
'get_subcondition_from_vk',
|
|
lambda x, y: []
|
|
)
|
|
fulfillment = {'current_owners': (None,)}
|
|
parsed_fulfillment = ThresholdSha256Fulfillment()
|
|
with pytest.raises(KeypairMismatchException):
|
|
fulfill_threshold_signature_fulfillment(
|
|
fulfillment, parsed_fulfillment, None, None)
|
|
|
|
|
|
def test_fulfill_threshold_signature_fulfillment_wrong_privkeys(monkeypatch):
|
|
from bigchaindb.exceptions import KeypairMismatchException
|
|
from bigchaindb.util import fulfill_threshold_signature_fulfillment
|
|
monkeypatch.setattr(
|
|
ThresholdSha256Fulfillment,
|
|
'get_subcondition_from_vk',
|
|
lambda x, y: (None,)
|
|
)
|
|
fulfillment = {'current_owners': ('alice-pub-key',)}
|
|
parsed_fulfillment = ThresholdSha256Fulfillment()
|
|
with pytest.raises(KeypairMismatchException):
|
|
fulfill_threshold_signature_fulfillment(
|
|
fulfillment, parsed_fulfillment, None, {})
|
|
|
|
|
|
def test_check_hash_and_signature_invalid_hash(monkeypatch):
|
|
from bigchaindb.exceptions import InvalidHash
|
|
from bigchaindb.util import check_hash_and_signature
|
|
transaction = {'id': 'txid'}
|
|
monkeypatch.setattr('bigchaindb.util.get_hash_data', lambda tx: 'txhash')
|
|
with pytest.raises(InvalidHash):
|
|
check_hash_and_signature(transaction)
|
|
|
|
|
|
def test_check_hash_and_signature_invalid_signature(monkeypatch):
|
|
from bigchaindb.exceptions import InvalidSignature
|
|
from bigchaindb.util import check_hash_and_signature
|
|
transaction = {'id': 'txid'}
|
|
monkeypatch.setattr('bigchaindb.util.get_hash_data', lambda tx: 'txid')
|
|
monkeypatch.setattr(
|
|
'bigchaindb.util.validate_fulfillments', lambda tx: False)
|
|
with pytest.raises(InvalidSignature):
|
|
check_hash_and_signature(transaction)
|