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_coderesponse.headersresponse.textresponse.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.AiohttpHttpClientcross_web.testing.clients.chalice.ChaliceHttpClientcross_web.testing.clients.django.DjangoHttpClientcross_web.testing.clients.django.AsyncDjangoHttpClientcross_web.testing.clients.flask.FlaskHttpClientcross_web.testing.clients.flask.AsyncFlaskHttpClientcross_web.testing.clients.litestar.LitestarHttpClientcross_web.testing.clients.quart.QuartHttpClientcross_web.testing.clients.sanic.SanicHttpClientcross_web.testing.clients.starlette.StarletteHttpClient
Notes
- All client methods are async, including clients wrapping sync frameworks.
files=uses(filename, content_bytes, content_type)tuples.ChaliceHttpClientsupports 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
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 cross_web-0.6.0.tar.gz.
File metadata
- Download URL: cross_web-0.6.0.tar.gz
- Upload date:
- Size: 331.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ae90570802615365ca1a781117b43bfd0d6cd3bf611649d24c3a206a82a693c9
|
|
| MD5 |
36074cf30dfd9b26bc9e8c116e563c86
|
|
| BLAKE2b-256 |
ad83b5ef04565acc065387dda3a4fbf0c4cfb6bab805c81b66b2bc5b5ac9a282
|
File details
Details for the file cross_web-0.6.0-py3-none-any.whl.
File metadata
- Download URL: cross_web-0.6.0-py3-none-any.whl
- Upload date:
- Size: 24.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
bdebf0c08d02f3a48cf67b6904d3a6d8fd8cab2cd905592ab96ab00b259cd582
|
|
| MD5 |
6dfb6c735f828aca68ee4730d0201b1e
|
|
| BLAKE2b-256 |
35a2dab06d9b80cb76c700883186a9a2e6fd103342c9b4def4d88f5787796e17
|