actions.txt 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  1. =============
  2. Admin actions
  3. =============
  4. .. currentmodule:: django.contrib.admin
  5. The basic workflow of Django's admin is, in a nutshell, "select an object,
  6. then change it." This works well for a majority of use cases. However, if you
  7. need to make the same change to many objects at once, this workflow can be
  8. quite tedious.
  9. In these cases, Django's admin lets you write and register "actions" --
  10. functions that get called with a list of objects selected on the change list
  11. page.
  12. If you look at any change list in the admin, you'll see this feature in
  13. action; Django ships with a "delete selected objects" action available to all
  14. models. For example, here's the user module from Django's built-in
  15. :mod:`django.contrib.auth` app:
  16. .. image:: _images/admin-actions.png
  17. .. warning::
  18. The "delete selected objects" action uses :meth:`QuerySet.delete()
  19. <django.db.models.query.QuerySet.delete>` for efficiency reasons, which
  20. has an important caveat: your model's ``delete()`` method will not be
  21. called.
  22. If you wish to override this behavior, you can override
  23. :meth:`.ModelAdmin.delete_queryset` or write a custom action which does
  24. deletion in your preferred manner -- for example, by calling
  25. ``Model.delete()`` for each of the selected items.
  26. For more background on bulk deletion, see the documentation on :ref:`object
  27. deletion <topics-db-queries-delete>`.
  28. Read on to find out how to add your own actions to this list.
  29. Writing actions
  30. ===============
  31. The easiest way to explain actions is by example, so let's dive in.
  32. A common use case for admin actions is the bulk updating of a model. Imagine a
  33. news application with an ``Article`` model::
  34. from django.db import models
  35. STATUS_CHOICES = [
  36. ('d', 'Draft'),
  37. ('p', 'Published'),
  38. ('w', 'Withdrawn'),
  39. ]
  40. class Article(models.Model):
  41. title = models.CharField(max_length=100)
  42. body = models.TextField()
  43. status = models.CharField(max_length=1, choices=STATUS_CHOICES)
  44. def __str__(self):
  45. return self.title
  46. A common task we might perform with a model like this is to update an
  47. article's status from "draft" to "published". We could easily do this in the
  48. admin one article at a time, but if we wanted to bulk-publish a group of
  49. articles, it'd be tedious. So, let's write an action that lets us change an
  50. article's status to "published."
  51. Writing action functions
  52. ------------------------
  53. First, we'll need to write a function that gets called when the action is
  54. triggered from the admin. Action functions are regular functions that take
  55. three arguments:
  56. * The current :class:`ModelAdmin`
  57. * An :class:`~django.http.HttpRequest` representing the current request,
  58. * A :class:`~django.db.models.query.QuerySet` containing the set of
  59. objects selected by the user.
  60. Our publish-these-articles function won't need the :class:`ModelAdmin` or the
  61. request object, but we will use the queryset::
  62. def make_published(modeladmin, request, queryset):
  63. queryset.update(status='p')
  64. .. note::
  65. For the best performance, we're using the queryset's :ref:`update method
  66. <topics-db-queries-update>`. Other types of actions might need to deal
  67. with each object individually; in these cases we'd iterate over the
  68. queryset::
  69. for obj in queryset:
  70. do_something_with(obj)
  71. That's actually all there is to writing an action! However, we'll take one
  72. more optional-but-useful step and give the action a "nice" title in the admin.
  73. By default, this action would appear in the action list as "Make published" --
  74. the function name, with underscores replaced by spaces. That's fine, but we
  75. can provide a better, more human-friendly name by giving the
  76. ``make_published`` function a ``short_description`` attribute::
  77. def make_published(modeladmin, request, queryset):
  78. queryset.update(status='p')
  79. make_published.short_description = "Mark selected stories as published"
  80. .. note::
  81. This might look familiar; the admin's ``list_display`` option uses the
  82. same technique to provide human-readable descriptions for callback
  83. functions registered there, too.
  84. Adding actions to the :class:`ModelAdmin`
  85. -----------------------------------------
  86. Next, we'll need to inform our :class:`ModelAdmin` of the action. This works
  87. just like any other configuration option. So, the complete ``admin.py`` with
  88. the action and its registration would look like::
  89. from django.contrib import admin
  90. from myapp.models import Article
  91. def make_published(modeladmin, request, queryset):
  92. queryset.update(status='p')
  93. make_published.short_description = "Mark selected stories as published"
  94. class ArticleAdmin(admin.ModelAdmin):
  95. list_display = ['title', 'status']
  96. ordering = ['title']
  97. actions = [make_published]
  98. admin.site.register(Article, ArticleAdmin)
  99. That code will give us an admin change list that looks something like this:
  100. .. image:: _images/adding-actions-to-the-modeladmin.png
  101. That's really all there is to it! If you're itching to write your own actions,
  102. you now know enough to get started. The rest of this document covers more
  103. advanced techniques.
  104. Handling errors in actions
  105. --------------------------
  106. If there are foreseeable error conditions that may occur while running your
  107. action, you should gracefully inform the user of the problem. This means
  108. handling exceptions and using
  109. :meth:`django.contrib.admin.ModelAdmin.message_user` to display a user friendly
  110. description of the problem in the response.
  111. Advanced action techniques
  112. ==========================
  113. There's a couple of extra options and possibilities you can exploit for more
  114. advanced options.
  115. Actions as :class:`ModelAdmin` methods
  116. --------------------------------------
  117. The example above shows the ``make_published`` action defined as a function.
  118. That's perfectly fine, but it's not perfect from a code design point of view:
  119. since the action is tightly coupled to the ``Article`` object, it makes sense
  120. to hook the action to the ``ArticleAdmin`` object itself.
  121. You can do it like this::
  122. class ArticleAdmin(admin.ModelAdmin):
  123. ...
  124. actions = ['make_published']
  125. def make_published(self, request, queryset):
  126. queryset.update(status='p')
  127. make_published.short_description = "Mark selected stories as published"
  128. Notice first that we've moved ``make_published`` into a method and renamed the
  129. ``modeladmin`` parameter to ``self``, and second that we've now put the string
  130. ``'make_published'`` in ``actions`` instead of a direct function reference. This
  131. tells the :class:`ModelAdmin` to look up the action as a method.
  132. Defining actions as methods gives the action more idiomatic access to the
  133. :class:`ModelAdmin` itself, allowing the action to call any of the methods
  134. provided by the admin.
  135. .. _custom-admin-action:
  136. For example, we can use ``self`` to flash a message to the user informing her
  137. that the action was successful::
  138. class ArticleAdmin(admin.ModelAdmin):
  139. ...
  140. def make_published(self, request, queryset):
  141. rows_updated = queryset.update(status='p')
  142. if rows_updated == 1:
  143. message_bit = "1 story was"
  144. else:
  145. message_bit = "%s stories were" % rows_updated
  146. self.message_user(request, "%s successfully marked as published." % message_bit)
  147. This make the action match what the admin itself does after successfully
  148. performing an action:
  149. .. image:: _images/actions-as-modeladmin-methods.png
  150. Actions that provide intermediate pages
  151. ---------------------------------------
  152. By default, after an action is performed the user is redirected back to the
  153. original change list page. However, some actions, especially more complex ones,
  154. will need to return intermediate pages. For example, the built-in delete action
  155. asks for confirmation before deleting the selected objects.
  156. To provide an intermediary page, return an :class:`~django.http.HttpResponse`
  157. (or subclass) from your action. For example, you might write a export function
  158. that uses Django's :doc:`serialization functions </topics/serialization>` to
  159. dump some selected objects as JSON::
  160. from django.core import serializers
  161. from django.http import HttpResponse
  162. def export_as_json(modeladmin, request, queryset):
  163. response = HttpResponse(content_type="application/json")
  164. serializers.serialize("json", queryset, stream=response)
  165. return response
  166. Generally, something like the above isn't considered a great idea. Most of the
  167. time, the best practice will be to return an
  168. :class:`~django.http.HttpResponseRedirect` and redirect the user to a view
  169. you've written, passing the list of selected objects in the GET query string.
  170. This allows you to provide complex interaction logic on the intermediary
  171. pages. For example, if you wanted to provide a more complete export function,
  172. you'd want to let the user choose a format, and possibly a list of fields to
  173. include in the export. The best thing to do would be to write a small action
  174. that redirects to your custom export view::
  175. from django.contrib import admin
  176. from django.contrib.contenttypes.models import ContentType
  177. from django.http import HttpResponseRedirect
  178. def export_selected_objects(modeladmin, request, queryset):
  179. selected = request.POST.getlist(admin.ACTION_CHECKBOX_NAME)
  180. ct = ContentType.objects.get_for_model(queryset.model)
  181. return HttpResponseRedirect("/export/?ct=%s&ids=%s" % (ct.pk, ",".join(selected)))
  182. As you can see, the action is rather short; all the complex logic would belong
  183. in your export view. This would need to deal with objects of any type, hence
  184. the business with the ``ContentType``.
  185. Writing this view is left as an exercise to the reader.
  186. .. _adminsite-actions:
  187. Making actions available site-wide
  188. ----------------------------------
  189. .. method:: AdminSite.add_action(action, name=None)
  190. Some actions are best if they're made available to *any* object in the admin
  191. site -- the export action defined above would be a good candidate. You can
  192. make an action globally available using :meth:`AdminSite.add_action()`. For
  193. example::
  194. from django.contrib import admin
  195. admin.site.add_action(export_selected_objects)
  196. This makes the ``export_selected_objects`` action globally available as an
  197. action named "export_selected_objects". You can explicitly give the action
  198. a name -- good if you later want to programmatically :ref:`remove the action
  199. <disabling-admin-actions>` -- by passing a second argument to
  200. :meth:`AdminSite.add_action()`::
  201. admin.site.add_action(export_selected_objects, 'export_selected')
  202. .. _disabling-admin-actions:
  203. Disabling actions
  204. -----------------
  205. Sometimes you need to disable certain actions -- especially those
  206. :ref:`registered site-wide <adminsite-actions>` -- for particular objects.
  207. There's a few ways you can disable actions:
  208. Disabling a site-wide action
  209. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  210. .. method:: AdminSite.disable_action(name)
  211. If you need to disable a :ref:`site-wide action <adminsite-actions>` you can
  212. call :meth:`AdminSite.disable_action()`.
  213. For example, you can use this method to remove the built-in "delete selected
  214. objects" action::
  215. admin.site.disable_action('delete_selected')
  216. Once you've done the above, that action will no longer be available
  217. site-wide.
  218. If, however, you need to re-enable a globally-disabled action for one
  219. particular model, list it explicitly in your ``ModelAdmin.actions`` list::
  220. # Globally disable delete selected
  221. admin.site.disable_action('delete_selected')
  222. # This ModelAdmin will not have delete_selected available
  223. class SomeModelAdmin(admin.ModelAdmin):
  224. actions = ['some_other_action']
  225. ...
  226. # This one will
  227. class AnotherModelAdmin(admin.ModelAdmin):
  228. actions = ['delete_selected', 'a_third_action']
  229. ...
  230. Disabling all actions for a particular :class:`ModelAdmin`
  231. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  232. If you want *no* bulk actions available for a given :class:`ModelAdmin`, set
  233. :attr:`ModelAdmin.actions` to ``None``::
  234. class MyModelAdmin(admin.ModelAdmin):
  235. actions = None
  236. This tells the :class:`ModelAdmin` to not display or allow any actions,
  237. including any :ref:`site-wide actions <adminsite-actions>`.
  238. Conditionally enabling or disabling actions
  239. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  240. .. method:: ModelAdmin.get_actions(request)
  241. Finally, you can conditionally enable or disable actions on a per-request
  242. (and hence per-user basis) by overriding :meth:`ModelAdmin.get_actions`.
  243. This returns a dictionary of actions allowed. The keys are action names, and
  244. the values are ``(function, name, short_description)`` tuples.
  245. For example, if you only want users whose names begin with 'J' to be able
  246. to delete objects in bulk::
  247. class MyModelAdmin(admin.ModelAdmin):
  248. ...
  249. def get_actions(self, request):
  250. actions = super().get_actions(request)
  251. if request.user.username[0].upper() != 'J':
  252. if 'delete_selected' in actions:
  253. del actions['delete_selected']
  254. return actions
  255. .. _admin-action-permissions:
  256. Setting permissions for actions
  257. -------------------------------
  258. Actions may limit their availability to users with specific permissions by
  259. setting an ``allowed_permissions`` attribute on the action function::
  260. def make_published(modeladmin, request, queryset):
  261. queryset.update(status='p')
  262. make_published.allowed_permissions = ('change',)
  263. The ``make_published()`` action will only be available to users that pass the
  264. :meth:`.ModelAdmin.has_change_permission` check.
  265. If ``allowed_permissions`` has more than one permission, the action will be
  266. available as long as the user passes at least one of the checks.
  267. Available values for ``allowed_permissions`` and the corresponding method
  268. checks are:
  269. - ``'add'``: :meth:`.ModelAdmin.has_add_permission`
  270. - ``'change'``: :meth:`.ModelAdmin.has_change_permission`
  271. - ``'delete'``: :meth:`.ModelAdmin.has_delete_permission`
  272. - ``'view'``: :meth:`.ModelAdmin.has_view_permission`
  273. You can specify any other value as long as you implement a corresponding
  274. ``has_<value>_permission(self, request)`` method on the ``ModelAdmin``.
  275. For example::
  276. from django.contrib import admin
  277. from django.contrib.auth import get_permission_codename
  278. class ArticleAdmin(admin.ModelAdmin):
  279. actions = ['make_published']
  280. def make_published(self, request, queryset):
  281. queryset.update(status='p')
  282. make_published.allowed_permissions = ('publish',)
  283. def has_publish_permission(self, request):
  284. """Does the user have the publish permission?"""
  285. opts = self.opts
  286. codename = get_permission_codename('publish', opts)
  287. return request.user.has_perm('%s.%s' % (opts.app_label, codename))