Skip to main content

Collection of pytest utils built on top of nbval

Project description

nbvalx

nbvalx is a collection of pytest utils built on top of nbval.

nbvalx is currently developed and maintained at Università Cattolica del Sacro Cuore by Prof. Francesco Ballarin.

nbvalx can be pip installed from its GitHub repository or from PyPI

Cell magics for conditional running based on tags

Add a cell with

%load_ext nbvalx

at the beginning of a notebook to load nbvalx IPython extension. The extension is implemented in nbvalx/jupyter_magics.py.

The extension allows to register a list of allowed tags, and their values

%%register_run_if_allowed_tags
bool_tag: True, False
int_tag: 2, 10
string_tag: "value1", "value2"

and set the current value of each tag with

%%register_run_if_current_tags
bool_tag = False
int_tag = 10
string_tag = "value1"

The tags can then be used to conditionally run cells. As an example, if the subsequent two cells were

%%run_if bool_tag and int_tag > 5
print("Cell with float and integer comparison")
%%run_if string_tag == "value1"
print("Cell with string comparison")

the string "Cell with float and integer comparison" would never get printed, while the "Cell with string comparison" would indeed be printed.

Without nbvalx, the four cells above could have equivalently been written as

bool_tag = False
int_tag = 10
string_tag = "value1"
if bool_tag and int_tag > 5:
    print("Cell with float and integer comparison")
if string_tag == "value1":
    print("Cell with string comparison")

The plain formulation is certainly less verbose and more compact than nbvalx equivalent one with four cells. However, it is less "testing-friendly", because the values of bool_tag, int_tag and string_tag are hardcoded in the notebook and cannot be easily varied. With a pytest terminology, nbvalx tags correspond to defining a parametrization of the notebook.

A similar statement can be used as an HTML comment in markdown cells

<!-- keep_if string_tag == "value1" -->
This is a markdown cell which explains a code cell that will be run only if string_tag == "value1".

Markdown cells will be stripped only when running through pytest with the --tag-collapse option, see below.

See tests/notebooks/data/tags for a few simple notebooks using tags.

Custom pytest hooks for jupyter notebooks

The file nbvalx/pytest_hooks_notebooks.py contains a few utility functions to be used in pytest configuration file for notebooks tests. The pytest hooks which can be customized in this way are:

  • pytest_addoption,
  • pytest_collect_file, and
  • pytest_sessionstart.

For clarity, the hooks implemented in nbvalx/pytest_hooks_notebooks.py do not have a pytest_ prefix, as it will be the user's responsability to pick them up and assign them to the corresponding pytest hook in a custom conftest.py, as show in tests/notebooks/conftest.py.

