Skip to main content

pytest plugin for HTTP testing using JSON files

Project description

image image image

pytest-httpchain

A pytest plugin for testing HTTP endpoints.

Overview

pytest-httpchain is an integration testing framework for HTTP APIs based on httpx lib.
It aims at helping with common HTTP API testing scenarios, where user needs to make several calls in specific order using data obtained along the way, like auth tokens or resource ids.

Why pytest-httpchain?

Testing HTTP APIs with plain pytest often leads to these pain points:

  • Boilerplate accumulates — Every test repeats the same setup: create client, set headers, make request, parse response, assert. The actual test intent gets buried.
  • Data threading is manual — When one call returns a token or ID needed by the next, you end up with fragile helper functions passing state around.
  • Common patterns get copy-pasted — Auth flows, base URLs, shared headers end up duplicated across test files. Fixtures might help, but they are not designed for that.
  • Code reviews are noisy — The actual test logic is rarely clear because of all the boilerplate and helpers, following changes gets overwhelming quickly.

pytest-httpchain offers a more structured approach.

Features

Declarative JSON format

Test scenarios are JSON documents that describe what to test, not how. No setup code to scroll through — the request and assertions are right there.

$ref with deep merging

Reuse arbitrary parts of your scenarios with JSONRef. Properties merge with type checking, so you can compose scenarios from shared fragments (auth flows, common headers, base URLs).

Multi-stage execution

Each scenario contains 1+ stages executed in order. One stage failure stops the chain. Use always_run for cleanup stages that should execute regardless.

Common data context

A key-value store persists throughout scenario execution. Variables, fixtures, and saved response data all live here. Use template expressions ({{ var }}) anywhere in your requests — substitution happens dynamically before each stage.

Response processing

  • JMESPath — Extract values from JSON responses directly
  • JSON Schema — Validate response structure against a schema
  • User functions — Call Python functions for custom extraction, verification, or authentication

Full pytest integration

Markers, fixtures, parametrization, and other plugins work as expected. You're not locked into a separate ecosystem.

Quick Start

Create a JSON test file named like test_<name>.<suffix>.json (default suffix is http):

# conftest.py
import pytest
from datetime import datetime

@pytest.fixture
def now_utc():
    return datetime.now()
{
    "substitutions": [
        {
            "vars": {
                "user_id": 1
            }
        }
    ],
    "stages": {
        "get_user": {
            "request": {
                "url": "https://api.example.com/users/{{ user_id }}"
            },
            "response": [
                {
                    "verify": {
                        "status": 200
                    }
                },
                {
                    "save": {
                        "jmespath": {
                            "user_name": "user.name"
                        }
                    }
                }
            ]
        },
        "update_user": {
            "fixtures": ["now_utc"],
            "request": {
                "url": "https://api.example.com/users/{{ user_id }}",
                "method": "PUT",
                "body": {
                    "json": {
                        "user": {
                            "name": "{{ user_name }}_updated",
                            "timestamp": "{{ str(now_utc) }}"
                        }
                    }
                }
            },
            "response": [
                {
                    "verify": {
                        "status": 200
                    }
                }
            ]
        },
        "cleanup": {
            "always_run": true,
            "request": {
                "url": "https://api.example.com/cleanup",
                "method": "POST"
            }
        }
    }
}

Scenario we created:

  • common data context is seeded with the first variable user_id
  • get_user
    url is assembled using user_id variable from common data context
    HTTP GET call is made
    we verify the call returned code 200
    assuming JSON body is returned, we extract a value by JMESPath expression user.name and save it to common data context under user_name key
  • update_user
    now_utc fixture value is injected into common data context
    url is assembled using user_id variable from common data context
    we create JSON body in place using values from common data context, note that now_utc is converted to string in place
    HTTP PUT call with body is made
    we verify the call returned code 200
  • cleanup
    finalizing call meant for graceful exit
    always_run parameter means this stage will be executed regardless of errors in previous stages

For detailed usage guide see the full documentation.

Installation

Install normally via package manager of your choice from PyPi:

pip install pytest-httpchain

or directly from Github, in case you need a particular ref:

pip install 'git+https://github.com/aeresov/pytest-httpchain@main'

