Skip to main content

vcr-style record/replay stubbing library for Python methods

Project description

lu — a vcr-style record/replay stubbing library for Python methods

lu (录) — Chinese for "recording".

lu provides a small context manager, record(), which can patch module-qualified functions or instance methods to record return values (or exceptions) to compressed pickle files and generate a manifest.json describing the recordings.

On the first run, the patched methods execute normally and their outputs are recorded. Subsequent runs replay the recorded return values (or re-raise recorded exceptions) instead of executing the original code — similar to how HTTP cassette libraries like vcrpy work, but for arbitrary Python methods.

Quickstart

Install from PyPI:

pip install lu-python

Basic usage:

import lu

with lu.record(
    target={
        'module1.Foo.expensive_method1': ['arg1', 'kwarg1'],
        'package2.module2.expensive_method2': ['arg2'],
    },
    recordings_dir='tests/fixtures/recordings/',
):
    from module1 import Foo
    foo = Foo()
    foo.expensive_method('bar', kwarg1=3)

    from package2 import module2
    module2.expensive_method2('baz')

See tests/test_lu.py for more examples.

Using a YAML Configuration File

You can also define your recording configuration in a lu.yaml file:

targets:
  module1.Foo.expensive_method1: [arg1, kwarg1]
  package2.module2.expensive_method2: [arg2]
recordings_dir: tests/fixtures/recordings/
manifest_file: tests/fixtures/recordings/recordings.json  # optional
short_hex_length: 6  # optional

Then use it in your code:

import lu

with lu.record2():
    # your code here
    ...

CLI Usage

lu provides a command-line interface for managing recordings:

Remove recordings

Remove recordings that match a specific pattern:

lu remove PATTERN

This will:

  1. Search for entries in the manifest file that contain the pattern (in entry IDs or entry data)
  2. Remove matching entries from the manifest
  3. Delete the corresponding physical recording files

Example:

# Remove all recordings related to a specific test
lu remove "test_user_login"

# Remove recordings for a specific method
lu remove "expensive_method1"

You can also specify a custom configuration file:

lu remove PATTERN --config path/to/custom.yaml

Using with pytest

Add an autouse fixture (for example in tests/conftest.py) to enable recording/replay for the test session:

import pytest
import lu

@pytest.fixture(scope='session', autouse=True)
def stub_calls():
    with lu.record(
        target={
            # 'module.Class.method': ['arg_name', 'kwarg_name'],
            # add targets you want recorded here
        },
        recordings_dir='tests/fixtures/recordings/',
    ):
        yield

How lu differs from other tools

I wrote lu because I couldn't find a simple record/replay tool for arbitrary Python methods. A few alternatives exist but they don't match this use case:

  • testcontainers: runs real services in Docker, which requires setup and infrastructure and does not provide automatic request/response recording.
  • keploy: captures and replays network calls at a lower level; powerful but heavier-weight.
  • vcrpy / pytest-recordings: focused on HTTP clients (requests/urllib3) only.
  • unittest.mock: flexible but manual — lu provides an automated record-and-replay workflow on top of method patching.

lu works at the method level, so you can record calls to client libraries (e.g., database drivers, HTTP clients) or your own functions.

flowchart LR
    subgraph L0 [Business Logic]
        A1[your code]
        B1{{unittest.mock, lu}}
    end

    subgraph L1 [Client Library]
        A[requests/urllib3/pyodbc]
        B{{vcrpy, pytest-recordings<br>unittest.mock,lu}}
    end

    subgraph L2 [Network]
        C[TCP/IP]
        D{{keploy}}
    end

    subgraph L3 [Data]
        E[Database]
        F{{testcontainers}}
    end

    %% Business logic
    A1 --> A
    A --> C
    C --> E

Limitations

  • Prefer using lu for read-only calls. While you can record methods with side effects (for example, database writes), lu does not maintain external state. Replaying a recorded write will not update a database or other external system.
  • No transaction or rollback support. lu records and replays return values and exceptions only.

Contributing

Contributions and bug reports are welcome. Please open an issue or submit a pull request with tests that demonstrate the change or fix.

License

This project is provided under the terms of the repository license. See the LICENSE file for details.

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

lu_python-0.0.9.tar.gz (15.3 kB view details)

Uploaded Source

Built Distribution

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

lu_python-0.0.9-py3-none-any.whl (12.2 kB view details)

Uploaded Python 3

File details

Details for the file lu_python-0.0.9.tar.gz.

File metadata

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

File hashes

Hashes for lu_python-0.0.9.tar.gz
Algorithm Hash digest
SHA256 fbfdc5e3ef84e0e7a02ce39e56c841839f47bf8675541e435c9038556705adae
MD5 fec9330c4c587809ab660ff35c8ac445
BLAKE2b-256 bc69f85c5ae5148e7fe393bceaad92420651181829424264e54036e2ddd51baa

See more details on using hashes here.

Provenance

The following attestation bundles were made for lu_python-0.0.9.tar.gz:

Publisher: python-publish.yml on jackxxu/lu

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

File details

Details for the file lu_python-0.0.9-py3-none-any.whl.

File metadata

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

File hashes

Hashes for lu_python-0.0.9-py3-none-any.whl
Algorithm Hash digest
SHA256 b7e1c262a7d49897688270d1a4a8e5a3dc91a75f596a67bf006be9c2082e381e
MD5 34aaeb56bf3b15dc8cfcd83284a63e0a
BLAKE2b-256 fcd413485b7ec38b54e87a44b7201bff8d2a0bc5567ef7ff24a755ba0b353b50

See more details on using hashes here.

Provenance

The following attestation bundles were made for lu_python-0.0.9-py3-none-any.whl:

Publisher: python-publish.yml on jackxxu/lu

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