Pydantic serialization for tortoise-orm
Project description
Tortoise Serializer
Motivation
This project was created to address some of the limitations of pydantic_model_creator, including:
- The ability to use a
contextin serialization at the field level. - Access to the actual Tortoise
Modelinstance during serialization. - Improved readability.
- Support for adding extra logic to specific serializers.
- The ability to document fields in a way that is visible in Swagger.
Usefull readings
Installation
pip add tortoise-serializer
Core concept
A Serializer does not need to know which model it will serialize. For example:
from tortoise_serializer import Serializer
class ItemByNameSerializer(Serializer):
id: int
name: str
products = await ItemByNameSerializer.from_queryset(Product.all())
users = await ItemByNameSerializer.from_queryset(User.all())
This is entirely valid.
Serializers are pydantic.BaseModel objects, which means you can directly return them from FastAPI endpoints or use any functionality provided by BaseModel.
Usage
Reading
from tortoise_serializer import Serializer
from tortoise import Model, fields
from pydantic import Field
from fastapi.routing import APIRouter
class MyUser(Model):
id = fields.IntegerField(primary_key=True)
name = fields.CharField(max_length=100, unique=True)
class MyUserSerializer(Serializer):
id: int
name: str = Field(max_length=100, description="User unique name")
router = APIRouter(prefix="/users")
@router.get("")
async def get_users() -> list[MyUserSerializer]:
return await MyUserSerializer.from_queryset(MyUser.all(), context={"user": ...})
(Note: You can specify a context to pass additional information to serializers, but it is not mandatory.)
Writing
from fastapi import Body
from pydantic import Field
class MyUserCreationSerializer(Serializer):
name: str = Field(max_length=200)
@router.post("")
async def create_user(user_serializer: MyUserCreationSerializer = Body(...)) -> MyUserSerializer:
user = await user_serializer.create_tortoise_instance(MyUser)
# Here you can also pass a `context=` to this function.
return await MyUserSerializer.from_tortoise_orm(user)
Note: It is currently not possible to handle ForeignKeys directly using serializers. You need to manage such logic in your views.
Context
The context in serializers is immutable.
Resolvers
Sometimes, you need to compute values or restrict access to sensitive data. This can be achieved with resolvers and context. Here's an example:
from tortoise_serializer import ContextType, Serializer, require_permission_or_unset
from tortoise import Model, fields
class UserModel(Model):
id = fields.IntegerField(primary_key=True)
address = fields.CharField(max_length=1000)
def is_self(instance: UserModel, context: ContextType) -> bool:
current_user = context.get("user")
if not current_user:
return False
return current_user.id == instance.id
class UserSerializer(Serializer):
id: int
# Default is set to None, but the field will be omitted.
address: str | None = None
@classmethod
@require_permission_or_unset(is_self)
async def resolve_address(cls, instance: UserModel, context: ContextType) -> str:
return instance.address
@app.get("/users", response_model_exclude_unset=True)
async def list_users(user: UserModel = Depends(...)) -> list[UserSerializer]:
return await UserSerializer.from_queryset(UserModel.all(), context={"user": user})
This ensures that the address field is not exposed to unauthorized users.
Async resolvers are called concurrently during serializer instantiation.
Relations
ForeignKeys & OneToOne
To serialize relations, declare a field in the serializer as another serializer:
from tortoise import Model, fields
from tortoise_serializer import Serializer
class BookShelf(Model):
id = fields.IntField(primary_key=True)
name = fields.CharField(unique=True)
class Book(Model):
id = fields.IntField(primary_key=True)
title = fields.CharField(db_index=True)
shelf = fields.ForeignKeyField(
"models.BookShelf",
on_delete=fields.SET_NULL,
null=True,
related_name="books",
)
class BookSerializer(Serializer):
id: int
title: str
class ShelfSerializer(Serializer):
id: int
name: str
books: list[BookSerializer] = []
# Prefetching related fields is optional but improves performance.
serializer = ShelfSerializer.from_queryset(
BookShelf.all().prefetch_related("books").order_by("name")
)
For a normal ForeignKey relationship:
class ShelfSerializer(Serializer):
id: int
name: str
class BookSerializer(Serializer):
id: int
title: str
shelf: ShelfSerializer | None
Reverse relations are list[Serializer]
Limitations: Limitations: You cannot declare a field like this:
class SerializerA(Serializer):
...
class SerializerB(Serializer):
...
class MyWrongSerializer(Serializer):
my_field = SerializerA | SerializerB
but you can still use None like:
class MySerializer(Serializer):
some_relation: SerializerA | None = None
Many2Many
There are two ways to handle Many-to-Many relationships:
- Use an intermediate Serializer with two ForeignKeys.
- Use a resolver in the serializer.
Computed fields
Serialization involves resolving fields in the following order:
- Resolvers (computed fields)
- ForeignKeys
- Model fields This order allows hiding fields based on the request.
Example of a computed field:
from pydantic import Field
from tortoise_serializer import Serializer, ContextType
from tortoise.queryset import QuerySet
class Book(Model):
id = fields.IntField(primary_key=True)
title = fields.CharField(db_index=True)
shelf = fields.ForeignKeyField(
"models.BookShelf",
on_delete=fields.SET_NULL,
null=True,
related_name="books",
)
class BookSerializer(Serializer):
id: int
title: str
path: str
# This description will appear in Swagger's schema.
answer_to_the_question: int = Field(description="The answer to the big question of life")
@classmethod
async def resolve_path(cls, instance: Book, context: ContextType) -> str:
if not instance.shelf:
return instance.title
if isinstance(instance.shelf, QuerySet):
await instance.fetch_related("shelf")
return f'{instance.shelf.name}/{instance.title}'
@classmethod
def resolve_answer_to_the_question(cls, instance: Book, context: ContextType) -> int:
return 42
main_shelf = await Shelf.create(title="main")
my_book = await Book.create(title="Serializers 101", shelf=main_shelf)
serializer = await BookSerializer.from_tortoise_orm(my_book)
assert serializer.path == "main/Serializers 101"
assert serializer.answer_to_the_question == 42
All async resolvers will be resolved in concurency in a asyncio.gather, non-async ones will be resolved one after the other
Model Serializers
Sometime it may be usefull or necessary to be able to create a row and it's related foreignkeys at once in one endpoint, to achieve that the ModelSerializer class exists
Models serializer can manage:
- Foreign keys
- Backward foreign key
- Many2Many relations
- One to one relationship
Basic Usage
from tortoise import Model, fields
from tortoise_serializer import ModelSerializer
class Book(Model):
id = fields.IntField(primary_key=True)
title = fields.CharField(db_index=True, max_length=200)
shelf = fields.ForeignKeyField(
"models.BookShelf",
on_delete=fields.SET_NULL,
null=True,
related_name="books",
)
class BookShelf(Model):
id = fields.IntField(primary_key=True)
name = fields.CharField(unique=True, max_length=200)
books: BackwardFKRelation[Book]
class ShelfCreationSerializer(ModelSerializer[BookShelf]):
name: str
class BookCreationSerializer(ModelSerializer[Book]):
title: str
# here ofc it's a bit weird to create the shelves with the books but
# it's only for the example
shelf: ShelfCreationSerializer
serializer = BookCreationSerializer(title="Some Title", shelv={"name": "where examples lie"})
example = await serializer.create_tortoise_instance()
# example will be an instance of `Book` here with it's related `shelf` realtion
assert await Book.filter(name="Some Title", shelv__name="where examples lie").exists()
FastAPI
Since Serializers inherit from pydantic.BaseModel it means you can safely use them with FastAPI without any extra effort
Fastapi Documentation: https://fastapi.tiangolo.com/
Example
from fastapi import status, Body, HTTPException
from fastapi.routing import APIRouter
from pydantic import Field
from tortoise import Model, fields
from tortoise.transaction import in_transaction
from tortoise_serializer import ModelSerializer
# Tortoise Models
class Author(Model):
id = models.IntegerField(primary_key=True)
name = models.CharField(max_length=200, unique=True)
class Book(Model):
id = fields.IntegerField(primary_key=True)
title = fields.CharField(max_length=200)
pages_count = fields.IntegerField()
author = fields.ForeignKeyField("models.Author", related_name="books")
# Serializer for creation
class AuthorCreationSerializer(ModelSerializer[Author]):
name: str
class BookCreationSerializer(ModelSerializer[Book]):
title: str = Field(max_length=200)
author: AuthorCreationSerializer
async def _get_or_create_author(self) -> Author:
# here's an example of get or create flow using the serializers
author = await Author.filter(name=self.author.name).get_or_none()
if not author:
author = await self.author.create_tortoise_instance()
return author
async def create_tortoise_instance(self, *args, **kwargs) -> Book:
kwargs["author"] = await self._get_or_create_author()
return await super().create_tortoise_instance(*args, **kwargs)
# Serializer for reading
class AuthorSerializer(ModelSerializer[Author]):
id: int
name: str
class BookSerializer(ModelSerializer[Book]):
id: int
title: str
author: AuthorSerializer
# Views to manage the books
router = APIRouter(prefix="/test")
@router.post("", status_code=status.HTTP_201_CREATED)
async def create_book(serializer: BookCreationSerializer = Body(...)) -> BookSerializer:
async with in_transaction():
book = await serializer.create_tortoise_instance()
return await BookSerializer.from_tortoise_orm(book)
@router.get("")
async def list_books() -> list[BookSerializer]:
queryset = Book.all().prefetch_related(*BookSerializer.get_prefetch_fields())
return await BookSerializer.from_queryset(queryset)
@router.get("/{book_id}")
async def get_book(book_id: int) -> BookSerializer:
book = await (
Book.filter(id=book_id)
.prefetch_related(*BookSerializer.get_prefetch_fields())
.get_or_none()
)
if not book:
raise HTTPException(status.HTTP_404_NOT_FOUND, "No such book")
return await BookSerializer.from_tortoise_orm(book)
@router.delete("/{book_id}", status_code=status.HTTP_204_NO_CONTENT)
async def delete_book(book_id: int) -> None:
await Book.filter(id=book_id).delete()
@router.patch("{book_id}")
async def update_book(book_id: int, update: BookCreationSerializer) -> BookSerializer:
book = await Book.get_or_none(id=book_id)
if not book:
raise HTTPException(status.HTTP_404_NOT_FOUND, "No such book")
book.author = await update._get_or_create_author()
update.partial_update_tortoise_instance(book)
await book.save().
return await BookSerializer.from_tortoise_orm(book)
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 tortoise_serializer-1.3.3.tar.gz.
File metadata
- Download URL: tortoise_serializer-1.3.3.tar.gz
- Upload date:
- Size: 14.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.0.1 CPython/3.13.2 Linux/6.14.0-arch1-1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f2881b2b80d19fea6a2d7907f9d33f806d1e0b82588fb4216329e6562c17b494
|
|
| MD5 |
41ec45170ba02e18a3517890bca23f56
|
|
| BLAKE2b-256 |
460f3823cb6748ef5fce67a20f56fcfe3a8a80d5e95ba03c54ebff1269068518
|
File details
Details for the file tortoise_serializer-1.3.3-py3-none-any.whl.
File metadata
- Download URL: tortoise_serializer-1.3.3-py3-none-any.whl
- Upload date:
- Size: 13.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.0.1 CPython/3.13.2 Linux/6.14.0-arch1-1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
24213696dad2840667205b35bca8af56bc401b20f8e41d31f1ee311fabfc957e
|
|
| MD5 |
f11975706454383c5af480faef5bce84
|
|
| BLAKE2b-256 |
4b4708c115425902d5c01fe0c6d23c050d6722c555ee22bb86e316f1968e9c3a
|