Testing

Testing an ape project is important and easy.

Test Structure

Tests must be located in a project’s tests/ directory. Each test file must start with test_ and have the .py extension, such as test_my_contract.py. Each test method within the file must also start with test_. The following is an example test:

def test_add():
    assert 1 + 1 == 2

NOTE: pytest assumes the actual value is on the left and the expected value is on the right.

Test Pattern

Tests are generally divisible into three parts:

  1. Set-up

  2. Invocation

  3. Assertion

In the example above, we created a fixture that deploys our smart-contract. This is an example of a ‘setup’ phase. Next, we need to call a method on our contract. Let’s assume there is an authorized_method() that requires the owner of the contract to make the transaction. If the sender of the transaction is not the owner, the transaction will fail to complete and will revert.

This is an example of how that test may look:

def test_authorization(my_contract, owner, not_owner):
    my_contract.set_owner(sender=owner)
    assert owner == my_contract.owner()

    with ape.reverts("!authorized"):
        my_contract.authorized_method(sender=not_owner)

Note

Ape has built-in test and fixture isolation for all pytest scopes. To disable isolation add the --disable-isolation flag when running ape test

Fixtures

Fixtures are any type of reusable instances of something with configurable scopes. pytest handles passing fixtures into each test method as test-time. To learn more about fixtures

Define fixtures for static data used by tests. This data can be accessed by all tests in the suite unless specified otherwise. This could be data as well as helpers of modules which will be passed to all tests.

A common place to define fixtures are in the conftest.py which should be saved under the test directory:

conftest.py is used to import external plugins or modules. By defining the following global variable, pytest will load the module and make it available for its test.

You can define your own fixtures or use existing ones. The ape-test plugin comes with fixtures you will likely want to use:

accounts fixture

You have access to test accounts. These accounts are automatically funded, and you can use them to transact in your tests. Access each test account by index from the accounts fixture:

def test_my_method(accounts):
    owner = accounts[0]
    receiver = accounts[1]

For code readability and sustainability, create your own fixtures using the accounts fixture:

import pytest

@pytest.fixture
def owner(accounts):
    return accounts[0]


@pytest.fixture
def receiver(accounts):
    return accounts[1]


def test_my_method(owner, receiver):
    ...

You can configure your accounts by changing the mnemonic or number_of_accounts settings in the test section of your ape-config.yaml file:

test:
  mnemonic: test test test test test test test test test test test junk
  number_of_accounts: 5

If you are using a fork-provider, such as Hardhat, you can use impersonated accounts by accessing random addresses off the fixture:

@pytest.fixture
def vitalik(accounts):
    return accounts["0xab5801a7d398351b8be11c439e05c5b3259aec9b"]

Using a fork-provider such as Hardhat, when using a contract instance as the sender in a transaction, it will be automatically impersonated:

def test_my_method(project, accounts):
    contract = project.MyContract.deploy(sender=accounts[0])
    other_contract = project.OtherContract.deploy(sender=accounts[0])
    contract.my_method(sender=other_contract)

It has the same interface as the TestAccountManager, (same as doing accounts.test_accounts in a script or the console).

chain fixture

Use the chain fixture to access the connected provider or adjust blockchain settings.

For example, increase the pending timestamp:

def test_in_future(chain):
    chain.pending_timestamp += 86000
    assert "Something"
    chain.pending_timestamp += 86000
    assert "Something else"

It has the same interface as the ChainManager.

networks fixture

Use the networks fixture to change the active provider in tests.

def test_multi_chain(networks):
    assert "Something"  # Make assertion in root network

    # NOTE: Assume have ecosystem named "foo" with network "local" and provider "bar"
    with networks.foo.local.use_provider("bar"):
        assert "Something else"

It has the same interface as the NetworkManager.

project fixture

You also have access to the project you are testing. You will need this to deploy your contracts in your tests.

import pytest


@pytest.fixture
def owner(accounts):
    return accounts[0]


@pytest.fixture
def my_contract(project, owner):
    #           ^ use the 'project' fixture from the 'ape-test' plugin
    return owner.deploy(project.MyContract)

