Litestar API generation for halfORM projects.
Project description
half-orm-litestar
A halfORM extension that generates a Litestar REST API from your halfORM relation classes by decorating their methods.
Installation
pip install half-orm-litestar
Requires half-orm-dev to be installed (for project introspection):
pip install half-orm-dev
Quick start
1. Decorate your halfORM methods
In any halfORM relation class, import tools and annotate the methods you
want to expose as API routes:
# myproject/actor/user.py
from half_orm_litestar import tools
class User(MODEL.get_relation_class('actor.user')):
@tools.api_get('/user/{id: uuid}', guards=['public'])
async def get_user(self, request: "Request"):
actor_id = request.path_params['id']
return self.load_user(actor_id)
@tools.api_get('/user/{user_id: uuid}/group_accesses', guards=['has_user_access'])
async def get_user_group_accesses(self, request: "Request"):
actor_id = request.path_params['user_id']
return [dict(row) for row in self.rfk_group_accesses(actor_id=actor_id)]
@tools.api_post('/users', guards=['connected'])
async def create_user(self, request: "Request"):
data = await request.json()
...
The decorators accept the same arguments as the corresponding Litestar
decorators (@get, @post, @put, @patch, @delete).
Available decorators:
| Decorator | HTTP method |
|---|---|
tools.api_get |
GET |
tools.api_post |
POST |
tools.api_put |
PUT |
tools.api_patch |
PATCH |
tools.api_delete |
DELETE |
2. Generate the API
From the root of your half-orm-dev project:
half_orm litestar generate
This command:
- Scans all halfORM relation classes for
@api_*decorated methods. - Creates missing scaffolding files in
api/(only on first run — existing files are never overwritten). - Writes
api/main.py, the ready-to-run Litestar application.
3. Run the API
uvicorn api.main:application --reload
Project structure after generate
myproject/
├── api/
│ ├── __init__.py
│ ├── guards.py ← authentication guards (edit this)
│ ├── main.py ← generated, do not edit by hand
│ └── custom/
│ ├── __init__.py
│ ├── routes.py ← hand-written extra routes
│ └── middlewares/
│ ├── __init__.py ← extra middlewares list
│ └── authorization.py ← authentication middleware (edit this)
Files in api/ (except main.py) are created once and never overwritten.
Regenerating only rewrites main.py.
Customisation
Guards (api/guards.py)
Guards are async callables that receive the ASGI connection and raise an
exception to deny access. Two minimal guards are provided out of the box:
public (allow all) and connected (require authenticated user).
Add your own project-specific guards in api/guards.py:
# api/guards.py
from litestar.connection import ASGIConnection
from litestar.handlers.base import BaseRouteHandler
from litestar.exceptions import HTTPException
async def has_user_access(
connection: ASGIConnection, handler: BaseRouteHandler = None
) -> None:
"""Allow only the user identified by the ``user_id`` path parameter."""
if not connection.user:
raise HTTPException(status_code=401)
if connection.user != str(connection.path_params.get('user_id')):
raise HTTPException(status_code=403)
Then reference the guard by name in your decorator:
@tools.api_get('/user/{user_id: uuid}/profile', guards=['has_user_access'])
async def get_profile(self, request: "Request"):
...
Authorization middleware (api/custom/middlewares/authorization.py)
Implement the Authorization class to decode tokens / sessions and populate
connection.user. When the file exists, Authorization is automatically
placed first in the middleware stack.
# api/custom/middlewares/authorization.py
import jwt
from litestar.middleware import AbstractAuthenticationMiddleware, AuthenticationResult
from litestar.connection import ASGIConnection
SECRET = "change-me"
class Authorization(AbstractAuthenticationMiddleware):
async def authenticate_request(self, connection: ASGIConnection) -> AuthenticationResult:
token = (
connection.headers.get("Authorization", "")
.removeprefix("Bearer ")
.strip()
)
try:
payload = jwt.decode(token, SECRET, algorithms=["HS256"])
return AuthenticationResult(user=payload["sub"], auth=token)
except Exception:
return AuthenticationResult(user=None, auth=None)
Extra routes (api/custom/routes.py)
Hand-written Litestar route handlers can be added here:
# api/custom/routes.py
from litestar import get
@get('/health')
async def health_check() -> dict:
return {'status': 'ok'}
routes = [health_check]
Extra middlewares (api/custom/middlewares/__init__.py)
# api/custom/middlewares/__init__.py
from .my_middleware import MyMiddleware
middlewares = [MyMiddleware]
Path prefix
The generated routes are automatically prefixed with the project's database
name. For a project named mydb, '/user/{id: uuid}' becomes
'/mydb/user/{id: uuid}'.
Advanced: using without half-orm-dev
GenApi can be used programmatically without a Repo object by supplying the
relation classes directly:
from half_orm_litestar.generate import GenApi
GenApi(
relation_classes=my_classes, # iterable of (RelationClass, type) pairs
module_name='mydb',
base_dir='/path/to/project',
)
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file half_orm_litestar-0.17.0a2.tar.gz.
File metadata
- Download URL: half_orm_litestar-0.17.0a2.tar.gz
- Upload date:
- Size: 27.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6f8ee1e30397462b3fc1f995382f0acbedc10f5918227debb2bf7d1a9ff476de
|
|
| MD5 |
a89e9bfc5451d657302f190cd0b98514
|
|
| BLAKE2b-256 |
3510ad89d23a17009615e5f2b3dcc1dd887b9d9d8843e1502cba04df8919bbc2
|
File details
Details for the file half_orm_litestar-0.17.0a2-py3-none-any.whl.
File metadata
- Download URL: half_orm_litestar-0.17.0a2-py3-none-any.whl
- Upload date:
- Size: 25.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ddfee1c574c73010e01182c06e77f5eb1f11a7fdfc515f52772223b3f1652122
|
|
| MD5 |
3b0ae2f18a9db377b1746921a5066f0a
|
|
| BLAKE2b-256 |
0b3cfa98d80bb5b395ce469101c42f39f1d47627c175d989f9f05a1cda7ab88b
|