Skip to main content

ASGI middleware for applying CORS headers to an ASGI application

Project description

asgi-cors

PyPI Tests Changelog License

ASGI middleware for applying CORS headers to an ASGI application.

Installation

pip install asgi-cors

Some background on CORS

CORS stands for Cross-Origin Resource Sharing. It is a web standard that allows applications to opt-in to allowing JavaScript running on other domains to make fetch() calls that can retrieve data from the application.

See MDN's CORS article for more background.

The easiest way to allow scripts running on other domains to access data from an application is to add the following HTTP header:

Access-Control-Allow-Origin: *

This will allow scripts running on ANY domain to make fetch() calls against the application. For public data this is often fine, but there are situations where this may not be what you want to do: one example might be code that runs behind a VPN and needs to allow specific, trusted hosts to load data without opening itself up to every site on the internet.

For these cases, the server needs to inspect the Origin header from the client and return that Origin in the above header. For example, an incoming request from http://localhost:8000 might be judged as trusted - in which case the application server needs to reply like so:

Access-Control-Allow-Origin: http://localhost:8000

Note that the Access-Control-Allow-Origin header can only return a single value. This means that if you want to allow requests from multiple origins you need to dynamically allowlist those origins and return a different header value depending on the incoming request.

Additionally if specific HTTP methods should be allowed an application should add:

Access-Control-Allow-Methods: GET, OPTIONS

Here GET and OPTIONS are allowed.

Similarly specific headers can be allowed:

Access-Control-Allow-Headers: content-type, Authorization

In this case content-type and Authorization headers are allowed to be sent to the server in a CORS request.

Verbs other than GET (such as POST) will trigger a preflight request. This is an OPTIONS request that the browser sends to the server to ask if the server will accept the request.

The access-control-max-age header can be used to specify how long the results of a preflight request can be cached. This can reduce the number of requests made to the server.

How to use this middleware

We will assume you have an existing ASGI app, in a variable called app.

First, import the asgi_cors function:

from asgi_cors import asgi_cors

To enable CORS headers for everywhere (by adding the Access-Control-Allow-Origin: * header to every request), do this:

app = asgi_cors(app, allow_all=True)

If you wish to only allow it from a specific host, use the following:

app = asgi_cors(app, hosts=[
    "https://www.example.com"
])

Now JavaScript executing on https://www.example.com will be able to call your API. You can test this out by opening up example.com in your browser, opening your browser's devtools console and pasting in the following JavaScript:

fetch("https://your-api.com/").then(r => r.json()).then(d => console.log(d))

You can include multiple hosts in the list.

If you want to open your application up to requests from a wildcard-defined selection of hosts, use the following:

app = asgi_cors(app, host_wildcards=[
    "http://localhost:800*",
    "http://*.example.com"
])

This will enable access from any JavaScript running on a local host server on ports 8000 through 8009 - or from any subdomain of example.com.

If you need to do something more complicated that cannot be expressed using the hosts= or host_wildcards= parameters, you can use callback= to specify a custom function. For example:

def validate_origin(origin):
    return origin.startswith("https://")

app = asgi_cors(app, callback=validate_origin)

Your callback function will be passed the Origin header that was passed in by the browser. Both regular and async functions are supported.

To add specific allowed headers or methods you can specify them with the headers= and methods= parameters:

app = asgi_cors(app, methods=[
    "GET", "OPTIONS"
], headers=[
    "Authorization","content-type"
])

To set a access-control-max-age header, use the max_age= parameter:

app = asgi_cors(app, host_wildcards=["*"], max_age=3600)

Using the middleware as a decorator

If you are defining your ASGI application directly as a function, you can use the asgi_cors_decorator function decorator like so:

from asgi_cors import asgi_cors_decorator


@asgi_cors_decorator(allow_all=True)
async def my_asgi_app(scope, receive, send):
    # Your app goes here

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

asgi-cors-1.0.1.tar.gz (8.2 kB view details)

Uploaded Source

Built Distribution

asgi_cors-1.0.1-py3-none-any.whl (8.5 kB view details)

Uploaded Python 3

File details

Details for the file asgi-cors-1.0.1.tar.gz.

File metadata

  • Download URL: asgi-cors-1.0.1.tar.gz
  • Upload date:
  • Size: 8.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.0.0 CPython/3.12.3

File hashes

Hashes for asgi-cors-1.0.1.tar.gz
Algorithm Hash digest
SHA256 068a88c824c513573041a4e26b0c2801fcefefe6d59bb470d475a84f59fd498b
MD5 d81895957804c985b62defa41b902719
BLAKE2b-256 4b7275047bc62e3d1bb2c7bc5cbabf7727b965e6a9130b6e1deac6ab72879c86

See more details on using hashes here.

File details

Details for the file asgi_cors-1.0.1-py3-none-any.whl.

File metadata

  • Download URL: asgi_cors-1.0.1-py3-none-any.whl
  • Upload date:
  • Size: 8.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.0.0 CPython/3.12.3

File hashes

Hashes for asgi_cors-1.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 075ee8359f2056ff8fa9d7751bdaeef30b40cf74040a25ffa3aaba61feca2c61
MD5 ab04855205dd30caf3650f5d59a9957d
BLAKE2b-256 b3653d13be6bf9b87ed327de72881bd9e3754789a57446962ff7d7cc1783b5cd

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page