Configuration

  • Test file discovery is based on this name pattern: test_<name>.<suffix>.json. The suffix is configurable as pytest ini option, default value is http.
  • $ref instructions can point to other files; absolute and relative paths are supported. You can limit the depth of relative path traversal using ref_parent_traversal_depth ini option, default value is 3.
  • Template expressions support list/dict comprehensions. You can limit the maximum comprehension length using max_comprehension_length ini option, default value is 50000.
  • Parallel stage iterations (repeat/foreach) have a safety limit configurable via max_parallel_iterations ini option, default value is 10000.

AI agent support

pytest-httpchain ships a scenario validator to help AI coding agents (and humans) author and check test scenarios.

Scenario validation

Validate scenario files for structure and common problems — undefined variables, variables referenced before they are saved (data-flow ordering), duplicate stage names, fixture/variable conflicts, no-op verify steps, and contradictory body checks:

uvx pytest-httpchain validate tests/test_login.http.json

Each finding carries a stable diagnostic code (HTTPCHAINxxx) and a severity. It exits non-zero when any file is invalid, so it doubles as a CI gate. Use --format json for machine-readable output (editor/CI integration):

uvx pytest-httpchain validate --format json tests/test_login.http.json

The same checks also run automatically at pytest collection time — semantic errors fail collection and warnings are reported — so pytest --collect-only validates every scenario in your suite.

For deeper, opt-in checks, add --deep: it imports your module:func references to confirm they resolve, checks their call signatures (including the injected response for save/verify functions), and verifies referenced files and schemas exist. Because it imports your code it is never run at collection time; pair it with --strict to fail CI on any warning, and --syspath to add import roots:

uvx pytest-httpchain validate --deep --strict tests/test_login.http.json

Editor schema

A JSON Schema is published for as-you-type validation and autocomplete. Reference it from your test files:

{
    "$schema": "https://aeresov.github.io/pytest-httpchain/schema/scenario.schema.json"
}

The hosted schema tracks the latest release; to pin the schema matching your installed version (e.g. for CI), emit it locally:

uvx pytest-httpchain schema --output scenario.schema.json

Inspecting scenarios

More read-only commands help author and debug scenarios offline — no network, no test run:

# Print a scenario with all $ref/$include/$merge inlined and deep-merged
uvx pytest-httpchain resolve tests/test_login.http.json

# Summarize stages and the variable data-flow (which stage saves what, who consumes it)
uvx pytest-httpchain show tests/test_login.http.json

# Render the stage data-flow as a Mermaid flowchart
uvx pytest-httpchain graph tests/test_login.http.json

Documentation

Thanks

This project was inspired by Tavern and pytest-play.

httpx does comms.
Pydantic keeps structure.
simpleeval powers templates.
pytest-order sorts chain.
pytest-datadir saved me a lot of elbow grease while testing.

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_httpchain-0.5.0.tar.gz (31.4 kB view details)

Uploaded Source

Built Distribution

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

pytest_httpchain-0.5.0-py3-none-any.whl (36.6 kB view details)

Uploaded Python 3

File details

Details for the file pytest_httpchain-0.5.0.tar.gz.

File metadata

  • Download URL: pytest_httpchain-0.5.0.tar.gz
  • Upload date:
  • Size: 31.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pytest_httpchain-0.5.0.tar.gz
Algorithm Hash digest
SHA256 4d69ffeae322363310d6d69795c2a4ea201ab77d2bc0a7eeff5a7c86c1ad3e4c
MD5 0af2d5eda98c0a0666349fd488cec620
BLAKE2b-256 caef6cf6db5cb3567d80b4fe32b245fd4b2afdc0c8d36520cf60709201be08ae

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_httpchain-0.5.0.tar.gz:

Publisher: publish.yml on aeresov/pytest-httpchain

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_httpchain-0.5.0-py3-none-any.whl.

File metadata

File hashes

Hashes for pytest_httpchain-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 a1c3dacfb2203aa12f5182006b2ac261687b8252682305166c5377eb821d3208
MD5 1d9f06b805c98a100e99215469302dfd
BLAKE2b-256 5e6c9e5b4a426cbec3173fbb82c335438c37380bf8325edd05217e39e798c373

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_httpchain-0.5.0-py3-none-any.whl:

Publisher: publish.yml on aeresov/pytest-httpchain

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