snappylapy


Namesnappylapy JSON
Version 0.2.1 PyPI version JSON
download
home_pagehttps://github.com/martinmoldrup/snappylapy
SummaryA snapshot library for python optimized for easy of use, human readable snapshots and enabling decoupling of chained integration tests.
upload_time2025-01-13 11:14:07
maintainerNone
docs_urlNone
authorMartin Møldrup
requires_python<4.0,>=3.9
licenseNone
keywords pytest snapshot testing snapshot-testing integration-testing unit-testing
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Snappylapy
   
Welcome to **Snappylapy**, a powerful and intuitive snapshot testing tool for Python's pytest framework. Snappylapy simplifies the process of capturing and verifying snapshots of your data, ensuring your code behaves as expected across different runs. With Snappylapy, you can save snapshots in a human-readable format and deserialize them for robust integration testing, providing a clear separation layer to help isolate errors and maintain code integrity.  

## Installation
To install Snappylapy, use the following command:

```bash
pip install snappylapy
```

## Key Features
Legend:
- ✅ Implemented
- ❌ Not implemented yet

The features are in development:
   
- **Human-Readable Snapshots**✅: Save snapshots in a format that's easy to read and understand, making it simpler to review changes and debug issues.  
- **Serialization and Deserialization**✅: Snapshots can be serialized and deserialized, allowing for flexible and reusable test cases.  
- **Integration Testing**✅: Use snapshots for integration testing with a clear separation layer, preventing interdependencies between code components and making it easier to isolate and identify errors.  
- **Easy to Use**✅: Seamlessly integrates with pytest, allowing you to start capturing and verifying snapshots with minimal setup. For a good developer experience the package is fully typed, with docstrings to provide good editor code completion.
- **Customizable Output**✅: Store snapshots in a location of your choice, enabling you to organize and manage your test data effectively.
- **Diff Report Generation**❌: Generate a diff report in html format for easy comparison between test results and snapshots.
- **Provides a wide set of assertions**❌: The package provides a wide set of assertions to make it easy to compare different types of data, for do fuzzy matching or ignore certain parts of the data that are variable.

## Benefits of Snapshot Testing
Snapshot testing is a powerful technique for verifying the output of your code by comparing it to a stored snapshot. This approach offers several benefits, including:

- Immutability Verification: Quickly detect unintended changes or regressions by comparing current output to stored snapshots.
- Faster Test Creation: Simplify the process of writing and maintaining tests by capturing snapshots once and letting the framework handle comparisons.
- Documentation: Use snapshots as a form of documentation, providing a clear record of expected output and behavior.
- Version Control Integration: Include snapshots in your version control system to aid in code reviews and track changes over time.
- Pull Request Reviews: Enhance PR reviews by showing exactly how changes affect the application's output, ensuring thorough and effective evaluations.
   
## Why Snappylapy?  
   
When working on a test suite for a project, it’s important to ensure tests are independent. This is to avoid situations where changes in one part of the code cause failures in tests for other unrelated areas, making it challenging to isolate and fix errors. Snappylapy addresses this by providing a mechanism to capture snapshots of your data and use them in your later tests, ensuring that each component can be tested independently. While also making sure that they are dependent enought to test the integration between them. It provides serialization and deserialization of the snapshots, making it easy to reuse them in different test cases. This is aimed at function working with large and complex data structures (dataframes or large nested dictionaries.)
   
### Example  
   
```python  
from snappylapy import Expect
from mypackage import my_function
   
def test_snapshot_dict(expect: Expect):
    """Test snapshot with dictionary data."""
    data: dict = my_function()
    expect.dict(data).to_match_snapshot()
```

Allows users full control to select output location for snapshots so they can be stored together with testcases.

```python
import pytest
import pathlib
from snappylapy import Expect

@pytest.mark.parametrize('case_dir', list(Path('test_cases').iterdir()))
def test_my_function(case_dir: pathlib.Path, expect: Expect):
    snapshot.snapshot_dir = case_dir / "__snapshots__"
    snapshot.test_results_dir = case_dir / "__test_results__"
    result = my_function(case_dir)
    expect.dict(result).to_match_snapshot()
```

