Skip to content

Commit

Permalink
feat: use provider in contract error
Browse files Browse the repository at this point in the history
  • Loading branch information
antazoey committed Nov 22, 2024
1 parent 9cc65bb commit 03ea7c2
Show file tree
Hide file tree
Showing 5 changed files with 39 additions and 34 deletions.
6 changes: 1 addition & 5 deletions src/ape/contracts/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -257,11 +257,7 @@ def decode_input(self, calldata: bytes) -> tuple[str, dict[str, Any]]:

def _validate_is_contract(self):
if not self.contract.is_contract:
raise ContractNotFoundError(
self.contract.address,
self.provider.network.explorer is not None,
self.provider.network_choice,
)
raise ContractNotFoundError(self.contract.address, provider=self.provider)


class ContractCallHandler(ContractMethodHandler):
Expand Down
36 changes: 23 additions & 13 deletions src/ape/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@
from ethpm_types.contract_type import ContractType

from ape.api.networks import NetworkAPI
from ape.api.providers import SubprocessProvider
from ape.api.providers import ProviderAPI, SubprocessProvider
from ape.api.trace import TraceAPI
from ape.api.transactions import ReceiptAPI, TransactionAPI
from ape.managers.project import ProjectManager
Expand Down Expand Up @@ -590,29 +590,39 @@ class ContractNotFoundError(ChainError):
Raised when a contract is not found at an address.
"""

# TODO: In 0.9, pass in provider object directly (instead of network choice + name)
def __init__(self, address: "AddressType", has_explorer: bool, network_choice: str):
def __init__(self, address: "AddressType", provider: Optional["ProviderAPI"] = None):
try:
msg = self._create_message(address, provider=provider)
except Exception as err:
# Don't let errors occurring within exception handling to
# ruin the exception completely.
logger.error(f"Failed to create proper error message because of: {err}")
msg = f"Failed to get contract type for address '{address}'."

super().__init__(msg)

@classmethod
def _create_message(
cls, address: "AddressType", provider: Optional["ProviderAPI"] = None
) -> str:
msg = f"Failed to get contract type for address '{address}'."
if not provider:
return msg

# NOTE: Network name is optional to avoid breaking change.
choice_parts = network_choice.split(":")
if len(choice_parts) > 1:
network_name = network_choice.split(":")[1]
else:
network_name = network_choice
network_name = provider.network_choice.split(":")[1]

if has_explorer:
msg += " Contract may need verification."
if provider.network.explorer:
msg += " Contract may need verification (if relying on an explorer)."
elif network_name != "local":
# Only bother mentioning explorer plugins if we are not the local network.
msg += (
f" Current network '{network_choice}' has no associated "
f" Current network '{provider.network_choice}' has no associated "
"explorer plugin. Try installing an explorer plugin using "
f"{click.style(text='ape plugins install etherscan', fg='green')}, "
"or using a network with explorer support."
)

super().__init__(msg)
return msg


class UnknownSnapshotError(ChainError):
Expand Down
10 changes: 2 additions & 8 deletions src/ape/managers/chain.py
Original file line number Diff line number Diff line change
Expand Up @@ -966,9 +966,7 @@ def __getitem__(self, address: AddressType) -> ContractType:
contract_type = self.get(address)
if not contract_type:
# Create error message from custom exception cls.
err = ContractNotFoundError(
address, self.provider.network.explorer is not None, self.provider.network_choice
)
err = ContractNotFoundError(address, provider=self.provider)
# Must raise KeyError.
raise KeyError(str(err))

Expand Down Expand Up @@ -1223,11 +1221,7 @@ def instance_at(
)

if not contract_type:
raise ContractNotFoundError(
contract_address,
self.provider.network.explorer is not None,
self.provider.network_choice,
)
raise ContractNotFoundError(contract_address, provider=self.provider)

elif not isinstance(contract_type, ContractType):
raise TypeError(
Expand Down
5 changes: 3 additions & 2 deletions src/ape_ethereum/provider.py
Original file line number Diff line number Diff line change
Expand Up @@ -1580,8 +1580,9 @@ def ots_get_contract_creator(self, address: "AddressType") -> Optional[dict]:

result = self.make_request("ots_getContractCreator", [address])
if result is None:
# NOTE: Skip the explorer part of the error message via `has_explorer=True`.
raise ContractNotFoundError(address, True, self.network_choice)
# Don't pass provider so the error message is simplifer in this case
# (avoids mentioning explorer plugins).
raise ContractNotFoundError(address)

return result

Expand Down
16 changes: 10 additions & 6 deletions tests/functional/test_exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
handle_ape_exception,
)
from ape.types.trace import SourceTraceback
from ape.utils.misc import LOCAL_NETWORK_NAME, ZERO_ADDRESS
from ape.utils.misc import ZERO_ADDRESS
from ape_ethereum.transactions import DynamicFeeTransaction, Receipt


Expand Down Expand Up @@ -229,20 +229,24 @@ def revert_type(self) -> Optional[str]:


class TestContractNotFoundError:
def test_local_network(self):
def test_local_network(self, eth_tester_provider):
"""
Testing we are NOT mentioning explorer plugins
for the local-network, as 99.9% of the time it is
confusing.
"""
err = ContractNotFoundError(ZERO_ADDRESS, False, f"ethereum:{LOCAL_NETWORK_NAME}:test")
err = ContractNotFoundError(ZERO_ADDRESS, provider=eth_tester_provider)
assert str(err) == f"Failed to get contract type for address '{ZERO_ADDRESS}'."

def test_fork_network(self):
err = ContractNotFoundError(ZERO_ADDRESS, False, "ethereum:sepolia-fork:test")
def test_fork_network(self, mocker, mock_sepolia):
provider = mocker.MagicMock()
provider.network = mock_sepolia
mock_sepolia.explorer = None
provider.network_choice = "ethereum:sepolia:node"
err = ContractNotFoundError(ZERO_ADDRESS, provider=provider)
assert str(err) == (
f"Failed to get contract type for address '{ZERO_ADDRESS}'. "
"Current network 'ethereum:sepolia-fork:test' has no associated explorer plugin. "
"Current network 'ethereum:sepolia:node' has no associated explorer plugin. "
"Try installing an explorer plugin using \x1b[32mape plugins install etherscan"
"\x1b[0m, or using a network with explorer support."
)

0 comments on commit 03ea7c2

Please sign in to comment.