nucypher/tests/utils/ape.py

218 lines
6.9 KiB
Python

import json
from copy import deepcopy
from pathlib import Path
from typing import Dict, List, Tuple, Union
from ape import config as ape_config
from ape import project
from ape.api import AccountAPI, DependencyAPI
from ape.contracts.base import ContractInstance
from ape_test.accounts import TestAccount
from eth_typing import ChecksumAddress
from eth_utils import to_checksum_address
from nucypher.blockchain.eth.agents import (
CoordinatorAgent,
NucypherTokenAgent,
SubscriptionManagerAgent,
TACoApplicationAgent,
)
from nucypher.blockchain.eth.registry import InMemoryContractRegistry
from tests.constants import (
CONDITION_NFT,
GLOBAL_ALLOW_LIST,
MOCK_STAKING_CONTRACT_NAME,
RITUAL_TOKEN,
STAKE_INFO,
T_TOKEN,
)
# order sensitive
_CONTRACTS_TO_DEPLOY_ON_TESTERCHAIN = (
RITUAL_TOKEN,
T_TOKEN,
NucypherTokenAgent.contract_name,
STAKE_INFO,
MOCK_STAKING_CONTRACT_NAME,
TACoApplicationAgent.contract_name,
SubscriptionManagerAgent.contract_name,
CoordinatorAgent.contract_name,
GLOBAL_ALLOW_LIST,
CONDITION_NFT,
)
VARIABLE_PREFIX_SYMBOL = "<"
VARIABLE_SUFFIX_SYMBOL = ">"
def get_ape_project_build_path(project) -> Path:
build_path = Path(project.path) / '.build'
return build_path
def _is_variable(param: Union[str, int, List[Union[str, int]]]) -> bool:
"""Check if param is a ape-config variable"""
return isinstance(param, str) and (
param.startswith(VARIABLE_PREFIX_SYMBOL)
and param.endswith(VARIABLE_SUFFIX_SYMBOL)
)
def _resolve_variable(
param: str,
contract_name: str,
deployments: Dict[str, ContractInstance],
accounts: List[TestAccount],
) -> Union[ChecksumAddress, str, int]:
"""Resolve a ape-config variable to a literal"""
dependency_expression = param.strip(VARIABLE_PREFIX_SYMBOL).strip(
VARIABLE_SUFFIX_SYMBOL
)
dependency_name, attribute_name = dependency_expression.split(".")
if dependency_name == "address":
try:
account = accounts[int(attribute_name)]
except ValueError:
raise ValueError(
f"Ape account must be accessed by an index; got '{attribute_name}'."
)
address = ChecksumAddress(account.address)
return address
try:
param = getattr(deployments[dependency_name], attribute_name)
except KeyError:
raise ValueError(f"Contract {contract_name} not found in deployments")
except AttributeError:
raise ValueError(f"Attribute {attribute_name} not found in {dependency_name}")
return param
def process_deployment_params(
contract_name: str,
params: Dict[str, Union[str, int, list]],
deployments: Dict[str, ContractInstance],
accounts: List[TestAccount],
) -> Dict[str, Union[ChecksumAddress, str, int]]:
"""Process deployment params for a contract."""
processed_params = dict()
for param_name, param_value in params.items():
if _is_variable(param_value):
param_value = _resolve_variable(
param=param_value,
contract_name=contract_name,
deployments=deployments,
accounts=accounts,
)
processed_params[param_name] = param_value
continue
elif isinstance(param_value, list):
value_list = list()
for param in param_value:
if _is_variable(param):
param = _resolve_variable(
param=param,
contract_name=contract_name,
deployments=deployments,
accounts=accounts,
)
value_list.append(param)
processed_params[param_name] = value_list
continue
else:
# this parameter is a literal
processed_params[param_name] = param_value
continue
return processed_params
def get_deployment_params(
contract_name: str,
config: Dict[str, Union[str, list]],
accounts: List[TestAccount],
deployments: Dict[str, ContractInstance],
) -> Tuple[Dict, AccountAPI]:
"""
Get deployment params for a contract.
"""
config = deepcopy(config)
while config:
params = config.pop()
deployer_address = accounts[params.pop("address")]
name = params.pop("contract_type")
if name == contract_name:
params = process_deployment_params(
contract_name=contract_name,
params=params,
deployments=deployments,
accounts=accounts,
)
return params, deployer_address
else:
# there are no deployment params for this contract; default to account at index 0
return dict(), accounts[0]
def deploy_contracts(
nucypher_contracts: DependencyAPI,
test_contracts: DependencyAPI,
accounts: List[TestAccount],
):
"""Deploy contracts o via ape's API for testing."""
config = ape_config.get_config("deployments")["ethereum"]["local"]
deployments = dict()
for name in _CONTRACTS_TO_DEPLOY_ON_TESTERCHAIN:
params, deployer_account = get_deployment_params(
name, deployments=deployments, config=config, accounts=accounts
)
try:
# this contract is a dependency
contract = getattr(nucypher_contracts, name)
except AttributeError:
# this contract is local to this project
try:
contract = getattr(project, name)
except AttributeError:
raise ValueError(
f"Contract {name} not found in project or in dependencies."
)
deployed_contract = deployer_account.deploy(contract, *params.values())
deployments[name] = deployed_contract
return deployments
def registry_from_ape_deployments(
nucypher_contracts: DependencyAPI, deployments: Dict[str, ContractInstance]
) -> InMemoryContractRegistry:
"""Creates a registry from ape deployments."""
local_contracts = project.contracts
# Get the raw abi from the cached dependency manifest
dependency_manifest = json.loads(nucypher_contracts.cached_manifest.json())
combined_contract_data = dependency_manifest["contractTypes"]
# Add the local contract ABIs to the data
for contract_name, local_contract_data in local_contracts.items():
contract_manifest = json.loads(local_contract_data.json())
contract_abi = contract_manifest["abi"]
combined_contract_data[contract_name] = {"abi": contract_abi}
data = list()
for contract_name, deployment in deployments.items():
abi = combined_contract_data[contract_name]["abi"]
entry = [
contract_name,
'v0.0.0', # TODO: get version from contract
to_checksum_address(deployment.address),
abi
]
data.append(entry)
registry = InMemoryContractRegistry()
registry.write(data)
return registry