You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Currently, the DRF stubs for GenericAPIView tie the serializer type to a model-based type variable. This approach works fine for model serializers but causes issues for non-model-based serializers that add custom methods (e.g., a to_dto() method). For example, when using a custom serializer that extends BaseSerializer[Any] with additional functionality, mypy complains that these methods do not exist.
The current stubs define the serializer class attribute as:
with _MT_co being a covariant TypeVar bounded to django.db.models.Model. This causes two problems:
The generic type parameter is forced to be a subtype of Model, which doesn’t fit non-model serializers.
Custom serializer methods (such as to_dto) are not recognized because the type is inferred as BaseSerializer[Any].
Steps to Reproduce
Create a custom serializer that extends BaseSerializer[Any] with an additional method:
from rest_framework.serializers import BaseSerializer
from rest_framework.generics import GenericAPIView
from rest_framework.response import Response
from rest_framework.request import Request
class MySerializer(BaseSerializer[Any]):
def to_dto(self) -> dict:
return {"data": "example"}
def to_dto(self, instance: Any) -> Any:
return instance
class MyView(GenericAPIView):
serializer_class = MySerializer
def post(self, request: Request) -> Response:
serializer = self.get_serializer(data=request.data)
serializer.is_valid(raise_exception=True)
# Mypy error: "BaseSerializer[Any]" has no attribute "to_dto"
data = serializer.to_dto()
return Response(data)
Run mypy. You will see an error indicating that the return type of get_serializer() does not have the custom to_dto attribute.
Expected Behavior
Mypy should infer the correct serializer type from the serializer_class attribute so that methods defined in custom serializers (like to_dto) are recognized. In other words, get_serializer() should return an instance of the actual serializer type instead of defaulting to BaseSerializer[Any].
Proposed Solution
Introduce a second generic type variable for the serializer and update the stubs to decouple the serializer type from the model type. For example, define a new type variable _S with a default value and bound to BaseSerializer[Any] while considering covariance for the model type:
For model-based serializers, users can continue not specifying the second type parameter, and _S will default to BaseSerializer[Any].
For custom serializers with additional methods, users can explicitly set the type, and mypy will infer the correct return type from get_serializer(), thereby recognizing the extra methods.
The text was updated successfully, but these errors were encountered:
Description
Currently, the DRF stubs for
GenericAPIView
tie the serializer type to a model-based type variable. This approach works fine for model serializers but causes issues for non-model-based serializers that add custom methods (e.g., ato_dto()
method). For example, when using a custom serializer that extendsBaseSerializer[Any]
with additional functionality, mypy complains that these methods do not exist.The current stubs define the serializer class attribute as:
and the return type of
get_serializer()
as:with
_MT_co
being a covariantTypeVar
bounded todjango.db.models.Model
. This causes two problems:Model
, which doesn’t fit non-model serializers.to_dto
) are not recognized because the type is inferred asBaseSerializer[Any]
.Steps to Reproduce
BaseSerializer[Any]
with an additional method:get_serializer()
does not have the custom to_dto attribute.Expected Behavior
Mypy should infer the correct serializer type from the
serializer_class
attribute so that methods defined in custom serializers (liketo_dto
) are recognized. In other words,get_serializer()
should return an instance of the actual serializer type instead of defaulting toBaseSerializer[Any]
.Proposed Solution
Introduce a second generic type variable for the serializer and update the stubs to decouple the serializer type from the model type. For example, define a new type variable
_S
with a default value and bound toBaseSerializer[Any]
while considering covariance for the model type:This change maintains backward compatibility:
_S
will default toBaseSerializer[Any]
.get_serializer()
, thereby recognizing the extra methods.The text was updated successfully, but these errors were encountered: