Skip to main content

A decorator for sharing costly resources between tests efficiently

Project description

Shared Resource Decorator

PyPI PyPI - Downloads Python Version

This package provides a simple decorator, @shared_resource(), which allows creation and sharing of resource instances between tests. Instead of repeatedly creating and then closing (or finalizing) the same resource in every test, a resource can be created once, shared across tests, and automatically finalized after all tests have finished.

Installation

Install the package via pip:

pip install vedro-shared-resource

Overview

In some testing scenarios, a resource is created and then closed or finalized separately in each test. For example:

# Test 1
resource = create_resource()
defer(resource.close)
# do something with resource

# Test 2
resource = create_resource()
defer(resource.close)
# do something with resource

(Note: Here, the defer function registers a callback to be executed after the current test scenario finishes)

Repeatedly creating the resource can lead to longer test execution times and increased resource consumption. By defining a shared resource that is created only once and cached using @shared_resource(), tests can run faster while ensuring that:

  • Performance Improvement: Expensive resource initialization is executed only once per unique set of arguments.
  • Consistency: All tests that request the resource with the same parameters receive the same shared instance.
  • Resource Management: The resource is finalized (closed) only once after all tests complete, reducing the risk of resource leaks.

Usage

1. Creating a Shared Resource

To share a resource, wrap the resource-creation function with the @shared_resource() decorator. Within the function, use defer_global to register the resource’s finalization callback.

from vedro_shared_resource import shared_resource
from vedro import defer_global

@shared_resource()
def create_shared_resource():
    resource = create_resource()
    defer_global(resource.close)
    return resource

(Note: While defer registers a callback for the current test scenario, defer_global registers a callback for the entire test suite)

2. Using the Shared Resource in Tests

When the decorated function is called in tests, the first call executes the function and caches its result. Subsequent calls return the cached resource immediately, avoiding redundant and expensive resource creation.

# Test 1
resource = create_shared_resource()
# perform operations with resource

# Test 2
resource = create_shared_resource()
# perform operations with resource

The @shared_resource() Decorator

The @shared_resource() decorator caches the result of the resource creation function:

  • First call: The resource creation function is executed, and its result is stored in a cache.
  • Subsequent calls: The cached resource is returned immediately, avoiding the expense of re-creating the resource.

If the function accepts arguments, they become part of the cache key, ensuring that each unique combination of arguments results in a separate cached resource.

Parameters

  • max_instances (int):
    Specifies the maximum number of unique cached results for the function. This limit is applied per resource function. Once the cache reaches this size, the least-recently-used entry is evicted.

  • type_sensitive (bool):
    If set to True, arguments of different types (for example, 1 vs. 1.0) are treated as distinct cache keys.

Under the Hood

  • For synchronous functions, the decorator leverages Python's built-in functools.lru_cache.
  • For asynchronous functions, it utilizes async_lru.alru_cache.

Use Cases Examples

Use Case 1: Sharing an Asynchronous Resource (HTTP Client)

An asynchronous HTTP client from the httpx library can be shared between tests. Although instantiating an AsyncClient might not be highly resource-intensive by itself, the official httpx documentation recommends reusing a single client instance to take full advantage of connection pooling. Instantiating multiple clients—especially inside a "hot loop"—can prevent efficient reuse of connections. Caching the client instance ensures that it is created only once and reused throughout the test suite, supporting optimal connection pooling.

from vedro_shared_resource import shared_resource
from vedro import defer_global
from httpx import AsyncClient

@shared_resource()
async def async_client() -> AsyncClient:
    client = AsyncClient()
    await client.__aenter__()

    defer_global(client.aclose)

    return client

Usage in tests:

client = await async_client()
response = await client.get("https://example.com")

(This is just an example. For better integration with httpx, consider using the vedro-httpx plugin)

Use Case 2: Sharing a Synchronous Resource (Web Browser)

Sharing a Chromium browser instance while accepting parameters allows different configurations to be cached separately. The resource creation function uses keyword arguments as part of the cache key, meaning that launching the Chromium browser with specified parameters (e.g., headless=False) is performed only once per configuration. Because launching a browser repeatedly can significantly slow down tests, caching the launched browser instance improves performance and ensures consistency across tests.

from vedro_shared_resource import shared_resource
from vedro import defer_global
from playwright.sync_api import sync_playwright, Browser

@shared_resource()
def chromium(**kwargs) -> Browser:
    playwright = sync_playwright().start()
    defer_global(playwright.stop)

    chromium = playwright.chromium.launch(**kwargs)
    defer_global(chromium.close)

    return chromium

Usage in tests:

browser = chromium(headless=False)
page = browser.new_page()
page.goto("https://example.com")

(This is just an example. For better integration with Playwright, consider using the vedro-pw plugin)

Caveats and Considerations

Introducing global variables and any kind of caching can sometimes lead to unexpected issues, such as shared state conflicts or challenges with test isolation. It is recommended to use resource caching only when resource creation is a true bottleneck and when sharing a resource does not compromise the reliability and correctness of tests.

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

vedro_shared_resource-0.2.1.tar.gz (8.9 kB view details)

Uploaded Source

Built Distribution

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

vedro_shared_resource-0.2.1-py3-none-any.whl (9.2 kB view details)

Uploaded Python 3

File details

Details for the file vedro_shared_resource-0.2.1.tar.gz.

File metadata

  • Download URL: vedro_shared_resource-0.2.1.tar.gz
  • Upload date:
  • Size: 8.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for vedro_shared_resource-0.2.1.tar.gz
Algorithm Hash digest
SHA256 1ac3d8e46e90c706b26dbb37cee4fe0cc870df9a7915647bdec936457c94cb6d
MD5 6208470d6a18eb8cb8f4e90c787d5df6
BLAKE2b-256 25b78fa49e83d2bbff4ebc220f0401cae69d7456a938821a75bcfef78ec27db3

See more details on using hashes here.

File details

Details for the file vedro_shared_resource-0.2.1-py3-none-any.whl.

File metadata

File hashes

Hashes for vedro_shared_resource-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 68064d305d1cb32358284b73f438c9996d4bda46fb16c48e4ed13d8f90c06c0c
MD5 c765b06b13c9abde7cef34132036a0ce
BLAKE2b-256 869cee2e6cfacd2d514766405f7ec989598374c3a741e27160bb87306966dadd

See more details on using hashes here.

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