In this example, `snappylapy` captures the output of `my_function` and compares it against a stored snapshot. If the output changes unexpectedly, pytest will flag the test, allowing you to review the differences and ensure your code behaves as expected.

Snappylapy can use the snapshots created for inputs in another test. You can think of it as automated/easier mock data generation and management.
```python
import pytest
from snappylapy import Expect, LoadSnapshot

def test_snapshot_dict(expect: Expect):
    """Test snapshot with dictionary data."""
    expect.dict({
        "name": "John Doe",
        "age": 31
    }).to_match_snapshot()

@pytest.mark.snappylapy(depends=[test_snapshot_dict])
def test_load_snapshot_from_file(load_snapshot: LoadSnapshot):
    """Test loading snapshot data created in test_snapshot_dict from a file using the deserializer."""
    data = load_snapshot.dict()
    assert data == {"name": "John Doe", "age": 31}
```

This can be great for external dependencies, for example an AI service, that might change response over time. With this approach we can isolate the changes to the service and still make succeding tests pass.

## Getting Started  
   
To get started with Snappylapy, install the package via pip:  
   
```bash  
pip install snappylapy  
```  
   
Add Snappylapy to your pytest configuration and start writing tests that capture and verify snapshots effortlessly.  

## The output structure

The results is split into two folders, for ease of comparison, and for handling stochastic/variable outputs (timestamps, generated ids, llm outputs, third party api responses etc).

- __test_results__: Updated every time the tests is ran. Compare with snapshots when doing snapshot style assertions. Add this to your .gitignore file.
- __snapshots__: Updated only when --snapshot-update flag is used when running the test suite. Commit this to your version control system.

## Usage
Update snapshots with:

```python
pytest --snapshot-update
```

A diff report in html can be generated with (not implemented yet ❌):

```python
pytest --snappylapy-html=report.html
```

## Fixtures and roadmap
Registers fixtures:
- expect ✅
- load_snapshot ✅

Supported data types
- .txt ✅
- .json ✅
- .csv ❌
- .yaml ❌
- .jsonl ❌

Planned data types:

| Python Type         | Default Output file type | Implementation Status |
|---------------------|--------------------------|-----------------------|
| bytes               | .txt                     | ✅                    |
| pd.DataFrame        | .csv                     | ❌                    |
| pd.Series           | .csv                     | ❌                    |
| np.ndarray          | .csv                     | ❌                    |
| dict                | .json                    | ✅                    |
| list                | .json                    | ✅                    |
| tuple               | .json                    | ❌                    |
| set                 | .json                    | ❌                    |
| str                 | .txt                     | ✅                    |
| int                 | .txt                     | ❌                    |
| float               | .txt                     | ❌                    |
| bool                | .txt                     | ❌                    |
| datetime.datetime   | .txt                     | ❌                    |
| datetime.date       | .txt                     | ❌                    |
| datetime.time       | .txt                     | ❌                    |
| pathlib.Path        | .txt                     | ❌                    |
| decimal.Decimal     | .txt                     | ❌                    |
| uuid.UUID           | .txt                     | ❌                    |



---  
   
Snappylapy is your go-to tool for efficient and reliable snapshot testing in Python. By maintaining clear boundaries between different parts of your code, Snappylapy helps you isolate errors, streamline debugging, and ensure your code remains robust and maintainable.

## Contributing
We welcome contributions to Snappylapy! If you have ideas for new features, improvements, or bug fixes, please open an issue or submit a pull request on our GitHub repository. We appreciate your feedback and support in making Snappylapy even better for the community.

# Change log
All notable changes to this project will be documented in this file.
## [0.2.1]
- Added missing dependency for typer to make the CLI work

## [0.2.0]
- Better error messages by using pytest assertion rewriting
- Allow users to set the snapshot directory when using the load_snapshot fixture
- Add CLI for for init and clear commands
- Added automated generation of documentation using mkdocs
  
## [0.1.1]
- Update dependencies with the lower bounds of package compatibility
- Refactor to make code easier for users of package to modify and extend

## [0.1.0]
- Added fixture for loading snapshots from previous tests (load_snapshot fixture)
- Added the snappylapy marker for tests that depend on previous tests (pytest.mark.snappylapy). This will be used for more advanced features in the future.