The hooks change the default behavior of nbval in the following ways:

  1. the options --nbval and --nbval-lax, which nbval requires to pass explicitly, are here enabled implicitly;
  2. support for MPI run by providing the --np option to pytest. When running pytest --np 2, nbvalx will start a ipyparallel.Cluster and run notebooks tests in parallel on 2 cores. In the default case one core is employed, and an ipyparallel.Cluster is not started;
  3. support for tags, as introduced in the previous section, as governed by two flags:
    • --tag-collapse: if enabled (default), strip all cells for which the %%run_if ... or <!-- keep_if ... --> conditions do not evaluate to True. This may be used to prepare notebook files to be read by the end user, as stripping unused cells may improve the readability of the notebook. If not enabled, all cells will be kept.
    • --ipynb-action: either collect-notebooks (default) or create-notebook. Both actions create several copies of the original notebook that differ by the currently enabled tag. For instance, if the original notebook in the section above is called notebook.ipynb and has two allowed tags, the action will generate a file notebook[tag1].ipynb where tag1 is assigned as the current value of the tag, and a file notebook[tag2].ipynb where tag2 is assigned as the current value of the tag. If tag-collapse is enabled, cells associated to all remaining tags are stripped. The create-notebook action only generates the tagged notebooks; instead, the collect-notebooks additionally also runs them through pytest;
  4. support for collecting cell outputs to log files, which are saved in a work directory provided by the user with the argument --work-dir. This is helpful to debug failures while testing notebooks. Log files are of two formats: a text log, with extension .log when running without --np or .log-{rank} when running in parallel; a notebook log, with extension .log.ipynb. If no work directory is specified, the default value is f".ipynb_pytest/np_{np}/collapse_{tag_collapse}". In case the notebook depends on additonal data files (e.g., local python modules), the flag --link-data-in-work-dir can be passed with glob patterns of data files that need to be symbolically linked in the work directory. The option can be passed multiple times in case multiple patterns are desired, and they will be joined with an or condition;
  5. the notebook is treated as if it were a demo or tutorial, rather than a collection of unit tests in different cells. For this reason, if a cell fails, the next cells will be skipped;
  6. a new # PYTEST_XFAIL marker is introduced to mark cells as expected to fail. The marker must be the first entry of the cell. A similar marker # PYTEST_XFAIL_AND_SKIP_NEXT marks the cell as expected to fail and interrupts execution of the subsequent cells. Both previous markers have a variant with XFAIL_IN_PARALLEL instead of XFAIL, that consider the cell to be expected to fail only when the value provided to --np is greater than one;
  7. support for running notebooks through coverage without having to install the pyvtest-cov plugin. Use flag --coverage-source to set the module name for which coverage testing is requested.

Custom pytest hooks for unit tests

The file nbvalx/pytest_unit_tests.py contains a few utility functions to be used in pytest configuration file for notebooks tests. The pytest hooks which can be customized in this way are:

  • pytest_runtest_setup, and
  • pytest_runtest_teardown.

For clarity, the hooks implemented in nbvalx/pytest_unit_tests.py do not have a pytest_ prefix, as it will be the user's responsability to pick them up and assign them to the corresponding pytest hook in a custom conftest.py, as show in tests/unit/conftest.py.

The hooks are typically employed to obtain a MPI-parallel safe execution of python unit tests by calling garbage collection and putting a MPI barrier after each test.

Custom pytest hooks for unit tests

The file nbvalx/tempfile.py contains MPI parallel-safe context managers to create temporary files and directories. Similarly to the tempfile module in the standard library, the following context managers are provided:

  • nbvalx.tempfile.TemporaryDirectory,
  • nbvalx.tempfile.TemporaryFile.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

nbvalx-0.3.2.tar.gz (20.6 kB view details)

Uploaded Source

Built Distribution

nbvalx-0.3.2-py3-none-any.whl (18.7 kB view details)

Uploaded Python 3

File details

Details for the file nbvalx-0.3.2.tar.gz.

File metadata

  • Download URL: nbvalx-0.3.2.tar.gz
  • Upload date:
  • Size: 20.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.0.0 CPython/3.12.2

File hashes

Hashes for nbvalx-0.3.2.tar.gz
Algorithm Hash digest
SHA256 0faa3dd0b69029c9378fc69c4846d30a579bc7ce5c3f540ee9aed7a42f6835a4
MD5 3082e4880339b6380a4fd39f0dd65b68
BLAKE2b-256 417b5c088571deff3badb6bd2b72aea7afa7dbdf3cef8330d9bac08166682acf

See more details on using hashes here.

Provenance

File details

Details for the file nbvalx-0.3.2-py3-none-any.whl.

File metadata

  • Download URL: nbvalx-0.3.2-py3-none-any.whl
  • Upload date:
  • Size: 18.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.0.0 CPython/3.12.2

File hashes

Hashes for nbvalx-0.3.2-py3-none-any.whl
Algorithm Hash digest
SHA256 1d9150c9b59e9af1d46380f3fccb67e0efd40e706830d397a2afe6dd1bf62912
MD5 6f25d8174244c53f83a4316c871c69ff
BLAKE2b-256 4c20a3da0db6246d613eda12de2a7f8f6a87406d23396419700458525024b72d

See more details on using hashes here.

Provenance

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page