Bocadillo 0.14 released!
April 21st, 2019 · Florimond Manca
All-async, app configuration, JSON data validation, route and query parameter validation, Bocadillo CLI, plugin framework, and more!
Warning: Bocadillo is now UNMAINTAINED. Users are recommended to migrate to a supported alternative, such as Starlette or FastAPI. Please see #344 for more information.
We're very excited to anounce that Bocadillo v0.14 is out! This release standardises how applications are structured and run, and brings new features that make working with Bocadillo even more productive.
If you have any questions or feedback about this release, feel free to get in touch!
IMPORTANT
Bocadillo 0.14.0 is incompatible with 0.13.x and earlier. We took the decision to introduce breaking changes to improve the API and the overall development workflow. You'll find tips on migrating from 0.13.x in this blog post.
Contents
Non backwards-compatible changes
All async
tl;dr
From 0.14 and onwards, when in doubt, use async def
.
Since the very first releases, Bocadillo was designed to support both synchronous and asynchronous syntax in a lot of places. This design principle was known as async-first.
However, this introduced complexity in the framework internals, and it was not clear where async was mandatory and where it was not. We realised that async-first was not the right approach.
For this reason, Bocadillo v0.14 and onwards will be all-async.
This means that synchronous syntax for the following features is not supported anymore: function-based HTTP views, methods of class-based HTTP views, error handlers, HTTP middleware callbacks, and HTTP hooks.
For example, you can't write this in Bocadillo v0.14+ anymore:
@app.route("/")
def index(req, res):
pass
Instead, you must use async def
:
@app.route("/")
async def index(req, res):
pass
To help with the transition:
- We've added checks to the framework that detect when you use a regular function whereas an asynchronous one is expected.
- We've added an async crash course with tips, common patterns, and resources for people getting started with Python async.
We hope this decision will foster your interest in learning about async Python, and make working with Bocadillo more straight-forward!
Application configuration
Before 0.14, configuration was primarily performed via arguments to App
, e.g. App(enable_hsts=True)
.
This proved to be hardly scalable, because:
- More and more parameters were added to
App
as we implemented new features. - Decomposing an app into multiple sub-apps required to configure them in the same way — when this didn't cause strange errors because of duplicated configuration.
To help with this, Bocadillo 0.14 completely changes the way applications are configured using a new configuration workflow supported by a lightweight plugin system.
This new way of doing things induces some necessary boilerplate, so to help you out we recommend you use the new Bocadillo CLI.
Configuration workflow
App configuration is now performed once and for all with the bocadillo.configure()
helper. It accepts various settings which are then made accessible anywhere using the bocadillo.settings
object.
As for where and how those settings should be defined, the TL;DR is: use a settings module.
To learn more, please refer to the new Configuration guide.
Plugins
Plugins are small pieces of functionality which are setup at configuration time.
Many features in Bocadillo are now implemented as plugins. As a result, App
does not take any parameter anymore. Instead, you should define plugin-specific settings in the settings module. These are specified in the plugins API reference.
What does this mean in practice? Well, instead of:
# project/app.py
from bocadillo import App
app = App(enable_hsts=True)
Use:
# project/settings.py
HSTS = True
# project/app.py
from bocadillo import App, configure
from . import settings
app = App()
configure(app, settings)
If this looks like writing more code, it's because it is — except you're now doing things right.
To help with this necessary boilerplate, be sure to check out Bocadillo CLI and its project creation command.
Serving apps
Previously, the idiomatic way to serve apps was to add
if __name__ == "__main__":
app.run()
at the end of the app.py
script and to run $ python app.py
.
We concluded that this early design decision was bad, because:
- It increases boilerplate.
- The
if __name__ == "__main__"
check can be confusing for beginners (and even more seasoned Python developers). app.run()
was merely a proxy foruvicorn.run()
with little to no added value.
For this reason, app.run()
has been removed entirely. This has two consequences:
- The official way to serve apps is now to use the
uvicorn
command. See Serving an application. - Debug mode has been removed. You can enable hot reload using the
--reload
option touvicorn
.
Here's a comparison of the minimum working application with hot reload enabled:
- Previously:
# app.py
from bocadillo import App
app = App()
if __name__ == "__main__":
app.run(debug=True)
python app.py
- Now:
from bocadillo import App, configure
app = App()
configure(app)
uvicorn app:app
res.json
Previously, sending a JSON response was performed using res.media
. We reckon that this was not very intuitive. Since media has been removed, we decided to replace res.media
by the more straight-forward res.json
:
@app.route("/health-check")
async def health_check(req, res):
# ❌ Previously:
res.media = {"status": "OK"}
# ✅ Now:
res.json = {"status": "OK"}
Redirect
exception
Previously, redirecting an HTTP request to another URL was performed using the app.redirect()
helper method. The fact that this method interrupted the execution of a view without a return
or a raise
felt like unnecessary magic. With the removal of named routes, we decided to fix this.
Redirections are now performed by raising a Redirect
exception (which is exactly what app.redirect()
did internally). Redirecting to another route by name is not supported anymore: you need to pass the full URL.
from bocadillo import App, Redirect
app = App()
@app.route("/hello/{name}")
async def hello(req, res, name: str):
res.text = f"Hello, {name}!"
@app.route("/")
async def index(req, res):
# ❌ Previously, one of:
app.redirect("hello", name="Stranger")
app.redirect(url="/hello/Stranger")
# ✅ Now:
raise Redirect("/hello/Stranger")
New features
Bocadillo CLI
Bocadillo CLI is a brand new CLI to help with bootstrapping Bocadillo projects. It aims at standardizing how Bocadillo projects are structured, and simplify the overall workflow, especially for beginners.
In particular, bocadillo create <PROJECT_NAME>
instantly generates a project with all the files you need to just start writing code, instead of figuring out how files should be structured! 🚀
Digging it? Give Bocadillo CLI a star!
Data validation
This one is pretty exciting. There's been a gap in the framework as to how JSON data should be validated, and with what tools.
In 0.14, you can now use the TypeSystem library to validate and serialize JSON data. 🎉
To learn more, read JSON validation.
Route parameter validation
Route parameters can now be validated using type annotations.
As a result, you can't use format specifiers anymore. So instead of {pk:d}
in the URL pattern, use:
@app.route("/users/{pk}")
async def get_user(req, res, pk: int):
pass
For advanced validation use cases, you can annotate the route parameter with a TypeSystem field.
Learn more in Route parameter validation and conversion.
Query parameter injection
Besides being available on req.query_params
, query parameters can now be injected into the view by declaring them as a parameter with a default.
Bonus points: query parameters injected like this are validated just like route parameters!
As a result, the following code:
@app.route("/users")
async def get_users(req, res):
token = req.query_params.get("token")
Can be refactored to:
@app.route("/users")
async def get_users(req, res, token: str = None):
pass
Learn more in Query parameters.
Exceptions in error handlers
Exceptions raised within an error handler are now fed back into the error handling pipeline. This allows to build modular and reusable error handlers.
For example, you can now re-raise an HTTPError
in a custom error handler:
from bocadillo import App, HTTPError
app = App()
class Nope(Exception):
pass
@app.error_handler(Nope)
async def on_nope(req, res, exc):
raise HTTPError(403, detail="We can't let you do this!")
Removed features
Media
The media framework allowed to configure the media type of an application globally and to abstract the exact serialization via media handlers. This feature brought inner complexity and quirks, and was very rarely used for anything else than JSON, the de-facto standard for Web APIs nowadays.
As a result, res.media
, the media_type
parameter to App
as well as @app.media_handler
and app.add_media_handler()
were removed.
To send JSON data in the response, use res.json instead.
Named routes
Previously, all HTTP routes were given a name which could be used to reverse URLs via the app.url_for()
helper.
Due to the complexity introduced compared to how useful this shortcut turns out to be, named routes and URL reversion have been removed. This means that:
@app.route()
does not have thename
andnamespace
parameters anymore.app.url_for()
and theurl_for
templates variable don't exist anymore.
If you want the full URL to a view, you'll simply need to build it yourself.
app.client
Test client with Accessing the test client via app.client
was deprecated in 0.13. It is now definitively removed. You must use bocadillo.create_client()
instead, as described in Testing.