CORS protection
Cross-Origin Resource Sharing (aka CORS) is a huge topic by itself. The documentation here cannot go into enough detail about what it is. You are highly encouraged to do some research on your own to understand the security problem presented by it, and the theory behind the solutions. MDN Web Docs are a great first step.
In super brief terms, CORS protection is a framework that browsers use to facilitate how and when a web page can access information from another domain. It is extremely relevant to anyone building a single-page application. Often times your frontend might be on a domain like https://portal.myapp.com
, but it needs to access the backend from https://api.myapp.com
.
The implementation here is heavily inspired by sanic-cors
, which is in turn based upon flask-cors
. It is therefore very likely that you can achieve a near drop-in replacement of sanic-cors
with sanic-ext
.
Basic implementation#
As shown in the example in the auto-endpoints example, Sanic Extensions will automatically enable CORS protection without further action. But, it does not offer too much out of the box.
At a bare minimum, it is highly recommended that you set config.CORS_ORIGINS
to the intended origin(s) that will be accessing the application.
from sanic import Sanic, text
from sanic_ext import Extend
app = Sanic(__name__)
app.config.CORS_ORIGINS = "http://foobar.com,http://bar.com"
Extend(app)
@app.get("/")
async def hello_world(request):
return text("Hello, world.")
$ curl localhost:8000 -X OPTIONS -i
HTTP/1.1 204 No Content
allow: GET,HEAD,OPTIONS
access-control-allow-origin: http://foobar.com
connection: keep-alive
Configuration#
The true power of CORS protection, however, comes into play once you start configuring it. Here is a table of all of the options.
Key | Type | Default | Description |
---|---|---|---|
CORS_ALLOW_HEADERS |
str or List[str] |
"*" |
The list of headers that will appear in access-control-allow-headers . |
CORS_ALWAYS_SEND |
bool |
True |
When True , will always set a value for access-control-allow-origin . When False , will only set it if there is an Origin header. |
CORS_AUTOMATIC_OPTIONS |
bool |
True |
When the incoming preflight request is received, whether to automatically set values for access-control-allow-headers , access-control-max-age , and access-control-allow-methods headers. If False these values will only be set on routes that are decorated with the @cors decorator. |
CORS_EXPOSE_HEADERS |
str or List[str] |
"" |
Specific list of headers to be set in access-control-expose-headers header. |
CORS_MAX_AGE |
str , int , timedelta |
0 |
The maximum number of seconds the preflight response may be cached using the access-control-max-age header. A falsey value will cause the header to not be set. |
CORS_METHODS |
str or List[str] |
"" |
The HTTP methods that the allowed origins can access, as set on the access-control-allow-methods header. |
CORS_ORIGINS |
str , List[str] , re.Pattern |
"*" |
The origins that are allowed to access the resource, as set on the access-control-allow-origin header. |
CORS_SEND_WILDCARD |
bool |
False |
If True , will send the wildcard * origin instead of the origin request header. |
CORS_SUPPORTS_CREDENTIALS |
bool |
False |
Whether to set the access-control-allow-credentials header. |
CORS_VARY_HEADER |
bool |
True |
Whether to add vary header, when appropriate. |
For the sake of brevity, where the above says List[str]
any instance of a list
, set
, frozenset
, or tuple
will be acceptable. Alternatively, if the value is a str
, it can be a comma delimited list.
Route level overrides#
It may sometimes be necessary to override app-wide settings for a specific route. To allow for this, you can use the @sanic_ext.cors()
decorator to set different route-specific values.
The values that can be overridden with this decorator are:
origin
expose_headers
allow_headers
allow_methods
supports_credentials
max_age
from sanic_ext import cors
app.config.CORS_ORIGINS = "https://foo.com"
@app.get("/", host="bar.com")
@cors(origin="https://bar.com")
async def hello_world(request):
return text("Hello, world.")