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.5.0.tar.gz (330.8 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.5.0-py3-none-any.whl (24.3 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: cross_web-0.5.0.tar.gz
  • Upload date:
  • Size: 330.8 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

Hashes for cross_web-0.5.0.tar.gz
Algorithm Hash digest
SHA256 ff122cc5a68db5ce010cf00e78705177026a2bfedcd2c36e133cf8ba53e172d7
MD5 f4b48ee6ac172633f21e1e2098c72344
BLAKE2b-256 6e674098b8c3a27b01f0120b2d362ac4b43f2c70fa4dafaece88793756789c32

See more details on using hashes here.

File details

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

File metadata

  • Download URL: cross_web-0.5.0-py3-none-any.whl
  • Upload date:
  • Size: 24.3 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

Hashes for cross_web-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 c9cc4daf366736d3ae5e619ad5c64907144ef3d705bd4d32544708671bbbc952
MD5 0ff1a64d88dded4bdbc715e49250c58a
BLAKE2b-256 cd33c8212606805ad0d5a3feb742aeb7f336cae4a3893f8287644725e0b2d0d9

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