首页 > 解决方案 > Django Serializer:如何验证模型实例?

问题描述

我对 Django 序列化程序还很陌生,但仍然对它们的工作方式感到困惑。

我有一个相当普遍的场景,其中我正在调用我的 api,它只是在我的模型对象中设置一个字段并保存它(假设记录已经存在并且它正在更新)。但是,我需要在保存之前验证这个模型对象。

api.py

@detail_route(methods=['POST'], url_path='submit-draft')
def submit_draft(self, request, *args, **kwargs):
    booking = self.get_object()

    # serializer with custom validations.
    serializer = self.get_serializer(booking)
    serializer.is_valid(raise_exception=True)

    booking.submit_draft(by=request.user)
    booking.save()

    data = serializers.BookingDetailSerializer(booking, context={'request': request}).data

    return response.Ok(data)

序列化程序.py

class BookingCreateUpdateSerializer(serializers.ModelSerializer):
    date = serializers.CharField()

    duration = serializers.IntegerField(required=True, )

    created_by = serializers.PrimaryKeyRelatedField(read_only=True, default=serializers.CurrentUserDefault(), )

    modified_by = serializers.PrimaryKeyRelatedField(read_only=True, default=serializers.CurrentUserDefault(), )
    ....

    class Meta:
    model = models.Booking
        fields = [
        'title',
        'date',
        'duration',
        'client',
        'created_by',
        'modified_by',
        ....
    ]

但是,我收到此错误:

AssertionError: Cannot call '.is_valid()' as no 'data=' keyword argument was passed when instantiating the serializer instance.

我知道序列化程序需要一个字典而不是实际的模态对象。但我不知道如何实现我想要的。验证模型对象。谁能建议正确的方法?

标签: pythondjangodjango-rest-frameworkdjango-serializer

解决方案


您可以使用序列化程序:

1)将您的对象序列化为字典。在这种情况下,您不需要调用 is_valid 因为对象已经创建并且它具有有效值。因此,您可以执行以下操作:

serializer = BookingCreateUpdateSerializer(booking)

return Response(serializer.data)

2)序列化输入并使用序列化程序创建新对象。在这种情况下,输入是一个字典,在调用 save 之前需要先验证它。你可以这样做:

serializer = BookingCreateUpdateSerializer(data=input_dict)
serializer.is_valid()
serializer.save()

更新以回答评论

您可以执行以下操作以使用序列化程序更新实例:

serializer = BookingCreateUpdateSerializer(booking, data=input_data_to_update, partial=True)
serializer.is_valid()

instance = serializer.save()

# and serialize the updated instance for response using another serializer
output_data = BookingDetailSerializer(instance).data

推荐阅读