Django REST Framework POST/PUT 的深度不同?

Posted

技术标签:

【中文标题】Django REST Framework POST/PUT 的深度不同?【英文标题】:Django REST Framework different depth for POST/PUT? 【发布时间】:2013-03-30 19:19:26 【问题描述】:

我正在使用 Django REST Framework 为我的 Web 应用程序创建 API。我有一个“评论”类,它在Meta 类中设置了depth=2。当GETComments 时,这很有效。当我尝试发送POSTPUT 请求时(即创建一个新的Comment),我被告知我需要包含对象而不是ForeignKey ID。

这是我的序列化程序类:

class CommentSerializer(serializers.ModelSerializer):
    class Meta:
        model = Comment
        depth = 2 

型号:

class Comment(models.Model):
    user = models.ForeignKey(User, null=True, blank=True,
        related_name='comments')
    budget = models.ForeignKey(Budget, related_name='comments')
    published = models.BooleanField(default=False)
    body = models.TextField()
    created = models.DateTimeField(auto_now_add=True)

查看代码:

class Comments(generics.ListCreateAPIView):

    model = Comment
    serializer_class = CommentSerializer

    def pre_save(self, obj):
        obj.user = self.request.user

输出(JSON)中显示的错误是:

"user": ["This field is required."], "budget": ["non_field_errors": ["Invalid data"]]

当这个原始数据被发送时:

"budget": 2, "published": true, "body": "Another comment"

【问题讨论】:

随机提问,发"budget_id": 2"budget": "id": 2怎么办? RE @Nathan Villaescusa。然后它会根据需要显示其他预算字段。 【参考方案1】:

我遇到了同样的问题,所以我解决了制作自定义泛型方法。这是上述答案的更好实现

class CustomListCreateAPIView(mixins.ListModelMixin,
                              mixins.CreateModelMixin,
                              generics.GenericAPIView):
    """
    Concrete view for listing a queryset or creating a model instance.
    """
    def get_serializer_class(self):
        method = self.request.method
        if method == 'PUT' or method == 'POST':
            return self.writeSerializers
        else:
            return self.readSerializers

    def get(self, request, *args, **kwargs):
        return self.list(request, *args, **kwargs)

    def post(self, request, *args, **kwargs):
        return self.create(request, *args, **kwargs)

类似的路德,

class CustomRetrieveUpdateDestroyAPIView(mixins.RetrieveModelMixin,
                                         mixins.UpdateModelMixin,
                                         mixins.DestroyModelMixin,
                                         generics.GenericAPIView):
    """
    Concrete view for retrieving, updating or deleting a model instance.
    """
    def get_serializer_class(self):
        method = self.request.method
        if method == 'PUT' or method == 'POST':
            return self.writeSerializers
        else:
            return self.readSerializers

    def get(self, request, *args, **kwargs):
        return self.retrieve(request, *args, **kwargs)

    def put(self, request, *args, **kwargs):
        return self.update(request, *args, **kwargs)

    def patch(self, request, *args, **kwargs):
        return self.partial_update(request, *args, **kwargs)

    def delete(self, request, *args, **kwargs):
        return self.destroy(request, *args, **kwargs)  # enter code here

现在我只在 Views.py 中给出 writeSerializersreadSerializers

还有一种创建读写序列化器的简单方法。

class employeeWriteSerializer(serializers.ModelSerializer):
    class Meta:
        model = employee
        fields = ('username','email',..)
        
class employeeReadSerializer(serializers.ModelSerializer):
     class Meta(employeeWriteSerializer.Meta):
         depth = 1

它可以节省时间和重复工作,您还可以在自定义通用 Api(Retitve 工作)中添加身份验证类。谢谢。

【讨论】:

【参考方案2】:

您可以通过覆盖get_serializer_class() 函数来设置不同的序列化程序,如下所示:

def get_serializer_class(self): method = self.request.method if method == 'PUT' or method == 'POST': return YourWriteSerializer else: return YourReadSerializer

我想添加这个,因为我在谷歌搜索了一段时间后来到这里。

【讨论】:

迄今为止最好和最简单的回应。【参考方案3】:

我相信定义引用外键关系的序列化程序字段的正确方法是通过serializers.PrimaryKeyRelatedField 之类的方法。我不相信模型序列化器会自动使用这个字段类而不在序列化器类中明确定义它。

http://www.django-rest-framework.org/api-guide/relations/#primarykeyrelatedfield

我想PrimaryKeyRelatedField 序列化程序可以正确处理 JSON 数据提交,就像您在示例中使用的那样。

【讨论】:

对于那些从 Google 来到这里的人,Tom Christie(DRF 的创建者)在这里写了更多关于此的内容:groups.google.com/d/msg/django-rest-framework/vCl2I_EzJnE/…【参考方案4】:

我知道这有点晚了,但我最终使用了 2 个这样的序列化程序:

class CommentReadSerializer(serializers.ModelSerializer):
    class Meta:
        model = Comment
        depth = 2

class CommentWriteSerializer(serializers.ModelSerializer):
    class Meta:
        model = Comment

然后这样使用:

class Comments(generics.ListCreateAPIView):

    model = Comment
    serializer_class = CommentReadSerializer

    def create(self, request, *args, **kwargs):
        serializer = CommentWriteSerializer(data=request.DATA, files=request.FILES)
        if serializer.is_valid():
            self.pre_save(serializer.object)
            self.object = serializer.save(force_insert=True)
            self.post_save(self.object, created=True)
            headers = self.get_success_headers(serializer.data)
            serializer = CommentReadSerializer(serializer.object)
            return Response(serializer.data, status=status.HTTP_201_CREATED, headers=headers)

        return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

【讨论】:

以上是关于Django REST Framework POST/PUT 的深度不同?的主要内容,如果未能解决你的问题,请参考以下文章

Django Rest Framework 和 django Rest Framework simplejwt 两因素身份验证

怎么安装django rest framework

django rest framework中文介绍

17-Django-Django REST framework-REST framework及RESTful简介

为啥 django-rest-framework 不显示 OneToOneField 数据 - django

Django:rest framework之分页(Pagination)