Skip to content

Commit 397c220

Browse files
[5.2.x] Fixed CVE-2026-4292 -- Disallowed instance creation via ModelAdmin.list_editable.
Thanks Natalia Bidart, Jake Howard, and Markus Holtermann for reviews. Backport of 6afe7ce from main.
1 parent 60ffa95 commit 397c220

5 files changed

Lines changed: 49 additions & 1 deletion

File tree

django/contrib/admin/options.py

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@
3333
from django.contrib.admin.widgets import AutocompleteSelect, AutocompleteSelectMultiple
3434
from django.contrib.auth import get_permission_codename
3535
from django.core.exceptions import (
36+
BadRequest,
3637
FieldDoesNotExist,
3738
FieldError,
3839
PermissionDenied,
@@ -2114,6 +2115,8 @@ def changelist_view(self, request, extra_context=None):
21142115
for form in formset.forms:
21152116
if form.has_changed():
21162117
obj = self.save_form(request, form, change=True)
2118+
if obj._state.adding:
2119+
raise BadRequest("list_editable does not allow adding.")
21172120
self.save_model(request, obj, form, change=True)
21182121
self.save_related(request, form, formsets=[], change=True)
21192122
change_msg = self.construct_change_message(

docs/releases/4.2.30.txt

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -36,3 +36,13 @@ forged ``POST`` data in
3636

3737
This issue has severity "low" according to the :ref:`Django security policy
3838
<security-disclosure>`.
39+
40+
CVE-2026-4292: Privilege abuse in ``ModelAdmin.list_editable``
41+
==============================================================
42+
43+
Admin changelist forms using
44+
:attr:`~django.contrib.admin.ModelAdmin.list_editable` incorrectly allowed new
45+
instances to be created via forged ``POST`` data.
46+
47+
This issue has severity "low" according to the :ref:`Django security policy
48+
<security-disclosure>`.

docs/releases/5.2.13.txt

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -36,3 +36,13 @@ forged ``POST`` data in
3636

3737
This issue has severity "low" according to the :ref:`Django security policy
3838
<security-disclosure>`.
39+
40+
CVE-2026-4292: Privilege abuse in ``ModelAdmin.list_editable``
41+
==============================================================
42+
43+
Admin changelist forms using
44+
:attr:`~django.contrib.admin.ModelAdmin.list_editable` incorrectly allowed new
45+
instances to be created via forged ``POST`` data.
46+
47+
This issue has severity "low" according to the :ref:`Django security policy
48+
<security-disclosure>`.

tests/admin_views/admin.py

Lines changed: 9 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -363,6 +363,14 @@ def get_queryset(self, request):
363363
return super().get_queryset(request).order_by("age")
364364

365365

366+
class ParentWithUUIDPKAdmin(admin.ModelAdmin):
367+
list_display = ("id", "title")
368+
list_editable = ("title",)
369+
370+
def has_add_permission(self, request):
371+
return False
372+
373+
366374
class FooAccountAdmin(admin.StackedInline):
367375
model = FooAccount
368376
extra = 1
@@ -1259,7 +1267,7 @@ class CamelCaseAdmin(admin.ModelAdmin):
12591267
site.register(InlineReferer, InlineRefererAdmin)
12601268
site.register(ReferencedByGenRel)
12611269
site.register(GenRelReference)
1262-
site.register(ParentWithUUIDPK)
1270+
site.register(ParentWithUUIDPK, ParentWithUUIDPKAdmin)
12631271
site.register(RelatedPrepopulated, search_fields=["name"])
12641272
site.register(RelatedWithUUIDPKModel)
12651273
site.register(ReadOnlyRelatedField, ReadOnlyRelatedFieldAdmin)

tests/admin_views/tests.py

Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
import re
44
import unittest
55
import zoneinfo
6+
from http import HTTPStatus
67
from unittest import mock
78
from urllib.parse import parse_qsl, urljoin, urlsplit
89

@@ -4335,6 +4336,22 @@ def test_post_submission(self):
43354336

43364337
self.assertIs(Person.objects.get(name="John Mauchly").alive, False)
43374338

4339+
def test_forged_post_submission_when_no_add_permission(self):
4340+
before_count = ParentWithUUIDPK.objects.count()
4341+
data = {
4342+
"form-TOTAL_FORMS": "1",
4343+
"form-INITIAL_FORMS": "0",
4344+
"form-MAX_NUM_FORMS": "0",
4345+
"form-0-title": "The News",
4346+
"form-0-id": "",
4347+
"_save": "Save",
4348+
}
4349+
# This model admin allows no add permissions.
4350+
changelist_url = reverse("admin:admin_views_parentwithuuidpk_changelist")
4351+
response = self.client.post(changelist_url, data)
4352+
self.assertEqual(response.status_code, HTTPStatus.BAD_REQUEST)
4353+
self.assertEqual(ParentWithUUIDPK.objects.count(), before_count)
4354+
43384355
def test_non_field_errors(self):
43394356
"""
43404357
Non-field errors are displayed for each of the forms in the

0 commit comments

Comments
 (0)