How to make a PATCH request using DJANGO REST framework

前端 未结 6 1903
北恋
北恋 2020-12-13 13:11

I am not very experience with Django REST framework and have been trying out many things but can not make my PATCH request work.

I have a Model serializer. This is

相关标签:
6条回答
  • 2020-12-13 13:11

    Another posiblity is to make the request by URL. For example, I have a model like this

          class Author(models.Model):
            FirstName = models.CharField(max_length=70)
            MiddleName = models.CharField(max_length=70)
            LastName = models.CharField(max_length=70)
            Gender = models.CharField(max_length=1, choices = GENDERS)
            user = models.ForeignKey(User, default = 1, on_delete = models.CASCADE, related_name='author_user')
            IsActive = models.BooleanField(default=True)
            class Meta:
              ordering = ['LastName']
    

    And a view like this

          class Author(viewsets.ModelViewSet):
            queryset = Author.objects.all()
            serializer_class = AuthorSerializer
    

    So can enter http://127.0.0.1:8000/author/ to get or post authors. If I want to make a PATCH request you can point to http://127.0.0.1:8000/author/ID_AUTHOR from your client. For example in angular2, you can have something like this

           patchRequest(item: any): Observable<Author> {
            return this.http.patch('http://127.0.0.1:8000/author/1', item);
           }
    

    It suppose you have configured your CORS and you have the same model in back and front. Hope it can be usefull.

    0 讨论(0)
  • 2020-12-13 13:16

    I ran into this issues as well, I solved it redefining the get_serializer_method and adding custom logic to handle the partial update. Python 3

     class ViewSet(viewsets.ModelViewSet):
         def get_serializer_class(self):
             if self.action == "partial_update":
                 return PartialUpdateSerializer
    

    Note: you may have to override the partial_update function on the serializer. Like so:

    class PartialUpdateSerializer(serializers.Serializer):
        def partial_update(self, instance, validated_data):
           *custom logic*
           return super().update(instance, validated_data)
    
    0 讨论(0)
  • 2020-12-13 13:19

    Make sure that you have "PATCH" in http_method_names. Alternatively you can write it like this:

    @property
    def allowed_methods(self):
        """
        Return the list of allowed HTTP methods, uppercased.
        """
        self.http_method_names.append("patch")
        return [method.upper() for method in self.http_method_names
                if hasattr(self, method)]
    

    As stated in documentation:

    By default, serializers must be passed values for all required fields or they will raise validation errors. You can use the partial argument in order to allow partial updates.

    Override update method in your view:

    def update(self, request, *args, **kwargs):
        instance = self.get_object()
        serializer = TimeSerializer(instance, data=request.data, partial=True)
        serializer.is_valid(raise_exception=True)
        serializer.save(customer_id=customer, **serializer.validated_data)
        return Response(serializer.validated_data)
    

    Or just override method partial_update in your view:

    def partial_update(self, request, *args, **kwargs):
        kwargs['partial'] = True
        return self.update(request, *args, **kwargs)
    

    Serializer calls update method of ModelSerializer(see sources):

    def update(self, instance, validated_data):
        raise_errors_on_nested_writes('update', self, validated_data)
    
        # Simply set each attribute on the instance, and then save it.
        # Note that unlike `.create()` we don't need to treat many-to-many
        # relationships as being a special case. During updates we already
        # have an instance pk for the relationships to be associated with.
        for attr, value in validated_data.items():
            setattr(instance, attr, value)
        instance.save()
    
        return instance
    

    Update pushes the validated_data values to the given instance. Note that update should not assume all the fields are available. This helps to deal with partial updates (PATCH requests).

    0 讨论(0)
  • 2020-12-13 13:19

    Use ModelViewSet instead and override perform_update method from UpdateModelMixin

    class UserViewSet(viewsets.ModelViewSet):
        queryset = TimeEntry.objects.all()
        serializer_class = TimeSerializer
    
        def perform_update(self, serializer):
            serializer.save()
            # you may also do additional things here
            # e.g.: signal other components about this update
    

    That's it. Don't return anything in this method. UpdateModelMixin has implemented update method to return updated data as response for you and also clears out _prefetched_objects_cache. See the source code here.

    0 讨论(0)
  • 2020-12-13 13:20
    class DetailView(APIView):
        def get_object(self, pk):
            return TestModel.objects.get(pk=pk)
    
        def patch(self, request, pk):
            testmodel_object = self.get_object(pk)
            serializer = TestModelSerializer(testmodel_object, data=request.data, partial=True) # set partial=True to update a data partially
            if serializer.is_valid():
                serializer.save()
                return JsonResponse(code=201, data=serializer.data)
            return JsonResponse(code=400, data="wrong parameters")
    

    Documentation
    You do not need to write the partial_update or overwrite the update method. Just use the patch method.

    0 讨论(0)
  • 2020-12-13 13:23

    The patch method is worked for me using viewset in DRF. I'm changing you code:

    class UserViewSet(viewsets.ModelViewSet):
        queryset = TimeEntry.objects.all()
        serializer_class = TimeSerializer
    
        def perform_update(self, serializer):
            user_instance = serializer.instance
            request = self.request
            serializer.save(**modified_attrs)
            return Response(status=status.HTTP_200_OK)
    
    0 讨论(0)
提交回复
热议问题