pytest-async-benchmark: Modern pytest benchmarking for async code. ๐
Project description
pytest-async-benchmark ๐
Modern pytest benchmarking for async code with beautiful terminal output and advanced comparison tools.
โจ Features
- ๐ฏ Async-First: Designed specifically for benchmarking
async deffunctions - ๐ Pytest Integration: Seamless integration as a pytest plugin with full pytest-asyncio support
- ๐จ Rich Output: Beautiful terminal reporting powered by Rich!
- ๐ Comprehensive Stats: Min, max, mean, median, std dev, percentiles, and more
- โ๏ธ A vs B Comparisons: Compare different implementations side-by-side
- ๐ Multi-Scenario Analysis: Benchmark multiple scenarios with detailed comparison tables
- ๐ฏ Performance Grading: Automatic performance scoring and analysis
- โก Auto Calibration: Intelligent round and iteration detection
- ๐ Quick Compare: One-line comparison utilities
- ๐ Winner Detection: Automatic identification of best-performing implementation
- ๐ Easy to Use: Simple fixture-based API with native
async/awaitsupport - ๐ง pytest-asyncio Compatible: Works perfectly with pytest-asyncio's event loop management
๐ฆ Installation
For Existing pytest-asyncio Users ๐
Already testing async APIs (FastAPI, Quart, aiohttp)? You're all set with the basic installation:
pip install pytest-async-benchmark
# or
uv add pytest-async-benchmark
You'll get the full async/await experience immediately since you already have pytest-asyncio!
For New Users
Choose your installation based on your needs:
# Full installation with async/await support (recommended)
pip install pytest-async-benchmark[asyncio]
uv add pytest-async-benchmark --optional asyncio
# Basic installation (simple interface)
pip install pytest-async-benchmark
uv add pytest-async-benchmark
Quick Check โ
Already using @pytest.mark.asyncio in your tests? Then the basic installation is all you need:
# If you already have tests like this:
@pytest.mark.asyncio
async def test_my_api():
# Your existing async test code
pass
# Then just add pytest-async-benchmark and use:
@pytest.mark.asyncio
async def test_my_api_performance(async_benchmark):
result = await async_benchmark(my_async_function)
assert result['mean'] < 0.01
๐ Quick Start
pytest-async-benchmark automatically adapts to your environment, providing two convenient interfaces:
Async Interface (with pytest-asyncio)
When pytest-asyncio is installed, use the natural async/await syntax:
import asyncio
import pytest
async def slow_async_operation():
await asyncio.sleep(0.01) # 10ms
return "result"
@pytest.mark.asyncio
@pytest.mark.async_benchmark(rounds=5, iterations=10)
async def test_async_performance(async_benchmark):
# Use await with pytest-asyncio for best experience
result = await async_benchmark(slow_async_operation)
# Your assertions here
assert result['mean'] < 0.02
Simple Interface (without pytest-asyncio)
For simpler setups, the sync interface works automatically:
import asyncio
import pytest
async def slow_async_operation():
await asyncio.sleep(0.01)
return "result"
@pytest.mark.async_benchmark(rounds=5, iterations=10)
def test_sync_performance(async_benchmark):
# No await needed - sync interface
result = async_benchmark(slow_async_operation)
# Your assertions here
assert result['mean'] < 0.02 # Should complete in under 20ms
Flexible Configuration Syntax
pytest-async-benchmark supports two syntax options for configuring benchmarks:
Option 1: Marker Syntax (Recommended)
@pytest.mark.async_benchmark(rounds=5, iterations=10)
async def test_with_marker(async_benchmark):
result = await async_benchmark(slow_async_operation)
assert result['rounds'] == 5 # From marker
Option 2: Function Parameter Syntax
async def test_with_parameters(async_benchmark):
result = await async_benchmark(slow_async_operation, rounds=5, iterations=10)
assert result['rounds'] == 5 # From function parameters
๐ง Interface Detection & Flexibility
pytest-async-benchmark automatically detects your environment and provides the best interface:
With pytest-asyncio (Recommended)
When pytest-asyncio is installed, use natural async/await syntax:
# Set in your pyproject.toml for automatic async test detection
[tool.pytest.ini_options]
asyncio_mode = "auto"
# Then use await syntax
@pytest.mark.asyncio
async def test_my_benchmark(async_benchmark):
result = await async_benchmark(my_async_function)
# Your assertions here
Benefits of pytest-asyncio integration:
- โ
Native
async/awaitsyntax support - โ Automatic event loop management
- โ
No
RuntimeError: cannot be called from a running event loop - โ Better compatibility with async frameworks like FastAPI, Quart, aiohttp
- โ Cleaner test code with standard async patterns
Without pytest-asyncio (Simple Setup)
When pytest-asyncio is not available, the simple interface works automatically:
# No pytest-asyncio required
def test_my_benchmark(async_benchmark):
result = async_benchmark(my_async_function) # No await needed
# Your assertions here
Benefits of simple interface:
- โ No additional dependencies required
- โ Simpler setup for basic use cases
- โ Perfect for getting started quickly
- โ Automatic event loop management internally
๐ฏ Core Usage Examples
Interface Flexibility
Async Interface (with pytest-asyncio)
@pytest.mark.asyncio
@pytest.mark.async_benchmark(rounds=10, iterations=100, warmup_rounds=2)
async def test_with_marker(async_benchmark):
"""Use marker for consistent, visible configuration."""
result = await async_benchmark(my_async_function)
assert result['rounds'] == 10 # Configuration is explicit and visible
Simple Interface (without pytest-asyncio)
@pytest.mark.async_benchmark(rounds=10, iterations=100, warmup_rounds=2)
def test_with_marker_sync(async_benchmark):
"""Use marker for consistent, visible configuration - simple style."""
result = async_benchmark(my_async_function) # No await needed
assert result['rounds'] == 10 # Configuration is explicit and visible
Two Configuration Syntaxes
Marker Syntax (Declarative)
# With pytest-asyncio
@pytest.mark.asyncio
@pytest.mark.async_benchmark(rounds=10, iterations=100, warmup_rounds=2)
async def test_with_marker_async(async_benchmark):
result = await async_benchmark(my_async_function)
assert result['rounds'] == 10
# Without pytest-asyncio
@pytest.mark.async_benchmark(rounds=10, iterations=100, warmup_rounds=2)
def test_with_marker_sync(async_benchmark):
result = async_benchmark(my_async_function)
assert result['rounds'] == 10
Function Parameter Syntax (Dynamic)
# With pytest-asyncio
@pytest.mark.asyncio
async def test_with_parameters_async(async_benchmark):
result = await async_benchmark(
my_async_function,
rounds=10,
iterations=100,
warmup_rounds=2
)
assert result['rounds'] == 10
# Without pytest-asyncio
def test_with_parameters_simple(async_benchmark):
result = async_benchmark(
my_async_function,
rounds=10,
iterations=100,
warmup_rounds=2
)
assert result['rounds'] == 10
Combined Syntax (Override)
Both interfaces support parameter precedence where function parameters override marker settings:
@pytest.mark.async_benchmark(rounds=5, iterations=50) # Default config
async def test_with_override(async_benchmark): # Works with or without @pytest.mark.asyncio
"""Function parameters override marker settings."""
result = await async_benchmark( # Use 'await' only with pytest-asyncio
my_async_function,
rounds=20 # This overrides marker's rounds=5
# iterations=50 comes from marker
)
assert result['rounds'] == 20 # Function parameter wins
assert result['iterations'] == 50 # From marker
Basic Benchmarking
@pytest.mark.asyncio
async def test_my_async_function(async_benchmark):
async def my_function():
# Your async code here
await some_async_operation()
return result
# Benchmark with default settings (5 rounds, 1 iteration each)
stats = await async_benchmark(my_function)
# Access comprehensive timing statistics
print(f"Mean execution time: {stats['mean']:.3f}s")
print(f"Standard deviation: {stats['stddev']:.3f}s")
print(f"95th percentile: {stats['p95']:.3f}s")
Advanced Configuration
๐ฏ Two Configuration Syntaxes
pytest-async-benchmark offers two flexible ways to configure your benchmarks:
๐ท๏ธ Marker Syntax (Recommended)
Use pytest markers for declarative, visible configuration:
@pytest.mark.asyncio
@pytest.mark.async_benchmark(rounds=10, iterations=100, warmup_rounds=2)
async def test_high_precision_benchmark(async_benchmark):
"""High precision benchmark with marker configuration."""
result = await async_benchmark(my_async_function)
# Configuration is visible and consistent
assert result['rounds'] == 10
assert result['iterations'] == 100
Benefits:
- โ Visible configuration - Parameters are clear at test level
- โ IDE support - Better tooling and autocomplete
- โ Test discovery - Easy to find all benchmark tests
- โ Consistent configs - Same settings across related tests
โ๏ธ Function Parameter Syntax
Use function parameters for dynamic, flexible configuration:
@pytest.mark.asyncio
async def test_dynamic_benchmark(async_benchmark):
"""Dynamic benchmark with runtime configuration."""
# Configuration can be computed or conditional
rounds = 20 if is_production else 5
result = await async_benchmark(
my_async_function,
rounds=rounds,
iterations=50,
warmup_rounds=1
)
Benefits:
- โ Dynamic configuration - Runtime parameter calculation
- โ Conditional logic - Different configs based on environment
- โ Per-call customization - Each benchmark call can differ
๐ Combined Syntax (Best of Both)
Function parameters override marker parameters:
@pytest.mark.asyncio
@pytest.mark.async_benchmark(rounds=5, iterations=50, warmup_rounds=1)
async def test_with_overrides(async_benchmark):
"""Use marker defaults with selective overrides."""
# Quick test with marker defaults
quick_result = await async_benchmark(fast_function)
# Precision test with overridden rounds
precise_result = await async_benchmark(
slow_function,
rounds=20 # Overrides marker's rounds=5
# iterations=50 and warmup_rounds=1 come from marker
)
assert quick_result['rounds'] == 5 # From marker
assert precise_result['rounds'] == 20 # From function override
Traditional Configuration
@pytest.mark.asyncio
async def test_with_custom_settings(async_benchmark):
result = await async_benchmark(
my_async_function,
rounds=10, # Number of rounds to run
iterations=5, # Iterations per round
warmup_rounds=2 # Warmup rounds before measurement
)
With Function Arguments
@pytest.mark.asyncio
async def test_with_args(async_benchmark):
async def process_data(data, multiplier=1):
# Process the data
await asyncio.sleep(0.01)
return len(data) * multiplier
result = await async_benchmark(
process_data,
"test_data", # positional arg
multiplier=2, # keyword arg
rounds=3
)
โ๏ธ A vs B Comparison Features
Quick Comparison
from pytest_async_benchmark import quick_compare
async def algorithm_v1():
await asyncio.sleep(0.002) # 2ms
return "v1_result"
async def algorithm_v2():
await asyncio.sleep(0.0015) # 1.5ms - optimized
return "v2_result"
# Quick one-liner comparison
def test_algorithm_comparison():
winner, results = quick_compare(algorithm_v1, algorithm_v2, rounds=5)
assert winner == "algorithm_v2" # v2 should be faster
Detailed A vs B Analysis
from pytest_async_benchmark import a_vs_b_comparison
def test_detailed_comparison():
# Compare with beautiful terminal output
a_vs_b_comparison(
"Original Algorithm", algorithm_v1,
"Optimized Algorithm", algorithm_v2,
rounds=8, iterations=20
)
Multi-Scenario Benchmarking
from pytest_async_benchmark import BenchmarkComparator
def test_multi_scenario():
comparator = BenchmarkComparator()
# Add multiple scenarios
comparator.add_scenario(
"Database Query v1", db_query_v1,
rounds=5, iterations=10,
description="Original database implementation"
)
comparator.add_scenario(
"Database Query v2", db_query_v2,
rounds=5, iterations=10,
description="Optimized with connection pooling"
)
# Run comparison and get results
results = comparator.run_comparison()
# Beautiful comparison table automatically displayed
# Access programmatic results
fastest = results.get_fastest_scenario()
assert fastest.name == "Database Query v2"
๐ Comprehensive Statistics
Each benchmark returns detailed statistics:
{
'min': 0.001234, # Minimum execution time
'max': 0.005678, # Maximum execution time
'mean': 0.002456, # Mean execution time
'median': 0.002123, # Median execution time
'stddev': 0.000234, # Standard deviation
'p50': 0.002123, # 50th percentile (median)
'p90': 0.003456, # 90th percentile
'p95': 0.004123, # 95th percentile
'p99': 0.004789, # 99th percentile
'rounds': 5, # Number of rounds executed
'iterations': 1, # Number of iterations per round
'raw_times': [...], # List of raw timing measurements
'grade': 'A', # Performance grade (A-F)
'grade_score': 87.5 # Numeric grade score (0-100)
}
๐จ Beautiful Terminal Output
Basic Benchmark Output
๐ Async Benchmark Results: test_my_function
โโโโโโโโโโโโโโโณโโโโโโโโโโโโโ
โ Metric โ Value โ
โกโโโโโโโโโโโโโโโโโโโโโโโโโโโฉ
โ Min โ 10.234ms โ
โ Max โ 15.678ms โ
โ Mean โ 12.456ms โ
โ Median โ 12.123ms โ
โ Std Dev โ 1.234ms โ
โ 95th %ile โ 14.567ms โ
โ 99th %ile โ 15.234ms โ
โ Grade โ A (87.5) โ
โ Rounds โ 5 โ
โ Iterations โ 1 โ
โโโโโโโโโโโโโโโดโโโโโโโโโโโโโ
โ
Benchmark completed successfully!
A vs B Comparison Output
โ๏ธ A vs B Comparison Results
โโโโโโโโโโโโโโโโโโโโโโโโโโโณโโโโโโโโโโโโโโณโโโโโโโโโโโโโโณโโโโโโโโโโโโ
โ Scenario โ Algorithm A โ Algorithm B โ Winner โ
โกโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฉ
โ Mean Time โ 2.456ms โ 1.789ms โ B ๐ โ
โ Median Time โ 2.234ms โ 1.678ms โ B ๐ โ
โ 95th Percentile โ 3.456ms โ 2.345ms โ B ๐ โ
โ Standard Deviation โ 0.567ms โ 0.234ms โ B ๐ โ
โ Performance Grade โ B (76.2) โ A (89.1) โ B ๐ โ
โ Improvement โ - โ 27.2% โ - โ
โโโโโโโโโโโโโโโโโโโโโโโโโโโดโโโโโโโโโโโโโโดโโโโโโโโโโโโโโดโโโโโโโโโโโโ
๐ Winner: Algorithm B (27.2% faster)
๐๏ธ Project Structure
pytest-async-benchmark/
โโโ src/
โ โโโ pytest_async_benchmark/
โ โโโ __init__.py # Main exports and API
โ โโโ plugin.py # Pytest plugin and fixtures
โ โโโ runner.py # Core benchmarking engine
โ โโโ display.py # Rich terminal output formatting
โ โโโ stats.py # Statistical calculations
โ โโโ utils.py # Utility functions
โ โโโ analytics.py # Performance analysis tools
โ โโโ comparison.py # A vs B comparison functionality
โโโ examples/
โ โโโ pytest_examples.py # Comprehensive pytest usage examples
โ โโโ quart_api_comparison.py # Real-world API endpoint comparison
โ โโโ comparison_examples.py # Advanced comparison features demo
โโโ tests/
โ โโโ test_async_bench.py # Core functionality tests
โ โโโ test_comparison.py # Comparison feature tests
โ โโโ test_demo.py # Demo test cases
โ โโโ conftest.py # Test configuration
โโโ pyproject.toml # Package configuration
โโโ README.md # This file
๐ Example Files Guide
๐ง examples/pytest_examples.py
Comprehensive pytest usage examples including:
- Basic benchmarking with the
async_benchmarkfixture - Advanced configuration options
- Performance assertions and testing patterns
- Using markers for benchmark organization
๐ examples/quart_api_comparison.py
Real-world API endpoint comparison demo featuring:
- Quart web framework setup
- API v1 vs v2 endpoint benchmarking
- Live server testing with actual HTTP requests
- Performance regression detection
โ๏ธ examples/comparison_examples.py
Advanced comparison features showcase:
- Multi-scenario benchmark comparisons
- A vs B testing with detailed analysis
- Performance grading and scoring
- Statistical comparison utilities
๐ Real-World Examples
FastAPI Endpoint Benchmarking
from fastapi import FastAPI
from fastapi.testclient import TestClient
import pytest
app = FastAPI()
@app.get("/api/data")
async def get_data():
# Simulate database query
await asyncio.sleep(0.005)
return {"data": "example"}
@pytest.mark.asyncio
async def test_fastapi_endpoint_performance(async_benchmark):
async def make_request():
with TestClient(app) as client:
response = client.get("/api/data")
return response.json()
result = await async_benchmark(make_request, rounds=10)
assert result['mean'] < 0.1 # Should respond within 100ms
assert result['grade'] in ['A', 'B'] # Should have good performance grade
Quart API Endpoint Comparison
See the complete example in examples/quart_api_comparison.py:
from pytest_async_benchmark import a_vs_b_comparison
import asyncio
import aiohttp
async def test_api_v1():
async with aiohttp.ClientSession() as session:
async with session.get('http://localhost:5000/api/v1/data') as resp:
return await resp.json()
async def test_api_v2():
async with aiohttp.ClientSession() as session:
async with session.get('http://localhost:5000/api/v2/data') as resp:
return await resp.json()
# Compare API versions
a_vs_b_comparison(
"API v1", test_api_v1,
"API v2 (Optimized)", test_api_v2,
rounds=10, iterations=5
)
Database Query Benchmarking
@pytest.mark.asyncio
async def test_database_query_performance(async_benchmark):
async def fetch_user_data(user_id):
async with database.connection() as conn:
return await conn.fetch_one(
"SELECT * FROM users WHERE id = ?", user_id
)
result = await async_benchmark(fetch_user_data, 123, rounds=5)
assert result['mean'] < 0.05 # Should complete within 50ms
assert result['p95'] < 0.1 # 95% of queries under 100ms
๐ฏ Using Markers
@pytest.mark.async_benchmark
@pytest.mark.asyncio
async def test_performance(async_benchmark):
# Your benchmark test
result = await async_benchmark(my_async_function)
assert result is not None
๐ API Reference
async_benchmark(func, *args, rounds=None, iterations=None, warmup_rounds=1, **kwargs)
Parameters:
func: The async function to benchmark*args: Positional arguments to pass to the functionrounds: Number of measurement rounds (default: 5)iterations: Number of iterations per round (default: 1)warmup_rounds: Number of warmup rounds before measurement (default: 1)**kwargs: Keyword arguments to pass to the function
Returns: A dictionary with comprehensive statistics including min, max, mean, median, stddev, percentiles, performance grade, and raw measurements.
Comparison Functions
quick_compare(func_a, func_b, **kwargs): Quick comparison returning winner and resultsa_vs_b_comparison(name_a, func_a, name_b, func_b, **kwargs): Detailed comparison with terminal outputBenchmarkComparator: Class for multi-scenario benchmarking and analysis
๐ Requirements
- Python โฅ 3.9
- pytest โฅ 8.3.5
- pytest-asyncio โฅ 0.23.0 (automatically installed)
Note: Rich (for beautiful terminal output) is automatically installed as a dependency.
๐ Development
# Clone the repository
git clone https://github.com/yourusername/pytest-async-benchmark.git
cd pytest-async-benchmark
# Install dependencies
uv sync
# Run tests
uv run pytest tests/ -v
# Run examples
uv run pytest examples/pytest_examples.py -v
# Test real-world Quart API comparison
uv run python examples/quart_api_comparison.py
# See advanced comparison features
uv run python examples/comparison_examples.py
๐ ๏ธ Code Quality and Formatting
This project uses Ruff for both linting and formatting:
# Check code for linting issues
uv run ruff check .
# Fix auto-fixable linting issues
uv run ruff check . --fix
# Check code formatting
uv run ruff format --check .
# Format code automatically
uv run ruff format .
# Run both linting and formatting in one go
uv run ruff check . --fix && uv run ruff format .
# Run all quality checks at once (linting, formatting, and tests)
uv run python scripts/quality-check.py
๐ Release Readiness Check
Before creating a release, verify everything is ready:
# Run comprehensive release check
uv run python scripts/release-check.py
# This checks:
# โ
Git repository status
# โ
Version consistency
# โ
Code formatting and linting
# โ
Test suite passes
# โ
Package builds successfully
# โ
All required files exist
๐ Quick Quality Check
Run all quality checks at once:
# Run linting, formatting, tests, and release checks
python scripts/quality-check.py
# This will:
# ๐ง Fix linting issues automatically
# ๐จ Format code with Ruff
# ๐งช Run the full test suite
# ๐ Check release readiness
๐ Automated Releases
This project uses GitHub Actions for automated testing and publishing to PyPI:
- Continuous Integration: Tests run on every push for Python 3.9-3.13
- Test Publishing: Automatic uploads to TestPyPI for testing releases
- Production Releases: Secure publishing to PyPI using trusted publishing
- Release Validation: Comprehensive checks ensure package quality
Creating a Release
- Update version in
pyproject.tomlandsrc/pytest_async_benchmark/__init__.py - Run
uv run python scripts/release-check.pyto verify readiness - Create a git tag:
git tag v1.0.0 && git push origin v1.0.0 - Create a GitHub release to trigger automated PyPI publishing
See RELEASE_GUIDE.md for detailed release instructions.
๐ค Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
๐ License
MIT License - see LICENSE file for details.
Built with โค๏ธ for the async Python community
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pytest_async_benchmark-0.2.0.tar.gz.
File metadata
- Download URL: pytest_async_benchmark-0.2.0.tar.gz
- Upload date:
- Size: 181.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eed7eb2ec810709f263db3234add2bc2f50ca5553c083f98bcce2ec86634df65
|
|
| MD5 |
ab446bd468800895b8db08be2a4b735c
|
|
| BLAKE2b-256 |
d5097034885e055ad6f449f97e0b9a0e06343ac17f01bf2a5cc94b5e546f7680
|
Provenance
The following attestation bundles were made for pytest_async_benchmark-0.2.0.tar.gz:
Publisher:
release.yml on mihailtd/pytest-async-benchmark
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pytest_async_benchmark-0.2.0.tar.gz -
Subject digest:
eed7eb2ec810709f263db3234add2bc2f50ca5553c083f98bcce2ec86634df65 - Sigstore transparency entry: 222864136
- Sigstore integration time:
-
Permalink:
mihailtd/pytest-async-benchmark@56376f653558cc6a04c98125ed8ae959e38522c4 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/mihailtd
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@56376f653558cc6a04c98125ed8ae959e38522c4 -
Trigger Event:
push
-
Statement type:
File details
Details for the file pytest_async_benchmark-0.2.0-py3-none-any.whl.
File metadata
- Download URL: pytest_async_benchmark-0.2.0-py3-none-any.whl
- Upload date:
- Size: 18.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7d2cd23c6c2ce2630849c38b9eb742df1b74ea378ecc53ca7b7754ba9bdc3683
|
|
| MD5 |
b99323946a936dfdce9362432bd54f80
|
|
| BLAKE2b-256 |
4b170ff79598ff109e8df60dcd455bcb98fef00281491a2f8a425a039815345d
|
Provenance
The following attestation bundles were made for pytest_async_benchmark-0.2.0-py3-none-any.whl:
Publisher:
release.yml on mihailtd/pytest-async-benchmark
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pytest_async_benchmark-0.2.0-py3-none-any.whl -
Subject digest:
7d2cd23c6c2ce2630849c38b9eb742df1b74ea378ecc53ca7b7754ba9bdc3683 - Sigstore transparency entry: 222864147
- Sigstore integration time:
-
Permalink:
mihailtd/pytest-async-benchmark@56376f653558cc6a04c98125ed8ae959e38522c4 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/mihailtd
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@56376f653558cc6a04c98125ed8ae959e38522c4 -
Trigger Event:
push
-
Statement type: