Metadata-Version: 2.4
Name: pytest-approval
Version: 0.16.2
Summary: A simple approval test library utilizing external diff programs such as PyCharm and Visual Studio Code to compare approved and received output.
Author: HeiGIT ohsome team
Author-email: HeiGIT ohsome team <ohsome@heigit.org>
License-Expression: LGPL-3.0-only
License-File: COPYING
License-File: COPYING.LESSER
Classifier: Framework :: Pytest
Classifier: Intended Audience :: Developers
Classifier: Programming Language :: Python :: 3 :: Only
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Topic :: Software Development :: Testing
Requires-Dist: pytest>=9.0.3
Requires-Dist: pytest-nodeid-to-filepath
Requires-Dist: numpy>=2.3.1 ; extra == 'image'
Requires-Dist: pillow>=11.3.0 ; extra == 'image'
Requires-Dist: kaleido>=1.2.0 ; extra == 'plotly'
Requires-Dist: plotly>=6.0 ; extra == 'plotly'
Requires-Python: >=3.11, <3.15
Project-URL: Documentation, https://github.com/GIScience/pytest-approval/blob/main/README.md
Project-URL: Homepage, https://ohsome.org/
Project-URL: Issues, https://github.com/GISciene/pytest-approval/issues
Project-URL: Repository, https://github.com/GIScience/pytest-approval
Provides-Extra: image
Provides-Extra: plotly
Description-Content-Type: text/markdown

# pytest approval

