nucypher/tests/unit/test_external_ip_utilities.py

239 lines
8.6 KiB
Python
Raw Normal View History

"""
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/>.
"""
from pathlib import Path
import pytest
from eth_utils import to_checksum_address
from nucypher_core import Address, NodeMetadata, NodeMetadataPayload
2021-12-25 20:56:36 +00:00
from nucypher_core.umbral import SecretKey, Signer
from nucypher.acumen.perception import FleetSensor
from nucypher.characters.lawful import Ursula
from nucypher.crypto.tls import generate_self_signed_certificate
from nucypher.network.exceptions import NodeSeemsToBeDown
2021-04-21 16:04:51 +00:00
from nucypher.network.middleware import NucypherMiddlewareClient
from nucypher.network.nodes import TEACHER_NODES
from nucypher.network.protocols import InterfaceInfo
from nucypher.utilities.networking import (
determine_external_ip_address,
get_external_ip_from_centralized_source,
get_external_ip_from_default_teacher,
get_external_ip_from_known_nodes,
CENTRALIZED_IP_ORACLE_URL,
UnknownIPAddress
)
from tests.constants import MOCK_IP_ADDRESS
2020-10-12 23:09:48 +00:00
MOCK_NETWORK = 'holodeck'
MOCK_PORT = 1111
2020-10-12 23:09:48 +00:00
class Dummy: # Teacher
2020-10-12 23:09:48 +00:00
2021-12-25 20:56:36 +00:00
def __init__(self, canonical_address):
self.canonical_address = canonical_address
self.checksum_address = to_checksum_address(canonical_address)
2020-10-12 23:09:48 +00:00
self.certificate_filepath = None
self.domain = MOCK_NETWORK
class GoodResponse:
status_code = 200
text = MOCK_IP_ADDRESS
class BadResponse:
status_code = 404
text = None
content = 'DUMMY 404'
def mature(self):
2020-10-12 23:09:48 +00:00
return self
def verify_node(self, *args, **kwargs):
pass
def rest_url(self):
return MOCK_IP_ADDRESS
@property
def rest_interface(self):
return InterfaceInfo(host=MOCK_IP_ADDRESS, port=MOCK_PORT)
2021-10-11 23:41:35 +00:00
def metadata(self):
2021-12-25 20:56:36 +00:00
signer = Signer(SecretKey.random())
# A dummy signature with the recovery byte
dummy_signature = bytes(signer.sign(b'whatever')) + b'\x00'
payload = NodeMetadataPayload(staking_provider_address=Address(self.canonical_address),
2021-12-25 20:56:36 +00:00
domain=':dummy:',
timestamp_epoch=0,
operator_signature=dummy_signature,
2021-12-25 20:56:36 +00:00
verifying_key=signer.verifying_key(),
encrypting_key=SecretKey.random().public_key(),
certificate_der=b'not a certificate',
host=MOCK_IP_ADDRESS,
port=MOCK_PORT,
2021-12-25 20:56:36 +00:00
)
return NodeMetadata(signer=signer,
payload=payload)
2020-10-12 23:09:48 +00:00
@pytest.fixture(autouse=True)
def mock_requests(mocker):
"""prevents making live HTTP requests from this module"""
make_request = 'nucypher.utilities.networking._request'
yield mocker.patch(make_request, return_value=None)
@pytest.fixture(autouse=True)
def mock_client(mocker):
cert, pk = generate_self_signed_certificate(host=MOCK_IP_ADDRESS)
mocker.patch.object(NucypherMiddlewareClient, 'get_certificate', return_value=(cert, Path()))
yield mocker.patch.object(NucypherMiddlewareClient, 'invoke_method', return_value=Dummy.GoodResponse)
@pytest.fixture(autouse=True)
2020-10-12 23:09:48 +00:00
def mock_default_teachers(mocker):
teachers = {MOCK_NETWORK: (f"{MOCK_IP_ADDRESS}:{MOCK_PORT}", )}
2021-04-22 10:21:39 +00:00
mocker.patch.dict(TEACHER_NODES, teachers, clear=True)
def test_get_external_ip_from_centralized_source(mock_requests):
get_external_ip_from_centralized_source()
mock_requests.assert_called_once_with(url=CENTRALIZED_IP_ORACLE_URL)
2020-10-12 23:09:48 +00:00
def test_get_external_ip_from_empty_known_nodes(mock_requests):
sensor = FleetSensor(domain=MOCK_NETWORK)
assert len(sensor) == 0
get_external_ip_from_known_nodes(known_nodes=sensor)
# skipped because there are no known nodes
mock_requests.assert_not_called()
2020-10-12 23:09:48 +00:00
def test_get_external_ip_from_known_nodes_with_one_known_node(mock_requests):
sensor = FleetSensor(domain=MOCK_NETWORK)
2021-12-25 20:56:36 +00:00
sensor.record_node(Dummy(b'deadbeefdeadbeefdead'))
2020-10-12 23:09:48 +00:00
sensor.record_fleet_state()
assert len(sensor) == 1
get_external_ip_from_known_nodes(known_nodes=sensor)
# skipped because there are too few known nodes
mock_requests.assert_not_called()
2020-10-12 23:09:48 +00:00
def test_get_external_ip_from_known_nodes(mock_client):
# Setup FleetSensor
2020-10-12 23:09:48 +00:00
sensor = FleetSensor(domain=MOCK_NETWORK)
sample_size = 3
2021-12-25 20:56:36 +00:00
sensor.record_node(Dummy(b'deadbeefdeadbeefdead'))
sensor.record_node(Dummy(b'deadllamadeadllamade'))
sensor.record_node(Dummy(b'deadmousedeadmousede'))
2020-10-12 23:09:48 +00:00
sensor.record_fleet_state()
assert len(sensor) == sample_size
# First sampled node replies
get_external_ip_from_known_nodes(known_nodes=sensor, sample_size=sample_size)
assert mock_client.call_count == 1
mock_client.call_count = 0 # reset
# All sampled nodes dont respond
mock_client.return_value = Dummy.BadResponse
get_external_ip_from_known_nodes(known_nodes=sensor, sample_size=sample_size)
assert mock_client.call_count == sample_size
2020-10-12 23:09:48 +00:00
def test_get_external_ip_from_known_nodes_client(mocker, mock_client):
# Setup FleetSensor
2020-10-12 23:09:48 +00:00
sensor = FleetSensor(domain=MOCK_NETWORK)
sample_size = 3
2021-12-25 20:56:36 +00:00
sensor.record_node(Dummy(b'deadbeefdeadbeefdead'))
sensor.record_node(Dummy(b'deadllamadeadllamade'))
sensor.record_node(Dummy(b'deadmousedeadmousede'))
2020-10-12 23:09:48 +00:00
sensor.record_fleet_state()
assert len(sensor) == sample_size
# Setup HTTP Client
2021-12-25 20:56:36 +00:00
mocker.patch.object(Ursula, 'from_teacher_uri', return_value=Dummy(b'deadporkdeadporkdead'))
2021-04-21 16:04:51 +00:00
teacher_uri = TEACHER_NODES[MOCK_NETWORK][0]
get_external_ip_from_known_nodes(known_nodes=sensor, sample_size=sample_size)
assert mock_client.call_count == 1 # first node responded
function, endpoint = mock_client.call_args[0]
assert function.__name__ == 'get'
assert endpoint == f'https://{teacher_uri}/ping'
2020-10-12 23:09:48 +00:00
def test_get_external_ip_default_teacher_unreachable(mocker):
for error in NodeSeemsToBeDown:
# Default seednode is down
mocker.patch.object(Ursula, 'from_teacher_uri', side_effect=error)
2020-10-12 23:09:48 +00:00
ip = get_external_ip_from_default_teacher(network=MOCK_NETWORK)
assert ip is None
2020-10-12 23:09:48 +00:00
def test_get_external_ip_from_default_teacher(mocker, mock_client, mock_requests):
mock_client.return_value = Dummy.GoodResponse
2021-04-21 16:04:51 +00:00
teacher_uri = TEACHER_NODES[MOCK_NETWORK][0]
2021-12-25 20:56:36 +00:00
mocker.patch.object(Ursula, 'from_teacher_uri', return_value=Dummy(b'deadbeefdeadbeefdead'))
# "Success"
2020-10-12 23:09:48 +00:00
ip = get_external_ip_from_default_teacher(network=MOCK_NETWORK)
assert ip == MOCK_IP_ADDRESS
# Check that the correct endpoint and function is targeted
mock_requests.assert_not_called()
mock_client.assert_called_once()
function, endpoint = mock_client.call_args[0]
assert function.__name__ == 'get'
assert endpoint == f'https://{teacher_uri}/ping'
def test_get_external_ip_default_unknown_network():
unknown_domain = 'thisisnotarealdomain'
# Without fleet sensor
with pytest.raises(UnknownIPAddress):
determine_external_ip_address(network=unknown_domain)
# with fleet sensor
sensor = FleetSensor(domain=unknown_domain)
with pytest.raises(UnknownIPAddress):
determine_external_ip_address(known_nodes=sensor, network=unknown_domain)
2020-10-12 23:09:48 +00:00
def test_get_external_ip_cascade_failure(mocker, mock_requests):
first = mocker.patch('nucypher.utilities.networking.get_external_ip_from_known_nodes', return_value=None)
second = mocker.patch('nucypher.utilities.networking.get_external_ip_from_default_teacher', return_value=None)
third = mocker.patch('nucypher.utilities.networking.get_external_ip_from_centralized_source', return_value=None)
2020-10-12 23:09:48 +00:00
sensor = FleetSensor(domain=MOCK_NETWORK)
2021-12-25 20:56:36 +00:00
sensor.record_node(Dummy(b'deadbeefdeadbeefdead'))
2020-10-12 23:09:48 +00:00
sensor.record_fleet_state()
with pytest.raises(UnknownIPAddress, match='External IP address detection failed'):
2020-10-12 23:09:48 +00:00
determine_external_ip_address(network=MOCK_NETWORK, known_nodes=sensor)
first.assert_called_once()
second.assert_called_once()
third.assert_called_once()