2012-04-29 17:51:12 +08:00
|
|
|
|
import datetime
|
2022-09-24 18:12:28 +08:00
|
|
|
|
from unittest import mock
|
2012-04-29 17:51:12 +08:00
|
|
|
|
|
2009-05-14 23:09:33 +08:00
|
|
|
|
from django.contrib import admin
|
2015-02-06 02:25:59 +08:00
|
|
|
|
from django.contrib.admin.models import LogEntry
|
2010-12-13 06:58:47 +08:00
|
|
|
|
from django.contrib.admin.options import IncorrectLookupParameters
|
2013-05-29 21:23:08 +08:00
|
|
|
|
from django.contrib.admin.templatetags.admin_list import pagination
|
2016-02-07 10:24:36 +08:00
|
|
|
|
from django.contrib.admin.tests import AdminSeleniumTestCase
|
2020-05-19 15:03:04 +08:00
|
|
|
|
from django.contrib.admin.views.main import (
|
|
|
|
|
ALL_VAR,
|
2023-02-16 20:23:24 +08:00
|
|
|
|
IS_FACETS_VAR,
|
2020-05-19 15:03:04 +08:00
|
|
|
|
IS_POPUP_VAR,
|
|
|
|
|
ORDER_VAR,
|
|
|
|
|
PAGE_VAR,
|
|
|
|
|
SEARCH_VAR,
|
|
|
|
|
TO_FIELD_VAR,
|
|
|
|
|
)
|
2011-10-14 02:51:33 +08:00
|
|
|
|
from django.contrib.auth.models import User
|
2015-02-06 02:25:59 +08:00
|
|
|
|
from django.contrib.contenttypes.models import ContentType
|
2019-08-22 20:09:49 +08:00
|
|
|
|
from django.contrib.messages.storage.cookie import CookieStorage
|
2022-09-24 18:12:28 +08:00
|
|
|
|
from django.db import DatabaseError, connection, models
|
2019-08-20 15:54:41 +08:00
|
|
|
|
from django.db.models import F, Field, IntegerField
|
2017-12-28 01:38:30 +08:00
|
|
|
|
from django.db.models.functions import Upper
|
2017-07-05 19:00:10 +08:00
|
|
|
|
from django.db.models.lookups import Contains, Exact
|
2018-03-25 21:02:07 +08:00
|
|
|
|
from django.template import Context, Template, TemplateSyntaxError
|
2022-09-24 18:12:28 +08:00
|
|
|
|
from django.test import TestCase, override_settings, skipUnlessDBFeature
|
2011-06-09 06:53:55 +08:00
|
|
|
|
from django.test.client import RequestFactory
|
2019-01-17 23:22:14 +08:00
|
|
|
|
from django.test.utils import CaptureQueriesContext, isolate_apps, register_lookup
|
2015-12-30 23:51:16 +08:00
|
|
|
|
from django.urls import reverse
|
2016-12-29 23:27:49 +08:00
|
|
|
|
from django.utils import formats
|
2015-01-28 20:35:27 +08:00
|
|
|
|
|
|
|
|
|
from .admin import (
|
2015-04-14 17:09:58 +08:00
|
|
|
|
BandAdmin,
|
|
|
|
|
ChildAdmin,
|
|
|
|
|
ChordsBandAdmin,
|
|
|
|
|
ConcertAdmin,
|
|
|
|
|
CustomPaginationAdmin,
|
|
|
|
|
CustomPaginator,
|
|
|
|
|
DynamicListDisplayChildAdmin,
|
2015-01-28 20:35:27 +08:00
|
|
|
|
DynamicListDisplayLinksChildAdmin,
|
|
|
|
|
DynamicListFilterChildAdmin,
|
2015-11-07 19:31:06 +08:00
|
|
|
|
DynamicSearchFieldsChildAdmin,
|
|
|
|
|
EmptyValueChildAdmin,
|
|
|
|
|
EventAdmin,
|
|
|
|
|
FilteredChildAdmin,
|
|
|
|
|
GroupAdmin,
|
|
|
|
|
InvitationAdmin,
|
2021-10-06 09:38:15 +08:00
|
|
|
|
NoListDisplayLinksParentAdmin,
|
|
|
|
|
ParentAdmin,
|
|
|
|
|
ParentAdminTwoSearchFields,
|
|
|
|
|
QuartetAdmin,
|
|
|
|
|
SwallowAdmin,
|
2015-01-28 20:35:27 +08:00
|
|
|
|
)
|
2016-02-07 10:24:36 +08:00
|
|
|
|
from .admin import site as custom_site
|
2015-01-28 20:35:27 +08:00
|
|
|
|
from .models import (
|
2017-07-05 19:00:10 +08:00
|
|
|
|
Band,
|
|
|
|
|
CharPK,
|
|
|
|
|
Child,
|
|
|
|
|
ChordsBand,
|
|
|
|
|
ChordsMusician,
|
|
|
|
|
Concert,
|
|
|
|
|
CustomIdUser,
|
|
|
|
|
Event,
|
|
|
|
|
Genre,
|
|
|
|
|
Group,
|
|
|
|
|
Invitation,
|
|
|
|
|
Membership,
|
|
|
|
|
Musician,
|
|
|
|
|
OrderedObject,
|
|
|
|
|
Parent,
|
|
|
|
|
Quartet,
|
|
|
|
|
Swallow,
|
|
|
|
|
SwallowOneToOne,
|
|
|
|
|
UnorderedObject,
|
2015-01-28 20:35:27 +08:00
|
|
|
|
)
|
2011-09-21 02:16:49 +08:00
|
|
|
|
|
2010-12-21 22:57:29 +08:00
|
|
|
|
|
2023-02-01 02:05:03 +08:00
|
|
|
|
def build_tbody_html(obj, href, extra_fields):
|
2017-08-14 18:46:02 +08:00
|
|
|
|
return (
|
2020-03-08 01:35:59 +08:00
|
|
|
|
"<tbody><tr>"
|
2017-08-14 18:46:02 +08:00
|
|
|
|
'<td class="action-checkbox">'
|
|
|
|
|
'<input type="checkbox" name="_selected_action" value="{}" '
|
2023-02-01 02:05:03 +08:00
|
|
|
|
'class="action-select" aria-label="Select this object for an action - {}"></td>'
|
2017-08-14 18:46:02 +08:00
|
|
|
|
'<th class="field-name"><a href="{}">name</a></th>'
|
|
|
|
|
"{}</tr></tbody>"
|
2023-02-01 02:05:03 +08:00
|
|
|
|
).format(obj.pk, str(obj), href, extra_fields)
|
2015-09-25 21:33:03 +08:00
|
|
|
|
|
|
|
|
|
|
2014-04-05 14:04:46 +08:00
|
|
|
|
@override_settings(ROOT_URLCONF="admin_changelist.urls")
|
2011-06-09 07:09:32 +08:00
|
|
|
|
class ChangeListTests(TestCase):
|
2018-11-27 03:01:27 +08:00
|
|
|
|
factory = RequestFactory()
|
2011-09-21 02:16:49 +08:00
|
|
|
|
|
2018-11-24 09:59:38 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
def setUpTestData(cls):
|
|
|
|
|
cls.superuser = User.objects.create_superuser(
|
|
|
|
|
username="super", email="a@b.com", password="xxx"
|
|
|
|
|
)
|
|
|
|
|
|
2011-11-18 05:30:07 +08:00
|
|
|
|
def _create_superuser(self, username):
|
2015-11-20 00:13:39 +08:00
|
|
|
|
return User.objects.create_superuser(
|
|
|
|
|
username=username, email="a@b.com", password="xxx"
|
|
|
|
|
)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
|
|
|
|
|
def _mocked_authenticated_request(self, url, user):
|
|
|
|
|
request = self.factory.get(url)
|
|
|
|
|
request.user = user
|
|
|
|
|
return request
|
|
|
|
|
|
2021-05-31 10:56:08 +08:00
|
|
|
|
def test_repr(self):
|
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
|
|
|
|
request = self.factory.get("/child/")
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(repr(cl), "<ChangeList: model=Child model_admin=ChildAdmin>")
|
|
|
|
|
|
2017-12-28 01:38:30 +08:00
|
|
|
|
def test_specified_ordering_by_f_expression(self):
|
|
|
|
|
class OrderedByFBandAdmin(admin.ModelAdmin):
|
|
|
|
|
list_display = ["name", "genres", "nr_of_members"]
|
|
|
|
|
ordering = (
|
|
|
|
|
F("nr_of_members").desc(nulls_last=True),
|
|
|
|
|
Upper(F("name")).asc(),
|
|
|
|
|
F("genres").asc(),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
m = OrderedByFBandAdmin(Band, custom_site)
|
|
|
|
|
request = self.factory.get("/band/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-12-28 01:38:30 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(cl.get_ordering_field_columns(), {3: "desc", 2: "asc"})
|
|
|
|
|
|
2018-06-15 02:22:04 +08:00
|
|
|
|
def test_specified_ordering_by_f_expression_without_asc_desc(self):
|
|
|
|
|
class OrderedByFBandAdmin(admin.ModelAdmin):
|
|
|
|
|
list_display = ["name", "genres", "nr_of_members"]
|
|
|
|
|
ordering = (F("nr_of_members"), Upper("name"), F("genres"))
|
|
|
|
|
|
|
|
|
|
m = OrderedByFBandAdmin(Band, custom_site)
|
|
|
|
|
request = self.factory.get("/band/")
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(cl.get_ordering_field_columns(), {3: "asc", 2: "asc"})
|
|
|
|
|
|
2009-05-14 23:09:33 +08:00
|
|
|
|
def test_select_related_preserved(self):
|
|
|
|
|
"""
|
2013-03-08 22:15:23 +08:00
|
|
|
|
Regression test for #10348: ChangeList.get_queryset() shouldn't
|
|
|
|
|
overwrite a custom select_related provided by ModelAdmin.get_queryset().
|
2009-05-14 23:09:33 +08:00
|
|
|
|
"""
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2015-10-10 09:03:04 +08:00
|
|
|
|
self.assertEqual(cl.queryset.query.select_related, {"parent": {}})
|
2013-06-05 05:35:11 +08:00
|
|
|
|
|
2021-04-26 18:07:16 +08:00
|
|
|
|
def test_select_related_preserved_when_multi_valued_in_search_fields(self):
|
|
|
|
|
parent = Parent.objects.create(name="Mary")
|
|
|
|
|
Child.objects.create(parent=parent, name="Danielle")
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel")
|
|
|
|
|
|
|
|
|
|
m = ParentAdmin(Parent, custom_site)
|
|
|
|
|
request = self.factory.get("/parent/", data={SEARCH_VAR: "daniel"})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
|
|
|
|
# select_related is preserved.
|
|
|
|
|
self.assertEqual(cl.queryset.query.select_related, {"child": {}})
|
|
|
|
|
|
2013-06-05 05:35:11 +08:00
|
|
|
|
def test_select_related_as_tuple(self):
|
2015-05-14 20:14:34 +08:00
|
|
|
|
ia = InvitationAdmin(Invitation, custom_site)
|
2013-06-05 05:35:11 +08:00
|
|
|
|
request = self.factory.get("/invitation/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = ia.get_changelist_instance(request)
|
2013-06-05 05:35:11 +08:00
|
|
|
|
self.assertEqual(cl.queryset.query.select_related, {"player": {}})
|
|
|
|
|
|
|
|
|
|
def test_select_related_as_empty_tuple(self):
|
2015-05-14 20:14:34 +08:00
|
|
|
|
ia = InvitationAdmin(Invitation, custom_site)
|
2013-06-05 05:35:11 +08:00
|
|
|
|
ia.list_select_related = ()
|
|
|
|
|
request = self.factory.get("/invitation/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = ia.get_changelist_instance(request)
|
2016-06-17 02:19:18 +08:00
|
|
|
|
self.assertIs(cl.queryset.query.select_related, False)
|
2013-06-05 05:35:11 +08:00
|
|
|
|
|
2015-03-16 18:00:16 +08:00
|
|
|
|
def test_get_select_related_custom_method(self):
|
|
|
|
|
class GetListSelectRelatedAdmin(admin.ModelAdmin):
|
|
|
|
|
list_display = ("band", "player")
|
|
|
|
|
|
|
|
|
|
def get_list_select_related(self, request):
|
|
|
|
|
return ("band", "player")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
ia = GetListSelectRelatedAdmin(Invitation, custom_site)
|
2015-03-16 18:00:16 +08:00
|
|
|
|
request = self.factory.get("/invitation/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = ia.get_changelist_instance(request)
|
2015-03-16 18:00:16 +08:00
|
|
|
|
self.assertEqual(cl.queryset.query.select_related, {"player": {}, "band": {}})
|
|
|
|
|
|
2021-10-06 09:38:15 +08:00
|
|
|
|
def test_many_search_terms(self):
|
|
|
|
|
parent = Parent.objects.create(name="Mary")
|
|
|
|
|
Child.objects.create(parent=parent, name="Danielle")
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel")
|
|
|
|
|
|
|
|
|
|
m = ParentAdmin(Parent, custom_site)
|
|
|
|
|
request = self.factory.get("/parent/", data={SEARCH_VAR: "daniel " * 80})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
with CaptureQueriesContext(connection) as context:
|
|
|
|
|
object_count = cl.queryset.count()
|
|
|
|
|
self.assertEqual(object_count, 1)
|
|
|
|
|
self.assertEqual(context.captured_queries[0]["sql"].count("JOIN"), 1)
|
|
|
|
|
|
|
|
|
|
def test_related_field_multiple_search_terms(self):
|
|
|
|
|
"""
|
|
|
|
|
Searches over multi-valued relationships return rows from related
|
|
|
|
|
models only when all searched fields match that row.
|
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="Mary")
|
|
|
|
|
Child.objects.create(parent=parent, name="Danielle", age=18)
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel", age=19)
|
|
|
|
|
|
|
|
|
|
m = ParentAdminTwoSearchFields(Parent, custom_site)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/parent/", data={SEARCH_VAR: "danielle 19"})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/parent/", data={SEARCH_VAR: "daniel 19"})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
|
|
|
|
|
2011-01-24 15:01:00 +08:00
|
|
|
|
def test_result_list_empty_changelist_value(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression test for #14982: EMPTY_CHANGELIST_VALUE should be honored
|
|
|
|
|
for relationship fields
|
|
|
|
|
"""
|
|
|
|
|
new_child = Child.objects.create(name="name", parent=None)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-01-24 15:01:00 +08:00
|
|
|
|
cl.formset = None
|
|
|
|
|
template = Template(
|
|
|
|
|
"{% load admin_list %}{% spaceless %}{% result_list cl %}{% endspaceless %}"
|
|
|
|
|
)
|
2017-01-13 00:06:00 +08:00
|
|
|
|
context = Context({"cl": cl, "opts": Child._meta})
|
2011-01-24 15:01:00 +08:00
|
|
|
|
table_output = template.render(context)
|
2012-09-25 09:02:59 +08:00
|
|
|
|
link = reverse("admin:admin_changelist_child_change", args=(new_child.id,))
|
2017-08-14 18:46:02 +08:00
|
|
|
|
row_html = build_tbody_html(
|
2023-02-01 02:05:03 +08:00
|
|
|
|
new_child, link, '<td class="field-parent nowrap">-</td>'
|
2017-08-14 18:46:02 +08:00
|
|
|
|
)
|
2016-04-08 10:04:45 +08:00
|
|
|
|
self.assertNotEqual(
|
|
|
|
|
table_output.find(row_html),
|
|
|
|
|
-1,
|
|
|
|
|
"Failed to find expected row element: %s" % table_output,
|
|
|
|
|
)
|
2015-03-13 18:08:03 +08:00
|
|
|
|
|
|
|
|
|
def test_result_list_set_empty_value_display_on_admin_site(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
Empty value display can be set on AdminSite.
|
2015-03-13 18:08:03 +08:00
|
|
|
|
"""
|
|
|
|
|
new_child = Child.objects.create(name="name", parent=None)
|
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-03-13 18:08:03 +08:00
|
|
|
|
# Set a new empty display value on AdminSite.
|
|
|
|
|
admin.site.empty_value_display = "???"
|
|
|
|
|
m = ChildAdmin(Child, admin.site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2015-03-13 18:08:03 +08:00
|
|
|
|
cl.formset = None
|
|
|
|
|
template = Template(
|
|
|
|
|
"{% load admin_list %}{% spaceless %}{% result_list cl %}{% endspaceless %}"
|
|
|
|
|
)
|
2017-01-13 00:06:00 +08:00
|
|
|
|
context = Context({"cl": cl, "opts": Child._meta})
|
2015-03-13 18:08:03 +08:00
|
|
|
|
table_output = template.render(context)
|
|
|
|
|
link = reverse("admin:admin_changelist_child_change", args=(new_child.id,))
|
2017-08-14 18:46:02 +08:00
|
|
|
|
row_html = build_tbody_html(
|
2023-02-01 02:05:03 +08:00
|
|
|
|
new_child, link, '<td class="field-parent nowrap">???</td>'
|
2017-08-14 18:46:02 +08:00
|
|
|
|
)
|
2016-04-08 10:04:45 +08:00
|
|
|
|
self.assertNotEqual(
|
|
|
|
|
table_output.find(row_html),
|
|
|
|
|
-1,
|
|
|
|
|
"Failed to find expected row element: %s" % table_output,
|
|
|
|
|
)
|
2015-03-13 18:08:03 +08:00
|
|
|
|
|
|
|
|
|
def test_result_list_set_empty_value_display_in_model_admin(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
Empty value display can be set in ModelAdmin or individual fields.
|
2015-03-13 18:08:03 +08:00
|
|
|
|
"""
|
|
|
|
|
new_child = Child.objects.create(name="name", parent=None)
|
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-03-13 18:08:03 +08:00
|
|
|
|
m = EmptyValueChildAdmin(Child, admin.site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2015-03-13 18:08:03 +08:00
|
|
|
|
cl.formset = None
|
|
|
|
|
template = Template(
|
|
|
|
|
"{% load admin_list %}{% spaceless %}{% result_list cl %}{% endspaceless %}"
|
|
|
|
|
)
|
2017-01-13 00:06:00 +08:00
|
|
|
|
context = Context({"cl": cl, "opts": Child._meta})
|
2015-03-13 18:08:03 +08:00
|
|
|
|
table_output = template.render(context)
|
|
|
|
|
link = reverse("admin:admin_changelist_child_change", args=(new_child.id,))
|
2017-08-14 18:46:02 +08:00
|
|
|
|
row_html = build_tbody_html(
|
2023-02-01 02:05:03 +08:00
|
|
|
|
new_child,
|
2017-08-14 18:46:02 +08:00
|
|
|
|
link,
|
|
|
|
|
'<td class="field-age_display">&dagger;</td>'
|
|
|
|
|
'<td class="field-age">-empty-</td>',
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2016-04-08 10:04:45 +08:00
|
|
|
|
self.assertNotEqual(
|
|
|
|
|
table_output.find(row_html),
|
2022-02-04 03:24:19 +08:00
|
|
|
|
-1,
|
2016-04-08 10:04:45 +08:00
|
|
|
|
"Failed to find expected row element: %s" % table_output,
|
2015-03-13 18:08:03 +08:00
|
|
|
|
)
|
2011-01-24 15:01:00 +08:00
|
|
|
|
|
2010-03-02 02:43:27 +08:00
|
|
|
|
def test_result_list_html(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
Inclusion tag result_list generates a table when with default
|
|
|
|
|
ModelAdmin settings.
|
2010-03-02 02:43:27 +08:00
|
|
|
|
"""
|
|
|
|
|
new_parent = Parent.objects.create(name="parent")
|
|
|
|
|
new_child = Child.objects.create(name="name", parent=new_parent)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2010-09-11 11:01:07 +08:00
|
|
|
|
cl.formset = None
|
2010-03-02 02:43:27 +08:00
|
|
|
|
template = Template(
|
|
|
|
|
"{% load admin_list %}{% spaceless %}{% result_list cl %}{% endspaceless %}"
|
|
|
|
|
)
|
2017-01-13 00:06:00 +08:00
|
|
|
|
context = Context({"cl": cl, "opts": Child._meta})
|
2010-03-02 02:43:27 +08:00
|
|
|
|
table_output = template.render(context)
|
2012-09-25 09:02:59 +08:00
|
|
|
|
link = reverse("admin:admin_changelist_child_change", args=(new_child.id,))
|
2017-08-14 18:46:02 +08:00
|
|
|
|
row_html = build_tbody_html(
|
2023-02-01 02:05:03 +08:00
|
|
|
|
new_child, link, '<td class="field-parent nowrap">%s</td>' % new_parent
|
2017-08-14 18:46:02 +08:00
|
|
|
|
)
|
2016-04-08 10:04:45 +08:00
|
|
|
|
self.assertNotEqual(
|
|
|
|
|
table_output.find(row_html),
|
|
|
|
|
-1,
|
|
|
|
|
"Failed to find expected row element: %s" % table_output,
|
|
|
|
|
)
|
2023-02-01 02:05:03 +08:00
|
|
|
|
self.assertInHTML(
|
|
|
|
|
'<input type="checkbox" id="action-toggle" '
|
|
|
|
|
'aria-label="Select all objects on this page for an action">',
|
|
|
|
|
table_output,
|
|
|
|
|
)
|
2010-09-11 11:01:07 +08:00
|
|
|
|
|
|
|
|
|
def test_result_list_editable_html(self):
|
|
|
|
|
"""
|
2010-12-04 15:28:12 +08:00
|
|
|
|
Regression tests for #11791: Inclusion tag result_list generates a
|
2010-09-11 11:01:07 +08:00
|
|
|
|
table and this checks that the items are nested within the table
|
|
|
|
|
element tags.
|
|
|
|
|
Also a regression test for #13599, verifies that hidden fields
|
2010-12-04 15:28:12 +08:00
|
|
|
|
when list_editable is enabled are rendered in a div outside the
|
2010-09-11 11:01:07 +08:00
|
|
|
|
table.
|
|
|
|
|
"""
|
|
|
|
|
new_parent = Parent.objects.create(name="parent")
|
|
|
|
|
new_child = Child.objects.create(name="name", parent=new_parent)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2010-03-02 02:43:27 +08:00
|
|
|
|
|
|
|
|
|
# Test with list_editable fields
|
|
|
|
|
m.list_display = ["id", "name", "parent"]
|
|
|
|
|
m.list_display_links = ["id"]
|
|
|
|
|
m.list_editable = ["name"]
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2010-03-02 02:43:27 +08:00
|
|
|
|
FormSet = m.get_changelist_formset(request)
|
|
|
|
|
cl.formset = FormSet(queryset=cl.result_list)
|
|
|
|
|
template = Template(
|
|
|
|
|
"{% load admin_list %}{% spaceless %}{% result_list cl %}{% endspaceless %}"
|
|
|
|
|
)
|
2017-01-13 00:06:00 +08:00
|
|
|
|
context = Context({"cl": cl, "opts": Child._meta})
|
2010-03-02 02:43:27 +08:00
|
|
|
|
table_output = template.render(context)
|
2010-09-11 11:01:07 +08:00
|
|
|
|
# make sure that hidden fields are in the correct place
|
2015-09-12 07:33:12 +08:00
|
|
|
|
hiddenfields_div = (
|
|
|
|
|
'<div class="hiddenfields">'
|
2018-01-21 15:09:10 +08:00
|
|
|
|
'<input type="hidden" name="form-0-id" value="%d" id="id_form-0-id">'
|
2015-09-12 07:33:12 +08:00
|
|
|
|
"</div>"
|
|
|
|
|
) % new_child.id
|
2012-09-28 15:20:01 +08:00
|
|
|
|
self.assertInHTML(
|
|
|
|
|
hiddenfields_div, table_output, msg_prefix="Failed to find hidden fields"
|
|
|
|
|
)
|
|
|
|
|
|
2010-09-11 11:01:07 +08:00
|
|
|
|
# make sure that list editable fields are rendered in divs correctly
|
2015-09-12 07:33:12 +08:00
|
|
|
|
editable_name_field = (
|
|
|
|
|
'<input name="form-0-name" value="name" class="vTextField" '
|
2018-01-21 15:09:10 +08:00
|
|
|
|
'maxlength="30" type="text" id="id_form-0-name">'
|
2015-09-12 07:33:12 +08:00
|
|
|
|
)
|
|
|
|
|
self.assertInHTML(
|
|
|
|
|
'<td class="field-name">%s</td>' % editable_name_field,
|
|
|
|
|
table_output,
|
|
|
|
|
msg_prefix='Failed to find "name" list_editable field',
|
|
|
|
|
)
|
2010-03-02 02:43:27 +08:00
|
|
|
|
|
2010-12-13 06:58:47 +08:00
|
|
|
|
def test_result_list_editable(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression test for #14312: list_editable with pagination
|
|
|
|
|
"""
|
|
|
|
|
new_parent = Parent.objects.create(name="parent")
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for i in range(1, 201):
|
2013-09-08 23:05:16 +08:00
|
|
|
|
Child.objects.create(name="name %s" % i, parent=new_parent)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/", data={"p": -1}) # Anything outside range
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2010-12-13 06:58:47 +08:00
|
|
|
|
|
|
|
|
|
# Test with list_editable fields
|
|
|
|
|
m.list_display = ["id", "name", "parent"]
|
|
|
|
|
m.list_display_links = ["id"]
|
|
|
|
|
m.list_editable = ["name"]
|
2015-09-25 21:33:03 +08:00
|
|
|
|
with self.assertRaises(IncorrectLookupParameters):
|
2017-08-14 18:46:02 +08:00
|
|
|
|
m.get_changelist_instance(request)
|
2010-12-21 22:57:29 +08:00
|
|
|
|
|
2022-09-24 18:12:28 +08:00
|
|
|
|
@skipUnlessDBFeature("supports_transactions")
|
|
|
|
|
def test_list_editable_atomicity(self):
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
b = Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
|
|
|
|
|
self.client.force_login(self.superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
data = {
|
|
|
|
|
"form-TOTAL_FORMS": "2",
|
|
|
|
|
"form-INITIAL_FORMS": "2",
|
|
|
|
|
"form-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form-MAX_NUM_FORMS": "1000",
|
|
|
|
|
"form-0-uuid": str(a.pk),
|
|
|
|
|
"form-1-uuid": str(b.pk),
|
|
|
|
|
"form-0-load": "9.0",
|
|
|
|
|
"form-0-speed": "3.0",
|
|
|
|
|
"form-1-load": "5.0",
|
|
|
|
|
"form-1-speed": "1.0",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
with mock.patch(
|
|
|
|
|
"django.contrib.admin.ModelAdmin.log_change", side_effect=DatabaseError
|
|
|
|
|
):
|
|
|
|
|
with self.assertRaises(DatabaseError):
|
|
|
|
|
self.client.post(changelist_url, data)
|
|
|
|
|
# Original values are preserved.
|
|
|
|
|
a.refresh_from_db()
|
|
|
|
|
self.assertEqual(a.load, 4)
|
|
|
|
|
self.assertEqual(a.speed, 1)
|
|
|
|
|
b.refresh_from_db()
|
|
|
|
|
self.assertEqual(b.load, 2)
|
|
|
|
|
self.assertEqual(b.speed, 2)
|
|
|
|
|
|
|
|
|
|
with mock.patch(
|
|
|
|
|
"django.contrib.admin.ModelAdmin.log_change",
|
|
|
|
|
side_effect=[None, DatabaseError],
|
|
|
|
|
):
|
|
|
|
|
with self.assertRaises(DatabaseError):
|
|
|
|
|
self.client.post(changelist_url, data)
|
|
|
|
|
# Original values are preserved.
|
|
|
|
|
a.refresh_from_db()
|
|
|
|
|
self.assertEqual(a.load, 4)
|
|
|
|
|
self.assertEqual(a.speed, 1)
|
|
|
|
|
b.refresh_from_db()
|
|
|
|
|
self.assertEqual(b.load, 2)
|
|
|
|
|
self.assertEqual(b.speed, 2)
|
|
|
|
|
|
2010-12-21 22:57:29 +08:00
|
|
|
|
def test_custom_paginator(self):
|
|
|
|
|
new_parent = Parent.objects.create(name="parent")
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for i in range(1, 201):
|
2013-09-08 23:05:16 +08:00
|
|
|
|
Child.objects.create(name="name %s" % i, parent=new_parent)
|
2010-12-21 22:57:29 +08:00
|
|
|
|
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = CustomPaginationAdmin(Child, custom_site)
|
2010-12-21 22:57:29 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2010-12-21 22:57:29 +08:00
|
|
|
|
cl.get_results(request)
|
|
|
|
|
self.assertIsInstance(cl.paginator, CustomPaginator)
|
2010-12-13 06:58:47 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_m2m_in_list_filter(self):
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
Regression test for #13902: When using a ManyToMany in list_filter,
|
2014-03-02 22:25:53 +08:00
|
|
|
|
results shouldn't appear more than once. Basic ManyToMany.
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
blues = Genre.objects.create(name="Blues")
|
|
|
|
|
band = Band.objects.create(name="B.B. King Review", nr_of_members=11)
|
|
|
|
|
|
|
|
|
|
band.genres.add(blues)
|
|
|
|
|
band.genres.add(blues)
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = BandAdmin(Band, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/band/", data={"genres": blues.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
cl.get_results(request)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
|
|
|
|
# There's only one Group instance
|
|
|
|
|
self.assertEqual(cl.result_count, 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_through_m2m_in_list_filter(self):
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
Regression test for #13902: When using a ManyToMany in list_filter,
|
2014-03-02 22:25:53 +08:00
|
|
|
|
results shouldn't appear more than once. With an intermediate model.
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
lead = Musician.objects.create(name="Vox")
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="lead voice")
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="bass player")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = GroupAdmin(Group, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/group/", data={"members": lead.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
cl.get_results(request)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
|
|
|
|
# There's only one Group instance
|
|
|
|
|
self.assertEqual(cl.result_count, 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_through_m2m_at_second_level_in_list_filter(self):
|
2015-04-14 17:09:58 +08:00
|
|
|
|
"""
|
|
|
|
|
When using a ManyToMany in list_filter at the second level behind a
|
2023-07-05 05:45:54 +08:00
|
|
|
|
ForeignKey, distinct() must be called and results shouldn't appear more
|
|
|
|
|
than once.
|
2015-04-14 17:09:58 +08:00
|
|
|
|
"""
|
|
|
|
|
lead = Musician.objects.create(name="Vox")
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
Concert.objects.create(name="Woodstock", group=band)
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="lead voice")
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="bass player")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
request = self.factory.get("/concert/", data={"group__members": lead.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-04-14 17:09:58 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
cl.get_results(request)
|
|
|
|
|
|
|
|
|
|
# There's only one Concert instance
|
|
|
|
|
self.assertEqual(cl.result_count, 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_inherited_m2m_in_list_filter(self):
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
Regression test for #13902: When using a ManyToMany in list_filter,
|
2014-03-02 22:25:53 +08:00
|
|
|
|
results shouldn't appear more than once. Model managed in the
|
2018-08-02 00:55:53 +08:00
|
|
|
|
admin inherits from the one that defines the relationship.
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
lead = Musician.objects.create(name="John")
|
|
|
|
|
four = Quartet.objects.create(name="The Beatles")
|
|
|
|
|
Membership.objects.create(group=four, music=lead, role="lead voice")
|
|
|
|
|
Membership.objects.create(group=four, music=lead, role="guitar player")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = QuartetAdmin(Quartet, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/quartet/", data={"members": lead.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
cl.get_results(request)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
|
|
|
|
# There's only one Quartet instance
|
|
|
|
|
self.assertEqual(cl.result_count, 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_m2m_to_inherited_in_list_filter(self):
|
2011-02-14 06:51:40 +08:00
|
|
|
|
"""
|
|
|
|
|
Regression test for #13902: When using a ManyToMany in list_filter,
|
2014-03-02 22:25:53 +08:00
|
|
|
|
results shouldn't appear more than once. Target of the relationship
|
2011-02-14 06:51:40 +08:00
|
|
|
|
inherits from another.
|
|
|
|
|
"""
|
|
|
|
|
lead = ChordsMusician.objects.create(name="Player A")
|
|
|
|
|
three = ChordsBand.objects.create(name="The Chords Trio")
|
|
|
|
|
Invitation.objects.create(band=three, player=lead, instrument="guitar")
|
|
|
|
|
Invitation.objects.create(band=three, player=lead, instrument="bass")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChordsBandAdmin(ChordsBand, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/chordsband/", data={"members": lead.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
cl.get_results(request)
|
2011-02-14 06:51:40 +08:00
|
|
|
|
|
|
|
|
|
# There's only one ChordsBand instance
|
|
|
|
|
self.assertEqual(cl.result_count, 1)
|
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_non_unique_related_object_in_list_filter(self):
|
2011-04-23 11:55:21 +08:00
|
|
|
|
"""
|
2023-07-05 05:45:54 +08:00
|
|
|
|
Regressions tests for #15819: If a field listed in list_filters
|
|
|
|
|
is a non-unique related object, distinct() must be called.
|
2011-04-23 11:55:21 +08:00
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="Mary")
|
|
|
|
|
# Two children with the same name
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel")
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ParentAdmin(Parent, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/parent/", data={"child__name": "Daniel"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2023-07-05 05:45:54 +08:00
|
|
|
|
# Make sure distinct() was called
|
2013-03-08 22:15:23 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2019-08-22 20:09:49 +08:00
|
|
|
|
def test_changelist_search_form_validation(self):
|
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
tests = [
|
|
|
|
|
({SEARCH_VAR: "\x00"}, "Null characters are not allowed."),
|
|
|
|
|
({SEARCH_VAR: "some\x00thing"}, "Null characters are not allowed."),
|
|
|
|
|
]
|
|
|
|
|
for case, error in tests:
|
|
|
|
|
with self.subTest(case=case):
|
|
|
|
|
request = self.factory.get("/concert/", case)
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
request._messages = CookieStorage(request)
|
|
|
|
|
m.get_changelist_instance(request)
|
|
|
|
|
messages = [m.message for m in request._messages]
|
|
|
|
|
self.assertEqual(1, len(messages))
|
|
|
|
|
self.assertEqual(error, messages[0])
|
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_non_unique_related_object_in_search_fields(self):
|
2011-04-23 11:55:21 +08:00
|
|
|
|
"""
|
|
|
|
|
Regressions tests for #15819: If a field listed in search_fields
|
2023-07-05 05:45:54 +08:00
|
|
|
|
is a non-unique related object, distinct() must be called.
|
2011-04-23 11:55:21 +08:00
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="Mary")
|
|
|
|
|
Child.objects.create(parent=parent, name="Danielle")
|
|
|
|
|
Child.objects.create(parent=parent, name="Daniel")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ParentAdmin(Parent, custom_site)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/parent/", data={SEARCH_VAR: "daniel"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2023-07-05 05:45:54 +08:00
|
|
|
|
# Make sure distinct() was called
|
2013-03-08 22:15:23 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2011-04-23 11:55:21 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_distinct_for_many_to_many_at_second_level_in_search_fields(self):
|
2015-04-14 17:09:58 +08:00
|
|
|
|
"""
|
|
|
|
|
When using a ManyToMany in search_fields at the second level behind a
|
2023-07-05 05:45:54 +08:00
|
|
|
|
ForeignKey, distinct() must be called and results shouldn't appear more
|
2015-04-14 17:09:58 +08:00
|
|
|
|
than once.
|
|
|
|
|
"""
|
|
|
|
|
lead = Musician.objects.create(name="Vox")
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
Concert.objects.create(name="Woodstock", group=band)
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="lead voice")
|
|
|
|
|
Membership.objects.create(group=band, music=lead, role="bass player")
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
request = self.factory.get("/concert/", data={SEARCH_VAR: "vox"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2015-04-14 17:09:58 +08:00
|
|
|
|
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
# There's only one Concert instance
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
2021-04-26 15:22:46 +08:00
|
|
|
|
# Queryset must be deletable.
|
|
|
|
|
cl.queryset.delete()
|
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
2015-04-14 17:09:58 +08:00
|
|
|
|
|
2021-10-28 03:30:03 +08:00
|
|
|
|
def test_multiple_search_fields(self):
|
|
|
|
|
"""
|
|
|
|
|
All rows containing each of the searched words are returned, where each
|
|
|
|
|
word must be in one of search_fields.
|
|
|
|
|
"""
|
|
|
|
|
band_duo = Group.objects.create(name="Duo")
|
|
|
|
|
band_hype = Group.objects.create(name="The Hype")
|
|
|
|
|
mary = Musician.objects.create(name="Mary Halvorson")
|
|
|
|
|
jonathan = Musician.objects.create(name="Jonathan Finlayson")
|
|
|
|
|
band_duo.members.set([mary, jonathan])
|
|
|
|
|
Concert.objects.create(name="Tiny desk concert", group=band_duo)
|
|
|
|
|
Concert.objects.create(name="Woodstock concert", group=band_hype)
|
|
|
|
|
# FK lookup.
|
|
|
|
|
concert_model_admin = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
concert_model_admin.search_fields = ["group__name", "name"]
|
|
|
|
|
# Reverse FK lookup.
|
|
|
|
|
group_model_admin = GroupAdmin(Group, custom_site)
|
|
|
|
|
group_model_admin.search_fields = ["name", "concert__name", "members__name"]
|
|
|
|
|
for search_string, result_count in (
|
|
|
|
|
("Duo Concert", 1),
|
|
|
|
|
("Tiny Desk Concert", 1),
|
|
|
|
|
("Concert", 2),
|
|
|
|
|
("Other Concert", 0),
|
|
|
|
|
("Duo Woodstock", 0),
|
|
|
|
|
):
|
|
|
|
|
with self.subTest(search_string=search_string):
|
|
|
|
|
# FK lookup.
|
|
|
|
|
request = self.factory.get(
|
|
|
|
|
"/concert/", data={SEARCH_VAR: search_string}
|
|
|
|
|
)
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
concert_changelist = concert_model_admin.get_changelist_instance(
|
|
|
|
|
request
|
|
|
|
|
)
|
|
|
|
|
self.assertEqual(concert_changelist.queryset.count(), result_count)
|
|
|
|
|
# Reverse FK lookup.
|
|
|
|
|
request = self.factory.get("/group/", data={SEARCH_VAR: search_string})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
group_changelist = group_model_admin.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(group_changelist.queryset.count(), result_count)
|
|
|
|
|
# Many-to-many lookup.
|
|
|
|
|
for search_string, result_count in (
|
|
|
|
|
("Finlayson Duo Tiny", 1),
|
|
|
|
|
("Finlayson", 1),
|
|
|
|
|
("Finlayson Hype", 0),
|
|
|
|
|
("Jonathan Finlayson Duo", 1),
|
2021-10-06 09:38:15 +08:00
|
|
|
|
("Mary Jonathan Duo", 0),
|
2021-10-28 03:30:03 +08:00
|
|
|
|
("Oscar Finlayson Duo", 0),
|
|
|
|
|
):
|
|
|
|
|
with self.subTest(search_string=search_string):
|
|
|
|
|
request = self.factory.get("/group/", data={SEARCH_VAR: search_string})
|
|
|
|
|
request.user = self.superuser
|
|
|
|
|
group_changelist = group_model_admin.get_changelist_instance(request)
|
|
|
|
|
self.assertEqual(group_changelist.queryset.count(), result_count)
|
|
|
|
|
|
2017-03-16 01:45:18 +08:00
|
|
|
|
def test_pk_in_search_fields(self):
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
Concert.objects.create(name="Woodstock", group=band)
|
|
|
|
|
|
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
m.search_fields = ["group__pk"]
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/concert/", data={SEARCH_VAR: band.pk})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2017-03-16 01:45:18 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 1)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/concert/", data={SEARCH_VAR: band.pk + 5})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2017-03-16 01:45:18 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 0)
|
|
|
|
|
|
2017-07-05 19:00:10 +08:00
|
|
|
|
def test_builtin_lookup_in_search_fields(self):
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
concert = Concert.objects.create(name="Woodstock", group=band)
|
|
|
|
|
|
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
m.search_fields = ["name__iexact"]
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "woodstock"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [concert])
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "wood"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [])
|
|
|
|
|
|
|
|
|
|
def test_custom_lookup_in_search_fields(self):
|
|
|
|
|
band = Group.objects.create(name="The Hype")
|
|
|
|
|
concert = Concert.objects.create(name="Woodstock", group=band)
|
|
|
|
|
|
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
m.search_fields = ["group__name__cc"]
|
2018-08-22 00:17:46 +08:00
|
|
|
|
with register_lookup(Field, Contains, lookup_name="cc"):
|
2017-07-05 19:00:10 +08:00
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "Hype"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [concert])
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "Woodstock"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [])
|
|
|
|
|
|
|
|
|
|
def test_spanning_relations_with_custom_lookup_in_search_fields(self):
|
|
|
|
|
hype = Group.objects.create(name="The Hype")
|
|
|
|
|
concert = Concert.objects.create(name="Woodstock", group=hype)
|
|
|
|
|
vox = Musician.objects.create(name="Vox", age=20)
|
|
|
|
|
Membership.objects.create(music=vox, group=hype)
|
|
|
|
|
# Register a custom lookup on IntegerField to ensure that field
|
|
|
|
|
# traversing logic in ModelAdmin.get_search_results() works.
|
2018-08-22 00:17:46 +08:00
|
|
|
|
with register_lookup(IntegerField, Exact, lookup_name="exactly"):
|
2017-07-05 19:00:10 +08:00
|
|
|
|
m = ConcertAdmin(Concert, custom_site)
|
|
|
|
|
m.search_fields = ["group__members__age__exactly"]
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "20"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [concert])
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "21"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [])
|
|
|
|
|
|
|
|
|
|
def test_custom_lookup_with_pk_shortcut(self):
|
|
|
|
|
self.assertEqual(CharPK._meta.pk.name, "char_pk") # Not equal to 'pk'.
|
|
|
|
|
m = admin.ModelAdmin(CustomIdUser, custom_site)
|
|
|
|
|
|
|
|
|
|
abc = CharPK.objects.create(char_pk="abc")
|
|
|
|
|
abcd = CharPK.objects.create(char_pk="abcd")
|
|
|
|
|
m = admin.ModelAdmin(CharPK, custom_site)
|
|
|
|
|
m.search_fields = ["pk__exact"]
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "abc"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [abc])
|
|
|
|
|
|
|
|
|
|
request = self.factory.get("/", data={SEARCH_VAR: "abcd"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-07-05 19:00:10 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
|
|
|
|
self.assertCountEqual(cl.queryset, [abcd])
|
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
def test_no_distinct_for_m2m_in_list_filter_without_params(self):
|
2016-10-23 19:55:53 +08:00
|
|
|
|
"""
|
|
|
|
|
If a ManyToManyField is in list_filter but isn't in any lookup params,
|
2023-07-05 05:45:54 +08:00
|
|
|
|
the changelist's query shouldn't have distinct.
|
2016-10-23 19:55:53 +08:00
|
|
|
|
"""
|
|
|
|
|
m = BandAdmin(Band, custom_site)
|
|
|
|
|
for lookup_params in ({}, {"name": "test"}):
|
|
|
|
|
request = self.factory.get("/band/", lookup_params)
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2023-07-05 05:45:54 +08:00
|
|
|
|
self.assertIs(cl.queryset.query.distinct, False)
|
2016-10-23 19:55:53 +08:00
|
|
|
|
|
2023-07-05 05:45:54 +08:00
|
|
|
|
# A ManyToManyField in params does have distinct applied.
|
2016-10-23 19:55:53 +08:00
|
|
|
|
request = self.factory.get("/band/", {"genres": "0"})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2023-07-05 05:45:54 +08:00
|
|
|
|
self.assertIs(cl.queryset.query.distinct, True)
|
2016-10-23 19:55:53 +08:00
|
|
|
|
|
2011-02-19 16:37:46 +08:00
|
|
|
|
def test_pagination(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression tests for #12893: Pagination in admins changelist doesn't
|
|
|
|
|
use queryset set by modeladmin.
|
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="anything")
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for i in range(1, 31):
|
2011-02-19 16:37:46 +08:00
|
|
|
|
Child.objects.create(name="name %s" % i, parent=parent)
|
|
|
|
|
Child.objects.create(name="filtered %s" % i, parent=parent)
|
|
|
|
|
|
2011-06-09 06:53:55 +08:00
|
|
|
|
request = self.factory.get("/child/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-02-19 16:37:46 +08:00
|
|
|
|
|
|
|
|
|
# Test default queryset
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2013-03-08 22:15:23 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 60)
|
2011-02-19 16:37:46 +08:00
|
|
|
|
self.assertEqual(cl.paginator.count, 60)
|
2012-08-16 11:09:01 +08:00
|
|
|
|
self.assertEqual(list(cl.paginator.page_range), [1, 2, 3, 4, 5, 6])
|
2011-02-19 16:37:46 +08:00
|
|
|
|
|
|
|
|
|
# Test custom queryset
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = FilteredChildAdmin(Child, custom_site)
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2013-03-08 22:15:23 +08:00
|
|
|
|
self.assertEqual(cl.queryset.count(), 30)
|
2011-02-19 16:37:46 +08:00
|
|
|
|
self.assertEqual(cl.paginator.count, 30)
|
2012-08-16 11:09:01 +08:00
|
|
|
|
self.assertEqual(list(cl.paginator.page_range), [1, 2, 3])
|
2011-02-19 16:37:46 +08:00
|
|
|
|
|
2012-04-29 17:51:12 +08:00
|
|
|
|
def test_computed_list_display_localization(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression test for #13196: output of functions should be localized
|
|
|
|
|
in the changelist.
|
|
|
|
|
"""
|
2018-05-02 16:39:12 +08:00
|
|
|
|
self.client.force_login(self.superuser)
|
2012-04-29 17:51:12 +08:00
|
|
|
|
event = Event.objects.create(date=datetime.date.today())
|
2015-02-07 06:25:15 +08:00
|
|
|
|
response = self.client.get(reverse("admin:admin_changelist_event_changelist"))
|
2012-04-29 17:51:12 +08:00
|
|
|
|
self.assertContains(response, formats.localize(event.date))
|
2016-12-29 23:27:49 +08:00
|
|
|
|
self.assertNotContains(response, str(event.date))
|
2012-04-29 17:51:12 +08:00
|
|
|
|
|
2011-06-09 06:53:55 +08:00
|
|
|
|
def test_dynamic_list_display(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression tests for #14206: dynamic list_display support.
|
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="parent")
|
|
|
|
|
for i in range(10):
|
|
|
|
|
Child.objects.create(name="child %s" % i, parent=parent)
|
|
|
|
|
|
2011-11-18 05:30:07 +08:00
|
|
|
|
user_noparents = self._create_superuser("noparents")
|
|
|
|
|
user_parents = self._create_superuser("parents")
|
2011-06-09 06:53:55 +08:00
|
|
|
|
|
|
|
|
|
# Test with user 'noparents'
|
2023-07-07 14:06:01 +08:00
|
|
|
|
m = custom_site.get_model_admin(Child)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", user_noparents)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertNotContains(response, "Parent object")
|
|
|
|
|
|
2011-10-26 20:19:18 +08:00
|
|
|
|
list_display = m.get_list_display(request)
|
|
|
|
|
list_display_links = m.get_list_display_links(request, list_display)
|
|
|
|
|
self.assertEqual(list_display, ["name", "age"])
|
|
|
|
|
self.assertEqual(list_display_links, ["name"])
|
|
|
|
|
|
2011-06-09 06:53:55 +08:00
|
|
|
|
# Test with user 'parents'
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = DynamicListDisplayChildAdmin(Child, custom_site)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", user_parents)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertContains(response, "Parent object")
|
|
|
|
|
|
2011-09-21 02:16:49 +08:00
|
|
|
|
custom_site.unregister(Child)
|
|
|
|
|
|
2011-10-26 20:19:18 +08:00
|
|
|
|
list_display = m.get_list_display(request)
|
|
|
|
|
list_display_links = m.get_list_display_links(request, list_display)
|
|
|
|
|
self.assertEqual(list_display, ("parent", "name", "age"))
|
|
|
|
|
self.assertEqual(list_display_links, ["parent"])
|
|
|
|
|
|
2011-06-09 06:53:55 +08:00
|
|
|
|
# Test default implementation
|
2011-09-21 02:16:49 +08:00
|
|
|
|
custom_site.register(Child, ChildAdmin)
|
2023-07-07 14:06:01 +08:00
|
|
|
|
m = custom_site.get_model_admin(Child)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", user_noparents)
|
2011-06-09 06:53:55 +08:00
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertContains(response, "Parent object")
|
|
|
|
|
|
2011-09-08 21:25:00 +08:00
|
|
|
|
def test_show_all(self):
|
|
|
|
|
parent = Parent.objects.create(name="anything")
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for i in range(1, 31):
|
2011-09-08 21:25:00 +08:00
|
|
|
|
Child.objects.create(name="name %s" % i, parent=parent)
|
|
|
|
|
Child.objects.create(name="filtered %s" % i, parent=parent)
|
|
|
|
|
|
|
|
|
|
# Add "show all" parameter to request
|
|
|
|
|
request = self.factory.get("/child/", data={ALL_VAR: ""})
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2011-09-08 21:25:00 +08:00
|
|
|
|
|
|
|
|
|
# Test valid "show all" request (number of total objects is under max)
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2015-09-25 21:33:03 +08:00
|
|
|
|
m.list_max_show_all = 200
|
2011-09-08 21:25:00 +08:00
|
|
|
|
# 200 is the max we'll pass to ChangeList
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-09-08 21:25:00 +08:00
|
|
|
|
cl.get_results(request)
|
|
|
|
|
self.assertEqual(len(cl.result_list), 60)
|
|
|
|
|
|
|
|
|
|
# Test invalid "show all" request (number of total objects over max)
|
|
|
|
|
# falls back to paginated pages
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = ChildAdmin(Child, custom_site)
|
2015-09-25 21:33:03 +08:00
|
|
|
|
m.list_max_show_all = 30
|
2011-09-08 21:25:00 +08:00
|
|
|
|
# 30 is the max we'll pass to ChangeList for this test
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2011-09-08 21:25:00 +08:00
|
|
|
|
cl.get_results(request)
|
|
|
|
|
self.assertEqual(len(cl.result_list), 10)
|
|
|
|
|
|
2011-10-26 20:19:18 +08:00
|
|
|
|
def test_dynamic_list_display_links(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression tests for #16257: dynamic list_display_links support.
|
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="parent")
|
|
|
|
|
for i in range(1, 10):
|
|
|
|
|
Child.objects.create(id=i, name="child %s" % i, parent=parent, age=i)
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = DynamicListDisplayLinksChildAdmin(Child, custom_site)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
request = self._mocked_authenticated_request("/child/", superuser)
|
2011-10-26 20:19:18 +08:00
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
for i in range(1, 10):
|
2012-09-25 09:02:59 +08:00
|
|
|
|
link = reverse("admin:admin_changelist_child_change", args=(i,))
|
|
|
|
|
self.assertContains(response, '<a href="%s">%s</a>' % (link, i))
|
2011-10-26 20:19:18 +08:00
|
|
|
|
|
|
|
|
|
list_display = m.get_list_display(request)
|
|
|
|
|
list_display_links = m.get_list_display_links(request, list_display)
|
|
|
|
|
self.assertEqual(list_display, ("parent", "name", "age"))
|
2011-11-18 05:30:07 +08:00
|
|
|
|
self.assertEqual(list_display_links, ["age"])
|
|
|
|
|
|
2013-09-07 02:25:13 +08:00
|
|
|
|
def test_no_list_display_links(self):
|
|
|
|
|
"""#15185 -- Allow no links from the 'change list' view grid."""
|
|
|
|
|
p = Parent.objects.create(name="parent")
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = NoListDisplayLinksParentAdmin(Parent, custom_site)
|
2013-09-07 02:25:13 +08:00
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
request = self._mocked_authenticated_request("/parent/", superuser)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
link = reverse("admin:admin_changelist_parent_change", args=(p.pk,))
|
|
|
|
|
self.assertNotContains(response, '<a href="%s">' % link)
|
|
|
|
|
|
2020-01-21 22:14:11 +08:00
|
|
|
|
def test_clear_all_filters_link(self):
|
|
|
|
|
self.client.force_login(self.superuser)
|
2020-05-19 15:03:04 +08:00
|
|
|
|
url = reverse("admin:auth_user_changelist")
|
|
|
|
|
response = self.client.get(url)
|
|
|
|
|
self.assertNotContains(response, "✖ Clear all filters")
|
|
|
|
|
link = '<a href="%s">✖ Clear all filters</a>'
|
|
|
|
|
for data, href in (
|
|
|
|
|
({"is_staff__exact": "0"}, "?"),
|
|
|
|
|
(
|
|
|
|
|
{"is_staff__exact": "0", "username__startswith": "test"},
|
|
|
|
|
"?username__startswith=test",
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
{"is_staff__exact": "0", SEARCH_VAR: "test"},
|
|
|
|
|
"?%s=test" % SEARCH_VAR,
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
{"is_staff__exact": "0", IS_POPUP_VAR: "id"},
|
|
|
|
|
"?%s=id" % IS_POPUP_VAR,
|
|
|
|
|
),
|
|
|
|
|
):
|
|
|
|
|
with self.subTest(data=data):
|
|
|
|
|
response = self.client.get(url, data=data)
|
|
|
|
|
self.assertContains(response, link % href)
|
|
|
|
|
|
|
|
|
|
def test_clear_all_filters_link_callable_filter(self):
|
|
|
|
|
self.client.force_login(self.superuser)
|
|
|
|
|
url = reverse("admin:admin_changelist_band_changelist")
|
|
|
|
|
response = self.client.get(url)
|
|
|
|
|
self.assertNotContains(response, "✖ Clear all filters")
|
|
|
|
|
link = '<a href="%s">✖ Clear all filters</a>'
|
|
|
|
|
for data, href in (
|
|
|
|
|
({"nr_of_members_partition": "5"}, "?"),
|
|
|
|
|
(
|
|
|
|
|
{"nr_of_members_partition": "more", "name__startswith": "test"},
|
|
|
|
|
"?name__startswith=test",
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
{"nr_of_members_partition": "5", IS_POPUP_VAR: "id"},
|
|
|
|
|
"?%s=id" % IS_POPUP_VAR,
|
|
|
|
|
),
|
|
|
|
|
):
|
|
|
|
|
with self.subTest(data=data):
|
|
|
|
|
response = self.client.get(url, data=data)
|
|
|
|
|
self.assertContains(response, link % href)
|
|
|
|
|
|
|
|
|
|
def test_no_clear_all_filters_link(self):
|
|
|
|
|
self.client.force_login(self.superuser)
|
|
|
|
|
url = reverse("admin:auth_user_changelist")
|
|
|
|
|
link = ">✖ Clear all filters</a>"
|
2020-01-21 22:14:11 +08:00
|
|
|
|
for data in (
|
|
|
|
|
{SEARCH_VAR: "test"},
|
2020-05-19 15:03:04 +08:00
|
|
|
|
{ORDER_VAR: "-1"},
|
|
|
|
|
{TO_FIELD_VAR: "id"},
|
|
|
|
|
{PAGE_VAR: "1"},
|
|
|
|
|
{IS_POPUP_VAR: "1"},
|
2023-02-16 20:23:24 +08:00
|
|
|
|
{IS_FACETS_VAR: ""},
|
2020-05-19 15:03:04 +08:00
|
|
|
|
{"username__startswith": "test"},
|
2020-01-21 22:14:11 +08:00
|
|
|
|
):
|
2020-05-19 15:03:04 +08:00
|
|
|
|
with self.subTest(data=data):
|
|
|
|
|
response = self.client.get(url, data=data)
|
|
|
|
|
self.assertNotContains(response, link)
|
2020-01-21 22:14:11 +08:00
|
|
|
|
|
2011-11-18 05:30:07 +08:00
|
|
|
|
def test_tuple_list_display(self):
|
2015-05-27 23:43:22 +08:00
|
|
|
|
swallow = Swallow.objects.create(origin="Africa", load="12.34", speed="22.2")
|
|
|
|
|
swallow2 = Swallow.objects.create(origin="Africa", load="12.34", speed="22.2")
|
|
|
|
|
swallow_o2o = SwallowOneToOne.objects.create(swallow=swallow2)
|
|
|
|
|
|
2015-05-14 20:14:34 +08:00
|
|
|
|
model_admin = SwallowAdmin(Swallow, custom_site)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
request = self._mocked_authenticated_request("/swallow/", superuser)
|
|
|
|
|
response = model_admin.changelist_view(request)
|
|
|
|
|
# just want to ensure it doesn't blow up during rendering
|
2016-12-29 23:27:49 +08:00
|
|
|
|
self.assertContains(response, str(swallow.origin))
|
|
|
|
|
self.assertContains(response, str(swallow.load))
|
|
|
|
|
self.assertContains(response, str(swallow.speed))
|
2015-05-27 23:43:22 +08:00
|
|
|
|
# Reverse one-to-one relations should work.
|
|
|
|
|
self.assertContains(response, '<td class="field-swallowonetoone">-</td>')
|
|
|
|
|
self.assertContains(
|
|
|
|
|
response, '<td class="field-swallowonetoone">%s</td>' % swallow_o2o
|
|
|
|
|
)
|
2011-11-18 05:30:07 +08:00
|
|
|
|
|
2015-11-20 00:13:39 +08:00
|
|
|
|
def test_multiuser_edit(self):
|
|
|
|
|
"""
|
|
|
|
|
Simultaneous edits of list_editable fields on the changelist by
|
|
|
|
|
different users must not result in one user's edits creating a new
|
|
|
|
|
object instead of modifying the correct existing object (#11313).
|
|
|
|
|
"""
|
|
|
|
|
# To replicate this issue, simulate the following steps:
|
|
|
|
|
# 1. User1 opens an admin changelist with list_editable fields.
|
|
|
|
|
# 2. User2 edits object "Foo" such that it moves to another page in
|
|
|
|
|
# the pagination order and saves.
|
|
|
|
|
# 3. User1 edits object "Foo" and saves.
|
|
|
|
|
# 4. The edit made by User1 does not get applied to object "Foo" but
|
|
|
|
|
# instead is used to create a new object (bug).
|
|
|
|
|
|
|
|
|
|
# For this test, order the changelist by the 'speed' attribute and
|
|
|
|
|
# display 3 objects per page (SwallowAdmin.list_per_page = 3).
|
|
|
|
|
|
|
|
|
|
# Setup the test to reflect the DB state after step 2 where User2 has
|
|
|
|
|
# edited the first swallow object's speed from '4' to '1'.
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
b = Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
c = Swallow.objects.create(origin="Swallow C", load=5, speed=5)
|
|
|
|
|
d = Swallow.objects.create(origin="Swallow D", load=9, speed=9)
|
|
|
|
|
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
self.client.force_login(superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
|
|
|
|
|
# Send the POST from User1 for step 3. It's still using the changelist
|
|
|
|
|
# ordering from before User2's edits in step 2.
|
|
|
|
|
data = {
|
|
|
|
|
"form-TOTAL_FORMS": "3",
|
|
|
|
|
"form-INITIAL_FORMS": "3",
|
|
|
|
|
"form-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form-MAX_NUM_FORMS": "1000",
|
2018-05-03 21:41:04 +08:00
|
|
|
|
"form-0-uuid": str(d.pk),
|
|
|
|
|
"form-1-uuid": str(c.pk),
|
|
|
|
|
"form-2-uuid": str(a.pk),
|
2015-11-20 00:13:39 +08:00
|
|
|
|
"form-0-load": "9.0",
|
|
|
|
|
"form-0-speed": "9.0",
|
|
|
|
|
"form-1-load": "5.0",
|
|
|
|
|
"form-1-speed": "5.0",
|
|
|
|
|
"form-2-load": "5.0",
|
|
|
|
|
"form-2-speed": "4.0",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
response = self.client.post(
|
|
|
|
|
changelist_url, data, follow=True, extra={"o": "-2"}
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# The object User1 edited in step 3 is displayed on the changelist and
|
|
|
|
|
# has the correct edits applied.
|
|
|
|
|
self.assertContains(response, "1 swallow was changed successfully.")
|
|
|
|
|
self.assertContains(response, a.origin)
|
|
|
|
|
a.refresh_from_db()
|
|
|
|
|
self.assertEqual(a.load, float(data["form-2-load"]))
|
|
|
|
|
self.assertEqual(a.speed, float(data["form-2-speed"]))
|
|
|
|
|
b.refresh_from_db()
|
|
|
|
|
self.assertEqual(b.load, 2)
|
|
|
|
|
self.assertEqual(b.speed, 2)
|
|
|
|
|
c.refresh_from_db()
|
|
|
|
|
self.assertEqual(c.load, float(data["form-1-load"]))
|
|
|
|
|
self.assertEqual(c.speed, float(data["form-1-speed"]))
|
|
|
|
|
d.refresh_from_db()
|
|
|
|
|
self.assertEqual(d.load, float(data["form-0-load"]))
|
|
|
|
|
self.assertEqual(d.speed, float(data["form-0-speed"]))
|
|
|
|
|
# No new swallows were created.
|
|
|
|
|
self.assertEqual(len(Swallow.objects.all()), 4)
|
|
|
|
|
|
2018-05-03 21:41:04 +08:00
|
|
|
|
def test_get_edited_object_ids(self):
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
b = Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
c = Swallow.objects.create(origin="Swallow C", load=5, speed=5)
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
self.client.force_login(superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
m = SwallowAdmin(Swallow, custom_site)
|
|
|
|
|
data = {
|
|
|
|
|
"form-TOTAL_FORMS": "3",
|
|
|
|
|
"form-INITIAL_FORMS": "3",
|
|
|
|
|
"form-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form-MAX_NUM_FORMS": "1000",
|
|
|
|
|
"form-0-uuid": str(a.pk),
|
|
|
|
|
"form-1-uuid": str(b.pk),
|
|
|
|
|
"form-2-uuid": str(c.pk),
|
|
|
|
|
"form-0-load": "9.0",
|
|
|
|
|
"form-0-speed": "9.0",
|
|
|
|
|
"form-1-load": "5.0",
|
|
|
|
|
"form-1-speed": "5.0",
|
|
|
|
|
"form-2-load": "5.0",
|
|
|
|
|
"form-2-speed": "4.0",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
request = self.factory.post(changelist_url, data=data)
|
|
|
|
|
pks = m._get_edited_object_pks(request, prefix="form")
|
|
|
|
|
self.assertEqual(sorted(pks), sorted([str(a.pk), str(b.pk), str(c.pk)]))
|
|
|
|
|
|
|
|
|
|
def test_get_list_editable_queryset(self):
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
data = {
|
|
|
|
|
"form-TOTAL_FORMS": "2",
|
|
|
|
|
"form-INITIAL_FORMS": "2",
|
|
|
|
|
"form-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form-MAX_NUM_FORMS": "1000",
|
|
|
|
|
"form-0-uuid": str(a.pk),
|
|
|
|
|
"form-0-load": "10",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
self.client.force_login(superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
m = SwallowAdmin(Swallow, custom_site)
|
|
|
|
|
request = self.factory.post(changelist_url, data=data)
|
|
|
|
|
queryset = m._get_list_editable_queryset(request, prefix="form")
|
|
|
|
|
self.assertEqual(queryset.count(), 1)
|
|
|
|
|
data["form-0-uuid"] = "INVALD_PRIMARY_KEY"
|
|
|
|
|
# The unfiltered queryset is returned if there's invalid data.
|
|
|
|
|
request = self.factory.post(changelist_url, data=data)
|
|
|
|
|
queryset = m._get_list_editable_queryset(request, prefix="form")
|
|
|
|
|
self.assertEqual(queryset.count(), 2)
|
|
|
|
|
|
2019-11-25 06:09:07 +08:00
|
|
|
|
def test_get_list_editable_queryset_with_regex_chars_in_prefix(self):
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
data = {
|
|
|
|
|
"form$-TOTAL_FORMS": "2",
|
|
|
|
|
"form$-INITIAL_FORMS": "2",
|
|
|
|
|
"form$-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form$-MAX_NUM_FORMS": "1000",
|
|
|
|
|
"form$-0-uuid": str(a.pk),
|
|
|
|
|
"form$-0-load": "10",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
self.client.force_login(superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
m = SwallowAdmin(Swallow, custom_site)
|
|
|
|
|
request = self.factory.post(changelist_url, data=data)
|
|
|
|
|
queryset = m._get_list_editable_queryset(request, prefix="form$")
|
|
|
|
|
self.assertEqual(queryset.count(), 1)
|
|
|
|
|
|
2018-05-03 21:41:04 +08:00
|
|
|
|
def test_changelist_view_list_editable_changed_objects_uses_filter(self):
|
|
|
|
|
"""list_editable edits use a filtered queryset to limit memory usage."""
|
|
|
|
|
a = Swallow.objects.create(origin="Swallow A", load=4, speed=1)
|
|
|
|
|
Swallow.objects.create(origin="Swallow B", load=2, speed=2)
|
|
|
|
|
data = {
|
|
|
|
|
"form-TOTAL_FORMS": "2",
|
|
|
|
|
"form-INITIAL_FORMS": "2",
|
|
|
|
|
"form-MIN_NUM_FORMS": "0",
|
|
|
|
|
"form-MAX_NUM_FORMS": "1000",
|
|
|
|
|
"form-0-uuid": str(a.pk),
|
|
|
|
|
"form-0-load": "10",
|
|
|
|
|
"_save": "Save",
|
|
|
|
|
}
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
self.client.force_login(superuser)
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_swallow_changelist")
|
|
|
|
|
with CaptureQueriesContext(connection) as context:
|
|
|
|
|
response = self.client.post(changelist_url, data=data)
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
self.assertIn("WHERE", context.captured_queries[4]["sql"])
|
|
|
|
|
self.assertIn("IN", context.captured_queries[4]["sql"])
|
|
|
|
|
# Check only the first few characters since the UUID may have dashes.
|
|
|
|
|
self.assertIn(str(a.pk)[:8], context.captured_queries[4]["sql"])
|
|
|
|
|
|
2012-03-03 10:13:35 +08:00
|
|
|
|
def test_deterministic_order_for_unordered_model(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
The primary key is used in the ordering of the changelist's results to
|
|
|
|
|
guarantee a deterministic order, even when the model doesn't have any
|
|
|
|
|
default ordering defined (#17198).
|
2012-03-03 10:13:35 +08:00
|
|
|
|
"""
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
|
|
|
|
|
for counter in range(1, 51):
|
|
|
|
|
UnorderedObject.objects.create(id=counter, bool=True)
|
|
|
|
|
|
|
|
|
|
class UnorderedObjectAdmin(admin.ModelAdmin):
|
|
|
|
|
list_per_page = 10
|
|
|
|
|
|
2012-04-09 12:28:32 +08:00
|
|
|
|
def check_results_order(ascending=False):
|
2015-05-14 20:14:34 +08:00
|
|
|
|
custom_site.register(UnorderedObject, UnorderedObjectAdmin)
|
|
|
|
|
model_admin = UnorderedObjectAdmin(UnorderedObject, custom_site)
|
2012-04-09 12:28:32 +08:00
|
|
|
|
counter = 0 if ascending else 51
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for page in range(1, 6):
|
2012-03-03 10:13:35 +08:00
|
|
|
|
request = self._mocked_authenticated_request(
|
|
|
|
|
"/unorderedobject/?p=%s" % page, superuser
|
|
|
|
|
)
|
|
|
|
|
response = model_admin.changelist_view(request)
|
|
|
|
|
for result in response.context_data["cl"].result_list:
|
2012-04-09 12:28:32 +08:00
|
|
|
|
counter += 1 if ascending else -1
|
2012-03-03 10:13:35 +08:00
|
|
|
|
self.assertEqual(result.id, counter)
|
2015-05-14 20:14:34 +08:00
|
|
|
|
custom_site.unregister(UnorderedObject)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
|
2012-04-09 12:28:32 +08:00
|
|
|
|
# When no order is defined at all, everything is ordered by '-pk'.
|
2012-03-03 10:13:35 +08:00
|
|
|
|
check_results_order()
|
|
|
|
|
|
|
|
|
|
# When an order field is defined but multiple records have the same
|
2012-04-09 12:28:32 +08:00
|
|
|
|
# value for that field, make sure everything gets ordered by -pk as well.
|
2012-03-03 10:13:35 +08:00
|
|
|
|
UnorderedObjectAdmin.ordering = ["bool"]
|
|
|
|
|
check_results_order()
|
|
|
|
|
|
|
|
|
|
# When order fields are defined, including the pk itself, use them.
|
|
|
|
|
UnorderedObjectAdmin.ordering = ["bool", "-pk"]
|
|
|
|
|
check_results_order()
|
2012-04-09 12:28:32 +08:00
|
|
|
|
UnorderedObjectAdmin.ordering = ["bool", "pk"]
|
|
|
|
|
check_results_order(ascending=True)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
UnorderedObjectAdmin.ordering = ["-id", "bool"]
|
|
|
|
|
check_results_order()
|
2012-04-09 12:28:32 +08:00
|
|
|
|
UnorderedObjectAdmin.ordering = ["id", "bool"]
|
|
|
|
|
check_results_order(ascending=True)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
|
|
|
|
|
def test_deterministic_order_for_model_ordered_by_its_manager(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
The primary key is used in the ordering of the changelist's results to
|
|
|
|
|
guarantee a deterministic order, even when the model has a manager that
|
|
|
|
|
defines a default ordering (#17198).
|
2012-03-03 10:13:35 +08:00
|
|
|
|
"""
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
|
|
|
|
|
for counter in range(1, 51):
|
|
|
|
|
OrderedObject.objects.create(id=counter, bool=True, number=counter)
|
|
|
|
|
|
|
|
|
|
class OrderedObjectAdmin(admin.ModelAdmin):
|
|
|
|
|
list_per_page = 10
|
|
|
|
|
|
2012-04-09 12:28:32 +08:00
|
|
|
|
def check_results_order(ascending=False):
|
2015-05-14 20:14:34 +08:00
|
|
|
|
custom_site.register(OrderedObject, OrderedObjectAdmin)
|
|
|
|
|
model_admin = OrderedObjectAdmin(OrderedObject, custom_site)
|
2012-04-09 12:28:32 +08:00
|
|
|
|
counter = 0 if ascending else 51
|
2020-07-30 19:45:32 +08:00
|
|
|
|
for page in range(1, 6):
|
2012-03-03 10:13:35 +08:00
|
|
|
|
request = self._mocked_authenticated_request(
|
|
|
|
|
"/orderedobject/?p=%s" % page, superuser
|
|
|
|
|
)
|
|
|
|
|
response = model_admin.changelist_view(request)
|
|
|
|
|
for result in response.context_data["cl"].result_list:
|
2012-04-09 12:28:32 +08:00
|
|
|
|
counter += 1 if ascending else -1
|
2012-03-03 10:13:35 +08:00
|
|
|
|
self.assertEqual(result.id, counter)
|
2015-05-14 20:14:34 +08:00
|
|
|
|
custom_site.unregister(OrderedObject)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
|
2012-04-09 12:28:32 +08:00
|
|
|
|
# When no order is defined at all, use the model's default ordering
|
|
|
|
|
# (i.e. 'number').
|
|
|
|
|
check_results_order(ascending=True)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
|
|
|
|
|
# When an order field is defined but multiple records have the same
|
2012-04-09 12:28:32 +08:00
|
|
|
|
# value for that field, make sure everything gets ordered by -pk as well.
|
2012-03-03 10:13:35 +08:00
|
|
|
|
OrderedObjectAdmin.ordering = ["bool"]
|
|
|
|
|
check_results_order()
|
|
|
|
|
|
|
|
|
|
# When order fields are defined, including the pk itself, use them.
|
|
|
|
|
OrderedObjectAdmin.ordering = ["bool", "-pk"]
|
|
|
|
|
check_results_order()
|
2012-04-09 12:28:32 +08:00
|
|
|
|
OrderedObjectAdmin.ordering = ["bool", "pk"]
|
|
|
|
|
check_results_order(ascending=True)
|
2012-03-03 10:13:35 +08:00
|
|
|
|
OrderedObjectAdmin.ordering = ["-id", "bool"]
|
2012-04-09 12:28:32 +08:00
|
|
|
|
check_results_order()
|
2012-03-03 10:13:35 +08:00
|
|
|
|
OrderedObjectAdmin.ordering = ["id", "bool"]
|
2012-04-29 17:51:12 +08:00
|
|
|
|
check_results_order(ascending=True)
|
2012-11-26 03:39:23 +08:00
|
|
|
|
|
2019-01-17 23:22:14 +08:00
|
|
|
|
@isolate_apps("admin_changelist")
|
|
|
|
|
def test_total_ordering_optimization(self):
|
|
|
|
|
class Related(models.Model):
|
|
|
|
|
unique_field = models.BooleanField(unique=True)
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
ordering = ("unique_field",)
|
|
|
|
|
|
|
|
|
|
class Model(models.Model):
|
|
|
|
|
unique_field = models.BooleanField(unique=True)
|
|
|
|
|
unique_nullable_field = models.BooleanField(unique=True, null=True)
|
|
|
|
|
related = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
other_related = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
related_unique = models.OneToOneField(Related, models.CASCADE)
|
|
|
|
|
field = models.BooleanField()
|
|
|
|
|
other_field = models.BooleanField()
|
|
|
|
|
null_field = models.BooleanField(null=True)
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
unique_together = {
|
|
|
|
|
("field", "other_field"),
|
|
|
|
|
("field", "null_field"),
|
|
|
|
|
("related", "other_related_id"),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class ModelAdmin(admin.ModelAdmin):
|
|
|
|
|
def get_queryset(self, request):
|
|
|
|
|
return Model.objects.none()
|
|
|
|
|
|
|
|
|
|
request = self._mocked_authenticated_request("/", self.superuser)
|
|
|
|
|
site = admin.AdminSite(name="admin")
|
|
|
|
|
model_admin = ModelAdmin(Model, site)
|
|
|
|
|
change_list = model_admin.get_changelist_instance(request)
|
|
|
|
|
tests = (
|
|
|
|
|
([], ["-pk"]),
|
|
|
|
|
# Unique non-nullable field.
|
|
|
|
|
(["unique_field"], ["unique_field"]),
|
|
|
|
|
(["-unique_field"], ["-unique_field"]),
|
|
|
|
|
# Unique nullable field.
|
|
|
|
|
(["unique_nullable_field"], ["unique_nullable_field", "-pk"]),
|
|
|
|
|
# Field.
|
|
|
|
|
(["field"], ["field", "-pk"]),
|
|
|
|
|
# Related field introspection is not implemented.
|
|
|
|
|
(["related__unique_field"], ["related__unique_field", "-pk"]),
|
|
|
|
|
# Related attname unique.
|
|
|
|
|
(["related_unique_id"], ["related_unique_id"]),
|
|
|
|
|
# Related ordering introspection is not implemented.
|
|
|
|
|
(["related_unique"], ["related_unique", "-pk"]),
|
|
|
|
|
# Composite unique.
|
|
|
|
|
(["field", "-other_field"], ["field", "-other_field"]),
|
|
|
|
|
# Composite unique nullable.
|
|
|
|
|
(["-field", "null_field"], ["-field", "null_field", "-pk"]),
|
|
|
|
|
# Composite unique and nullable.
|
|
|
|
|
(
|
|
|
|
|
["-field", "null_field", "other_field"],
|
|
|
|
|
["-field", "null_field", "other_field"],
|
|
|
|
|
),
|
|
|
|
|
# Composite unique attnames.
|
|
|
|
|
(["related_id", "-other_related_id"], ["related_id", "-other_related_id"]),
|
|
|
|
|
# Composite unique names.
|
|
|
|
|
(["related", "-other_related_id"], ["related", "-other_related_id", "-pk"]),
|
|
|
|
|
)
|
|
|
|
|
# F() objects composite unique.
|
|
|
|
|
total_ordering = [F("field"), F("other_field").desc(nulls_last=True)]
|
|
|
|
|
# F() objects composite unique nullable.
|
|
|
|
|
non_total_ordering = [F("field"), F("null_field").desc(nulls_last=True)]
|
|
|
|
|
tests += (
|
|
|
|
|
(total_ordering, total_ordering),
|
|
|
|
|
(non_total_ordering, non_total_ordering + ["-pk"]),
|
|
|
|
|
)
|
|
|
|
|
for ordering, expected in tests:
|
|
|
|
|
with self.subTest(ordering=ordering):
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
change_list._get_deterministic_ordering(ordering), expected
|
|
|
|
|
)
|
|
|
|
|
|
2020-01-21 03:50:30 +08:00
|
|
|
|
@isolate_apps("admin_changelist")
|
|
|
|
|
def test_total_ordering_optimization_meta_constraints(self):
|
|
|
|
|
class Related(models.Model):
|
|
|
|
|
unique_field = models.BooleanField(unique=True)
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
ordering = ("unique_field",)
|
|
|
|
|
|
|
|
|
|
class Model(models.Model):
|
|
|
|
|
field_1 = models.BooleanField()
|
|
|
|
|
field_2 = models.BooleanField()
|
|
|
|
|
field_3 = models.BooleanField()
|
|
|
|
|
field_4 = models.BooleanField()
|
|
|
|
|
field_5 = models.BooleanField()
|
|
|
|
|
field_6 = models.BooleanField()
|
|
|
|
|
nullable_1 = models.BooleanField(null=True)
|
|
|
|
|
nullable_2 = models.BooleanField(null=True)
|
|
|
|
|
related_1 = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
related_2 = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
related_3 = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
related_4 = models.ForeignKey(Related, models.CASCADE)
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
constraints = [
|
|
|
|
|
*[
|
|
|
|
|
models.UniqueConstraint(fields=fields, name="".join(fields))
|
|
|
|
|
for fields in (
|
|
|
|
|
["field_1"],
|
|
|
|
|
["nullable_1"],
|
|
|
|
|
["related_1"],
|
|
|
|
|
["related_2_id"],
|
|
|
|
|
["field_2", "field_3"],
|
|
|
|
|
["field_2", "nullable_2"],
|
|
|
|
|
["field_2", "related_3"],
|
|
|
|
|
["field_3", "related_4_id"],
|
|
|
|
|
)
|
|
|
|
|
],
|
|
|
|
|
models.CheckConstraint(check=models.Q(id__gt=0), name="foo"),
|
|
|
|
|
models.UniqueConstraint(
|
|
|
|
|
fields=["field_5"],
|
|
|
|
|
condition=models.Q(id__gt=10),
|
|
|
|
|
name="total_ordering_1",
|
|
|
|
|
),
|
|
|
|
|
models.UniqueConstraint(
|
|
|
|
|
fields=["field_6"],
|
|
|
|
|
condition=models.Q(),
|
|
|
|
|
name="total_ordering",
|
|
|
|
|
),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
class ModelAdmin(admin.ModelAdmin):
|
|
|
|
|
def get_queryset(self, request):
|
|
|
|
|
return Model.objects.none()
|
|
|
|
|
|
|
|
|
|
request = self._mocked_authenticated_request("/", self.superuser)
|
|
|
|
|
site = admin.AdminSite(name="admin")
|
|
|
|
|
model_admin = ModelAdmin(Model, site)
|
|
|
|
|
change_list = model_admin.get_changelist_instance(request)
|
|
|
|
|
tests = (
|
|
|
|
|
# Unique non-nullable field.
|
|
|
|
|
(["field_1"], ["field_1"]),
|
|
|
|
|
# Unique nullable field.
|
|
|
|
|
(["nullable_1"], ["nullable_1", "-pk"]),
|
|
|
|
|
# Related attname unique.
|
|
|
|
|
(["related_1_id"], ["related_1_id"]),
|
|
|
|
|
(["related_2_id"], ["related_2_id"]),
|
|
|
|
|
# Related ordering introspection is not implemented.
|
|
|
|
|
(["related_1"], ["related_1", "-pk"]),
|
|
|
|
|
# Composite unique.
|
|
|
|
|
(["-field_2", "field_3"], ["-field_2", "field_3"]),
|
|
|
|
|
# Composite unique nullable.
|
|
|
|
|
(["field_2", "-nullable_2"], ["field_2", "-nullable_2", "-pk"]),
|
|
|
|
|
# Composite unique and nullable.
|
|
|
|
|
(
|
|
|
|
|
["field_2", "-nullable_2", "field_3"],
|
|
|
|
|
["field_2", "-nullable_2", "field_3"],
|
|
|
|
|
),
|
|
|
|
|
# Composite field and related field name.
|
|
|
|
|
(["field_2", "-related_3"], ["field_2", "-related_3", "-pk"]),
|
|
|
|
|
(["field_3", "related_4"], ["field_3", "related_4", "-pk"]),
|
|
|
|
|
# Composite field and related field attname.
|
|
|
|
|
(["field_2", "related_3_id"], ["field_2", "related_3_id"]),
|
|
|
|
|
(["field_3", "-related_4_id"], ["field_3", "-related_4_id"]),
|
|
|
|
|
# Partial unique constraint is ignored.
|
|
|
|
|
(["field_5"], ["field_5", "-pk"]),
|
|
|
|
|
# Unique constraint with an empty condition.
|
|
|
|
|
(["field_6"], ["field_6"]),
|
|
|
|
|
)
|
|
|
|
|
for ordering, expected in tests:
|
|
|
|
|
with self.subTest(ordering=ordering):
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
change_list._get_deterministic_ordering(ordering), expected
|
|
|
|
|
)
|
|
|
|
|
|
2012-11-26 03:39:23 +08:00
|
|
|
|
def test_dynamic_list_filter(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression tests for ticket #17646: dynamic list_filter support.
|
|
|
|
|
"""
|
|
|
|
|
parent = Parent.objects.create(name="parent")
|
|
|
|
|
for i in range(10):
|
|
|
|
|
Child.objects.create(name="child %s" % i, parent=parent)
|
|
|
|
|
|
|
|
|
|
user_noparents = self._create_superuser("noparents")
|
|
|
|
|
user_parents = self._create_superuser("parents")
|
|
|
|
|
|
|
|
|
|
# Test with user 'noparents'
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = DynamicListFilterChildAdmin(Child, custom_site)
|
2012-11-26 03:39:23 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", user_noparents)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertEqual(response.context_data["cl"].list_filter, ["name", "age"])
|
|
|
|
|
|
|
|
|
|
# Test with user 'parents'
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = DynamicListFilterChildAdmin(Child, custom_site)
|
2012-11-26 03:39:23 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", user_parents)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
response.context_data["cl"].list_filter, ("parent", "name", "age")
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2013-03-20 06:17:31 +08:00
|
|
|
|
|
2013-08-04 00:15:15 +08:00
|
|
|
|
def test_dynamic_search_fields(self):
|
|
|
|
|
child = self._create_superuser("child")
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = DynamicSearchFieldsChildAdmin(Child, custom_site)
|
2013-08-04 00:15:15 +08:00
|
|
|
|
request = self._mocked_authenticated_request("/child/", child)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertEqual(response.context_data["cl"].search_fields, ("name", "age"))
|
|
|
|
|
|
2013-05-29 21:23:08 +08:00
|
|
|
|
def test_pagination_page_range(self):
|
|
|
|
|
"""
|
|
|
|
|
Regression tests for ticket #15653: ensure the number of pages
|
|
|
|
|
generated for changelist views are correct.
|
|
|
|
|
"""
|
|
|
|
|
# instantiating and setting up ChangeList object
|
2015-05-14 20:14:34 +08:00
|
|
|
|
m = GroupAdmin(Group, custom_site)
|
2013-05-29 21:23:08 +08:00
|
|
|
|
request = self.factory.get("/group/")
|
2018-05-02 16:39:12 +08:00
|
|
|
|
request.user = self.superuser
|
2017-08-14 18:46:02 +08:00
|
|
|
|
cl = m.get_changelist_instance(request)
|
2020-08-06 17:43:42 +08:00
|
|
|
|
cl.list_per_page = 10
|
|
|
|
|
|
|
|
|
|
ELLIPSIS = cl.paginator.ELLIPSIS
|
|
|
|
|
for number, pages, expected in [
|
|
|
|
|
(1, 1, []),
|
|
|
|
|
(1, 2, [1, 2]),
|
|
|
|
|
(6, 11, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]),
|
|
|
|
|
(6, 12, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]),
|
|
|
|
|
(6, 13, [1, 2, 3, 4, 5, 6, 7, 8, 9, ELLIPSIS, 12, 13]),
|
|
|
|
|
(7, 12, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]),
|
|
|
|
|
(7, 13, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13]),
|
|
|
|
|
(7, 14, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, ELLIPSIS, 13, 14]),
|
|
|
|
|
(8, 13, [1, 2, ELLIPSIS, 5, 6, 7, 8, 9, 10, 11, 12, 13]),
|
|
|
|
|
(8, 14, [1, 2, ELLIPSIS, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]),
|
|
|
|
|
(8, 15, [1, 2, ELLIPSIS, 5, 6, 7, 8, 9, 10, 11, ELLIPSIS, 14, 15]),
|
2013-05-29 21:23:08 +08:00
|
|
|
|
]:
|
2020-07-31 06:22:34 +08:00
|
|
|
|
with self.subTest(number=number, pages=pages):
|
2020-08-06 17:43:42 +08:00
|
|
|
|
# assuming exactly `pages * cl.list_per_page` objects
|
2020-07-31 06:22:34 +08:00
|
|
|
|
Group.objects.all().delete()
|
2020-08-06 17:43:42 +08:00
|
|
|
|
for i in range(pages * cl.list_per_page):
|
2020-07-31 06:22:34 +08:00
|
|
|
|
Group.objects.create(name="test band")
|
|
|
|
|
|
|
|
|
|
# setting page number and calculating page range
|
2020-08-06 17:43:42 +08:00
|
|
|
|
cl.page_num = number
|
2020-07-31 06:22:34 +08:00
|
|
|
|
cl.get_results(request)
|
2020-08-06 17:43:42 +08:00
|
|
|
|
self.assertEqual(list(pagination(cl)["page_range"]), expected)
|
2013-05-29 21:23:08 +08:00
|
|
|
|
|
2015-11-07 19:31:06 +08:00
|
|
|
|
def test_object_tools_displayed_no_add_permission(self):
|
|
|
|
|
"""
|
|
|
|
|
When ModelAdmin.has_add_permission() returns False, the object-tools
|
|
|
|
|
block is still shown.
|
|
|
|
|
"""
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
m = EventAdmin(Event, custom_site)
|
|
|
|
|
request = self._mocked_authenticated_request("/event/", superuser)
|
|
|
|
|
self.assertFalse(m.has_add_permission(request))
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertIn('<ul class="object-tools">', response.rendered_content)
|
|
|
|
|
# The "Add" button inside the object-tools shouldn't appear.
|
2016-01-29 02:27:25 +08:00
|
|
|
|
self.assertNotIn("Add ", response.rendered_content)
|
2015-11-07 19:31:06 +08:00
|
|
|
|
|
2021-05-20 05:22:26 +08:00
|
|
|
|
def test_search_help_text(self):
|
|
|
|
|
superuser = self._create_superuser("superuser")
|
|
|
|
|
m = BandAdmin(Band, custom_site)
|
|
|
|
|
# search_fields without search_help_text.
|
|
|
|
|
m.search_fields = ["name"]
|
|
|
|
|
request = self._mocked_authenticated_request("/band/", superuser)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertIsNone(response.context_data["cl"].search_help_text)
|
2022-03-30 16:10:24 +08:00
|
|
|
|
self.assertNotContains(response, '<div class="help id="searchbar_helptext">')
|
2021-05-20 05:22:26 +08:00
|
|
|
|
# search_fields with search_help_text.
|
|
|
|
|
m.search_help_text = "Search help text"
|
|
|
|
|
request = self._mocked_authenticated_request("/band/", superuser)
|
|
|
|
|
response = m.changelist_view(request)
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
response.context_data["cl"].search_help_text, "Search help text"
|
|
|
|
|
)
|
2022-03-30 16:10:24 +08:00
|
|
|
|
self.assertContains(
|
|
|
|
|
response, '<div class="help" id="searchbar_helptext">Search help text</div>'
|
|
|
|
|
)
|
|
|
|
|
self.assertContains(
|
|
|
|
|
response,
|
|
|
|
|
'<input type="text" size="40" name="q" value="" id="searchbar" '
|
2022-09-27 03:06:48 +08:00
|
|
|
|
'aria-describedby="searchbar_helptext">',
|
2022-03-30 16:10:24 +08:00
|
|
|
|
)
|
2021-05-20 05:22:26 +08:00
|
|
|
|
|
2023-03-03 03:06:33 +08:00
|
|
|
|
def test_search_bar_total_link_preserves_options(self):
|
|
|
|
|
self.client.force_login(self.superuser)
|
|
|
|
|
url = reverse("admin:auth_user_changelist")
|
|
|
|
|
for data, href in (
|
|
|
|
|
({"is_staff__exact": "0"}, "?"),
|
|
|
|
|
({"is_staff__exact": "0", IS_POPUP_VAR: "1"}, f"?{IS_POPUP_VAR}=1"),
|
2023-02-16 20:23:24 +08:00
|
|
|
|
({"is_staff__exact": "0", IS_FACETS_VAR: ""}, f"?{IS_FACETS_VAR}"),
|
|
|
|
|
(
|
|
|
|
|
{"is_staff__exact": "0", IS_POPUP_VAR: "1", IS_FACETS_VAR: ""},
|
|
|
|
|
f"?{IS_POPUP_VAR}=1&{IS_FACETS_VAR}",
|
|
|
|
|
),
|
2023-03-03 03:06:33 +08:00
|
|
|
|
):
|
|
|
|
|
with self.subTest(data=data):
|
|
|
|
|
response = self.client.get(url, data=data)
|
|
|
|
|
self.assertContains(
|
|
|
|
|
response, f'0 results (<a href="{href}">1 total</a>)'
|
|
|
|
|
)
|
|
|
|
|
|
2013-05-29 21:23:08 +08:00
|
|
|
|
|
2018-03-25 21:02:07 +08:00
|
|
|
|
class GetAdminLogTests(TestCase):
|
|
|
|
|
def test_custom_user_pk_not_named_id(self):
|
2013-03-20 06:17:31 +08:00
|
|
|
|
"""
|
2018-03-25 21:02:07 +08:00
|
|
|
|
{% get_admin_log %} works if the user model's primary key isn't named
|
|
|
|
|
'id'.
|
2013-03-20 06:17:31 +08:00
|
|
|
|
"""
|
2023-02-09 01:37:32 +08:00
|
|
|
|
context = Context(
|
|
|
|
|
{
|
|
|
|
|
"user": CustomIdUser(),
|
|
|
|
|
"log_entries": LogEntry.objects.all(),
|
|
|
|
|
}
|
|
|
|
|
)
|
2018-03-25 21:02:07 +08:00
|
|
|
|
template = Template(
|
|
|
|
|
"{% load log %}{% get_admin_log 10 as admin_log for_user user %}"
|
|
|
|
|
)
|
2018-03-21 20:15:07 +08:00
|
|
|
|
# This template tag just logs.
|
2013-03-20 10:16:48 +08:00
|
|
|
|
self.assertEqual(template.render(context), "")
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
2018-03-25 21:02:07 +08:00
|
|
|
|
def test_no_user(self):
|
|
|
|
|
"""{% get_admin_log %} works without specifying a user."""
|
2015-02-06 02:25:59 +08:00
|
|
|
|
user = User(username="jondoe", password="secret", email="super@example.com")
|
|
|
|
|
user.save()
|
|
|
|
|
ct = ContentType.objects.get_for_model(User)
|
|
|
|
|
LogEntry.objects.log_action(user.pk, ct.pk, user.pk, repr(user), 1)
|
2023-02-09 01:37:32 +08:00
|
|
|
|
context = Context({"log_entries": LogEntry.objects.all()})
|
2015-02-06 02:25:59 +08:00
|
|
|
|
t = Template(
|
|
|
|
|
"{% load log %}"
|
|
|
|
|
"{% get_admin_log 100 as admin_log %}"
|
|
|
|
|
"{% for entry in admin_log %}"
|
|
|
|
|
"{{ entry|safe }}"
|
|
|
|
|
"{% endfor %}"
|
|
|
|
|
)
|
2023-02-09 01:37:32 +08:00
|
|
|
|
self.assertEqual(t.render(context), "Added “<User: jondoe>”.")
|
2015-02-06 02:25:59 +08:00
|
|
|
|
|
2018-03-25 21:02:07 +08:00
|
|
|
|
def test_missing_args(self):
|
|
|
|
|
msg = "'get_admin_log' statements require two arguments"
|
|
|
|
|
with self.assertRaisesMessage(TemplateSyntaxError, msg):
|
|
|
|
|
Template("{% load log %}{% get_admin_log 10 as %}")
|
|
|
|
|
|
|
|
|
|
def test_non_integer_limit(self):
|
|
|
|
|
msg = "First argument to 'get_admin_log' must be an integer"
|
|
|
|
|
with self.assertRaisesMessage(TemplateSyntaxError, msg):
|
|
|
|
|
Template(
|
|
|
|
|
'{% load log %}{% get_admin_log "10" as admin_log for_user user %}'
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def test_without_as(self):
|
|
|
|
|
msg = "Second argument to 'get_admin_log' must be 'as'"
|
|
|
|
|
with self.assertRaisesMessage(TemplateSyntaxError, msg):
|
|
|
|
|
Template("{% load log %}{% get_admin_log 10 ad admin_log for_user user %}")
|
|
|
|
|
|
|
|
|
|
def test_without_for_user(self):
|
|
|
|
|
msg = "Fourth argument to 'get_admin_log' must be 'for_user'"
|
|
|
|
|
with self.assertRaisesMessage(TemplateSyntaxError, msg):
|
|
|
|
|
Template("{% load log %}{% get_admin_log 10 as admin_log foruser user %}")
|
|
|
|
|
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
2016-02-06 04:56:52 +08:00
|
|
|
|
@override_settings(ROOT_URLCONF="admin_changelist.urls")
|
2016-02-07 10:24:36 +08:00
|
|
|
|
class SeleniumTests(AdminSeleniumTestCase):
|
|
|
|
|
available_apps = ["admin_changelist"] + AdminSeleniumTestCase.available_apps
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
2015-03-05 21:21:29 +08:00
|
|
|
|
def setUp(self):
|
2016-02-06 04:56:52 +08:00
|
|
|
|
User.objects.create_superuser(username="super", password="secret", email=None)
|
2015-02-23 08:53:57 +08:00
|
|
|
|
|
2014-02-22 18:36:15 +08:00
|
|
|
|
def test_add_row_selection(self):
|
|
|
|
|
"""
|
2016-10-27 15:53:39 +08:00
|
|
|
|
The status line for selected rows gets updated correctly (#22038).
|
2014-02-22 18:36:15 +08:00
|
|
|
|
"""
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.admin_login(username="super", password="secret")
|
2016-04-08 10:04:45 +08:00
|
|
|
|
self.selenium.get(self.live_server_url + reverse("admin:auth_user_changelist"))
|
2022-02-04 03:24:19 +08:00
|
|
|
|
|
2014-02-22 18:36:15 +08:00
|
|
|
|
form_id = "#changelist-form"
|
|
|
|
|
|
|
|
|
|
# Test amount of rows in the Changelist
|
2021-10-20 21:54:30 +08:00
|
|
|
|
rows = self.selenium.find_elements(
|
2020-05-14 20:58:51 +08:00
|
|
|
|
By.CSS_SELECTOR, "%s #result_list tbody tr" % form_id
|
|
|
|
|
)
|
2014-02-22 18:36:15 +08:00
|
|
|
|
self.assertEqual(len(rows), 1)
|
2020-04-26 08:15:16 +08:00
|
|
|
|
row = rows[0]
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
selection_indicator = self.selenium.find_element(
|
2020-05-14 20:58:51 +08:00
|
|
|
|
By.CSS_SELECTOR, "%s .action-counter" % form_id
|
|
|
|
|
)
|
2021-10-20 21:54:30 +08:00
|
|
|
|
all_selector = self.selenium.find_element(By.ID, "action-toggle")
|
|
|
|
|
row_selector = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR,
|
2020-04-26 08:15:16 +08:00
|
|
|
|
"%s #result_list tbody tr:first-child .action-select" % form_id,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Test current selection
|
2014-02-22 18:36:15 +08:00
|
|
|
|
self.assertEqual(selection_indicator.text, "0 of 1 selected")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(all_selector.get_property("checked"), False)
|
|
|
|
|
self.assertEqual(row.get_attribute("class"), "")
|
2014-02-22 18:36:15 +08:00
|
|
|
|
|
|
|
|
|
# Select a row and check again
|
|
|
|
|
row_selector.click()
|
|
|
|
|
self.assertEqual(selection_indicator.text, "1 of 1 selected")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(all_selector.get_property("checked"), True)
|
|
|
|
|
self.assertEqual(row.get_attribute("class"), "selected")
|
|
|
|
|
|
|
|
|
|
# Deselect a row and check again
|
|
|
|
|
row_selector.click()
|
|
|
|
|
self.assertEqual(selection_indicator.text, "0 of 1 selected")
|
|
|
|
|
self.assertIs(all_selector.get_property("checked"), False)
|
|
|
|
|
self.assertEqual(row.get_attribute("class"), "")
|
|
|
|
|
|
2021-04-15 23:15:28 +08:00
|
|
|
|
def test_modifier_allows_multiple_section(self):
|
|
|
|
|
"""
|
|
|
|
|
Selecting a row and then selecting another row whilst holding shift
|
|
|
|
|
should select all rows in-between.
|
|
|
|
|
"""
|
|
|
|
|
from selenium.webdriver.common.action_chains import ActionChains
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2021-04-15 23:15:28 +08:00
|
|
|
|
from selenium.webdriver.common.keys import Keys
|
|
|
|
|
|
|
|
|
|
Parent.objects.bulk_create([Parent(name="parent%d" % i) for i in range(5)])
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
2021-10-20 21:54:30 +08:00
|
|
|
|
)
|
|
|
|
|
checkboxes = self.selenium.find_elements(
|
|
|
|
|
By.CSS_SELECTOR, "tr input.action-select"
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2021-04-15 23:15:28 +08:00
|
|
|
|
self.assertEqual(len(checkboxes), 5)
|
|
|
|
|
for c in checkboxes:
|
|
|
|
|
self.assertIs(c.get_property("checked"), False)
|
|
|
|
|
# Check first row. Hold-shift and check next-to-last row.
|
|
|
|
|
checkboxes[0].click()
|
|
|
|
|
ActionChains(self.selenium).key_down(Keys.SHIFT).click(checkboxes[-2]).key_up(
|
|
|
|
|
Keys.SHIFT
|
|
|
|
|
).perform()
|
|
|
|
|
for c in checkboxes[:-2]:
|
|
|
|
|
self.assertIs(c.get_property("checked"), True)
|
|
|
|
|
self.assertIs(checkboxes[-1].get_property("checked"), False)
|
|
|
|
|
|
2023-07-07 21:58:07 +08:00
|
|
|
|
def test_selection_counter_is_synced_when_page_is_shown(self):
|
|
|
|
|
from selenium.webdriver.common.by import By
|
|
|
|
|
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(self.live_server_url + reverse("admin:auth_user_changelist"))
|
|
|
|
|
|
|
|
|
|
form_id = "#changelist-form"
|
|
|
|
|
first_row_checkbox_selector = (
|
|
|
|
|
f"{form_id} #result_list tbody tr:first-child .action-select"
|
|
|
|
|
)
|
|
|
|
|
selection_indicator_selector = f"{form_id} .action-counter"
|
|
|
|
|
selection_indicator = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, selection_indicator_selector
|
|
|
|
|
)
|
|
|
|
|
row_checkbox = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, first_row_checkbox_selector
|
|
|
|
|
)
|
|
|
|
|
# Select a row.
|
|
|
|
|
row_checkbox.click()
|
|
|
|
|
self.assertEqual(selection_indicator.text, "1 of 1 selected")
|
|
|
|
|
# Go to another page and get back.
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
|
|
|
|
)
|
|
|
|
|
self.selenium.back()
|
|
|
|
|
# The selection indicator is synced with the selected checkboxes.
|
|
|
|
|
selection_indicator = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, selection_indicator_selector
|
|
|
|
|
)
|
|
|
|
|
row_checkbox = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, first_row_checkbox_selector
|
|
|
|
|
)
|
|
|
|
|
selected_rows = 1 if row_checkbox.is_selected() else 0
|
|
|
|
|
self.assertEqual(selection_indicator.text, f"{selected_rows} of 1 selected")
|
|
|
|
|
|
2020-04-26 08:15:16 +08:00
|
|
|
|
def test_select_all_across_pages(self):
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
Parent.objects.bulk_create([Parent(name="parent%d" % i) for i in range(101)])
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
selection_indicator = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, ".action-counter"
|
|
|
|
|
)
|
|
|
|
|
select_all_indicator = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, ".actions .all"
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2021-10-20 21:54:30 +08:00
|
|
|
|
question = self.selenium.find_element(By.CSS_SELECTOR, ".actions > .question")
|
|
|
|
|
clear = self.selenium.find_element(By.CSS_SELECTOR, ".actions > .clear")
|
|
|
|
|
select_all = self.selenium.find_element(By.ID, "action-toggle")
|
|
|
|
|
select_across = self.selenium.find_elements(By.NAME, "select_across")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
|
|
|
|
self.assertIs(question.is_displayed(), False)
|
|
|
|
|
self.assertIs(clear.is_displayed(), False)
|
|
|
|
|
self.assertIs(select_all.get_property("checked"), False)
|
2021-09-22 01:58:00 +08:00
|
|
|
|
for hidden_input in select_across:
|
|
|
|
|
self.assertEqual(hidden_input.get_property("value"), "0")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(selection_indicator.is_displayed(), True)
|
|
|
|
|
self.assertEqual(selection_indicator.text, "0 of 100 selected")
|
|
|
|
|
self.assertIs(select_all_indicator.is_displayed(), False)
|
|
|
|
|
|
|
|
|
|
select_all.click()
|
|
|
|
|
self.assertIs(question.is_displayed(), True)
|
|
|
|
|
self.assertIs(clear.is_displayed(), False)
|
|
|
|
|
self.assertIs(select_all.get_property("checked"), True)
|
2021-09-22 01:58:00 +08:00
|
|
|
|
for hidden_input in select_across:
|
|
|
|
|
self.assertEqual(hidden_input.get_property("value"), "0")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(selection_indicator.is_displayed(), True)
|
|
|
|
|
self.assertEqual(selection_indicator.text, "100 of 100 selected")
|
|
|
|
|
self.assertIs(select_all_indicator.is_displayed(), False)
|
|
|
|
|
|
|
|
|
|
question.click()
|
|
|
|
|
self.assertIs(question.is_displayed(), False)
|
|
|
|
|
self.assertIs(clear.is_displayed(), True)
|
|
|
|
|
self.assertIs(select_all.get_property("checked"), True)
|
2021-09-22 01:58:00 +08:00
|
|
|
|
for hidden_input in select_across:
|
|
|
|
|
self.assertEqual(hidden_input.get_property("value"), "1")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(selection_indicator.is_displayed(), False)
|
|
|
|
|
self.assertIs(select_all_indicator.is_displayed(), True)
|
|
|
|
|
|
|
|
|
|
clear.click()
|
|
|
|
|
self.assertIs(question.is_displayed(), False)
|
|
|
|
|
self.assertIs(clear.is_displayed(), False)
|
|
|
|
|
self.assertIs(select_all.get_property("checked"), False)
|
2021-09-22 01:58:00 +08:00
|
|
|
|
for hidden_input in select_across:
|
|
|
|
|
self.assertEqual(hidden_input.get_property("value"), "0")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.assertIs(selection_indicator.is_displayed(), True)
|
|
|
|
|
self.assertEqual(selection_indicator.text, "0 of 100 selected")
|
|
|
|
|
self.assertIs(select_all_indicator.is_displayed(), False)
|
|
|
|
|
|
|
|
|
|
def test_actions_warn_on_pending_edits(self):
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
|
|
|
|
Parent.objects.create(name="foo")
|
|
|
|
|
|
2020-04-26 08:15:16 +08:00
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
name_input = self.selenium.find_element(By.ID, "id_form-0-name")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
name_input.clear()
|
|
|
|
|
name_input.send_keys("bar")
|
2021-10-20 21:54:30 +08:00
|
|
|
|
self.selenium.find_element(By.ID, "action-toggle").click()
|
|
|
|
|
self.selenium.find_element(By.NAME, "index").click() # Go
|
2020-04-26 08:15:16 +08:00
|
|
|
|
alert = self.selenium.switch_to.alert
|
|
|
|
|
try:
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
alert.text,
|
|
|
|
|
"You have unsaved changes on individual editable fields. If you "
|
|
|
|
|
"run an action, your unsaved changes will be lost.",
|
|
|
|
|
)
|
|
|
|
|
finally:
|
|
|
|
|
alert.dismiss()
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
|
|
|
|
def test_save_with_changes_warns_on_pending_action(self):
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2020-04-26 08:15:16 +08:00
|
|
|
|
from selenium.webdriver.support.ui import Select
|
|
|
|
|
|
|
|
|
|
Parent.objects.create(name="parent")
|
|
|
|
|
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
name_input = self.selenium.find_element(By.ID, "id_form-0-name")
|
2020-04-26 08:15:16 +08:00
|
|
|
|
name_input.clear()
|
|
|
|
|
name_input.send_keys("other name")
|
2021-10-20 21:54:30 +08:00
|
|
|
|
Select(self.selenium.find_element(By.NAME, "action")).select_by_value(
|
2020-04-26 08:15:16 +08:00
|
|
|
|
"delete_selected"
|
|
|
|
|
)
|
2021-10-20 21:54:30 +08:00
|
|
|
|
self.selenium.find_element(By.NAME, "_save").click()
|
2020-04-26 08:15:16 +08:00
|
|
|
|
alert = self.selenium.switch_to.alert
|
|
|
|
|
try:
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
alert.text,
|
2020-04-29 13:00:20 +08:00
|
|
|
|
"You have selected an action, but you haven’t saved your "
|
2020-04-26 08:15:16 +08:00
|
|
|
|
"changes to individual fields yet. Please click OK to save. "
|
2020-04-29 13:00:20 +08:00
|
|
|
|
"You’ll need to re-run the action.",
|
2020-04-26 08:15:16 +08:00
|
|
|
|
)
|
|
|
|
|
finally:
|
|
|
|
|
alert.dismiss()
|
|
|
|
|
|
|
|
|
|
def test_save_without_changes_warns_on_pending_action(self):
|
2021-10-20 21:54:30 +08:00
|
|
|
|
from selenium.webdriver.common.by import By
|
2020-04-26 08:15:16 +08:00
|
|
|
|
from selenium.webdriver.support.ui import Select
|
|
|
|
|
|
|
|
|
|
Parent.objects.create(name="parent")
|
|
|
|
|
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_parent_changelist")
|
2022-02-04 03:24:19 +08:00
|
|
|
|
)
|
2020-04-26 08:15:16 +08:00
|
|
|
|
|
2021-10-20 21:54:30 +08:00
|
|
|
|
Select(self.selenium.find_element(By.NAME, "action")).select_by_value(
|
2020-04-26 08:15:16 +08:00
|
|
|
|
"delete_selected"
|
|
|
|
|
)
|
2021-10-20 21:54:30 +08:00
|
|
|
|
self.selenium.find_element(By.NAME, "_save").click()
|
2020-04-26 08:15:16 +08:00
|
|
|
|
alert = self.selenium.switch_to.alert
|
|
|
|
|
try:
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
alert.text,
|
2020-04-29 13:00:20 +08:00
|
|
|
|
"You have selected an action, and you haven’t made any "
|
|
|
|
|
"changes on individual fields. You’re probably looking for "
|
2020-04-26 08:15:16 +08:00
|
|
|
|
"the Go button rather than the Save button.",
|
|
|
|
|
)
|
|
|
|
|
finally:
|
|
|
|
|
alert.dismiss()
|
2022-02-14 11:03:53 +08:00
|
|
|
|
|
|
|
|
|
def test_collapse_filters(self):
|
|
|
|
|
from selenium.webdriver.common.by import By
|
|
|
|
|
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
self.selenium.get(self.live_server_url + reverse("admin:auth_user_changelist"))
|
|
|
|
|
|
|
|
|
|
# The UserAdmin has 3 field filters by default: "staff status",
|
|
|
|
|
# "superuser status", and "active".
|
|
|
|
|
details = self.selenium.find_elements(By.CSS_SELECTOR, "details")
|
|
|
|
|
# All filters are opened at first.
|
|
|
|
|
for detail in details:
|
|
|
|
|
self.assertTrue(detail.get_attribute("open"))
|
|
|
|
|
# Collapse "staff' and "superuser" filters.
|
|
|
|
|
for detail in details[:2]:
|
|
|
|
|
summary = detail.find_element(By.CSS_SELECTOR, "summary")
|
|
|
|
|
summary.click()
|
|
|
|
|
self.assertFalse(detail.get_attribute("open"))
|
|
|
|
|
# Filters are in the same state after refresh.
|
|
|
|
|
self.selenium.refresh()
|
|
|
|
|
self.assertFalse(
|
|
|
|
|
self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, "[data-filter-title='staff status']"
|
|
|
|
|
).get_attribute("open")
|
|
|
|
|
)
|
|
|
|
|
self.assertFalse(
|
|
|
|
|
self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, "[data-filter-title='superuser status']"
|
|
|
|
|
).get_attribute("open")
|
|
|
|
|
)
|
|
|
|
|
self.assertTrue(
|
|
|
|
|
self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, "[data-filter-title='active']"
|
|
|
|
|
).get_attribute("open")
|
|
|
|
|
)
|
|
|
|
|
# Collapse a filter on another view (Bands).
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_band_changelist")
|
|
|
|
|
)
|
|
|
|
|
self.selenium.find_element(By.CSS_SELECTOR, "summary").click()
|
|
|
|
|
# Go to Users view and then, back again to Bands view.
|
|
|
|
|
self.selenium.get(self.live_server_url + reverse("admin:auth_user_changelist"))
|
|
|
|
|
self.selenium.get(
|
|
|
|
|
self.live_server_url + reverse("admin:admin_changelist_band_changelist")
|
|
|
|
|
)
|
|
|
|
|
# The filter remains in the same state.
|
|
|
|
|
self.assertFalse(
|
|
|
|
|
self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR,
|
|
|
|
|
"[data-filter-title='number of members']",
|
|
|
|
|
).get_attribute("open")
|
|
|
|
|
)
|
2023-01-26 03:53:38 +08:00
|
|
|
|
|
|
|
|
|
def test_collapse_filter_with_unescaped_title(self):
|
|
|
|
|
from selenium.webdriver.common.by import By
|
|
|
|
|
|
|
|
|
|
self.admin_login(username="super", password="secret")
|
|
|
|
|
changelist_url = reverse("admin:admin_changelist_proxyuser_changelist")
|
|
|
|
|
self.selenium.get(self.live_server_url + changelist_url)
|
|
|
|
|
# Title is escaped.
|
|
|
|
|
filter_title = self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, "[data-filter-title='It\\'s OK']"
|
|
|
|
|
)
|
|
|
|
|
filter_title.find_element(By.CSS_SELECTOR, "summary").click()
|
|
|
|
|
self.assertFalse(filter_title.get_attribute("open"))
|
|
|
|
|
# Filter is in the same state after refresh.
|
|
|
|
|
self.selenium.refresh()
|
|
|
|
|
self.assertFalse(
|
|
|
|
|
self.selenium.find_element(
|
|
|
|
|
By.CSS_SELECTOR, "[data-filter-title='It\\'s OK']"
|
|
|
|
|
).get_attribute("open")
|
|
|
|
|
)
|