Skip to main content

Snapshot testing pytest plugin with minimal ceremony and flexible persistence formats.

Project description

pytest-ditto

PyPI version Continuous Integration

Snapshot testing pytest plugin with minimal ceremony and flexible recorders.

Introduction

The pytest-ditto plugin is intended to be used for snapshot/regression testing. There are two key components: the snapshot fixture and snapshot recorders.

The snapshot Fixture

In the following basic example, the function to test is fn, the test is using the snapshot fixture and it is asserting that the result of calling fn with the value of x does not change.

import ditto


def fn(x: int) -> int:
    return x + 1  # original implementation
    # return x + 2  # new implementation


def test_fn(snapshot) -> None:
    x = 1
    result = fn(x)
    assert result == snapshot(result, key="fn")

The first time the test is run, the snapshot fixture takes the data passed to it and persists it to a .ditto directory in the same location as the test module. Subsequent test runs load the stored file and use that value for comparison.

By default, snapshot data is persisted using pickle; however, a range of recorders can be selected per test using ditto marks.

@ditto Marks

If the default recorder (pickle) isn't appropriate, a different recorder can be specified per test using ditto marks — customised pytest mark decorators.

The built-in recorders are:

Mark Recorder File extension
@ditto.pickle pickle .pkl
@ditto.yaml yaml .yaml
@ditto.json json .json
@ditto.record("name") any registered recorder varies

@ditto.pickle, @ditto.yaml, and @ditto.json are convenience shorthands for @ditto.record("pickle"), @ditto.record("yaml"), and @ditto.record("json") respectively.

Additional recorders can be installed via plugins:

Recorders Plugin Marks
pandas pytest-ditto-pandas
  • @ditto.pandas.parquet
  • @ditto.pandas.json
  • @ditto.pandas.csv
pyarrow pytest-ditto-pyarrow
  • @ditto.pyarrow.parquet
  • @ditto.pyarrow.feather
  • @ditto.pyarrow.csv

Usage

pd.DataFrame

Install pytest-ditto[pandas] to make pandas recorders available.

import pandas as pd

import ditto


def awesome_fn_to_test(df: pd.DataFrame):
    df.loc[:, "a"] *= 2
    return df


@ditto.pandas.parquet
def test_fn_with_parquet_dataframe_snapshot(snapshot):
    input_data = pd.DataFrame({"a": [1, 2, 3], "b": [4, 5, 9]})
    result = awesome_fn_to_test(input_data)
    pd.testing.assert_frame_equal(result, snapshot(result, key="ab_dataframe"))


@ditto.pandas.json
def test_fn_with_json_dataframe_snapshot(snapshot):
    input_data = pd.DataFrame({"a": [1, 2, 3], "b": [4, 5, 9]})
    result = awesome_fn_to_test(input_data)
    pd.testing.assert_frame_equal(result, snapshot(result, key="ab_dataframe"))

For the above example the snapshot files would be found in the following locations:

  • .ditto/test_fn_with_parquet_dataframe_snapshot@ab_dataframe.pandas.parquet
  • .ditto/test_fn_with_json_dataframe_snapshot@ab_dataframe.pandas.json

pyarrow.Table

Install pytest-ditto[pyarrow] to make pyarrow recorders available.

import pyarrow as pa
import pyarrow.compute as pc
import ditto
import pytest


@pytest.fixture
def table() -> pa.Table:
    return pa.table(
        [
            [1, 2, 3, 4],
            [4.5, 5.2, 6.8, 3.5],
            [7, 8.5, None, None],
            [True, False, True, True],
            ["a", "b", "c", "x"],
        ],
        names=list("abcde"),
    )


def fn(x: pa.Table):
    even_filter = (pc.bit_wise_and(pc.field("a"), pc.scalar(1)) == pc.scalar(0))
    return x.filter(even_filter)


@ditto.pyarrow.parquet
def test_fn_with_pyarrow_parquet_snapshot(snapshot, table):
    result = fn(table)
    assert result.equals(snapshot(result, key="filtered"))

For the above example the snapshot files would be found in the following location:

  • .ditto/test_fn_with_pyarrow_parquet_snapshot@filtered.pyarrow.parquet

unittest.TestCase

DittoTestCase provides the snapshot fixture as a cached_property for use with unittest.TestCase:

import unittest
from ditto import DittoTestCase


def fn(x: int) -> int:
    return x + 1


class TestFn(DittoTestCase):
    def test_fn(self):
        result = fn(1)
        assert result == self.snapshot(result, key="fn")

Snapshot files are placed in a .ditto directory adjacent to the test file, using the fully-qualified test method name as the group name.

Custom Recorders

A Recorder is a frozen dataclass pairing a file extension with save and load functions. Plugin packages register Recorder instances via the ditto_recorders entry point group.

from pathlib import Path
from ditto.recorders import Recorder


def _save(data: MyType, filepath: Path) -> None:
    ...  # write data to filepath


def _load(filepath: Path) -> MyType:
    ...  # read and return data from filepath


my_recorder: Recorder[MyType] = Recorder(
    extension="myformat",
    save=_save,
    load=_load,
)

Register it in pyproject.toml:

[project.entry-points.ditto_recorders]
my_recorder = "my_package.recorders:my_recorder"

Once registered, the recorder is available by name via @ditto.record("my_recorder"). Plugin marks (e.g. @ditto.myplugin.myformat) can also be registered via the ditto_marks entry point group.

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

pytest_ditto-1.0.0.tar.gz (10.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pytest_ditto-1.0.0-py3-none-any.whl (14.8 kB view details)

Uploaded Python 3

File details

Details for the file pytest_ditto-1.0.0.tar.gz.

File metadata

  • Download URL: pytest_ditto-1.0.0.tar.gz
  • Upload date:
  • Size: 10.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pytest_ditto-1.0.0.tar.gz
Algorithm Hash digest
SHA256 952edb8329b32511441696a60050806453c624ccff8dba286675cd6683ffff54
MD5 443ed5ab430493bff14653522967f23d
BLAKE2b-256 ffb29bbf6d8608f1410836659f7797706deaa2f964fd12c67554efbf76d435a3

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_ditto-1.0.0.tar.gz:

Publisher: release.yml on owlowlyowl/pytest-ditto

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pytest_ditto-1.0.0-py3-none-any.whl.

File metadata

  • Download URL: pytest_ditto-1.0.0-py3-none-any.whl
  • Upload date:
  • Size: 14.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pytest_ditto-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 64c7dadb32f120460431d8a0cf646c4bca9249f24fe31c03e86f60ef5b2efd73
MD5 2a149845a80b911d12186271d34126d7
BLAKE2b-256 2eb3cd8bac33c2aa4663f00d6097c4807dde384a87b68bf8a50c0ab24261b24f

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_ditto-1.0.0-py3-none-any.whl:

Publisher: release.yml on owlowlyowl/pytest-ditto

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

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