It has the same interface as the ProjectManager.

Ape testing commands

ape test

To run a particular test:

ape test test_my_contract

Use ape test -I to open the interactive mode at the point of exception. This allows the user to inspect the point of failure in your tests.

ape test test_my_contract -I -s

Test Providers

Out-of-the-box, your tests run using the eth-tester provider, which comes bundled with ape. If you have geth installed, you can use the ape-geth plugin that also comes with ape.

ape test --network ethereum:local:geth

Each testing plugin should work the same way. You will have access to the same test accounts.

Another option for testing providers is the ape-hardhat plugin, which does not come with ape but can be installed by including it in the plugins list in your ape-config.yaml file or manually installing it using the command:

ape plugins install hardhat

Advanced Testing Tips

If you want to use sample projects, follow this link to Ape Academy.

project                     # The root project directory
└── tests/                  # Project tests folder, ran using the 'ape test' command to run all tests within the folder.
    └── conftest.py         # A file to define global variable for testing
    └── test_accounts.py    # A test file, if you want to ONLY run one test file you can use 'ape test test_accounts.py' command
    └── test_mint.py        # A test file

Here is an example of a test function from a sample NFT project

def test_account_balance(project, owner, receiver, nft):
    quantity = 1
    nft.mint(receiver, quantity, ["0"], value=nft.PRICE() * quantity, sender=owner)
    actual = project.balanceOf(receiver)
    expect = quantity
    assert actual == expect

Testing Transaction Failures

Similar to pytest.raises(), you can use ape.reverts() to assert that contract transactions fail and revert.

From our earlier example we can see this in action:

def test_authorization(my_contract, owner, not_owner):
    my_contract.set_owner(sender=owner)
    assert owner == my_contract.owner()

    with ape.reverts("!authorized"):
        my_contract.authorized_method(sender=not_owner)

reverts() takes two optional parameters:

expected_message

This is the expected revert reason given when the transaction fails. If the message in the ContractLogicError raised by the transaction failure is empty or does not match the expected_message, then ape.reverts() will raise an AssertionError.

You may also supply an re.Pattern object to assert on a message pattern, rather than on an exact match.

# Matches explicitly "foo" or "bar"
with ape.reverts(re.compile(r"^(foo|bar)$")):
    ...

dev_message

This is the expected dev message corresponding to the line in the contract’s source code where the error occurred. These can be helpful in optimizing for gas usage and keeping revert reason strings shorter.

Dev messages take the form of a comment in Vyper, and should be placed on the line that may cause a transaction revert:

assert x != 0  # dev: invalid value

Take for example:

# @version 0.3.7

@external
def check_value(_value: uint256) -> bool:
    assert _value != 0  # dev: invalid value
    return True

We can explicitly cause a transaction revert and check the failed line by supplying an expected dev_message:

def test_authorization(my_contract, owner):
    with ape.reverts(dev_message="dev: invalid value"):
        my_contract.check_value(sender=owner)

When the transaction reverts and ContractLogicError is raised, ape.reverts() will check the source contract to see if the failed line contains a message.

There are a few scenarios where AssertionError will be raised when using dev_message:

  • If the line in the source contract has a different dev message or no dev message

  • If the contract source cannot be obtained

  • If the transaction trace cannot be obtained

Because dev_message relies on transaction tracing to function, you must use a provider like ape-hardhat when testing with dev_message.

You may also supply an re.Pattern object to assert on a dev message pattern, rather than on an exact match.

# Matches explictly "dev: foo" or "dev: bar"
with ape.reverts(dev_message=re.compile(r"^dev: (foo|bar)$")):
    ...

Caveats

Language Support

As of ape version 0.5.6, dev_messages assertions are available for contracts compiled with ape-vyper, but not for those compiled with ape-solidity or ape-cairo.

Inlining

Due to function inlining, the position of the # dev: ... message may sometimes be one line higher than expected:

@external
def foo(_x: decimal) -> decimal:  # dev: correct location
    return sqrt(_x)  # dev: incorrect location

This typically only applies when trying to add dev messages to statements containing built-in function calls.