[![Build Status](https://jenkins.heigit.org/buildStatus/icon?job=pytest-approval/main)](https://jenkins.heigit.org/job/pytest-approval/job/main/)
[![Sonarcloud Status](https://sonarcloud.io/api/project_badges/measure?project=pytest-approval&metric=alert_status)](https://sonarcloud.io/dashboard?id=pytest-approval)
[![PyPI - Version](https://img.shields.io/pypi/v/pytest-approval)](https://pypi.org/project/pytest-approval/)
[![LICENSE](https://img.shields.io/github/license/GIScience/pytest-approval)](COPYING)
[![status: active](https://github.com/GIScience/badges/raw/master/status/active.svg)](https://github.com/GIScience/badges#active)

A simple approval test library utilizing external diff programs such as
Meld, PyCharm and Visual Studio Code to compare approved and received output.

## About

Approval tests capture the output (a snapshot) of a piece of code and compare it
with a previously approved version of the output (the expected result).

It's most useful in environments where frequent changes are common or where
the output is of a complex nature but can be easily verified by humans, aided for
example by a diff-tool or a visual representation of the output (think of an image).

Once the output has been *approved* then as long as the output stays the same
the test will pass. A test fails if the *received* output is not identical to
the approved version. In that case, the difference between the received and the
approved output is reported to the tester.

For outputs that can be represented by text, a report can be as simple as
printing the difference to the terminal. Using diff programs with a graphical
user interface such as Meld, PyCharm or Visual Studio Code as *reporter* not
only helps to visualize the difference, but they can also be used as *approver*
by applying the changes of the received output to the approved output.

Not all data can or should be represented by text. In many cases an
image is the best and most easily verifiable representation.
PyCharm and Visual Studio Code can work with images as well.

> A picture’s worth a 1000 tests ([approvaltests.com](https://approvaltests.com/)).


<!-- ## Features -->
<!---->
<!-- - Auto approval mode: Helps cleaning up approval files -->
<!-- - Generated names of approval files are based on pytest nodeid -->


## Requirements

OS
- Linux/Unix
- macOS

One of following programs installed:
- PyCharm
- Visual Studio Code
- Meld
- GNU Diffutils (`diff`)


## Installation

```sh
uv add pytest-approval

# Including image support
uv add --optional image pytest-approval

# Including plotly support
uv add --optional plotly pytest-approval
```


## Usage

Verify text:

```python
from pytest_approval import verify, verify_json


def test_verify_string():
    assert verify("Hello World!")


def test_verify_dict():
    # automatic conversion to JSON
    assert verify_json({"msg": "Hello World!"})
    # works with string as well
    assert verify_json('{"msg": "Hello World!"}')
```


To verify binary files such as an image PyCharm or Visual Studio Code needs to
be installed. Examples:

```python
from PIL import Image
from pytest_approval import verify_binary, verify_image, verify_image_pillow


def test_verify_binary(image):
    with open("my_image.jpg", "rb") as file:
        buffer = file.read()
    assert verify_binary(buffer, extension=".jpg")


def test_verify_image(image):
    image = Image.open("my_image.jpg")
    assert verify_image(image, extension=".jpg", content_only=True)


def test_verify_image_pillow(image):
    image = Image.open("my_image.jpg")
    assert verify_image_pillow(image, extension=".jpg")
```


Plotly figures can be verified as well. For comparison the JSON representation of a Plotly figure is used and for reporting the image representation.

```python
from pytest_approval import verify_plotly
import plotly.graph_objects as go

FIGURE = go.Figure(
    data=go.Contour(
        z=[
            [10, 10.625, 12.5, 15.625, 20],
            [5.625, 6.25, 8.125, 11.25, 15.625],
            [2.5, 3.125, 5.0, 8.125, 12.5],
            [0.625, 1.25, 3.125, 6.25, 10.625],
            [0, 0.625, 2.5, 5.625, 10],
        ]
    )
)


def test_verify_plotly():
    assert verify_plotly(FIGURE)
```


During development its sometimes helpful to show received and approved output,
to report, even though both are equal:

```python
from pytest_approval import verify


def test_verify_string():
    assert verify("Hello World!", report_always=True)
```

### Auto approval

It is possible to run auto approve every approval tests:
```shell
uv run pytest --auto-approve
```

This is useful for elimination of approval files which are not in use anymore.
1. Make sure tests are green.
2. Then remove all approval files.
3. Run pytest in auto approval mode.


## Configuration

Approved and received files are stored next to the test file per default.
If you want to save those files in a specific directory instead, please set the `approvals-dir` key in your `pyproject.toml`:

```toml
[tool.pytest-approval]
"approvals-dir"="tests/approvals"  
```

The path is relative to pytest root (usually `pyproject.toml`).

<!-- ## Configuration -->
<!---->
<!-- ### Approver/Reporter -->
<!---->
<!-- Per default `pytest-approval` tries a list of diff programs as reporters until a working one is found. -->
<!---->
<!-- You can provide your own list in the `pyproject.toml` file: -->
<!---->
<!-- ```toml -->
<!-- [tool.pytest-approval] -->
<!-- reporters = [ -->
<!--     [ -->
<!--         "meld", -->
<!--         "%received", -->
<!--         "%approved", -->
<!--     ], -->
<!--     [ -->
<!--         "diff", -->
<!--         "--unified", -->
<!--         "--color", -->
<!--         "--suppress-common-lines", -->
<!--         "--label", -->
<!--         "received", -->
<!--         "--label", -->
<!--         "approved", -->
<!--         "%received", -->
<!--         "%approved", -->
<!--     ], -->
<!-- ] -->
<!-- ``` -->
<!---->
<!-- This list will be put in front of the [list of default reporters](pytest_approval/definitions.py). -->

## Development

```sh
uv sync --all-extras
uv run prek install  # pre-commit hooks
uv run pytest
uv run pytest --markdown-docs -m markdown-docs README.md
```

### Release

This project uses [SemVer](https://semver.org/).

To make a new release run `./scripts/release.sh <version>`.


## Alternatives

- [Syrupy](https://github.com/syrupy-project/syrupy) is a zero-dependency pytest snapshot plugin. It enables developers to write tests which assert immutability of computed results.
- [ApprovalTests.Python](https://github.com/approvals/ApprovalTests.Python) is an open source assertion/verification library to aid testing.

### Comparison with ApprovalTests.Python

ApprovalTests.Python and [ApprovalTests](https://approvaltests.com/) in general
are the main inspiration for this library. The goal of this library is to
provide a much smaller, simpler and maintainable code base while at the same
time providing a simpler interface.

In contrast to ApprovalTests.Python this library features:
- Default naming of files are based on PyTest node ID and works with multiple calls to `verify` and with parametrized tests out of the box: [ApprovalTests.Python Documentation](https://github.com/approvals/ApprovalTests.Python/blob/main/docs/how_to/multiple_approvals_per_test.md)
- Reporters are blocking: [ApprovalTests.Python Issue](https://github.com/approvals/ApprovalTests.Python/issues/95)
- A list of programs for different operating systems are provided as reporters and the first working reporter is used without any configuration: -> [ApprovalTests.Python Issue](https://github.com/approvals/ApprovalTests.Python/issues/198)
- On approve the test goes green immediately instead of failing first and succeeding only after another run.
- No HTTP request to fetch empty binary files during testing: [ApprovalTests.Python Call to EmptyFiles](https://github.com/approvals/ApprovalTests.Python/blob/f1ba04383d7c76c627ef42faef35d1b96f765a2a/approval_utilities/utils.py#L63) [](https://github.com/approvals/EmptyFiles.Python/blob/master/empty_files/empty_files.py#L10) & [EmptyFiles HTTP Request](https://github.com/approvals/EmptyFiles.Python/blob/c1f744ae97e341abb3692908b5df368c6bbc8516/empty_files/empty_files.py#L10)
- Modern Python project: Usage of `uv`, `ruff` and `pytest` (No shell scripts needed for setup or running tests.)
- Configuration happens through `pyproject.toml`: [ApprovalTests.Python Documentation](https://github.com/approvals/ApprovalTests.Python/blob/main/docs/configuration.md)
- Auto approve mode via parameter to PyTest: `pytest --auto-approve`

### Comparison with Syrupy

- Approval in syrupy happens by passing a command line argument `--snapshot-update` to PyTest. 
- Syrupy has not built-in diff reporter for images (See issues [#886](https://github.com/syrupy-project/syrupy/issues/886) and [#566](https://github.com/syrupy-project/syrupy/issues/566)).
