Traditional Django REST Framework serializers often lead to what’s known as “serializer hell” - a situation where developers:
- Create numerous serializer variations for slightly different API endpoints
- Duplicate code for simple field variations
- Struggle with rigid and complex nested relationships
- Maintain sprawling serializer classes that become hard to manage
drf-shapeless-serializers was created to solve these pain points by introducing dynamic runtime configuration capabilities, allowing you to eliminate up to 80% of your serializer code while gaining unprecedented flexibility.
https://drf-shapeless-serializers.readthedocs.io/en/latest/
drf-shapeless-serializers provides powerful mixins that extend Django REST Framework's serializers with dynamic configuration capabilities. By inheriting from our base classes, you can select fields at runtime, rename output keys dynamically, modify field attributes per-request, and add configured nested relationships on-the-fly.
pip install drf-shapeless-serializers
Add to your Django settings:
INSTALLED_APPS = [
# ... other apps
'shapeless_serializers',
]- Define your shapeless serializer:
from shapeless_serializers.serializers import ShapelessModelSerializer
class UserSerializer(ShapelessModelSerializer):
class Meta:
model = User
fields = '__all__'
class AuthorSerializer(ShapelessModelSerializer):
class Meta:
model = Author
fields = '__all__'
class BookSerializer(ShapelessModelSerializer):
class Meta:
model = Book
fields = '__all__'- Configure dynamically in views:
Instead of passing dictionaries, you now pass instantiated serializers into the
nestedparameter, allowing for a more Pythonic and type-safe configuration.
@api_view(['GET'])
def book_detail(request, pk):
book = Book.objects.get(pk=pk)
serializer = BookSerializer(
book,
fields=['id', 'title', 'price', 'author'],
rename_fields={'price': 'retail_price', 'id': 'book_id'},
nested={
'author': AuthorSerializer(
fields=['id', 'bio', 'user'],
rename_fields={'bio': 'biography'},
nested={
'user': UserSerializer(fields=['id', 'username', 'email'])
}
)
}
)
return Response(serializer.data)The ShapelessViewMixin allows you to define dynamic configurations directly on your ViewSets. This keeps your view logic clean and centralized.
from shapeless_serializers.views import ShapelessViewMixin
class BookViewSet(ShapelessViewMixin, viewsets.ModelViewSet):
queryset = Book.objects.all()
serializer_class = BookSerializer
def get_serializer_fields(self):
if self.action == 'list':
return ['id', 'title', 'price']
return ['id', 'title', 'description', 'author', 'comments']
def get_serializer_nested(self):
if self.action == 'retrieve':
return {
'author': AuthorSerializer(fields=['name', 'bio']),
'comments': CommentSerializer(fields=['content', 'user'], many=True)
}
return {}The fields parameter lets you cherry-pick exactly which fields to include.
AuthorSerializer(author, fields=['id', 'name', 'birth_date'])Pass standard DRF serializer params at runtime.
AuthorSerializer(
author,
field_attributes={
'bio': {'help_text': 'Author biography'},
'address': {'write_only': True}
}
)Nested configuration supports unlimited depth. Each level can be customized with its own fields, renaming, and attributes.
AuthorSerializer(
author,
nested={
'books': BookSerializer(
fields=['title', 'publish_year', 'publisher'],
nested={
'publisher': PublisherSerializer(fields=['name', 'country'])
}
)
}
)For extremely complex structures, the syntax remains readable:
serializer = DynamicBlogPostSerializer(
posts,
fields=["id", "title", "author", "comments"],
rename_fields={"id": "post_identifier"},
nested={
"author": DynamicAuthorProfileSerializer(
fields=["bio", "is_verified", "user"],
rename_fields={"bio": "author_biography"},
nested={
"user": UserSerializer(
fields=["id", "username"],
rename_fields={"username": "user_login"}
)
}
),
"comments": DynamicCommentSerializer(
fields=["id", "content", "user", "replies"],
instance=posts.comments.filter(is_approved=True, parent__isnull=True),
rename_fields={"content": "comment_text"},
nested={
"user": UserSerializer(fields=["id", "username"]),
"replies": DynamicCommentSerializer(
fields=["id", "content", "user"],
instance=lambda instance, ctx: instance.replies.filter(is_approved=True),
rename_fields={"content": "reply_text"}
)
}
)
}
)Include fields based on runtime logic (like user permissions):
AuthorSerializer(
author,
conditional_fields={
'email': lambda instance, ctx: ctx['request'].user.is_staff
}
)Create serializers on-the-fly without defining a class—perfect for one-off needs:
serializer = InlineShapelessModelSerializer(
book,
model=Book,
fields=['title', 'author'],
nested={
'author': InlineShapelessModelSerializer(
model=Author,
fields=['name', 'bio']
)
}
)- Building public APIs with multiple versions.
- Projects needing different "views" of the same data (e.g., Summary vs. Detail).
- Rapidly evolving API requirements where creating new classes is a bottleneck.
- Any project suffering from "Serializer Bloat."
We welcome contributions! Please check the CONTRIBUTING.md file.
This project is licensed under the MIT License. See the LICENSE file for details.
Inspired by the need for high-flexibility API systems. Special thanks to the Django REST Framework community.
If you find this package useful, please consider supporting its development:
- USDT (TRC20):
TEitNDQMm4upYmNvFeMpxTRGEJGdord3S5 - USDT (BEP20):
0xc491a2ba6f386ddbf26cdc906939230036473f5d - BTC:
13X8aZ23pFNCH2FPW6YpRTw4PGxo7AvFkN