-
Notifications
You must be signed in to change notification settings - Fork 87
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
✨ feat(et): add
et make test
for interactively creating new tests (#…
…950) Co-authored-by: rahul <[email protected]> Co-authored-by: danceratopz <[email protected]>
- Loading branch information
1 parent
1b30c33
commit 8539345
Showing
16 changed files
with
401 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Binary file not shown.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,22 @@ | ||
# Writing Tests | ||
|
||
The best way to get started is to use one of the existing test modules for inspiration. A good simple example is [tests.berlin.eip2930_access_list.test_acl.test_access_list](../tests/berlin/eip2930_access_list/test_acl/test_access_list.md). | ||
The easiest way to get started is to use the interactive CLI: | ||
|
||
```console | ||
uv run et make test | ||
``` | ||
|
||
and modify the generated test module to suit your needs. | ||
|
||
<figure class="video_container"> | ||
<video controls="true" allowfullscreen="true"> | ||
<source src="./img/et_make_test.mp4" type="video/mp4"> | ||
</video> | ||
</figure> | ||
|
||
For help deciding which test format to select, see [Types of Tests](./types_of_tests.md), in particular [Deciding on a Test Type](./types_of_tests.md#deciding-on-a-test-type). Otherwise, some simple test case examples to get started with are: | ||
|
||
- [tests.berlin.eip2930_access_list.test_acl.test_access_list](../tests/berlin/eip2930_access_list/test_acl/test_access_list.md). | ||
- [tests.istanbul.eip1344_chainid.test_chainid.test_chainid](../tests/istanbul/eip1344_chainid/test_chainid/test_chainid.md). | ||
|
||
Please check that your code adheres to the repo's [Coding Standards](./code_standards.md) and read the other pages in this section for more background and an explanation of how to implement state transition and blockchain tests. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
""" | ||
This module is the entry point for the `et` command line interface. | ||
""" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
""" | ||
`et` is the dev CLI for EEST. It provides commands to help developers write tests. | ||
Invoke using `uv run et`. | ||
""" | ||
|
||
import click | ||
|
||
from cli.et.make.cli import make | ||
|
||
|
||
@click.group() | ||
def et(): | ||
""" | ||
`et` 👽 is the dev CLI for EEST. It provides commands to help developers write tests. | ||
""" | ||
pass | ||
|
||
|
||
""" | ||
################################ | ||
|| || | ||
|| Command Registration || | ||
|| || | ||
################################ | ||
Register nested commands here. For more information, see Click documentation: | ||
https://click.palletsprojects.com/en/8.0.x/commands/#nested-handling-and-contexts | ||
""" | ||
et.add_command(make) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
""" | ||
Make CLI | ||
This module provides the `make` CLI command that helps you quickly scaffold | ||
files. | ||
""" | ||
|
||
from .cli import test | ||
|
||
__all__ = ["test"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
""" | ||
The `make` CLI streamlines the process of scaffolding tasks, such as generating new test files, | ||
enabling developers to concentrate on the core aspects of specification testing. | ||
The module calls the appropriate function for the subcommand. If an invalid subcommand | ||
is chosen, it throws an error and shows a list of valid subcommands. If no subcommand | ||
is present, it shows a list of valid subcommands to choose from. | ||
""" | ||
|
||
import click | ||
|
||
from .commands import test | ||
|
||
|
||
@click.group() | ||
def make(): | ||
""" | ||
Generate project files from the CLI. | ||
""" | ||
pass | ||
|
||
|
||
""" | ||
################################ | ||
|| || | ||
|| Command Registration || | ||
|| || | ||
################################ | ||
Register nested commands here. For more information, see Click documentation: | ||
https://click.palletsprojects.com/en/8.0.x/commands/#nested-handling-and-contexts | ||
""" | ||
make.add_command(test) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
""" | ||
This subpackage holds subcommands for the make command. New subcommands must be created as | ||
modules and exported from this package, then registered under the make command in | ||
`cli.py`. | ||
""" | ||
|
||
from .test import test | ||
|
||
__all__ = ["test"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,117 @@ | ||
""" | ||
This module provides a CLI command to scaffold a test file. | ||
The `test` command guides the user through a series of prompts to generate a test file | ||
based on the selected test type, fork, EIP number, and EIP name. The generated test file | ||
is saved in the appropriate directory with a rendered template using Jinja2. | ||
""" | ||
|
||
import os | ||
import sys | ||
from pathlib import Path | ||
|
||
import click | ||
import jinja2 | ||
|
||
from cli.input import input_select, input_text | ||
from config.docs import DocsConfig | ||
from ethereum_test_forks import get_development_forks, get_forks | ||
|
||
template_loader = jinja2.PackageLoader("cli.et.make") | ||
template_env = jinja2.Environment( | ||
loader=template_loader, keep_trailing_newline=True, trim_blocks=True, lstrip_blocks=True | ||
) | ||
|
||
|
||
@click.command() | ||
def test(): | ||
""" | ||
Create a new specification test file for an EIP. | ||
This function guides the user through a series of prompts to generate a test file | ||
for Ethereum execution specifications. The user is prompted to select the type of test, | ||
the fork to use, and to provide the EIP number and name. Based on the inputs, a test file | ||
is created in the appropriate directory with a rendered template. | ||
Prompts: | ||
- Choose the type of test to generate (State or Blockchain) | ||
- Select the fork to use (Prague or Osaka) | ||
- Enter the EIP number | ||
- Enter the EIP name | ||
The generated test file is saved in the following format: | ||
`tests/{fork}/eip{eip_number}_{eip_name}/test_{eip_name}.py` | ||
Example: | ||
If the user selects "State" as the test type, "Prague" as the fork, | ||
enters "1234" as the EIP number, | ||
and "Sample EIP" as the EIP name, the generated file will be: | ||
`tests/prague/eip1234_sample_eip/test_sample_eip.py` | ||
The function uses Jinja2 templates to render the content of the test file. | ||
Raises: | ||
- FileNotFoundError: If the template file does not exist. | ||
- IOError: If there is an error writing the file. | ||
""" | ||
test_type = input_select( | ||
"Choose the type of test to generate", choices=["State", "Blockchain"] | ||
) | ||
|
||
fork_choices = [str(fork) for fork in get_forks()] | ||
fork = input_select( | ||
"Select the fork where this functionality was introduced", choices=fork_choices | ||
) | ||
|
||
eip_number = input_text("Enter the EIP number").strip() | ||
|
||
# TODO: Perhaps get the EIP name from the number using an API? | ||
eip_name = input_text("Enter the EIP name").strip() | ||
|
||
test_name = eip_name.lower().replace(" ", "_") | ||
|
||
file_name = f"test_{test_name}.py" | ||
|
||
directory_path = Path("tests") / fork.lower() / f"eip{eip_number}_{test_name}" | ||
|
||
file_path = directory_path / file_name | ||
|
||
if file_path.exists(): | ||
click.echo( | ||
click.style(f"\n 🛑 The target test module {file_path} already exists!", fg="red"), | ||
err=True, | ||
) | ||
sys.exit(1) | ||
|
||
# Create directories if they don't exist | ||
os.makedirs(directory_path, exist_ok=True) | ||
|
||
template = template_env.get_template(f"{test_type.lower()}_test.py.j2") | ||
rendered_template = template.render( | ||
fork=fork, | ||
eip_number=eip_number, | ||
eip_name=eip_name, | ||
test_name=test_name, | ||
) | ||
|
||
with open(file_path, "w") as file: | ||
file.write(rendered_template) | ||
|
||
click.echo( | ||
click.style( | ||
f"\n 🎉 Success! Test file created at: {file_path}", | ||
fg="green", | ||
) | ||
) | ||
|
||
fork_option = "" | ||
if fork in [dev_fork.name() for dev_fork in get_development_forks()]: | ||
fork_option = f" --until={fork}" | ||
|
||
click.echo( | ||
click.style( | ||
f"\n 📝 Get started with tests: {DocsConfig().DOCS_URL__WRITING_TESTS}" | ||
f"\n ⛽ To fill this test, run: `uv run fill {file_path}{fork_option}`", | ||
fg="cyan", | ||
) | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
""" | ||
A blockchain test for [EIP-{{eip_number}} {{eip_name}}](https://eips.ethereum.org/EIPS/eip-{{eip_number}}). | ||
""" | ||
|
||
import pytest | ||
|
||
from ethereum_test_tools import Account, Alloc, Block, BlockchainTestFiller, Transaction | ||
from ethereum_test_tools.vm.opcode import Opcodes as Op | ||
|
||
REFERENCE_SPEC_GIT_PATH = "DUMMY/eip-DUMMY.md" | ||
REFERENCE_SPEC_VERSION = "DUMMY_VERSION" | ||
|
||
@pytest.mark.valid_from("{{fork}}") | ||
def test_{{test_name}}(blockchain_test: BlockchainTestFiller, pre: Alloc): | ||
""" | ||
TODO: Enter a one-line test summary here. | ||
|
||
TODO: (Optional) Enter a more detailed test function description here. | ||
""" | ||
# TODO: Delete this explanation. | ||
# In this demo test, the pre-state contains one EOA and one very simple | ||
# smart contract. The EOA, `sender`, executes the smart contract, which | ||
# simply sets the value of the contract's storage slot. | ||
# The (non-exhaustive) post-state verifies that the storage slot was set | ||
# correctly - this is checked when filling the test. | ||
# | ||
# One gotcha is ensuring that the transaction `gas_limit` is set high | ||
# enough to cover the gas cost of the contract execution. | ||
|
||
storage_slot: int = 1 | ||
|
||
# TODO: Modify pre-state allocations here. | ||
sender = pre.fund_eoa() | ||
contract_address = pre.deploy_contract( | ||
code=Op.SSTORE(storage_slot, 0x2) + Op.STOP, | ||
storage={storage_slot: 0x1}, | ||
) | ||
|
||
tx = Transaction( | ||
to=contract_address, | ||
gas_limit=100000000, | ||
data=b"", | ||
value=0, | ||
sender=sender, | ||
{% if fork in ["Frontier", "Homestead"] %} | ||
protected=False, | ||
{% endif %} | ||
) | ||
|
||
# TODO: Modify post-state allocations here. | ||
post = {contract_address: Account(storage={storage_slot: 0x2})} | ||
|
||
blockchain_test(pre=pre, blocks=[Block(txs=[tx])], post=post) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
""" | ||
A state test for [EIP-{{eip_number}} {{eip_name}}](https://eips.ethereum.org/EIPS/eip-{{eip_number}}). | ||
""" | ||
|
||
import pytest | ||
|
||
from ethereum_test_tools import Account, Alloc, Environment, StateTestFiller, Transaction | ||
from ethereum_test_tools.vm.opcode import Opcodes as Op | ||
|
||
REFERENCE_SPEC_GIT_PATH = "DUMMY/eip-DUMMY.md" | ||
REFERENCE_SPEC_VERSION = "DUMMY_VERSION" | ||
|
||
@pytest.mark.valid_from("{{fork}}") | ||
def test_{{test_name}}(state_test: StateTestFiller, pre: Alloc): | ||
""" | ||
TODO: Enter a one-line test summary here. | ||
|
||
TODO: (Optional) Enter a more detailed test function description here. | ||
""" | ||
env = Environment() | ||
|
||
# TODO: Delete this explanation. | ||
# In this demo test, the pre-state contains one EOA and one very simple | ||
# smart contract. The EOA, `sender`, executes the smart contract, which | ||
# simply sets the value of the contract's storage slot. | ||
# The (non-exhaustive) post-state verifies that the storage slot was set | ||
# correctly - this is checked when filling the test. | ||
# | ||
# One gotcha is ensuring that the transaction `gas_limit` is set high | ||
# enough to cover the gas cost of the contract execution. | ||
|
||
storage_slot: int = 1 | ||
|
||
# TODO: Modify pre-state allocations here. | ||
sender = pre.fund_eoa() | ||
contract_address = pre.deploy_contract( | ||
code=Op.SSTORE(storage_slot, 0x2) + Op.STOP, | ||
storage={storage_slot: 0x1}, | ||
) | ||
|
||
tx = Transaction( | ||
to=contract_address, | ||
gas_limit=100000000, | ||
data=b"", | ||
value=0, | ||
sender=sender, | ||
{% if fork in ["Frontier", "Homestead"] %} | ||
protected=False, | ||
{% endif %} | ||
) | ||
|
||
# TODO: Modify post-state allocations here. | ||
post = {contract_address: Account(storage={storage_slot: 0x2})} | ||
|
||
state_test(env=env, pre=pre, post=post, tx=tx) |
Oops, something went wrong.