PageRenderTime 76ms CodeModel.GetById 17ms RepoModel.GetById 0ms app.codeStats 0ms

/rest_framework/generics.py

https://github.com/grimborg/django-rest-framework
Python | 453 lines | 246 code | 75 blank | 132 comment | 32 complexity | c4aa3bc583053c7e1088555d4698cf93 MD5 | raw file
  1. """
  2. Generic views that provide commonly needed behaviour.
  3. """
  4. from __future__ import unicode_literals
  5. from django.core.exceptions import ImproperlyConfigured
  6. from django.core.paginator import Paginator, InvalidPage
  7. from django.http import Http404
  8. from django.shortcuts import get_object_or_404
  9. from django.utils.translation import ugettext as _
  10. from rest_framework import views, mixins
  11. from rest_framework.exceptions import ConfigurationError
  12. from rest_framework.settings import api_settings
  13. import warnings
  14. class GenericAPIView(views.APIView):
  15. """
  16. Base class for all other generic views.
  17. """
  18. # You'll need to either set these attributes,
  19. # or override `get_queryset()`/`get_serializer_class()`.
  20. queryset = None
  21. serializer_class = None
  22. # This shortcut may be used instead of setting either or both
  23. # of the `queryset`/`serializer_class` attributes, although using
  24. # the explicit style is generally preferred.
  25. model = None
  26. # If you want to use object lookups other than pk, set this attribute.
  27. # For more complex lookup requirements override `get_object()`.
  28. lookup_field = 'pk'
  29. # Pagination settings
  30. paginate_by = api_settings.PAGINATE_BY
  31. paginate_by_param = api_settings.PAGINATE_BY_PARAM
  32. pagination_serializer_class = api_settings.DEFAULT_PAGINATION_SERIALIZER_CLASS
  33. page_kwarg = 'page'
  34. # The filter backend classes to use for queryset filtering
  35. filter_backends = api_settings.DEFAULT_FILTER_BACKENDS
  36. # The following attributes may be subject to change,
  37. # and should be considered private API.
  38. model_serializer_class = api_settings.DEFAULT_MODEL_SERIALIZER_CLASS
  39. paginator_class = Paginator
  40. ######################################
  41. # These are pending deprecation...
  42. pk_url_kwarg = 'pk'
  43. slug_url_kwarg = 'slug'
  44. slug_field = 'slug'
  45. allow_empty = True
  46. filter_backend = api_settings.FILTER_BACKEND
  47. def get_serializer_context(self):
  48. """
  49. Extra context provided to the serializer class.
  50. """
  51. return {
  52. 'request': self.request,
  53. 'format': self.format_kwarg,
  54. 'view': self
  55. }
  56. def get_serializer(self, instance=None, data=None,
  57. files=None, many=False, partial=False):
  58. """
  59. Return the serializer instance that should be used for validating and
  60. deserializing input, and for serializing output.
  61. """
  62. serializer_class = self.get_serializer_class()
  63. context = self.get_serializer_context()
  64. return serializer_class(instance, data=data, files=files,
  65. many=many, partial=partial, context=context)
  66. def get_pagination_serializer(self, page):
  67. """
  68. Return a serializer instance to use with paginated data.
  69. """
  70. class SerializerClass(self.pagination_serializer_class):
  71. class Meta:
  72. object_serializer_class = self.get_serializer_class()
  73. pagination_serializer_class = SerializerClass
  74. context = self.get_serializer_context()
  75. return pagination_serializer_class(instance=page, context=context)
  76. def paginate_queryset(self, queryset, page_size=None):
  77. """
  78. Paginate a queryset if required, either returning a page object,
  79. or `None` if pagination is not configured for this view.
  80. """
  81. deprecated_style = False
  82. if page_size is not None:
  83. warnings.warn('The `page_size` parameter to `paginate_queryset()` '
  84. 'is due to be deprecated. '
  85. 'Note that the return style of this method is also '
  86. 'changed, and will simply return a page object '
  87. 'when called without a `page_size` argument.',
  88. PendingDeprecationWarning, stacklevel=2)
  89. deprecated_style = True
  90. else:
  91. # Determine the required page size.
  92. # If pagination is not configured, simply return None.
  93. page_size = self.get_paginate_by()
  94. if not page_size:
  95. return None
  96. if not self.allow_empty:
  97. warnings.warn(
  98. 'The `allow_empty` parameter is due to be deprecated. '
  99. 'To use `allow_empty=False` style behavior, You should override '
  100. '`get_queryset()` and explicitly raise a 404 on empty querysets.',
  101. PendingDeprecationWarning, stacklevel=2
  102. )
  103. paginator = self.paginator_class(queryset, page_size,
  104. allow_empty_first_page=self.allow_empty)
  105. page_kwarg = self.kwargs.get(self.page_kwarg)
  106. page_query_param = self.request.QUERY_PARAMS.get(self.page_kwarg)
  107. page = page_kwarg or page_query_param or 1
  108. try:
  109. page_number = int(page)
  110. except ValueError:
  111. if page == 'last':
  112. page_number = paginator.num_pages
  113. else:
  114. raise Http404(_("Page is not 'last', nor can it be converted to an int."))
  115. try:
  116. page = paginator.page(page_number)
  117. except InvalidPage as e:
  118. raise Http404(_('Invalid page (%(page_number)s): %(message)s') % {
  119. 'page_number': page_number,
  120. 'message': str(e)
  121. })
  122. if deprecated_style:
  123. return (paginator, page, page.object_list, page.has_other_pages())
  124. return page
  125. def filter_queryset(self, queryset):
  126. """
  127. Given a queryset, filter it with whichever filter backend is in use.
  128. You are unlikely to want to override this method, although you may need
  129. to call it either from a list view, or from a custom `get_object`
  130. method if you want to apply the configured filtering backend to the
  131. default queryset.
  132. """
  133. filter_backends = self.filter_backends or []
  134. if not filter_backends and self.filter_backend:
  135. warnings.warn(
  136. 'The `filter_backend` attribute and `FILTER_BACKEND` setting '
  137. 'are due to be deprecated in favor of a `filter_backends` '
  138. 'attribute and `DEFAULT_FILTER_BACKENDS` setting, that take '
  139. 'a *list* of filter backend classes.',
  140. PendingDeprecationWarning, stacklevel=2
  141. )
  142. filter_backends = [self.filter_backend]
  143. for backend in filter_backends:
  144. queryset = backend().filter_queryset(self.request, queryset, self)
  145. return queryset
  146. ########################
  147. ### The following methods provide default implementations
  148. ### that you may want to override for more complex cases.
  149. def get_paginate_by(self, queryset=None):
  150. """
  151. Return the size of pages to use with pagination.
  152. If `PAGINATE_BY_PARAM` is set it will attempt to get the page size
  153. from a named query parameter in the url, eg. ?page_size=100
  154. Otherwise defaults to using `self.paginate_by`.
  155. """
  156. if queryset is not None:
  157. warnings.warn('The `queryset` parameter to `get_paginate_by()` '
  158. 'is due to be deprecated.',
  159. PendingDeprecationWarning, stacklevel=2)
  160. if self.paginate_by_param:
  161. query_params = self.request.QUERY_PARAMS
  162. try:
  163. return int(query_params[self.paginate_by_param])
  164. except (KeyError, ValueError):
  165. pass
  166. return self.paginate_by
  167. def get_serializer_class(self):
  168. """
  169. Return the class to use for the serializer.
  170. Defaults to using `self.serializer_class`.
  171. You may want to override this if you need to provide different
  172. serializations depending on the incoming request.
  173. (Eg. admins get full serialization, others get basic serilization)
  174. """
  175. serializer_class = self.serializer_class
  176. if serializer_class is not None:
  177. return serializer_class
  178. assert self.model is not None, \
  179. "'%s' should either include a 'serializer_class' attribute, " \
  180. "or use the 'model' attribute as a shortcut for " \
  181. "automatically generating a serializer class." \
  182. % self.__class__.__name__
  183. class DefaultSerializer(self.model_serializer_class):
  184. class Meta:
  185. model = self.model
  186. return DefaultSerializer
  187. def get_queryset(self):
  188. """
  189. Get the list of items for this view.
  190. This must be an iterable, and may be a queryset.
  191. Defaults to using `self.queryset`.
  192. You may want to override this if you need to provide different
  193. querysets depending on the incoming request.
  194. (Eg. return a list of items that is specific to the user)
  195. """
  196. if self.queryset is not None:
  197. return self.queryset._clone()
  198. if self.model is not None:
  199. return self.model._default_manager.all()
  200. raise ImproperlyConfigured("'%s' must define 'queryset' or 'model'"
  201. % self.__class__.__name__)
  202. def get_object(self, queryset=None):
  203. """
  204. Returns the object the view is displaying.
  205. You may want to override this if you need to provide non-standard
  206. queryset lookups. Eg if objects are referenced using multiple
  207. keyword arguments in the url conf.
  208. """
  209. # Determine the base queryset to use.
  210. if queryset is None:
  211. queryset = self.filter_queryset(self.get_queryset())
  212. else:
  213. pass # Deprecation warning
  214. # Perform the lookup filtering.
  215. pk = self.kwargs.get(self.pk_url_kwarg, None)
  216. slug = self.kwargs.get(self.slug_url_kwarg, None)
  217. lookup = self.kwargs.get(self.lookup_field, None)
  218. if lookup is not None:
  219. filter_kwargs = {self.lookup_field: lookup}
  220. elif pk is not None and self.lookup_field == 'pk':
  221. warnings.warn(
  222. 'The `pk_url_kwarg` attribute is due to be deprecated. '
  223. 'Use the `lookup_field` attribute instead',
  224. PendingDeprecationWarning
  225. )
  226. filter_kwargs = {'pk': pk}
  227. elif slug is not None and self.lookup_field == 'pk':
  228. warnings.warn(
  229. 'The `slug_url_kwarg` attribute is due to be deprecated. '
  230. 'Use the `lookup_field` attribute instead',
  231. PendingDeprecationWarning
  232. )
  233. filter_kwargs = {self.slug_field: slug}
  234. else:
  235. raise ConfigurationError(
  236. 'Expected view %s to be called with a URL keyword argument '
  237. 'named "%s". Fix your URL conf, or set the `.lookup_field` '
  238. 'attribute on the view correctly.' %
  239. (self.__class__.__name__, self.lookup_field)
  240. )
  241. obj = get_object_or_404(queryset, **filter_kwargs)
  242. # May raise a permission denied
  243. self.check_object_permissions(self.request, obj)
  244. return obj
  245. ########################
  246. ### The following are placeholder methods,
  247. ### and are intended to be overridden.
  248. ###
  249. ### The are not called by GenericAPIView directly,
  250. ### but are used by the mixin methods.
  251. def pre_save(self, obj):
  252. """
  253. Placeholder method for calling before saving an object.
  254. May be used to set attributes on the object that are implicit
  255. in either the request, or the url.
  256. """
  257. pass
  258. def post_save(self, obj, created=False):
  259. """
  260. Placeholder method for calling after saving an object.
  261. """
  262. pass
  263. ##########################################################
  264. ### Concrete view classes that provide method handlers ###
  265. ### by composing the mixin classes with the base view. ###
  266. ##########################################################
  267. class CreateAPIView(mixins.CreateModelMixin,
  268. GenericAPIView):
  269. """
  270. Concrete view for creating a model instance.
  271. """
  272. def post(self, request, *args, **kwargs):
  273. return self.create(request, *args, **kwargs)
  274. class ListAPIView(mixins.ListModelMixin,
  275. GenericAPIView):
  276. """
  277. Concrete view for listing a queryset.
  278. """
  279. def get(self, request, *args, **kwargs):
  280. return self.list(request, *args, **kwargs)
  281. class RetrieveAPIView(mixins.RetrieveModelMixin,
  282. GenericAPIView):
  283. """
  284. Concrete view for retrieving a model instance.
  285. """
  286. def get(self, request, *args, **kwargs):
  287. return self.retrieve(request, *args, **kwargs)
  288. class DestroyAPIView(mixins.DestroyModelMixin,
  289. GenericAPIView):
  290. """
  291. Concrete view for deleting a model instance.
  292. """
  293. def delete(self, request, *args, **kwargs):
  294. return self.destroy(request, *args, **kwargs)
  295. class UpdateAPIView(mixins.UpdateModelMixin,
  296. GenericAPIView):
  297. """
  298. Concrete view for updating a model instance.
  299. """
  300. def put(self, request, *args, **kwargs):
  301. return self.update(request, *args, **kwargs)
  302. def patch(self, request, *args, **kwargs):
  303. return self.partial_update(request, *args, **kwargs)
  304. class ListCreateAPIView(mixins.ListModelMixin,
  305. mixins.CreateModelMixin,
  306. GenericAPIView):
  307. """
  308. Concrete view for listing a queryset or creating a model instance.
  309. """
  310. def get(self, request, *args, **kwargs):
  311. return self.list(request, *args, **kwargs)
  312. def post(self, request, *args, **kwargs):
  313. return self.create(request, *args, **kwargs)
  314. class RetrieveUpdateAPIView(mixins.RetrieveModelMixin,
  315. mixins.UpdateModelMixin,
  316. GenericAPIView):
  317. """
  318. Concrete view for retrieving, updating a model instance.
  319. """
  320. def get(self, request, *args, **kwargs):
  321. return self.retrieve(request, *args, **kwargs)
  322. def put(self, request, *args, **kwargs):
  323. return self.update(request, *args, **kwargs)
  324. def patch(self, request, *args, **kwargs):
  325. return self.partial_update(request, *args, **kwargs)
  326. class RetrieveDestroyAPIView(mixins.RetrieveModelMixin,
  327. mixins.DestroyModelMixin,
  328. GenericAPIView):
  329. """
  330. Concrete view for retrieving or deleting a model instance.
  331. """
  332. def get(self, request, *args, **kwargs):
  333. return self.retrieve(request, *args, **kwargs)
  334. def delete(self, request, *args, **kwargs):
  335. return self.destroy(request, *args, **kwargs)
  336. class RetrieveUpdateDestroyAPIView(mixins.RetrieveModelMixin,
  337. mixins.UpdateModelMixin,
  338. mixins.DestroyModelMixin,
  339. GenericAPIView):
  340. """
  341. Concrete view for retrieving, updating or deleting a model instance.
  342. """
  343. def get(self, request, *args, **kwargs):
  344. return self.retrieve(request, *args, **kwargs)
  345. def put(self, request, *args, **kwargs):
  346. return self.update(request, *args, **kwargs)
  347. def patch(self, request, *args, **kwargs):
  348. return self.partial_update(request, *args, **kwargs)
  349. def delete(self, request, *args, **kwargs):
  350. return self.destroy(request, *args, **kwargs)
  351. ##########################
  352. ### Deprecated classes ###
  353. ##########################
  354. class MultipleObjectAPIView(GenericAPIView):
  355. def __init__(self, *args, **kwargs):
  356. warnings.warn(
  357. 'Subclassing `MultipleObjectAPIView` is due to be deprecated. '
  358. 'You should simply subclass `GenericAPIView` instead.',
  359. PendingDeprecationWarning, stacklevel=2
  360. )
  361. super(MultipleObjectAPIView, self).__init__(*args, **kwargs)
  362. class SingleObjectAPIView(GenericAPIView):
  363. def __init__(self, *args, **kwargs):
  364. warnings.warn(
  365. 'Subclassing `SingleObjectAPIView` is due to be deprecated. '
  366. 'You should simply subclass `GenericAPIView` instead.',
  367. PendingDeprecationWarning, stacklevel=2
  368. )
  369. super(SingleObjectAPIView, self).__init__(*args, **kwargs)