Give your Sanic API a UI and OpenAPI documentation, all for the price of free!
pip install sanic-openapi
Add Swagger UI with the OpenAPI spec:
from sanic_openapi import swagger_blueprint
app.blueprint(swagger_blueprint)
You'll now have a Swagger UI at the URL /swagger/
and an OpenAPI 2.0 spec at /swagger/swagger.json
.
Your routes will be automatically categorized by their blueprints.
For an example Swagger UI, see the Pet Store
from sanic_openapi import doc
@app.get("/user/<user_id:int>")
@doc.summary("Fetches a user by ID")
@doc.produces({ "user": { "name": str, "id": int } })
async def get_user(request, user_id):
...
@app.post("/user")
@doc.summary("Creates a user")
@doc.consumes(doc.JsonBody({"user": { "name": str }}), location="body")
async def create_user(request):
...
class Car:
make = str
model = str
year = int
class Garage:
spaces = int
cars = [Car]
@app.get("/garage")
@doc.summary("Gets the whole garage")
@doc.produces(Garage)
async def get_garage(request):
return json({
"spaces": 2,
"cars": [{"make": "Nissan", "model": "370Z"}]
})
class Car:
make = doc.String("Who made the car")
model = doc.String("Type of car. This will vary by make")
year = doc.Integer("4-digit year of the car", required=False)
class Garage:
spaces = doc.Integer("How many cars can fit in the garage")
cars = doc.List(Car, description="All cars in the garage")
garage = doc.JsonBody({
"spaces": doc.Integer,
"cars": [
{
"make": doc.String,
"model": doc.String,
"year": doc.Integer
}
]
})
@app.post("/store/garage")
@doc.summary("Stores a garage object")
@doc.consumes(garage, content_type="application/json", location="body")
async def store_garage(request):
store_garage(request.json)
return json(request.json)
app.config.API_VERSION = '1.0.0'
app.config.API_TITLE = 'Car API'
app.config.API_DESCRIPTION = 'Car API'
app.config.API_TERMS_OF_SERVICE = 'Use with caution!'
app.config.API_PRODUCES_CONTENT_TYPES = ['application/json']
app.config.API_CONTACT_EMAIL = '[email protected]'
By default, Sanic registers URIs both with and without a trailing /
. You may specify the type of the shown URIs by setting app.config.API_URI_FILTER
to one of the following values:
all
: Include both types of URIs.slash
: Only include URIs with a trailing/
.- All other values (and default): Only include URIs without a trailing
/
.
Just follow the OpenAPI 2.0 specification on this
app.config.API_HOST = 'subdomain.host.ext'
app.config.API_BASEPATH = '/v2/api/'
app.config.API_SECURITY = [
{
'authToken': []
}
]
app.config.API_SECURITY_DEFINITIONS = {
'authToken': {
'type': 'apiKey',
'in': 'header',
'name': 'Authorization',
'description': 'Paste your auth token and do not forget to add "Bearer " in front of it'
},
'OAuth2': {
'type': 'oauth2',
'flow': 'application',
'tokenUrl': 'https://your.authserver.ext/v1/token',
'scopes': {
'some_scope': 'Grants access to this API'
}
}
}
Here you can set any configuration described in the swagger-ui documentation
app.config.SWAGGER_UI_CONFIGURATION = {
'validatorUrl': None, # Disable Swagger validator
'displayRequestDuration': True,
'docExpansion': 'full'
}
@app.get("/garage/<id>")
@doc.summary("Gets the whole garage")
@doc.produces(Garage)
@doc.response(404, {"message": str}, description="When the garage cannot be found")
async def get_garage(request, id):
garage = some_fetch_function(id)
return json(garage)