nucypher/tests/integration/porter/test_porter_specifications.py

205 lines
7.8 KiB
Python

"""
This file is part of nucypher.
nucypher is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
nucypher is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with nucypher. If not, see <https://www.gnu.org/licenses/>.
"""
import random
import pytest
from nucypher_core.umbral import SecretKey
from nucypher.characters.control.specifications.fields import Key
from nucypher.control.specifications.exceptions import InvalidArgumentCombo, InvalidInputData
from nucypher.utilities.porter.control.specifications.fields import UrsulaInfoSchema, RetrievalResultSchema
from nucypher.utilities.porter.control.specifications.porter_schema import (
AliceGetUrsulas,
BobRetrieveCFrags
)
from nucypher.utilities.porter.porter import Porter
from tests.utils.policy import retrieval_request_setup
def test_alice_get_ursulas_schema(get_random_checksum_address):
#
# Input i.e. load
#
# no args
with pytest.raises(InvalidInputData):
AliceGetUrsulas().load({})
quantity = 10
required_data = {
'quantity': quantity,
}
# required args
AliceGetUrsulas().load(required_data)
# missing required args
updated_data = {k: v for k, v in required_data.items() if k != 'quantity'}
with pytest.raises(InvalidInputData):
AliceGetUrsulas().load(updated_data)
# optional components
# only exclude
updated_data = dict(required_data)
exclude_ursulas = []
for i in range(2):
exclude_ursulas.append(get_random_checksum_address())
updated_data['exclude_ursulas'] = exclude_ursulas
AliceGetUrsulas().load(updated_data)
# only include
updated_data = dict(required_data)
include_ursulas = []
for i in range(3):
include_ursulas.append(get_random_checksum_address())
updated_data['include_ursulas'] = include_ursulas
AliceGetUrsulas().load(updated_data)
# both exclude and include
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = exclude_ursulas
updated_data['include_ursulas'] = include_ursulas
AliceGetUrsulas().load(updated_data)
# list input formatted as ',' separated strings
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = ','.join(exclude_ursulas)
updated_data['include_ursulas'] = ','.join(include_ursulas)
data = AliceGetUrsulas().load(updated_data)
assert data['exclude_ursulas'] == exclude_ursulas
assert data['include_ursulas'] == include_ursulas
# single value as string cast to list
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = exclude_ursulas[0]
updated_data['include_ursulas'] = include_ursulas[0]
data = AliceGetUrsulas().load(updated_data)
assert data['exclude_ursulas'] == [exclude_ursulas[0]]
assert data['include_ursulas'] == [include_ursulas[0]]
# invalid include entry
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = exclude_ursulas
updated_data['include_ursulas'] = list(include_ursulas) # make copy to modify
updated_data['include_ursulas'].append("0xdeadbeef")
with pytest.raises(InvalidInputData):
AliceGetUrsulas().load(updated_data)
# invalid exclude entry
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = list(exclude_ursulas) # make copy to modify
updated_data['exclude_ursulas'].append("0xdeadbeef")
updated_data['include_ursulas'] = include_ursulas
with pytest.raises(InvalidInputData):
AliceGetUrsulas().load(updated_data)
# too many ursulas to include
updated_data = dict(required_data)
too_many_ursulas_to_include = []
while len(too_many_ursulas_to_include) <= quantity:
too_many_ursulas_to_include.append(get_random_checksum_address())
updated_data['include_ursulas'] = too_many_ursulas_to_include
with pytest.raises(InvalidArgumentCombo):
# number of ursulas to include exceeds quantity to sample
AliceGetUrsulas().load(updated_data)
# include and exclude addresses are not mutually exclusive - include has common entry
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = exclude_ursulas
updated_data['include_ursulas'] = list(include_ursulas) # make copy to modify
updated_data['include_ursulas'].append(exclude_ursulas[0]) # one address that overlaps
with pytest.raises(InvalidArgumentCombo):
# 1 address in both include and exclude lists
AliceGetUrsulas().load(updated_data)
# include and exclude addresses are not mutually exclusive - exclude has common entry
updated_data = dict(required_data)
updated_data['exclude_ursulas'] = list(exclude_ursulas) # make copy to modify
updated_data['exclude_ursulas'].append(include_ursulas[0]) # on address that overlaps
updated_data['include_ursulas'] = include_ursulas
with pytest.raises(InvalidArgumentCombo):
# 1 address in both include and exclude lists
AliceGetUrsulas().load(updated_data)
#
# Output i.e. dump
#
ursulas_info = []
expected_ursulas_info = []
port = 11500
for i in range(3):
ursula_info = Porter.UrsulaInfo(get_random_checksum_address(),
f"https://127.0.0.1:{port+i}",
SecretKey.random().public_key())
ursulas_info.append(ursula_info)
# use schema to determine expected output (encrypting key gets changed to hex)
expected_ursulas_info.append(UrsulaInfoSchema().dump(ursula_info))
output = AliceGetUrsulas().dump(obj={'ursulas': ursulas_info})
assert output == {"ursulas": expected_ursulas_info}
def test_alice_revoke():
pass # TODO
def test_bob_retrieve_cfrags(federated_porter,
enacted_federated_policy,
federated_bob,
federated_alice):
bob_retrieve_cfrags_schema = BobRetrieveCFrags()
# no args
with pytest.raises(InvalidInputData):
bob_retrieve_cfrags_schema.load({})
# Setup
retrieval_args, _ = retrieval_request_setup(enacted_federated_policy,
federated_bob,
federated_alice,
encode_for_rest=True)
bob_retrieve_cfrags_schema.load(retrieval_args)
# missing required argument
updated_data = dict(retrieval_args)
key_to_remove = random.choice(list(updated_data.keys()))
del updated_data[key_to_remove]
with pytest.raises(InvalidInputData):
# missing arg
bob_retrieve_cfrags_schema.load(updated_data)
#
# Output i.e. dump
#
non_encoded_retrieval_args, _ = retrieval_request_setup(enacted_federated_policy,
federated_bob,
federated_alice,
encode_for_rest=False)
retrieval_results = federated_porter.retrieve_cfrags(**non_encoded_retrieval_args)
expected_retrieval_results_json = []
retrieval_result_schema = RetrievalResultSchema()
for result in retrieval_results:
data = retrieval_result_schema.dump(result)
expected_retrieval_results_json.append(data)
output = bob_retrieve_cfrags_schema.dump(obj={'retrieval_results': retrieval_results})
assert output == {"retrieval_results": expected_retrieval_results_json}