2013-02-23 19:02:01 +01:00
|
|
|
from django.contrib import admin
|
2010-09-27 17:11:29 +02:00
|
|
|
from django.contrib.admin.options import ModelAdmin
|
2011-10-13 20:51:33 +02:00
|
|
|
from django.contrib.auth.models import User
|
2017-12-27 19:47:14 +01:00
|
|
|
from django.db.models import F
|
2015-01-28 13:35:27 +01:00
|
|
|
from django.test import RequestFactory, TestCase
|
2010-09-27 17:11:29 +02:00
|
|
|
|
2015-01-28 13:35:27 +01:00
|
|
|
from .models import (
|
|
|
|
Band, DynOrderingBandAdmin, Song, SongInlineDefaultOrdering,
|
|
|
|
SongInlineNewOrdering,
|
|
|
|
)
|
2010-09-27 17:11:29 +02:00
|
|
|
|
2011-10-07 02:41:25 +02:00
|
|
|
|
2017-01-19 08:39:46 +01:00
|
|
|
class MockRequest:
|
2011-10-07 02:41:25 +02:00
|
|
|
pass
|
|
|
|
|
2013-11-03 05:36:09 +01:00
|
|
|
|
2017-01-19 08:39:46 +01:00
|
|
|
class MockSuperUser:
|
2020-09-14 12:28:17 +02:00
|
|
|
def has_perm(self, perm, obj=None):
|
2011-10-07 02:41:25 +02:00
|
|
|
return True
|
|
|
|
|
2014-05-28 18:07:27 +02:00
|
|
|
def has_module_perms(self, module):
|
|
|
|
return True
|
|
|
|
|
2016-11-12 18:11:23 +01:00
|
|
|
|
2011-10-07 02:41:25 +02:00
|
|
|
request = MockRequest()
|
|
|
|
request.user = MockSuperUser()
|
|
|
|
|
2015-05-14 14:14:34 +02:00
|
|
|
site = admin.AdminSite()
|
|
|
|
|
2011-10-07 02:41:25 +02:00
|
|
|
|
2010-09-27 17:11:29 +02:00
|
|
|
class TestAdminOrdering(TestCase):
|
|
|
|
"""
|
2013-03-08 15:15:23 +01:00
|
|
|
Let's make sure that ModelAdmin.get_queryset uses the ordering we define
|
|
|
|
in ModelAdmin rather that ordering defined in the model's inner Meta
|
2010-09-27 17:11:29 +02:00
|
|
|
class.
|
|
|
|
"""
|
2018-11-26 20:01:27 +01:00
|
|
|
request_factory = RequestFactory()
|
2010-09-27 17:11:29 +02:00
|
|
|
|
2018-11-24 02:59:38 +01:00
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
2013-11-21 20:57:58 +01:00
|
|
|
Band.objects.bulk_create([
|
|
|
|
Band(name='Aerosmith', bio='', rank=3),
|
|
|
|
Band(name='Radiohead', bio='', rank=1),
|
|
|
|
Band(name='Van Halen', bio='', rank=2),
|
|
|
|
])
|
2010-09-27 17:11:29 +02:00
|
|
|
|
|
|
|
def test_default_ordering(self):
|
|
|
|
"""
|
|
|
|
The default ordering should be by name, as specified in the inner Meta
|
|
|
|
class.
|
|
|
|
"""
|
2015-05-14 14:14:34 +02:00
|
|
|
ma = ModelAdmin(Band, site)
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [b.name for b in ma.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Aerosmith', 'Radiohead', 'Van Halen'], names)
|
2010-09-27 17:11:29 +02:00
|
|
|
|
|
|
|
def test_specified_ordering(self):
|
|
|
|
"""
|
|
|
|
Let's use a custom ModelAdmin that changes the ordering, and make sure
|
|
|
|
it actually changes.
|
|
|
|
"""
|
|
|
|
class BandAdmin(ModelAdmin):
|
2013-11-02 22:02:56 +01:00
|
|
|
ordering = ('rank',) # default ordering is ('name',)
|
2015-05-14 14:14:34 +02:00
|
|
|
ma = BandAdmin(Band, site)
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [b.name for b in ma.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Radiohead', 'Van Halen', 'Aerosmith'], names)
|
2010-12-12 23:56:01 +01:00
|
|
|
|
2017-12-27 19:47:14 +01:00
|
|
|
def test_specified_ordering_by_f_expression(self):
|
|
|
|
class BandAdmin(ModelAdmin):
|
|
|
|
ordering = (F('rank').desc(nulls_last=True),)
|
|
|
|
band_admin = BandAdmin(Band, site)
|
|
|
|
names = [b.name for b in band_admin.get_queryset(request)]
|
|
|
|
self.assertEqual(['Aerosmith', 'Van Halen', 'Radiohead'], names)
|
|
|
|
|
2011-06-12 15:04:53 +02:00
|
|
|
def test_dynamic_ordering(self):
|
|
|
|
"""
|
2014-03-02 15:25:53 +01:00
|
|
|
Let's use a custom ModelAdmin that changes the ordering dynamically.
|
2011-06-12 15:04:53 +02:00
|
|
|
"""
|
|
|
|
super_user = User.objects.create(username='admin', is_superuser=True)
|
|
|
|
other_user = User.objects.create(username='other')
|
|
|
|
request = self.request_factory.get('/')
|
|
|
|
request.user = super_user
|
2015-05-14 14:14:34 +02:00
|
|
|
ma = DynOrderingBandAdmin(Band, site)
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [b.name for b in ma.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Radiohead', 'Van Halen', 'Aerosmith'], names)
|
2011-06-12 15:04:53 +02:00
|
|
|
request.user = other_user
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [b.name for b in ma.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Aerosmith', 'Radiohead', 'Van Halen'], names)
|
2011-06-12 15:04:53 +02:00
|
|
|
|
|
|
|
|
2010-12-12 23:56:01 +01:00
|
|
|
class TestInlineModelAdminOrdering(TestCase):
|
|
|
|
"""
|
2013-03-08 15:15:23 +01:00
|
|
|
Let's make sure that InlineModelAdmin.get_queryset uses the ordering we
|
|
|
|
define in InlineModelAdmin.
|
2010-12-12 23:56:01 +01:00
|
|
|
"""
|
|
|
|
|
2018-11-24 02:59:38 +01:00
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
cls.band = Band.objects.create(name='Aerosmith', bio='', rank=3)
|
2013-11-21 20:57:58 +01:00
|
|
|
Song.objects.bulk_create([
|
2018-11-24 02:59:38 +01:00
|
|
|
Song(band=cls.band, name='Pink', duration=235),
|
|
|
|
Song(band=cls.band, name='Dude (Looks Like a Lady)', duration=264),
|
|
|
|
Song(band=cls.band, name='Jaded', duration=214),
|
2013-11-21 20:57:58 +01:00
|
|
|
])
|
2010-12-12 23:56:01 +01:00
|
|
|
|
|
|
|
def test_default_ordering(self):
|
|
|
|
"""
|
|
|
|
The default ordering should be by name, as specified in the inner Meta
|
|
|
|
class.
|
|
|
|
"""
|
2015-05-14 14:14:34 +02:00
|
|
|
inline = SongInlineDefaultOrdering(self.band, site)
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [s.name for s in inline.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Dude (Looks Like a Lady)', 'Jaded', 'Pink'], names)
|
2010-12-12 23:56:01 +01:00
|
|
|
|
|
|
|
def test_specified_ordering(self):
|
|
|
|
"""
|
|
|
|
Let's check with ordering set to something different than the default.
|
|
|
|
"""
|
2015-05-14 14:14:34 +02:00
|
|
|
inline = SongInlineNewOrdering(self.band, site)
|
2013-03-08 15:15:23 +01:00
|
|
|
names = [s.name for s in inline.get_queryset(request)]
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(['Jaded', 'Pink', 'Dude (Looks Like a Lady)'], names)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
|
|
|
|
|
|
|
class TestRelatedFieldsAdminOrdering(TestCase):
|
2018-11-24 02:59:38 +01:00
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
cls.b1 = Band.objects.create(name='Pink Floyd', bio='', rank=1)
|
|
|
|
cls.b2 = Band.objects.create(name='Foo Fighters', bio='', rank=5)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
2018-11-24 02:59:38 +01:00
|
|
|
def setUp(self):
|
2013-02-23 19:02:01 +01:00
|
|
|
# we need to register a custom ModelAdmin (instead of just using
|
|
|
|
# ModelAdmin) because the field creator tries to find the ModelAdmin
|
|
|
|
# for the related model
|
|
|
|
class SongAdmin(admin.ModelAdmin):
|
|
|
|
pass
|
2015-05-14 14:14:34 +02:00
|
|
|
site.register(Song, SongAdmin)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
2013-11-21 20:57:58 +01:00
|
|
|
def tearDown(self):
|
2015-05-14 14:14:34 +02:00
|
|
|
site.unregister(Song)
|
|
|
|
if Band in site._registry:
|
|
|
|
site.unregister(Band)
|
2013-11-21 20:57:58 +01:00
|
|
|
|
2013-02-23 19:02:01 +01:00
|
|
|
def check_ordering_of_field_choices(self, correct_ordering):
|
2015-12-24 16:25:59 +01:00
|
|
|
fk_field = site._registry[Song].formfield_for_foreignkey(Song.band.field, request=None)
|
|
|
|
m2m_field = site._registry[Song].formfield_for_manytomany(Song.other_interpreters.field, request=None)
|
2017-03-17 12:51:48 +01:00
|
|
|
self.assertEqual(list(fk_field.queryset), correct_ordering)
|
|
|
|
self.assertEqual(list(m2m_field.queryset), correct_ordering)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
|
|
|
def test_no_admin_fallback_to_model_ordering(self):
|
|
|
|
# should be ordered by name (as defined by the model)
|
|
|
|
self.check_ordering_of_field_choices([self.b2, self.b1])
|
|
|
|
|
|
|
|
def test_admin_with_no_ordering_fallback_to_model_ordering(self):
|
|
|
|
class NoOrderingBandAdmin(admin.ModelAdmin):
|
|
|
|
pass
|
2015-05-14 14:14:34 +02:00
|
|
|
site.register(Band, NoOrderingBandAdmin)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
|
|
|
# should be ordered by name (as defined by the model)
|
|
|
|
self.check_ordering_of_field_choices([self.b2, self.b1])
|
|
|
|
|
|
|
|
def test_admin_ordering_beats_model_ordering(self):
|
|
|
|
class StaticOrderingBandAdmin(admin.ModelAdmin):
|
2013-11-21 20:57:58 +01:00
|
|
|
ordering = ('rank',)
|
2015-05-14 14:14:34 +02:00
|
|
|
site.register(Band, StaticOrderingBandAdmin)
|
2013-02-23 19:02:01 +01:00
|
|
|
|
|
|
|
# should be ordered by rank (defined by the ModelAdmin)
|
|
|
|
self.check_ordering_of_field_choices([self.b1, self.b2])
|
2013-11-21 22:18:52 +01:00
|
|
|
|
|
|
|
def test_custom_queryset_still_wins(self):
|
2016-10-27 09:53:39 +02:00
|
|
|
"""Custom queryset has still precedence (#21405)"""
|
2013-11-21 22:18:52 +01:00
|
|
|
class SongAdmin(admin.ModelAdmin):
|
|
|
|
# Exclude one of the two Bands from the querysets
|
2015-12-24 16:25:59 +01:00
|
|
|
def formfield_for_foreignkey(self, db_field, request, **kwargs):
|
2013-11-21 22:18:52 +01:00
|
|
|
if db_field.name == 'band':
|
|
|
|
kwargs["queryset"] = Band.objects.filter(rank__gt=2)
|
2017-01-21 14:13:44 +01:00
|
|
|
return super().formfield_for_foreignkey(db_field, request, **kwargs)
|
2013-11-22 18:10:18 +01:00
|
|
|
|
2015-12-24 16:25:59 +01:00
|
|
|
def formfield_for_manytomany(self, db_field, request, **kwargs):
|
2013-11-21 22:18:52 +01:00
|
|
|
if db_field.name == 'other_interpreters':
|
|
|
|
kwargs["queryset"] = Band.objects.filter(rank__gt=2)
|
2017-01-21 14:13:44 +01:00
|
|
|
return super().formfield_for_foreignkey(db_field, request, **kwargs)
|
2013-11-21 22:18:52 +01:00
|
|
|
|
|
|
|
class StaticOrderingBandAdmin(admin.ModelAdmin):
|
|
|
|
ordering = ('rank',)
|
|
|
|
|
2015-05-14 14:14:34 +02:00
|
|
|
site.unregister(Song)
|
|
|
|
site.register(Song, SongAdmin)
|
|
|
|
site.register(Band, StaticOrderingBandAdmin)
|
2013-11-21 22:18:52 +01:00
|
|
|
|
|
|
|
self.check_ordering_of_field_choices([self.b2])
|