## [0.0.2]
- 🐞 Added fix for python 3.9, by refactoring incompatible type annotation
- Loosened the version requirements for pytest (until the lower bound have been discovered, with automated testing)
- Improved metadata for pypi

## [0.0.1]
- Initial release of Snappylapy
- Implemented basic snapshot testing functionality for dict, list, bytes and str data types

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/martinmoldrup/snappylapy",
    "name": "snappylapy",
    "maintainer": null,
    "docs_url": null,
    "requires_python": "<4.0,>=3.9",
    "maintainer_email": null,
    "keywords": "pytest, snapshot, testing, snapshot-testing, integration-testing, unit-testing",
    "author": "Martin M\u00f8ldrup",
    "author_email": null,
    "download_url": "https://files.pythonhosted.org/packages/12/2a/012e8910edb11587a166bc8b6950a90b2db2820174d0c48fbcfb688d0d78/snappylapy-0.2.1.tar.gz",
    "platform": null,
    "description": "# Snappylapy\n   \nWelcome to **Snappylapy**, a powerful and intuitive snapshot testing tool for Python's pytest framework. Snappylapy simplifies the process of capturing and verifying snapshots of your data, ensuring your code behaves as expected across different runs. With Snappylapy, you can save snapshots in a human-readable format and deserialize them for robust integration testing, providing a clear separation layer to help isolate errors and maintain code integrity.  \n\n## Installation\nTo install Snappylapy, use the following command:\n\n```bash\npip install snappylapy\n```\n\n## Key Features\nLegend:\n- \u2705 Implemented\n- \u274c Not implemented yet\n\nThe features are in development:\n   \n- **Human-Readable Snapshots**\u2705: Save snapshots in a format that's easy to read and understand, making it simpler to review changes and debug issues.  \n- **Serialization and Deserialization**\u2705: Snapshots can be serialized and deserialized, allowing for flexible and reusable test cases.  \n- **Integration Testing**\u2705: Use snapshots for integration testing with a clear separation layer, preventing interdependencies between code components and making it easier to isolate and identify errors.  \n- **Easy to Use**\u2705: Seamlessly integrates with pytest, allowing you to start capturing and verifying snapshots with minimal setup. For a good developer experience the package is fully typed, with docstrings to provide good editor code completion.\n- **Customizable Output**\u2705: Store snapshots in a location of your choice, enabling you to organize and manage your test data effectively.\n- **Diff Report Generation**\u274c: Generate a diff report in html format for easy comparison between test results and snapshots.\n- **Provides a wide set of assertions**\u274c: The package provides a wide set of assertions to make it easy to compare different types of data, for do fuzzy matching or ignore certain parts of the data that are variable.\n\n## Benefits of Snapshot Testing\nSnapshot testing is a powerful technique for verifying the output of your code by comparing it to a stored snapshot. This approach offers several benefits, including:\n\n- Immutability Verification: Quickly detect unintended changes or regressions by comparing current output to stored snapshots.\n- Faster Test Creation: Simplify the process of writing and maintaining tests by capturing snapshots once and letting the framework handle comparisons.\n- Documentation: Use snapshots as a form of documentation, providing a clear record of expected output and behavior.\n- Version Control Integration: Include snapshots in your version control system to aid in code reviews and track changes over time.\n- Pull Request Reviews: Enhance PR reviews by showing exactly how changes affect the application's output, ensuring thorough and effective evaluations.\n   \n## Why Snappylapy?  \n   \nWhen working on a test suite for a project, it\u2019s important to ensure tests are independent. This is to avoid situations where changes in one part of the code cause failures in tests for other unrelated areas, making it challenging to isolate and fix errors. Snappylapy addresses this by providing a mechanism to capture snapshots of your data and use them in your later tests, ensuring that each component can be tested independently. While also making sure that they are dependent enought to test the integration between them. It provides serialization and deserialization of the snapshots, making it easy to reuse them in different test cases. This is aimed at function working with large and complex data structures (dataframes or large nested dictionaries.)\n   \n### Example  \n   \n```python  \nfrom snappylapy import Expect\nfrom mypackage import my_function\n   \ndef test_snapshot_dict(expect: Expect):\n    \"\"\"Test snapshot with dictionary data.\"\"\"\n    data: dict = my_function()\n    expect.dict(data).to_match_snapshot()\n```\n\nAllows users full control to select output location for snapshots so they can be stored together with testcases.\n\n```python\nimport pytest\nimport pathlib\nfrom snappylapy import Expect\n\n@pytest.mark.parametrize('case_dir', list(Path('test_cases').iterdir()))\ndef test_my_function(case_dir: pathlib.Path, expect: Expect):\n    snapshot.snapshot_dir = case_dir / \"__snapshots__\"\n    snapshot.test_results_dir = case_dir / \"__test_results__\"\n    result = my_function(case_dir)\n    expect.dict(result).to_match_snapshot()\n```\n\nIn this example, `snappylapy` captures the output of `my_function` and compares it against a stored snapshot. If the output changes unexpectedly, pytest will flag the test, allowing you to review the differences and ensure your code behaves as expected.\n\nSnappylapy can use the snapshots created for inputs in another test. You can think of it as automated/easier mock data generation and management.\n```python\nimport pytest\nfrom snappylapy import Expect, LoadSnapshot\n\ndef test_snapshot_dict(expect: Expect):\n    \"\"\"Test snapshot with dictionary data.\"\"\"\n    expect.dict({\n        \"name\": \"John Doe\",\n        \"age\": 31\n    }).to_match_snapshot()\n\n@pytest.mark.snappylapy(depends=[test_snapshot_dict])\ndef test_load_snapshot_from_file(load_snapshot: LoadSnapshot):\n    \"\"\"Test loading snapshot data created in test_snapshot_dict from a file using the deserializer.\"\"\"\n    data = load_snapshot.dict()\n    assert data == {\"name\": \"John Doe\", \"age\": 31}\n```\n\nThis can be great for external dependencies, for example an AI service, that might change response over time. With this approach we can isolate the changes to the service and still make succeding tests pass.\n\n## Getting Started  \n   \nTo get started with Snappylapy, install the package via pip:  \n   \n```bash  \npip install snappylapy  \n```  \n   \nAdd Snappylapy to your pytest configuration and start writing tests that capture and verify snapshots effortlessly.  \n\n## The output structure\n\nThe results is split into two folders, for ease of comparison, and for handling stochastic/variable outputs (timestamps, generated ids, llm outputs, third party api responses etc).\n\n- __test_results__: Updated every time the tests is ran. Compare with snapshots when doing snapshot style assertions. Add this to your .gitignore file.\n- __snapshots__: Updated only when --snapshot-update flag is used when running the test suite. Commit this to your version control system.\n\n## Usage\nUpdate snapshots with:\n\n```python\npytest --snapshot-update\n```\n\nA diff report in html can be generated with (not implemented yet \u274c):\n\n```python\npytest --snappylapy-html=report.html\n```\n\n## Fixtures and roadmap\nRegisters fixtures:\n- expect \u2705\n- load_snapshot \u2705\n\nSupported data types\n- .txt \u2705\n- .json \u2705\n- .csv \u274c\n- .yaml \u274c\n- .jsonl \u274c\n\nPlanned data types:\n\n| Python Type         | Default Output file type | Implementation Status |\n|---------------------|--------------------------|-----------------------|\n| bytes               | .txt                     | \u2705                    |\n| pd.DataFrame        | .csv                     | \u274c                    |\n| pd.Series           | .csv                     | \u274c                    |\n| np.ndarray          | .csv                     | \u274c                    |\n| dict                | .json                    | \u2705                    |\n| list                | .json                    | \u2705                    |\n| tuple               | .json                    | \u274c                    |\n| set                 | .json                    | \u274c                    |\n| str                 | .txt                     | \u2705                    |\n| int                 | .txt                     | \u274c                    |\n| float               | .txt                     | \u274c                    |\n| bool                | .txt                     | \u274c                    |\n| datetime.datetime   | .txt                     | \u274c                    |\n| datetime.date       | .txt                     | \u274c                    |\n| datetime.time       | .txt                     | \u274c                    |\n| pathlib.Path        | .txt                     | \u274c                    |\n| decimal.Decimal     | .txt                     | \u274c                    |\n| uuid.UUID           | .txt                     | \u274c                    |\n\n\n\n---  \n   \nSnappylapy is your go-to tool for efficient and reliable snapshot testing in Python. By maintaining clear boundaries between different parts of your code, Snappylapy helps you isolate errors, streamline debugging, and ensure your code remains robust and maintainable.\n\n## Contributing\nWe welcome contributions to Snappylapy! If you have ideas for new features, improvements, or bug fixes, please open an issue or submit a pull request on our GitHub repository. We appreciate your feedback and support in making Snappylapy even better for the community.\n\n# Change log\nAll notable changes to this project will be documented in this file.\n## [0.2.1]\n- Added missing dependency for typer to make the CLI work\n\n## [0.2.0]\n- Better error messages by using pytest assertion rewriting\n- Allow users to set the snapshot directory when using the load_snapshot fixture\n- Add CLI for for init and clear commands\n- Added automated generation of documentation using mkdocs\n  \n## [0.1.1]\n- Update dependencies with the lower bounds of package compatibility\n- Refactor to make code easier for users of package to modify and extend\n\n## [0.1.0]\n- Added fixture for loading snapshots from previous tests (load_snapshot fixture)\n- Added the snappylapy marker for tests that depend on previous tests (pytest.mark.snappylapy). This will be used for more advanced features in the future.\n\n## [0.0.2]\n- \ud83d\udc1e Added fix for python 3.9, by refactoring incompatible type annotation\n- Loosened the version requirements for pytest (until the lower bound have been discovered, with automated testing)\n- Improved metadata for pypi\n\n## [0.0.1]\n- Initial release of Snappylapy\n- Implemented basic snapshot testing functionality for dict, list, bytes and str data types\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "A snapshot library for python optimized for easy of use, human readable snapshots and enabling decoupling of chained integration tests.",
    "version": "0.2.1",
    "project_urls": {
        "Bug Tracker": "https://github.com/martinmoldrup/snappylapy/issues",
        "Documentation": "https://martinmoldrup.github.io/snappylapy",
        "Homepage": "https://github.com/martinmoldrup/snappylapy",
        "Repository": "https://github.com/martinmoldrup/snappylapy"
    },
    "split_keywords": [
        "pytest",
        " snapshot",
        " testing",
        " snapshot-testing",
        " integration-testing",
        " unit-testing"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "96e65b9e879d916619d4db7007c75d5b0c9d31fbbb1d30cabbab087bf405301c",
                "md5": "634d523caa5d368e7046550ec683baa2",
                "sha256": "f2a89fc587d6b2afa3f3c2d3280af8ed25d0f56198b7c5ddc95354e5430bcb80"
            },
            "downloads": -1,
            "filename": "snappylapy-0.2.1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "634d523caa5d368e7046550ec683baa2",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<4.0,>=3.9",
            "size": 26142,
            "upload_time": "2025-01-13T11:14:05",
            "upload_time_iso_8601": "2025-01-13T11:14:05.536330Z",
            "url": "https://files.pythonhosted.org/packages/96/e6/5b9e879d916619d4db7007c75d5b0c9d31fbbb1d30cabbab087bf405301c/snappylapy-0.2.1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "122a012e8910edb11587a166bc8b6950a90b2db2820174d0c48fbcfb688d0d78",
                "md5": "c3f61486e7856343c428ea58bc5e57ab",
                "sha256": "b48280b6b2cea796d36dc095bd581f18e87885545742c6dce317f2eed1d59897"
            },
            "downloads": -1,
            "filename": "snappylapy-0.2.1.tar.gz",
            "has_sig": false,
            "md5_digest": "c3f61486e7856343c428ea58bc5e57ab",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": "<4.0,>=3.9",
            "size": 24725,
            "upload_time": "2025-01-13T11:14:07",
            "upload_time_iso_8601": "2025-01-13T11:14:07.344217Z",
            "url": "https://files.pythonhosted.org/packages/12/2a/012e8910edb11587a166bc8b6950a90b2db2820174d0c48fbcfb688d0d78/snappylapy-0.2.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-01-13 11:14:07",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "martinmoldrup",
    "github_project": "snappylapy",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "snappylapy"
}
        
Elapsed time: 0.41710s