diff --git a/AUTHORS b/AUTHORS
index a52ba34954..51bc2f592a 100644
--- a/AUTHORS
+++ b/AUTHORS
@@ -748,6 +748,7 @@ answer newbie questions, and generally made Django that much better:
Sam Newman
Sander Dijkhuis
Sanket Saurav
+ Sanyam Khurana
Sarthak Mehrish
schwank@gmail.com
Scot Hacker
diff --git a/django/contrib/admin/checks.py b/django/contrib/admin/checks.py
index 4007a781fb..3f8abdd476 100644
--- a/django/contrib/admin/checks.py
+++ b/django/contrib/admin/checks.py
@@ -20,6 +20,17 @@ from django.utils.deprecation import RemovedInDjango30Warning
from django.utils.inspect import get_func_args
+def _issubclass(cls, classinfo):
+ """
+ issubclass() variant that doesn't raise an exception if cls isn't a
+ class.
+ """
+ try:
+ return issubclass(cls, classinfo)
+ except TypeError:
+ return False
+
+
def check_admin_app(app_configs, **kwargs):
from django.contrib.admin.sites import all_sites
errors = []
@@ -341,7 +352,7 @@ class BaseModelAdminChecks:
def _check_form(self, obj):
""" Check that form subclasses BaseModelForm. """
- if not issubclass(obj.form, BaseModelForm):
+ if not _issubclass(obj.form, BaseModelForm):
return must_inherit_from(parent='BaseModelForm', option='form',
obj=obj, id='admin.E016')
else:
@@ -640,11 +651,20 @@ class ModelAdminChecks(BaseModelAdminChecks):
def _check_inlines_item(self, obj, inline, label):
""" Check one inline model admin. """
- inline_label = inline.__module__ + '.' + inline.__name__
+ try:
+ inline_label = inline.__module__ + '.' + inline.__name__
+ except AttributeError:
+ return [
+ checks.Error(
+ "'%s' must inherit from 'InlineModelAdmin'." % obj,
+ obj=obj.__class__,
+ id='admin.E104',
+ )
+ ]
from django.contrib.admin.options import InlineModelAdmin
- if not issubclass(inline, InlineModelAdmin):
+ if not _issubclass(inline, InlineModelAdmin):
return [
checks.Error(
"'%s' must inherit from 'InlineModelAdmin'." % inline_label,
@@ -660,7 +680,7 @@ class ModelAdminChecks(BaseModelAdminChecks):
id='admin.E105',
)
]
- elif not issubclass(inline.model, models.Model):
+ elif not _issubclass(inline.model, models.Model):
return must_be('a Model', option='%s.model' % inline_label, obj=obj, id='admin.E106')
else:
return inline(obj.model, obj.admin_site).check()
@@ -763,7 +783,7 @@ class ModelAdminChecks(BaseModelAdminChecks):
if callable(item) and not isinstance(item, models.Field):
# If item is option 3, it should be a ListFilter...
- if not issubclass(item, ListFilter):
+ if not _issubclass(item, ListFilter):
return must_inherit_from(parent='ListFilter', option=label,
obj=obj, id='admin.E113')
# ... but not a FieldListFilter.
@@ -780,7 +800,7 @@ class ModelAdminChecks(BaseModelAdminChecks):
elif isinstance(item, (tuple, list)):
# item is option #2
field, list_filter_class = item
- if not issubclass(list_filter_class, FieldListFilter):
+ if not _issubclass(list_filter_class, FieldListFilter):
return must_inherit_from(parent='FieldListFilter', option='%s[1]' % label, obj=obj, id='admin.E115')
else:
return []
@@ -1041,7 +1061,7 @@ class InlineModelAdminChecks(BaseModelAdminChecks):
def _check_formset(self, obj):
""" Check formset is a subclass of BaseModelFormSet. """
- if not issubclass(obj.formset, BaseModelFormSet):
+ if not _issubclass(obj.formset, BaseModelFormSet):
return must_inherit_from(parent='BaseModelFormSet', option='formset', obj=obj, id='admin.E206')
else:
return []
diff --git a/tests/modeladmin/test_checks.py b/tests/modeladmin/test_checks.py
index 89fde35d3c..a1b7001f68 100644
--- a/tests/modeladmin/test_checks.py
+++ b/tests/modeladmin/test_checks.py
@@ -225,11 +225,16 @@ class FormCheckTests(CheckTestCase):
class TestModelAdmin(ModelAdmin):
form = FakeForm
- self.assertIsInvalid(
- TestModelAdmin, ValidationTestModel,
- "The value of 'form' must inherit from 'BaseModelForm'.",
- 'admin.E016'
- )
+ class TestModelAdminWithNoForm(ModelAdmin):
+ form = 'not a form'
+
+ for model_admin in (TestModelAdmin, TestModelAdminWithNoForm):
+ with self.subTest(model_admin):
+ self.assertIsInvalid(
+ model_admin, ValidationTestModel,
+ "The value of 'form' must inherit from 'BaseModelForm'.",
+ 'admin.E016'
+ )
def test_fieldsets_with_custom_form_validation(self):
@@ -598,6 +603,40 @@ class ListFilterTests(CheckTestCase):
'admin.E112'
)
+ def test_not_list_filter_class(self):
+ class TestModelAdmin(ModelAdmin):
+ list_filter = ['RandomClass']
+
+ self.assertIsInvalid(
+ TestModelAdmin, ValidationTestModel,
+ "The value of 'list_filter[0]' refers to 'RandomClass', which "
+ "does not refer to a Field.",
+ 'admin.E116'
+ )
+
+ def test_callable(self):
+ def random_callable():
+ pass
+
+ class TestModelAdmin(ModelAdmin):
+ list_filter = [random_callable]
+
+ self.assertIsInvalid(
+ TestModelAdmin, ValidationTestModel,
+ "The value of 'list_filter[0]' must inherit from 'ListFilter'.",
+ 'admin.E113'
+ )
+
+ def test_not_callable(self):
+ class TestModelAdmin(ModelAdmin):
+ list_filter = [[42, 42]]
+
+ self.assertIsInvalid(
+ TestModelAdmin, ValidationTestModel,
+ "The value of 'list_filter[0][1]' must inherit from 'FieldListFilter'.",
+ 'admin.E115'
+ )
+
def test_missing_field(self):
class TestModelAdmin(ModelAdmin):
list_filter = ('non_existent_field',)
@@ -655,6 +694,19 @@ class ListFilterTests(CheckTestCase):
'admin.E115'
)
+ def test_list_filter_is_func(self):
+ def get_filter():
+ pass
+
+ class TestModelAdmin(ModelAdmin):
+ list_filter = [get_filter]
+
+ self.assertIsInvalid(
+ TestModelAdmin, ValidationTestModel,
+ "The value of 'list_filter[0]' must inherit from 'ListFilter'.",
+ 'admin.E113'
+ )
+
def test_not_associated_with_field_name(self):
class TestModelAdmin(ModelAdmin):
list_filter = (BooleanFieldListFilter,)
@@ -918,6 +970,16 @@ class InlinesCheckTests(CheckTestCase):
'admin.E103'
)
+ def test_not_correct_inline_field(self):
+ class TestModelAdmin(ModelAdmin):
+ inlines = [42]
+
+ self.assertIsInvalidRegexp(
+ TestModelAdmin, ValidationTestModel,
+ r"'.*\.TestModelAdmin' must inherit from 'InlineModelAdmin'\.",
+ 'admin.E104'
+ )
+
def test_not_model_admin(self):
class ValidationTestInline:
pass
@@ -960,6 +1022,32 @@ class InlinesCheckTests(CheckTestCase):
'admin.E106'
)
+ def test_invalid_model(self):
+ class ValidationTestInline(TabularInline):
+ model = 'Not a class'
+
+ class TestModelAdmin(ModelAdmin):
+ inlines = [ValidationTestInline]
+
+ self.assertIsInvalidRegexp(
+ TestModelAdmin, ValidationTestModel,
+ r"The value of '.*\.ValidationTestInline.model' must be a Model\.",
+ 'admin.E106'
+ )
+
+ def test_invalid_callable(self):
+ def random_obj():
+ pass
+
+ class TestModelAdmin(ModelAdmin):
+ inlines = [random_obj]
+
+ self.assertIsInvalidRegexp(
+ TestModelAdmin, ValidationTestModel,
+ r"'.*\.random_obj' must inherit from 'InlineModelAdmin'\.",
+ 'admin.E104'
+ )
+
def test_valid_case(self):
class ValidationTestInline(TabularInline):
model = ValidationTestInlineModel
@@ -1102,6 +1190,21 @@ class FormsetCheckTests(CheckTestCase):
invalid_obj=ValidationTestInline
)
+ def test_inline_without_formset_class(self):
+ class ValidationTestInlineWithoutFormsetClass(TabularInline):
+ model = ValidationTestInlineModel
+ formset = 'Not a FormSet Class'
+
+ class TestModelAdminWithoutFormsetClass(ModelAdmin):
+ inlines = [ValidationTestInlineWithoutFormsetClass]
+
+ self.assertIsInvalid(
+ TestModelAdminWithoutFormsetClass, ValidationTestModel,
+ "The value of 'formset' must inherit from 'BaseModelFormSet'.",
+ 'admin.E206',
+ invalid_obj=ValidationTestInlineWithoutFormsetClass
+ )
+
def test_valid_case(self):
class RealModelFormSet(BaseModelFormSet):
pass