Non-reentrant Functions

Similarly, if you require dev assertions for non-reentrant functions you must be sure to leave the comment on the function that should not have reentry:

@internal
@nonreentrant('lock')
def _foo_internal():  # dev: correct location
    pass

@external
@nonreentrant('lock')
def foo():
    self._foo_internal()  # dev: incorrect location

Multi-chain Testing

The Ape framework supports connecting to alternative providers in tests. The easiest way to achieve this is to use the networks provider context-manager.

# Switch to Fantom mid test
def test_my_fantom_test(networks):
    # The test starts in 1 ecosystem but switches to another
    assert networks.provider.network.ecosystem.name == "ethereum"

    with networks.fantom.local.use_provider("test") as provider:
        assert provider.network.ecosystem.name == "fantom"

    # You can also use the context manager like this:
    with networks.parse_network_choice("fantom:local:test") as provider:
       assert provider.network.ecosystem.name == "fantom"

You can also set the network context in a context-manager pytest fixture:

import pytest


@pytest.fixture
def stark_contract(networks, project):
    with networks.parse_network_choice("starknet:local"):
        yield project.MyStarknetContract.deploy()


def test_starknet_thing(stark_contract, stark_account):
    # Uses the starknet connection via the stark_contract fixture
    receipt = stark_contract.my_method(sender=stark_account)
    assert not receipt.failed

When you exit a provider’s context, Ape does not disconnect the provider. When you re-enter that provider’s context, Ape uses the previously-connected provider. At the end of the tests, Ape disconnects all the providers. Thus, you can enter and exit a provider’s context as much as you need in tests.

Gas Reporting

To include a gas report at the end of your tests, you can use the --gas flag. NOTE: This feature requires using a provider with tracing support, such as ape-hardhat.

ape test --network ethereum:local:hardhat --gas

At the end of test suite, you will see tables such as:

                            FundMe Gas

  Method           Times called    Min.    Max.    Mean   Median
 ────────────────────────────────────────────────────────────────
  fund                        8   57198   91398   82848    91398
  withdraw                    2   28307   38679   33493    33493
  changeOnStatus              2   23827   45739   34783    34783
  getSecret                   1   24564   24564   24564    24564

                  Transferring ETH Gas

  Method     Times called   Min.   Max.   Mean   Median
 ───────────────────────────────────────────────────────
  to:test0              2   2400   9100   5750     5750

                     TestContract Gas

  Method      Times called    Min.    Max.    Mean   Median
 ───────────────────────────────────────────────────────────
  setNumber              1   51021   51021   51021    51021

The following demonstrates how to use the ape-config.yaml file to exclude contracts and / or methods from the gas report:

test:
  gas:
    exclude:
      - method_name: DEBUG_*         # Exclude all methods starting with `DEBUG_`.
      - contract_name: MockToken     # Exclude all methods in contract named `MockToken`.
      - contract_name: PoolContract  # Exclude methods starting with `reset_` in `PoolContract`.
        method_name: reset_*

Similarly, you can exclude sources via the CLI option --gas-exclude. The value --gas-exclude takes is a comma-separated list of colon-separated values representing the structure similar as above, except you must explicitly use * where meaning “all”. For example to exclude all methods starting with DEBUG_, you would do:

ape test --gas --gas-exclude "*:DEBUG_*".

To exclude all methods in the MockToken contract, do:

ape test --gas --gas-exclude MockToken

And finally, to exclude all methods starting with reset_ in PoolContract, do:

ape test --gas --gas-exclude "PoolContract:reset_*"

Iterative Testing

Ape has a set of flags that controls running your test suite locally in a “watch” mode, which means watching for updates to files in your project and re-triggering the test suite.

To enable this mode, run ape test --watch to set up this mode using the default settings. While in this mode, any time a .py file (i.e. your tests) or smart contract source file (i.e. any files that get compiled using your installed compiler plugins) is added, removed, or changed, then the ape test task will be re-triggered, based on a polling interval.

To exit this mode, press Ctrl+D (on Linux or macOS) to stop the execution and undo it.