mixins.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. from __future__ import print_function, unicode_literals
  2. import traceback
  3. from rest_framework import status
  4. from rest_framework.mixins import CreateModelMixin
  5. from rest_framework.response import Response
  6. __all__ = [
  7. 'BulkCreateModelMixin',
  8. 'BulkDestroyModelMixin',
  9. 'BulkUpdateModelMixin',
  10. ]
  11. class BulkCreateModelMixin(CreateModelMixin):
  12. """
  13. Either create a single or many model instances in bulk by using the
  14. Serializer's ``many=True`` ability from Django REST >= 2.2.5.
  15. .. note::
  16. This mixin uses the same method to create model instances
  17. as ``CreateModelMixin`` because both non-bulk and bulk
  18. requests will use ``POST`` request method.
  19. """
  20. def create(self, request, *args, **kwargs):
  21. bulk = isinstance(request.data, list)
  22. if not bulk:
  23. return super(BulkCreateModelMixin, self).create(request, *args, **kwargs)
  24. else:
  25. serializer = self.get_serializer(data=request.data, many=True)
  26. serializer.is_valid(raise_exception=True)
  27. self.perform_create(serializer)
  28. return Response(serializer.data, status=status.HTTP_201_CREATED)
  29. class BulkUpdateModelMixin(object):
  30. """
  31. Update model instances in bulk by using the Serializer's
  32. ``many=True`` ability from Django REST >= 2.2.5.
  33. """
  34. def get_object(self):
  35. try:
  36. super(BulkUpdateModelMixin, self).get_object()
  37. except AssertionError:
  38. # probably happened when called get_object() within options()
  39. # via self.metadata_class which is not allowed on list viewset
  40. # however since we are enabling PUT here, we should handle the
  41. # exception if called within options()
  42. # We can simply swallow the exception since that method
  43. # does not actually do anythingwith the returned object
  44. for file, line, function, code in traceback.extract_stack():
  45. if all((file.endswith('rest_framework/views.py'),
  46. function == 'options')):
  47. return
  48. # not called inside metadata() so probably something went
  49. # wrong and so we should reraise exception
  50. raise
  51. def bulk_update(self, request, *args, **kwargs):
  52. partial = kwargs.pop('partial', False)
  53. # restrict the update to the filtered queryset
  54. serializer = self.get_serializer(
  55. self.filter_queryset(self.get_queryset()),
  56. data=request.data,
  57. many=True,
  58. partial=partial,
  59. )
  60. serializer.is_valid(raise_exception=True)
  61. self.perform_update(serializer)
  62. return Response(serializer.data, status=status.HTTP_200_OK)
  63. def perform_update(self, serializer):
  64. serializer.save()
  65. def partial_bulk_update(self, request, *args, **kwargs):
  66. kwargs['partial'] = True
  67. return self.bulk_update(request, *args, **kwargs)
  68. class BulkDestroyModelMixin(object):
  69. """
  70. Destroy model instances.
  71. """
  72. def allow_bulk_destroy(self, qs, filtered):
  73. """
  74. Hook to ensure that the bulk destroy should be allowed.
  75. By default this checks that the destroy is only applied to
  76. filtered querysets.
  77. """
  78. return qs is not filtered
  79. def bulk_destroy(self, request, *args, **kwargs):
  80. qs = self.get_queryset()
  81. filtered = self.filter_queryset(qs)
  82. if not self.allow_bulk_destroy(qs, filtered):
  83. return Response(status=status.HTTP_400_BAD_REQUEST)
  84. for obj in filtered:
  85. self.perform_destroy(obj)
  86. return Response(status=status.HTTP_204_NO_CONTENT)
  87. def perform_destroy(self, instance):
  88. instance.delete()