* We actually need Python 3.7 because of dataclasses * Rerun 'pipenv lock' under Python 3.7 and add 'pipenv' to dev deps * Update docs on developing for Python 3.7 * CircleCI: use Python 3.7 via Docker image instead of Orb
Zenith test runner
This directory contains integration tests.
Prerequisites:
- Python 3.7 or later
- Development headers may also be needed to build
psycopg2from source. - Python 3.7 is recommended if you want to update tests.
- Development headers may also be needed to build
- Dependencies: install them via
pipenv install. Note that Debian/Ubuntu packages are stale, as it commonly happens, so manual installation is not recommended. Exact version ofpipenvis not important unless you change dependencies. Runpipenv shellto activate the venv or usepipenv runto run a single command in the venv, e.g.pipenv run pytest. - Zenith and Postgres binaries
- See the root README.md for build directions
- Tests can be run from the git tree; or see the environment variables below to run from other directories.
- The zenith git repo, including the postgres submodule (for some tests, e.g. pg_regress)
Test Organization
The tests are divided into a few batches, such that each batch takes roughly the same amount of time. The batches can be run in parallel, to minimize total runtime. Currently, there are only two batches:
- test_batch_pg_regress: Runs PostgreSQL regression tests
- test_others: All other tests
Running the tests
Because pytest will search all subdirectories for tests, it's easiest to
run the tests from within the test_runner directory.
Test state (postgres data, pageserver state, and log files) will
be stored under a directory test_output.
You can run all the tests with:
pytest
If you want to run all the tests in a particular file:
pytest test_pgbench.py
If you want to run all tests that have the string "bench" in their names:
pytest -k bench
Useful environment variables:
ZENITH_BIN: The directory where zenith binaries can be found.
POSTGRES_DISTRIB_DIR: The directory where postgres distribution can be found.
TEST_OUTPUT: Set the directory where test state and test output files
should go.
TEST_SHARED_FIXTURES: Try to re-use a single pageserver for all the tests.
Let stdout, stderr and INFO log messages go to the terminal instead of capturing them:
pytest -s --log-cli-level=INFO ...
(Note many tests capture subprocess outputs separately, so this may not
show much.)
Exit after the first test failure:
pytest -x ...
(there are many more pytest options; run pytest -h to see them.)
Building new tests
The tests make heavy use of pytest fixtures. You can read about how they work here: https://docs.pytest.org/en/stable/fixture.html
Essentially, this means that each time you see a fixture named as an input parameter, the function with that name will be run and passed as a parameter to the function.
So this code:
def test_something(zenith_cli, pg_bin):
pass
... will run the fixtures called zenith_cli and pg_bin and deliver those results to the test function.
Fixtures can't be imported using the normal python syntax. Instead, use this:
pytest_plugins = ("fixtures.something")
That will make all the fixtures in the fixtures/something.py file available.
Anything that's likely to be used in multiple tests should be built into a fixture.
Note that fixtures can clean up after themselves if they use the yield syntax.
Cleanup will happen even if the test fails (raises an unhandled exception).
Python destructors, e.g. __del__() aren't recommended for cleanup.
Before submitting a patch
Obligatory checks
Install dev dependencies via pipenv --python 3.7 install --dev (better)
or pipenv install --dev (if you don't have Python 3.7 and don't need to change dependencies).
We force code formatting via yapf and type hints via mypy.
Run the following commands in the test_runner/ directory:
pipenv run yapf -ri . # All code is reformatted
pipenv run mypy . # Ensure there are no typing errors
Advisable actions
- Writing a couple of docstrings to clarify the reasoning behind a new test.
- Running
flake8(or a linter of your choice, e.g.pycodestyle) and fixing possible defects, if any. - Adding more type hints to your code to avoid
Any, especially:- For fixture parameters, they are not automatically deduced.
- For function arguments and return values.
Changing dependencies
You have to update Pipfile.lock if you have changed Pipfile:
pipenv --python 3.7 install --dev # Re-create venv for Python 3.7 and install recent pipenv inside
pipenv run pipenv --version # Should be at least 2021.5.29
pipenv run pipenv lock # Regenerate Pipfile.lock
As the minimal supported version is Python 3.7 and we use it in CI,
you have to use a Python 3.7 environment when updating Pipfile.lock.
Otherwise some back-compatibility packages will be missing.
It is also important to run recent pipenv.
Older versions remove markers from Pipfile.lock.
If you don't have Python 3.7, you should install it and its headers (for psycopg2)
separately, e.g.:
# In Ubuntu
sudo add-apt-repository ppa:deadsnakes/ppa
sudo apt update
sudo apt install python3.7 python3.7-dev