nucypher/tests/cli/test_felix.py

170 lines
6.4 KiB
Python

import os
from unittest import mock
import pytest_twisted
from nucypher.blockchain.eth.registry import LocalContractRegistry
from nucypher.cli.actions import SUCCESSFUL_DESTRUCTION
from twisted.internet import threads
from twisted.internet.task import Clock
from nucypher.blockchain.eth.actors import Staker
from nucypher.blockchain.eth.token import NU
from nucypher.characters.chaotic import Felix
from nucypher.cli.main import nucypher_cli
from nucypher.config.characters import FelixConfiguration
from nucypher.config.constants import NUCYPHER_ENVVAR_KEYRING_PASSWORD
from nucypher.utilities.sandbox.constants import (
TEMPORARY_DOMAIN,
TEST_PROVIDER_URI,
INSECURE_DEVELOPMENT_PASSWORD,
MOCK_CUSTOM_INSTALLATION_PATH_2,
)
@mock.patch('nucypher.config.characters.FelixConfiguration.default_filepath', return_value='/non/existent/file')
def test_missing_configuration_file(default_filepath_mock, click_runner):
cmd_args = ('felix', 'view')
result = click_runner.invoke(nucypher_cli, cmd_args, catch_exceptions=False)
assert result.exit_code != 0
assert default_filepath_mock.called
assert "run: 'nucypher felix init'" in result.output
@pytest_twisted.inlineCallbacks
def test_run_felix(click_runner, testerchain, agency_local_registry, deploy_user_input):
clock = Clock()
Felix._CLOCK = clock
Felix.DISTRIBUTION_INTERVAL = 5 # seconds
Felix.DISBURSEMENT_INTERVAL = 0.01 # hours
Felix.STAGING_DELAY = 2 # seconds
# Main thread (Flask)
os.environ['NUCYPHER_FELIX_DB_SECRET'] = INSECURE_DEVELOPMENT_PASSWORD
# Test subproc (Click)
envvars = {NUCYPHER_ENVVAR_KEYRING_PASSWORD: INSECURE_DEVELOPMENT_PASSWORD,
'NUCYPHER_FELIX_DB_SECRET': INSECURE_DEVELOPMENT_PASSWORD,
'NUCYPHER_WORKER_ETH_PASSWORD': INSECURE_DEVELOPMENT_PASSWORD,
'FLASK_DEBUG': '1'}
# Felix creates a system configuration
init_args = ('felix', 'init',
'--debug',
'--registry-filepath', agency_local_registry.filepath,
'--checksum-address', testerchain.client.accounts[0],
'--config-root', MOCK_CUSTOM_INSTALLATION_PATH_2,
'--network', TEMPORARY_DOMAIN,
'--provider', TEST_PROVIDER_URI)
_original_read_function = LocalContractRegistry.read
result = click_runner.invoke(nucypher_cli, init_args, catch_exceptions=False, env=envvars)
assert result.exit_code == 0
configuration_file_location = os.path.join(MOCK_CUSTOM_INSTALLATION_PATH_2, FelixConfiguration.generate_filename())
# Felix Creates a Database
db_args = ('felix', 'createdb',
'--debug',
'--config-file', configuration_file_location,
'--provider', TEST_PROVIDER_URI)
result = click_runner.invoke(nucypher_cli, db_args, catch_exceptions=False, env=envvars)
assert result.exit_code == 0
# Felix Runs Web Services
def run_felix():
args = ('felix', 'run',
'--debug',
'--config-file', configuration_file_location,
'--provider', TEST_PROVIDER_URI,
'--dry-run')
run_result = click_runner.invoke(nucypher_cli, args, catch_exceptions=False, env=envvars)
assert run_result.exit_code == 0
return run_result
# A (mocked) client requests Felix's services
def request_felix_landing_page(_result):
# Init an equal Felix to the already running one.
felix_config = FelixConfiguration.from_configuration_file(filepath=configuration_file_location,
registry_filepath=agency_local_registry.filepath)
felix_config.attach_keyring()
felix_config.keyring.unlock(password=INSECURE_DEVELOPMENT_PASSWORD)
felix = felix_config.produce()
# Make a flask app
web_app = felix.make_web_app()
test_client = web_app.test_client()
# Register a new recipient
response = test_client.post('/register', data={'address': testerchain.client.accounts[-1]})
assert response.status_code == 200
return
def time_travel(_result):
clock.advance(amount=60)
# Record starting ether balance
recipient = testerchain.client.accounts[-1]
staker = Staker(checksum_address=recipient,
registry=agency_local_registry,
is_me=True)
original_eth_balance = staker.eth_balance
# Run the callbacks
d = threads.deferToThread(run_felix)
d.addCallback(request_felix_landing_page)
d.addCallback(time_travel)
yield d
def confirm_airdrop(_results):
recipient = testerchain.client.accounts[-1]
staker = Staker(checksum_address=recipient,
registry=agency_local_registry,
is_me=True)
assert staker.token_balance == NU(45000, 'NU')
# TODO: Airdrop Testnet Ethers?
new_eth_balance = original_eth_balance + testerchain.w3.fromWei(Felix.ETHER_AIRDROP_AMOUNT, 'ether')
assert staker.eth_balance == new_eth_balance
staged_airdrops = Felix._AIRDROP_QUEUE
next_airdrop = staged_airdrops[0]
next_airdrop.addCallback(confirm_airdrop)
yield next_airdrop
# Felix view
view_args = ('felix', 'view',
'--config-file', configuration_file_location,
'--provider', TEST_PROVIDER_URI)
result = click_runner.invoke(nucypher_cli, view_args, catch_exceptions=False, env=envvars)
assert result.exit_code == 0
assert "Address" in result.output
assert "NU" in result.output
assert "ETH" in result.output
# Felix accounts
accounts_args = ('felix', 'accounts',
'--config-file', configuration_file_location,
'--provider', TEST_PROVIDER_URI)
result = click_runner.invoke(nucypher_cli, accounts_args, catch_exceptions=False, env=envvars)
assert result.exit_code == 0
assert testerchain.client.accounts[-1] in result.output
# Felix destroy
destroy_args = ('felix', 'destroy',
'--config-file', configuration_file_location,
'--provider', TEST_PROVIDER_URI,
'--force')
result = click_runner.invoke(nucypher_cli, destroy_args, catch_exceptions=False, env=envvars)
assert result.exit_code == 0
assert SUCCESSFUL_DESTRUCTION in result.output
assert not os.path.exists(configuration_file_location), "Felix configuration file was deleted"