django-hatchway
Hatchway is an API framework inspired by the likes of FastAPI, but while trying to keep API views as much like standard Django views as possible.
It was built for, and extracted from, TakahΔ; if you want to see an example of it being used, browse its api app.
Installation
Install Hatchway from PyPI:
pip install django-hatchway
And add it to your INSTALLED_APPS
:
INSTALLED_APPS = [ ... "hatchway", ]
Usage
To make a view an API endpoint, you should write a standard function-based
view, and decorate it with @api_view.get
, @api_view.post
or similar:
from hatchway import api_view @api_view.get def my_api_endpoint(request, id: int, limit: int = 100) -> list[str]: ...
The types of your function arguments matter; Hatchway will use them to work out
where to get their values from and how to parse them. All the standard Python
types are supported, plus Pydantic-style models
(which ideally you should build based on the hatchway.Schema
base class,
as it understands how to load things from Django model instances).
Your return type also matters - this is what Hatchway uses to work out how to
format/validate the return value. You can leave it off, or set it to Any
,
if you don't want any return validation.
URL Patterns
You add API views in your urls.py
file like any other view:
urlpatterns = [ ... path("api/test/", my_api_endpoint), ]
The view will only accept the method it was decorated with (e.g. GET
for
api_view.get
).
If you want to have two or more views on the same URL but responding to
different methods, use Hatchway's methods
object:
from hatchway import methods urlpatterns = [ ... path( "api/post/<id>/", methods( get=posts.post_get, delete=posts.posts_delete, ), ), ]
Argument Sourcing
There are four places that input arguments can be sourced from:
- Path: The URL of the view, as provided via kwargs from the URL resolver
- Query: Query parameters (
request.GET
) - Body: The body of a request, in either JSON, formdata, or multipart format
- File: Uploaded files, as part of a multipart body
By default, Hatchway will pull arguments from these sources:
- Standard Python singular types (
int
,str
,float
, etc.): Path first, and then Query - Python collection types (
list[int]
, etc.): Query only, with implicit list conversion of either one or multiple values hatchway.Schema
/Pydantic BaseModel subclasses: Body only (see Model Sourcing below)django.core.files.File
: File only
You can override where Hatchway pulls an argument from by using one of the
Path
, Query
, Body
, File
, QueryOrBody
, PathOrQuery
,
or BodyDirect
annotations:
from hatchway import api_view, Path, QueryOrBody @api_view.post def my_api_endpoint(request, id: Path[int], limit: QueryOrBody[int] = 100) -> dict: ...
While Path
, Query
, Body
and File
force the argument to be
picked from only that source, there are some more complex ones in there:
PathOrQuery
first tries the Path, then tries the Query (the default for simple types)QueryOrBody
first tries the Query, then tries the BodyBodyDirect
forces top-level population of a model - see Model Sourcing, below.
Model Sourcing
When you define a hatchway.Schema
subclass (or any other pydantic model
subclass), Hatchway will presume that it should pull it from the POST/PUT/etc.
body.
How it pulls it depends on how many body-sourced arguments you have:
- If you just have one, it will feed it the top-level keys in the body data as its internal values.
- If you have more than one, it will look for its data in a sub-key named the same as the argument name.
For example, this function has two body-sourced things (one implicit, one explicit):
@api_view.post def my_api_endpoint(request, thing: schemas.MyInputSchema, limit: Body[int] = 100): ...
This means Hatchway will feed the schemas.MyInputSchema
model whatever it
finds under the thing
key in the request body as its input, and limit
will come from the limit
key.
If limit
wasn't specified, then there would be only one body-sourced item,
and Hatchway would feed schemas.MyInputSchema
the entire request body as
its input.
You can force a schema subclass to be fed the entire request body by using the
BodyDirect[MySchemaClass]
annotation on its type.
Return Values
The return value of an API view, if provided, is used to validate and coerce the type of the response:
@api_view.delete def my_api_endpoint(request) -> int: ...
It can be either a normal Python type, or a hatchway.Schema
subclass. If
it is a Schema subclass, the response will be fed to it for coercion, and ORM
objects are supported - returning a model instance, a dict with the model
instance values, or an instance of the schema are all equivalent.
A typechecker will honour these too, so we generally recommend returning instances of your Schema so that your entire view benefits from typechecking, rather than relying on the coercion. You'll get typechecking in your Schema subclass constructors, and then typechecking that you're always returnining the right things from the view.
You can also use generics like list[MySchemaClass]
or
dict[str, MySchemaClass]
as a response type; generally, anything Pydantic
allows, we do as well.
Adding Headers/Status Codes to the Response
If you want to do more to your response than just sling some data back at your client, you can return an ApiResponse object instead of a plain value:
from hatchway import api_view, ApiResponse @api_view.delete def my_api_endpoint(request) -> ApiResponse[int]: ... return ApiResponse(42, headers={"X-Safe-Delete": "no"})
ApiResponse
is a standard Django HTTPResponse
subclass, so accepts
almost all of the same arguments, and has most of the same methods. Just don't
edit its .content
value; if you want to mutate the data you passed into
it, that is stored in .data
.
Note that we also changed the return type of the view so that it would pass
typechecking; ApiResponse
accepts any response type as its argument and
passes it through to the same validation layer.
Auto-Collections
Hatchway allows you to say that Schema subclasses can pull their values from individual query parameters or body values; these are normally flat strings, though, unless you're looking at a JSON-encoded body, or multiple repeated query parameters.
However, it will respect the use of name[]
to make lists, and name[key]
to make dicts. Some examples:
- A
a=Query[list[int]]
argument will seeurl?a=1
as[1]
,url?a=1&a=2
as[1, 2]
, andurl?a[]=1&a[]=2
as[1, 2]
. - A
b=Body[dict[str, int]]
argument will correctly accept the POST datab[age]=30&b[height]=180
and give you{"age": 30, "height": 180}
.
These will also work in JSON bodies too, though of course you don't need them there; nevertheless, they still work for compatibility reasons.