2019-03-05 20:08:42 +00:00
|
|
|
import os
|
|
|
|
|
|
|
|
import pytest_twisted
|
|
|
|
from twisted.internet import threads
|
2019-03-21 22:51:51 +00:00
|
|
|
from twisted.internet.task import Clock
|
2019-03-05 20:08:42 +00:00
|
|
|
|
2019-03-21 22:51:51 +00:00
|
|
|
from nucypher.blockchain.eth.actors import Miner
|
|
|
|
from nucypher.blockchain.eth.token import NU
|
2019-03-21 21:01:30 +00:00
|
|
|
from nucypher.characters.chaotic import Felix
|
2019-03-07 07:43:57 +00:00
|
|
|
from nucypher.cli import deploy
|
2019-03-05 20:08:42 +00:00
|
|
|
from nucypher.cli.main import nucypher_cli
|
|
|
|
from nucypher.config.characters import FelixConfiguration
|
|
|
|
from nucypher.utilities.sandbox.constants import (
|
|
|
|
TEMPORARY_DOMAIN,
|
|
|
|
TEST_PROVIDER_URI,
|
|
|
|
INSECURE_DEVELOPMENT_PASSWORD,
|
|
|
|
MOCK_CUSTOM_INSTALLATION_PATH_2
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest_twisted.inlineCallbacks
|
2019-03-21 21:01:56 +00:00
|
|
|
def test_run_felix(click_runner, testerchain, federated_ursulas, mock_primary_registry_filepath):
|
|
|
|
|
2019-03-21 22:51:51 +00:00
|
|
|
clock = Clock()
|
|
|
|
Felix._CLOCK = clock
|
2019-03-21 21:01:56 +00:00
|
|
|
Felix.DISTRIBUTION_INTERVAL = 5 # seconds
|
|
|
|
Felix.DISBURSEMENT_INTERVAL = 0.01 # hours
|
2019-03-21 22:51:51 +00:00
|
|
|
Felix.STAGING_DELAY = 2 # seconds
|
2019-03-05 20:08:42 +00:00
|
|
|
|
2019-03-06 18:15:19 +00:00
|
|
|
# Main thread (Flask)
|
|
|
|
os.environ['NUCYPHER_FELIX_DB_SECRET'] = INSECURE_DEVELOPMENT_PASSWORD
|
|
|
|
|
|
|
|
# Test subproc (Click)
|
|
|
|
envvars = {'NUCYPHER_KEYRING_PASSWORD': INSECURE_DEVELOPMENT_PASSWORD,
|
2019-03-21 21:01:56 +00:00
|
|
|
'NUCYPHER_FELIX_DB_SECRET': INSECURE_DEVELOPMENT_PASSWORD,
|
|
|
|
'FLASK_DEBUG': '1'}
|
2019-03-06 18:15:19 +00:00
|
|
|
|
2019-03-07 07:43:57 +00:00
|
|
|
# Deploy contracts
|
2019-04-19 07:17:25 +00:00
|
|
|
deploy_args = ('deploy',
|
2019-03-07 07:43:57 +00:00
|
|
|
'--registry-outfile', mock_primary_registry_filepath,
|
|
|
|
'--provider-uri', TEST_PROVIDER_URI,
|
|
|
|
'--poa')
|
|
|
|
|
2019-03-20 04:22:23 +00:00
|
|
|
user_input = 'Y\n'+f'{INSECURE_DEVELOPMENT_PASSWORD}\n'*8 # TODO: Use Env Vars
|
2019-03-07 07:43:57 +00:00
|
|
|
result = click_runner.invoke(deploy.deploy, deploy_args, input=user_input, catch_exceptions=False, env=envvars)
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2019-03-05 22:01:38 +00:00
|
|
|
# Felix creates a system configuration
|
2019-03-05 21:50:14 +00:00
|
|
|
init_args = ('felix', 'init',
|
2019-03-21 21:01:30 +00:00
|
|
|
'--checksum-address', testerchain.interface.w3.eth.accounts[0],
|
2019-03-05 21:50:14 +00:00
|
|
|
'--config-root', MOCK_CUSTOM_INSTALLATION_PATH_2,
|
|
|
|
'--network', TEMPORARY_DOMAIN,
|
|
|
|
'--no-registry',
|
|
|
|
'--provider-uri', TEST_PROVIDER_URI)
|
2019-03-05 20:08:42 +00:00
|
|
|
|
2019-03-06 18:15:19 +00:00
|
|
|
result = click_runner.invoke(nucypher_cli, init_args, catch_exceptions=False, env=envvars)
|
2019-03-05 20:08:42 +00:00
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
configuration_file_location = os.path.join(MOCK_CUSTOM_INSTALLATION_PATH_2, 'felix.config')
|
|
|
|
|
2019-03-05 22:01:38 +00:00
|
|
|
# Felix Creates a Database
|
|
|
|
db_args = ('felix', 'createdb',
|
|
|
|
'--config-file', configuration_file_location,
|
|
|
|
'--provider-uri', TEST_PROVIDER_URI)
|
|
|
|
|
2019-03-06 18:15:19 +00:00
|
|
|
result = click_runner.invoke(nucypher_cli, db_args, catch_exceptions=False, env=envvars)
|
2019-03-05 22:01:38 +00:00
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Felix Runs Web Services
|
2019-03-05 20:08:42 +00:00
|
|
|
def run_felix():
|
2019-03-21 21:01:30 +00:00
|
|
|
args = ('--debug',
|
|
|
|
'felix', 'run',
|
2019-03-05 20:08:42 +00:00
|
|
|
'--config-file', configuration_file_location,
|
|
|
|
'--provider-uri', TEST_PROVIDER_URI,
|
|
|
|
'--dry-run',
|
|
|
|
'--no-registry')
|
|
|
|
|
2019-03-06 18:15:19 +00:00
|
|
|
result = click_runner.invoke(nucypher_cli, args, catch_exceptions=False, env=envvars)
|
2019-03-05 20:08:42 +00:00
|
|
|
assert result.exit_code == 0
|
|
|
|
return result
|
|
|
|
|
2019-03-06 07:16:49 +00:00
|
|
|
# A (mocked) client requests Felix's services
|
2019-03-06 18:15:19 +00:00
|
|
|
def request_felix_landing_page(_result):
|
2019-03-05 20:08:42 +00:00
|
|
|
|
|
|
|
# Init an equal Felix to the already running one.
|
|
|
|
felix_config = FelixConfiguration.from_configuration_file(filepath=configuration_file_location)
|
|
|
|
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()
|
|
|
|
|
2019-03-05 22:01:38 +00:00
|
|
|
# Load the landing page
|
2019-03-05 20:08:42 +00:00
|
|
|
response = test_client.get('/')
|
2019-03-05 21:50:14 +00:00
|
|
|
assert response.status_code == 200
|
|
|
|
|
2019-03-05 22:01:38 +00:00
|
|
|
# Register a new recipient
|
2019-03-21 22:51:51 +00:00
|
|
|
response = test_client.post('/register', data={'address': felix.blockchain.interface.w3.eth.accounts[-1]})
|
2019-03-05 21:50:14 +00:00
|
|
|
assert response.status_code == 200
|
2019-03-05 20:08:42 +00:00
|
|
|
|
2019-03-06 07:16:49 +00:00
|
|
|
return
|
|
|
|
|
2019-03-21 22:51:51 +00:00
|
|
|
def time_travel(_result):
|
|
|
|
clock.advance(amount=60)
|
|
|
|
|
2019-03-05 22:01:38 +00:00
|
|
|
# Run the callbacks
|
2019-03-05 20:08:42 +00:00
|
|
|
d = threads.deferToThread(run_felix)
|
|
|
|
d.addCallback(request_felix_landing_page)
|
2019-03-21 22:51:51 +00:00
|
|
|
d.addCallback(time_travel)
|
|
|
|
|
2019-03-05 20:08:42 +00:00
|
|
|
yield d
|
2019-03-21 22:51:51 +00:00
|
|
|
|
|
|
|
def confirm_airdrop(_results):
|
|
|
|
recipient = testerchain.interface.w3.eth.accounts[-1]
|
|
|
|
miner = Miner(checksum_address=recipient,
|
|
|
|
blockchain=testerchain,
|
|
|
|
is_me=True)
|
|
|
|
|
|
|
|
assert miner.token_balance == NU(15000, 'NU')
|
|
|
|
|
|
|
|
staged_airdrops = Felix._AIRDROP_QUEUE
|
|
|
|
next_airdrop = staged_airdrops[0]
|
|
|
|
next_airdrop.addCallback(confirm_airdrop)
|
|
|
|
yield next_airdrop
|