Skip to main content

A library for working with web frameworks

Project description

Cross

Write once, run everywhere - A universal web framework adapter for Python that lets you write code once and use it across multiple web frameworks.

Installation

uv add cross-web
from cross_web import Response

Overview

Cross provides a unified interface for common web framework operations, allowing you to write framework-agnostic code that can be easily adapted to work with FastAPI, Flask, Django, and other popular Python web frameworks.

Testing

This project is in early development! Cross also ships framework-specific test clients under cross_web.testing.clients.

Import a concrete client from its module:

from cross_web.testing.clients.starlette import StarletteHttpClient
from cross_web.testing.clients.flask import FlaskHttpClient
from cross_web.testing.clients.django import DjangoHttpClient

Do not import client classes from cross_web.testing.clients directly. The package only exports the shared base types so importing it does not pull optional framework dependencies.

The shared testing API lives in cross_web.testing:

from cross_web.testing import HttpClient, Response

Every concrete client exposes the same async interface:

  • await client.request(url, method, headers=None, **kwargs)
  • await client.get(url, headers=None, **kwargs)
  • await client.post(url, data=None, json=None, files=None, headers=None, **kwargs)

Response exposes:

  • response.status_code
  • response.headers
  • response.text
  • response.json

Example

import json

import pytest
from starlette.applications import Starlette
from starlette.requests import Request
from starlette.responses import JSONResponse
from starlette.routing import Route

from cross_web.request._starlette import StarletteRequestAdapter
from cross_web.testing.clients.starlette import StarletteHttpClient


async def echo(request: Request) -> JSONResponse:
    adapter = StarletteRequestAdapter(request)
    body = await adapter.get_body()

    return JSONResponse(
        {
            "method": adapter.method,
            "query": dict(adapter.query_params),
            "body": json.loads(body.decode()),
        }
    )


app = Starlette(routes=[Route("/echo", echo, methods=["POST"])])


@pytest.mark.asyncio
async def test_echo() -> None:
    client = StarletteHttpClient(app)

    response = await client.post("/echo?debug=1", json={"hello": "world"})

    assert response.status_code == 200
    assert response.json == {
        "method": "POST",
        "query": {"debug": "1"},
        "body": {"hello": "world"},
    }

Django

DjangoHttpClient and AsyncDjangoHttpClient take a Django view callable, not a Django app object.

Django must already be configured by your test environment. The recommended setup is pytest-django with a test settings module:

[tool.pytest.ini_options]
DJANGO_SETTINGS_MODULE = "myproject.test_settings"

Available clients

  • cross_web.testing.clients.aiohttp.AiohttpHttpClient
  • cross_web.testing.clients.chalice.ChaliceHttpClient
  • cross_web.testing.clients.django.DjangoHttpClient
  • cross_web.testing.clients.django.AsyncDjangoHttpClient
  • cross_web.testing.clients.flask.FlaskHttpClient
  • cross_web.testing.clients.flask.AsyncFlaskHttpClient
  • cross_web.testing.clients.litestar.LitestarHttpClient
  • cross_web.testing.clients.quart.QuartHttpClient
  • cross_web.testing.clients.sanic.SanicHttpClient
  • cross_web.testing.clients.starlette.StarletteHttpClient

Notes

  • All client methods are async, including clients wrapping sync frameworks.
  • files= uses (filename, content_bytes, content_type) tuples.
  • ChaliceHttpClient supports JSON requests but does not support form data or file uploads.
  • The testing clients are intended for tests and integration-style handler checks, not as production HTTP clients.

Documentation website

The repository now includes a Cross-Docs-powered website in website/.

cd website
just setup
just dev

That setup uses cross-docs for the Python side, @usecross/docs on the frontend, the Inertia Vite plugin, and Inertia v3 packages.

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

cross_web-0.7.0.tar.gz (332.4 kB view details)

Uploaded Source

Built Distribution

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

cross_web-0.7.0-py3-none-any.whl (25.2 kB view details)

Uploaded Python 3

File details

Details for the file cross_web-0.7.0.tar.gz.

File metadata

  • Download URL: cross_web-0.7.0.tar.gz
  • Upload date:
  • Size: 332.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.15 {"installer":{"name":"uv","version":"0.11.15","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for cross_web-0.7.0.tar.gz
Algorithm Hash digest
SHA256 15fbc8b9a824a055db8127fd6e43e0773074f620fdecb6b2b587d3d0a2bdd459
MD5 2108303f470a2fe114ebe200ed4e904f
BLAKE2b-256 4aa0bdb8370215987cc5bde497cb8b976b17ab14841566ecef2aab6ae3e6c7b0

See more details on using hashes here.

File details

Details for the file cross_web-0.7.0-py3-none-any.whl.

File metadata

  • Download URL: cross_web-0.7.0-py3-none-any.whl
  • Upload date:
  • Size: 25.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.15 {"installer":{"name":"uv","version":"0.11.15","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for cross_web-0.7.0-py3-none-any.whl
Algorithm Hash digest
SHA256 ddea9be3c68b48eaf16561847a5831a559786949c544b3701432e00a4e8d19d9
MD5 50ea1a5dbe5ae2c3a62bfea24e19ac4b
BLAKE2b-256 7a4a78b52ec2edcbd9b123638f4e0421fd8699ebe653ebf63ac5f82abd2665bc

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