Django - DRF 删除/检索/补丁返回 404 { 详细信息:“未找到”}

Vik*_*lev 5 django django-queryset django-views python-3.x django-rest-framework

经过大约4-5个小时的密集调试。我放弃试图找出导致这个错误的原因,可能非常简单。尝试使用 Update 和 Patch/Put Mixins 进行此操作,但效果不佳。100%是因为queryset的原因,但是我找不到问题所在?尝试使用 .get() 和我能想到的其他一切。

我的观点非常简单:

class RemoveModel3D(generics.DestroyAPIView):

    serializer_class = Model3DSerializer

    def get_queryset(self):
        user_pk = self.kwargs["pk"]
        return Model3D.objects.filter(owners__in=[user_pk])
Run Code Online (Sandbox Code Playgroud)

PS 该查询集与 ListModelMixin 完美配合。我读到 List 用于集合,而 Retrieve/Destroy/Update 用于单个模型实例,但是如何使查询集成为单个模型实例?我在任何地方都找不到

编辑:添加与问题相关的序列化器和模型片段

序列化器:

class Model3DSerializer(serializers.ModelSerializer):

    User = get_user_model()

    commits = CommitSerializer(many=True, required=False, read_only=True)
    favorited_by = UserSerializer(many=True, required=False, read_only=True)

    date_uploaded = serializers.DateTimeField(read_only=True)
    owners = serializers.PrimaryKeyRelatedField(many=True, read_only=True)

    class Meta:
        model = Model3D
        fields = (
            'id',
            'title',
            'owners',
            'description',
            'date_uploaded',
            'favorited_by',
            'commits'
        )
Run Code Online (Sandbox Code Playgroud)

模型:

class Model3D(models.Model):

    title = models.CharField(max_length=64)
    # Many models many owners, seems reasonable to me
    owners = models.ManyToManyField(User, related_name='owners')
    description = models.TextField(null=True)
    date_uploaded = models.DateTimeField(auto_now_add=True)

    # Many models many people who like them.
    favorited_by = models.ManyToManyField(User, related_name='favorited_by')
Run Code Online (Sandbox Code Playgroud)

浏览次数:

class ListAllModels3D(generics.ListAPIView):

    serializer_class = Model3DSerializer

    def get_queryset(self):
        queryset = Model3D.objects.all()
        model_id = self.request.query_params.get('id', None)

        if model_id is not None:
            queryset = queryset.filter(pk=model_id)

        return queryset

class RemoveModel3D(generics.DestroyAPIView):

    serializer_class = Model3DSerializer

    def get_queryset(self):
        queryset = Model3D.objects.all()
        model_id = self.request.query_params.get('id', None)

        if model_id is not None:
            queryset = queryset.filter(pk=model_id)

        return queryset


class Models3D( mixins.ListModelMixin,
            mixins.CreateModelMixin,
            generics.GenericAPIView,
        ):

    serializer_class = Model3DSerializer

    def get_queryset(self):
        user_pk = self.kwargs["pk"]
        return Model3D.objects.filter(owners__in=[user_pk])

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

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

    def perform_create(self, serializer):
        # FIXME: this is a bad way to set the value, but ...
        user_id = self.kwargs["pk"]
        serializer.validated_data['owners'] = [user_id]

        serializer.save()
Run Code Online (Sandbox Code Playgroud)

Håk*_*Lid 2

lookup_url_kwargGenericApiView 的默认值是pk. 但是您使用它来过滤User,因此在查询 Owner 和 Model3D 时将使用相同的 pk 值。

结果类似于此伪代码,除非用户和 model3d 具有相同的 pk,否则会导致 404 响应。

kwargs = {'pk': 20} 
try:
   Model3D.objects.get(owners__in=[kwargs['pk']], pk=kwargs['pk'])
except ObjectNotFound:
   raise Http404Exception('Not found')
Run Code Online (Sandbox Code Playgroud)

要解决此问题,请使用支持嵌套 api 路由的插件。或者您可以重写 viewsetget_object方法,这是引发 404 错误的地方。

def get_object(self):
    queryset = self.get_queryset()
    pk = self.request.query_params.get('id', None)
    obj = get_object_or_404(queryset, pk=pk)
    self.check_object_permissions(self.request, obj)
    return obj
Run Code Online (Sandbox Code Playgroud)

http://www.django-rest-framework.org/api-guide/generic-views/#get_objectself

该方法由(GET) PATCH/PUT 和(DELETE)get_object使用,默认用作查找值。retrieve()update()destroy()self.kwargs['pk']