{
  "instance_id": "django__django-13447",
  "repo": "django/django",
  "created_at": "2020-09-22T08:49:25Z",
  "problem_statement": "Added model class to app_list context\nDescription\n\t \n\t\t(last modified by Raffaele Salmaso)\n\t \nI need to manipulate the app_list in my custom admin view, and the easiest way to get the result is to have access to the model class (currently the dictionary is a serialized model).\nIn addition I would make the _build_app_dict method public, as it is used by the two views index and app_index.\n",
  "patch": "diff --git a/django/contrib/admin/sites.py b/django/contrib/admin/sites.py\n--- a/django/contrib/admin/sites.py\n+++ b/django/contrib/admin/sites.py\n@@ -461,6 +461,7 @@ def _build_app_dict(self, request, label=None):\n \n             info = (app_label, model._meta.model_name)\n             model_dict = {\n+                'model': model,\n                 'name': capfirst(model._meta.verbose_name_plural),\n                 'object_name': model._meta.object_name,\n                 'perms': perms,\n",
  "similar_bug_items": [
    {
      "pr_number": 6774,
      "pr_title": "Fixed #26171 -- Made MySQL create an index on ForeignKeys with db_contraint=False.",
      "pr_body": "Refactor \"Prevented unneeded index creation on MySQL-InnoDB\" (2ceb10f)\nto avoid setting db_index = False. Check db_constraint=True before\nskipping the index creation, fixes #26171.\n",
      "issue_id": 26171,
      "issue_title": "ForeignKey with db_constraint=False doesn't generate an index on MySQL",
      "issue_body": "I have a model that does not want to use a constraint on a foreign key, but still use a database index:\nclass Category(models.Model):\n    text = models.CharField(max_length=3)\n\nclass Message(models.Model):\n    cat = models.ForeignKey(Category, db_constraint=False)\n\nclass IndexMessage(models.Model):\n    cat = models.ForeignKey(Category, db_constraint=False, db_index=True)\n\nclass StrongMessage(models.Model):\n    cat = models.ForeignKey(Category)\nThe SQLite backend generates an index on the FK column for both models:\n$ python manage.py sqlmigrate boohoo 0001_initial\nBEGIN;\n--\n-- Create model Category\n--\nCREATE TABLE \"boohoo_category\" (\"id\" integer NOT NULL PRIMARY KEY AUTOINCREMENT, \"text\" varchar(3) NOT NULL);\n--\n-- Create model IndexMessage\n--\nCREATE TABLE \"boohoo_indexmessage\" (\"id\" integer NOT NULL PRIMARY KEY AUTOINCREMENT, \"cat_id\" integer NOT NULL);\n--\n-- Create model Message\n--\nCREATE TABLE \"boohoo_message\" (\"id\" integer NOT NULL PRIMARY KEY AUTOINCREMENT, \"cat_id\" integer NOT NULL);\n--\n-- Create model StrongMessage\n--\nCREATE TABLE \"boohoo_strongmessage\" (\"id\" integer NOT NULL PRIMARY KEY AUTOINCREMENT, \"cat_id\" integer NOT NULL REFERENCES \"boohoo_category\" (\"id\"));\nCREATE INDEX \"boohoo_indexmessage_05e7bb57\" ON \"boohoo_indexmessage\" (\"cat_id\");\nCREATE INDEX \"boohoo_message_05e7bb57\" ON \"boohoo_message\" (\"cat_id\");\nCREATE INDEX \"boohoo_strongmessage_05e7bb57\" ON \"boohoo_strongmessage\" (\"cat_id\");\n\nCOMMIT;\nWith the MySQL backend, this does not create an index on the FK:\n$ python manage.py sqlmigrate boohoo 0001_initial\nBEGIN;\n--\n-- Create model Category\n--\nCREATE TABLE `boohoo_category` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `text` varchar(3) NOT NULL);\n--\n-- Create model IndexMessage\n--\nCREATE TABLE `boohoo_indexmessage` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `cat_id` integer NOT NULL);\n--\n-- Create model Message\n--\nCREATE TABLE `boohoo_message` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `cat_id` integer NOT NULL);\n--\n-- Create model StrongMessage\n--\nCREATE TABLE `boohoo_strongmessage` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `cat_id` integer NOT NULL);\nALTER TABLE `boohoo_strongmessage` ADD CONSTRAINT `boohoo_strongmessage_cat_id_c843b68a_fk_boohoo_category_id` FOREIGN KEY (`cat_id`) REFERENCES `boohoo_category` (`id`);\n\nCOMMIT;\nI would think that specifying db_constraint=false would only leave out the constraint, and not also the index; Especially with <field>.db_index still set to true. Adding db_index=True does not help, probably because that is the default setting on the FK field object.\nThis also applies to earlier django versions.\nA simple workaround is to use\nindex_together = (('cat', ), )\non the models.\nI'm inclined to blame this on django.db.backends.mysql.schema.DatabaseSchemaEditor#_model_indexes_sql,\nwhich may need an extra check for db_constraint being used. This fixes my problem (but changes current django behavior):\n--- django/db/backends/mysql/schema.py.orig     2016-02-03 12:01:10.000000000 +0100\n+++ django/db/backends/mysql/schema.py  2016-02-03 12:00:19.000000000 +0100\n@@ -64,7 +64,7 @@\n         )\n         if storage == \"InnoDB\":\n             for field in model._meta.local_fields:\n-                if field.db_index and not field.unique and field.get_internal_type() == \"ForeignKey\":\n+                if field.db_index and not field.unique and field.get_internal_type() == \"ForeignKey\" and field.db_constraint:\n                     # Temporary setting db_index to False (in memory) to disable\n                     # index creation for FKs (index automatically created by MySQL)\n                     field.db_index = False",
      "issue_closed_at": "2016-06-28T07:19:11",
      "base_commit": "5fe1c92250017110430c7c2153cfd8776e4c7064",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_model_indexes_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _model_indexes_sql(self, model):\n        \"\"\"\n        Return all index SQL statements (field indexes, index_together) for the\n        specified model, as a list.\n        \"\"\"\n        if not model._meta.managed or model._meta.proxy or model._meta.swapped:\n            return []\n        output = []\n        for field in model._meta.local_fields:\n            if field.db_index and not field.unique:\n                output.append(self._create_index_sql(model, [field], suffix=\"\"))\n\n        for field_names in model._meta.index_together:\n            fields = [model._meta.get_field(field) for field in field_names]\n            output.append(self._create_index_sql(model, fields, suffix=\"_idx\"))\n        return output"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_model_indexes_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _model_indexes_sql(self, model):\n        \"\"\"\n        Return all index SQL statements (field indexes, index_together) for the\n        specified model, as a list.\n        \"\"\"\n        if not model._meta.managed or model._meta.proxy or model._meta.swapped:\n            return []\n        output = []\n        for field in model._meta.local_fields:\n            if field.db_index and not field.unique:\n                output.append(self._create_index_sql(model, [field], suffix=\"\"))\n\n        for field_names in model._meta.index_together:\n            fields = [model._meta.get_field(field) for field in field_names]\n            output.append(self._create_index_sql(model, fields, suffix=\"_idx\"))\n        return output"
        },
        {
          "file": "django/db/backends/mysql/schema.py",
          "type": "function",
          "name": "add_field",
          "class_name": "DatabaseSchemaEditor",
          "code": "def add_field(self, model, field):\n        super(DatabaseSchemaEditor, self).add_field(model, field)\n\n        # Simulate the effect of a one-off default.\n        # field.default may be unhashable, so a set isn't used for \"in\" check.\n        if self.skip_default(field) and field.default not in (None, NOT_PROVIDED):\n            effective_default = self.effective_default(field)\n            self.execute('UPDATE %(table)s SET %(column)s = %%s' % {\n                'table': self.quote_name(model._meta.db_table),\n                'column': self.quote_name(field.column),\n            }, [effective_default])"
        }
      ]
    },
    {
      "pr_number": 7097,
      "pr_title": "Fixed #27068 -- Unified form field initial data retrieval.",
      "pr_body": "https://code.djangoproject.com/ticket/27068\n",
      "issue_id": 27068,
      "issue_title": "Acquire form's initial data more consistently",
      "issue_body": "In\nBoundField\n, sometimes initial data\n​\nhandles callables\n. While other times,\n​\nit doesn't\n. This can lead to a theoretical bug when the initial data is a callable, but the field is disabled.\nOther times, the initial callable is handled, but microseconds are\n​\nnot chopped\nleading to theoretical false positives that data has changed.\nInitial data should be handled more consistently across all cases.\nMarking as a bug due to the theoretical edge cases that can produce unexpected results. Noticed these inconsistencies while working on\n#27037\n.",
      "issue_closed_at": "2016-08-18T19:51:32",
      "base_commit": "13857b45ca54a519a58361238442af84262c0d23",
      "changes": [
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "value",
          "class_name": "BoundField",
          "code": "def value(self):\n        \"\"\"\n        Returns the value for this BoundField, using the initial value if\n        the form is not bound or the data otherwise.\n        \"\"\"\n        if not self.form.is_bound:\n            data = self.initial\n        else:\n            data = self.field.bound_data(\n                self.data, self.form.initial.get(self.name, self.field.initial)\n            )\n        return self.field.prepare_value(data)"
        },
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "id_for_label",
          "class_name": "BoundField",
          "code": "def id_for_label(self):\n        \"\"\"\n        Wrapper around the field widget's `id_for_label` method.\n        Useful, for example, for focusing on this field regardless of whether\n        it has a single widget or a MultiWidget.\n        \"\"\"\n        widget = self.field.widget\n        id_ = widget.attrs.get('id') or self.auto_id\n        return widget.id_for_label(id_)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "_clean_fields",
          "class_name": "BaseForm",
          "code": "def _clean_fields(self):\n        for name, field in self.fields.items():\n            # value_from_datadict() gets the data from the data dictionaries.\n            # Each widget type knows how to retrieve its own data, because some\n            # widgets split data over several HTML fields.\n            if field.disabled:\n                value = self.initial.get(name, field.initial)\n            else:\n                value = field.widget.value_from_datadict(self.data, self.files, self.add_prefix(name))\n            try:\n                if isinstance(field, FileField):\n                    initial = self.initial.get(name, field.initial)\n                    value = field.clean(value, initial)\n                else:\n                    value = field.clean(value)\n                self.cleaned_data[name] = value\n                if hasattr(self, 'clean_%s' % name):\n                    value = getattr(self, 'clean_%s' % name)()\n                    self.cleaned_data[name] = value\n            except ValidationError as e:\n                self.add_error(name, e)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "changed_data",
          "class_name": "BaseForm",
          "code": "def changed_data(self):\n        data = []\n        for name, field in self.fields.items():\n            prefixed_name = self.add_prefix(name)\n            data_value = field.widget.value_from_datadict(self.data, self.files, prefixed_name)\n            if not field.show_hidden_initial:\n                initial_value = self.initial.get(name, field.initial)\n                if callable(initial_value):\n                    initial_value = initial_value()\n            else:\n                initial_prefixed_name = self.add_initial_prefix(name)\n                hidden_widget = field.hidden_widget()\n                try:\n                    initial_value = field.to_python(hidden_widget.value_from_datadict(\n                        self.data, self.files, initial_prefixed_name))\n                except ValidationError:\n                    # Always assume data has changed if validation fails.\n                    data.append(name)\n                    continue\n            if field.has_changed(initial_value, data_value):\n                data.append(name)\n        return data"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "visible_fields",
          "class_name": "BaseForm",
          "code": "def visible_fields(self):\n        \"\"\"\n        Returns a list of BoundField objects that aren't hidden fields.\n        The opposite of the hidden_fields() method.\n        \"\"\"\n        return [field for field in self if not field.is_hidden]"
        }
      ]
    },
    {
      "pr_number": 10355,
      "pr_title": " Fixed #29723 -- Fixed crash if InlineModelAdmin.has_add_permission() doesn't accept the obj argument.",
      "pr_body": "https://code.djangoproject.com/ticket/29723",
      "issue_id": 29723,
      "issue_title": "Admin crashes if InlineModelAdmin.has_add_permission() doesn't accept the obj argument",
      "issue_body": "The release notes suggest that InlineModelAdmin.has_add_permission() methods that don’t accept obj as the second positional argument will be supported until Django 3.0:\nSupport for InlineModelAdmin.has_add_permission() methods that don’t accept obj as the second positional argument will be removed in Django 3.0.\nThis doesn't appear to be true in my experience.\nI have this method defined on an InlineModelAdmin:\ndef has_add_permission(self, request):\n        return False\nI'm getting this traceback with Django 2.1:\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/core/handlers/exception.py\" in inner\n  34.             response = get_response(request)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/core/handlers/base.py\" in _get_response\n  126.                 response = self.process_exception_by_middleware(e, request)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/core/handlers/base.py\" in _get_response\n  124.                 response = wrapped_callback(request, *callback_args, **callback_kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in wrapper\n  607.                 return self.admin_site.admin_view(view)(*args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/utils/decorators.py\" in _wrapped_view\n  142.                     response = view_func(request, *args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/views/decorators/cache.py\" in _wrapped_view_func\n  44.         response = view_func(request, *args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/sites.py\" in inner\n  223.             return view(request, *args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in add_view\n  1647.         return self.changeform_view(request, None, form_url, extra_context)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/utils/decorators.py\" in _wrapper\n  45.         return bound_method(*args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/utils/decorators.py\" in _wrapped_view\n  142.                     response = view_func(request, *args, **kwargs)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in changeform_view\n  1536.             return self._changeform_view(request, object_id, form_url, extra_context)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in _changeform_view\n  1590.                 formsets, inline_instances = self._create_formsets(request, form.instance, change=False)\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in _create_formsets\n  1945.         for FormSet, inline in self.get_formsets_with_inlines(*get_formsets_args):\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in get_formsets_with_inlines\n  795.             yield inline.get_formset(request, obj), inline\n\nFile \"/home/ubuntu/.pyenv/versions/3.6.2/envs/cupido/lib/python3.6/site-packages/django/contrib/admin/options.py\" in get_formset\n  2055.         can_add = self.has_add_permission(request, obj) if request else True\n\nException Type: TypeError at /admin/cupido/notification/add/\nException Value: has_add_permission() takes 2 positional arguments but 3 were given",
      "issue_closed_at": "2018-08-30T04:23:35",
      "base_commit": "54b331451cb22ee354beadf31ee42cbd714877f0",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "get_inline_instances",
          "class_name": "ModelAdmin",
          "code": "def get_inline_instances(self, request, obj=None):\n        inline_instances = []\n        for inline_class in self.inlines:\n            inline = inline_class(self.model, self.admin_site)\n            # RemovedInDjango30Warning: obj will be a required argument.\n            args = get_func_args(inline.has_add_permission)\n            if 'obj' in args:\n                inline_has_add_permission = inline.has_add_permission(request, obj)\n            else:\n                inline_has_add_permission = inline.has_add_permission(request)\n            if request:\n                if not (inline.has_view_or_change_permission(request, obj) or\n                        inline_has_add_permission or\n                        inline.has_delete_permission(request, obj)):\n                    continue\n                if not inline_has_add_permission:\n                    inline.max_num = 0\n            inline_instances.append(inline)\n\n        return inline_instances"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "get_inline_formsets",
          "class_name": "ModelAdmin",
          "code": "def get_inline_formsets(self, request, formsets, inline_instances, obj=None):\n        inline_admin_formsets = []\n        for inline, formset in zip(inline_instances, formsets):\n            fieldsets = list(inline.get_fieldsets(request, obj))\n            readonly = list(inline.get_readonly_fields(request, obj))\n            has_add_permission = inline.has_add_permission(request, obj)\n            has_change_permission = inline.has_change_permission(request, obj)\n            has_delete_permission = inline.has_delete_permission(request, obj)\n            has_view_permission = inline.has_view_permission(request, obj)\n            prepopulated = dict(inline.get_prepopulated_fields(request, obj))\n            inline_admin_formset = helpers.InlineAdminFormSet(\n                inline, formset, fieldsets, prepopulated, readonly, model_admin=self,\n                has_add_permission=has_add_permission, has_change_permission=has_change_permission,\n                has_delete_permission=has_delete_permission, has_view_permission=has_view_permission,\n            )\n            inline_admin_formsets.append(inline_admin_formset)\n        return inline_admin_formsets"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "media",
          "class_name": "InlineModelAdmin",
          "code": "def media(self):\n        extra = '' if settings.DEBUG else '.min'\n        js = ['vendor/jquery/jquery%s.js' % extra, 'jquery.init.js',\n              'inlines%s.js' % extra]\n        if self.filter_vertical or self.filter_horizontal:\n            js.extend(['SelectBox.js', 'SelectFilter2.js'])\n        if self.classes and 'collapse' in self.classes:\n            js.append('collapse%s.js' % extra)\n        return forms.Media(js=['admin/js/%s' % url for url in js])"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "get_formset",
          "class_name": "InlineModelAdmin",
          "code": "def get_formset(self, request, obj=None, **kwargs):\n        \"\"\"Return a BaseInlineFormSet class for use in admin add/change views.\"\"\"\n        if 'fields' in kwargs:\n            fields = kwargs.pop('fields')\n        else:\n            fields = flatten_fieldsets(self.get_fieldsets(request, obj))\n        excluded = self.get_exclude(request, obj)\n        exclude = [] if excluded is None else list(excluded)\n        exclude.extend(self.get_readonly_fields(request, obj))\n        if excluded is None and hasattr(self.form, '_meta') and self.form._meta.exclude:\n            # Take the custom ModelForm's Meta.exclude into account only if the\n            # InlineModelAdmin doesn't define its own.\n            exclude.extend(self.form._meta.exclude)\n        # If exclude is an empty list we use None, since that's the actual\n        # default.\n        exclude = exclude or None\n        can_delete = self.can_delete and self.has_delete_permission(request, obj)\n        defaults = {\n            'form': self.form,\n            'formset': self.formset,\n            'fk_name': self.fk_name,\n            'fields': fields,\n            'exclude': exclude,\n            'formfield_callback': partial(self.formfield_for_dbfield, request=request),\n            'extra': self.get_extra(request, obj, **kwargs),\n            'min_num': self.get_min_num(request, obj, **kwargs),\n            'max_num': self.get_max_num(request, obj, **kwargs),\n            'can_delete': can_delete,\n            **kwargs,\n        }\n\n        base_model_form = defaults['form']\n        can_change = self.has_change_permission(request, obj) if request else True\n        can_add = self.has_add_permission(request, obj) if request else True\n\n        class DeleteProtectedModelForm(base_model_form):\n\n            def hand_clean_DELETE(self):\n                \"\"\"\n                We don't validate the 'DELETE' field itself because on\n                templates it's not rendered using the field information, but\n                just using a generic \"deletion_field\" of the InlineModelAdmin.\n                \"\"\"\n                if self.cleaned_data.get(DELETION_FIELD_NAME, False):\n                    using = router.db_for_write(self._meta.model)\n                    collector = NestedObjects(using=using)\n                    if self.instance._state.adding:\n                        return\n                    collector.collect([self.instance])\n                    if collector.protected:\n                        objs = []\n                        for p in collector.protected:\n                            objs.append(\n                                # Translators: Model verbose name and instance representation,\n                                # suitable to be an item in a list.\n                                _('%(class_name)s %(instance)s') % {\n                                    'class_name': p._meta.verbose_name,\n                                    'instance': p}\n                            )\n                        params = {'class_name': self._meta.model._meta.verbose_name,\n                                  'instance': self.instance,\n                                  'related_objects': get_text_list(objs, _('and'))}\n                        msg = _(\"Deleting %(class_name)s %(instance)s would require \"\n                                \"deleting the following protected related objects: \"\n                                \"%(related_objects)s\")\n                        raise ValidationError(msg, code='deleting_protected', params=params)\n\n            def is_valid(self):\n                result = super().is_valid()\n                self.hand_clean_DELETE()\n                return result\n\n            def has_changed(self):\n                # Protect against unauthorized edits.\n                if not can_change and not self.instance._state.adding:\n                    return False\n                if not can_add and self.instance._state.adding:\n                    return False\n                return super().has_changed()\n\n        defaults['form'] = DeleteProtectedModelForm\n\n        if defaults['fields'] is None and not modelform_defines_fields(defaults['form']):\n            defaults['fields'] = forms.ALL_FIELDS\n\n        return inlineformset_factory(self.parent_model, self.model, **defaults)"
        }
      ]
    },
    {
      "pr_number": 7149,
      "pr_title": "Fixed #26816 -- Corrected an admin check to require inlines to subclass InlineModelAdmin.",
      "pr_body": "https://code.djangoproject.com/ticket/26816\n",
      "issue_id": 26816,
      "issue_title": "BaseModelAdminChecks._check_inlines_item may raise AttributeError",
      "issue_body": "given something like:\nclass MyInline(object): pass\n\nclass MyModelAdmin(admin.ModelAdmin):\n    inlines = [MyInline]\nadmin.E104\nwill be added to the error stack because its not a subclass (ugh) of BaseModelAdmin.\nchanging to:\nclass MyInline(BaseModelAdmin): pass\nwill throw:\nAttributeError: type object 'MyInline'  has no attribute 'model'\nbecause its not until\nModelAdmin\nor\nInlineModelAdmin\nthat the attribute is set (in\n__init__\nfor\nModelAdmin\n, as a class attribute for\nInlineModelAdmin\n)\nEncountered on 1.9, but looks to be the same in master.",
      "issue_closed_at": "2016-08-24T16:30:04",
      "base_commit": "a3abbe1fcb10e9ed4502c22748eb825170f41f23",
      "changes": [
        {
          "file": "django/contrib/admin/checks.py",
          "type": "function",
          "name": "_check_inlines_item",
          "class_name": "ModelAdminChecks",
          "code": "def _check_inlines_item(self, obj, model, inline, label):\n        \"\"\" Check one inline model admin. \"\"\"\n        inline_label = '.'.join([inline.__module__, inline.__name__])\n\n        from django.contrib.admin.options import BaseModelAdmin\n\n        if not issubclass(inline, BaseModelAdmin):\n            return [\n                checks.Error(\n                    \"'%s' must inherit from 'BaseModelAdmin'.\" % inline_label,\n                    obj=obj.__class__,\n                    id='admin.E104',\n                )\n            ]\n        elif not inline.model:\n            return [\n                checks.Error(\n                    \"'%s' must have a 'model' attribute.\" % inline_label,\n                    obj=obj.__class__,\n                    id='admin.E105',\n                )\n            ]\n        elif not issubclass(inline.model, models.Model):\n            return must_be('a Model', option='%s.model' % inline_label, obj=obj, id='admin.E106')\n        else:\n            return inline(model, obj.admin_site).check()"
        }
      ]
    },
    {
      "pr_number": 9861,
      "pr_title": "Fixed #29296 -- Fix handling of callable object views in admindocs",
      "pr_body": "See https://code.djangoproject.com/ticket/29296\r\n\r\nI'm not sure how you normally handle backports, but a backported version of this commit can be found here:\r\nhttps://github.com/PaulSD/django/commit/b2128a2afa14c64b39a5fae1f9a63b2959c4f752",
      "issue_id": 29296,
      "issue_title": "admindocs ViewIndexView crashes if a syndication Feed (or something without __qualname__) is configured",
      "issue_body": "If a Feed (\n​\nhttps://docs.djangoproject.com/en/2.0/ref/contrib/syndication/\n) is configured anywhere in the project, then /admin/doc/views/ (\n​\nhttps://docs.djangoproject.com/en/2.0/ref/contrib/admin/admindocs/\n) crashes with AttributeError:\n'GlobalFeed' object has no attribute '__qualname__'\nThe problem is that Feed is a callable object (\n​\nhttps://github.com/django/django/blob/master/django/contrib/syndication/views.py#L34\n) and the documentation uses that class directly as a View (\npath('latest/feed/', LatestEntriesFeed()),\n), but admindocs assumes that all views are functions and it does not work properly with a callable object (\n​\nhttps://github.com/django/django/blob/master/django/contrib/admindocs/views.py#L130\n).\nIn Django 1.11 and earlier, admindocs supported callable objects on Python 2 because it fell back to\nview.__class__.__name__\n, but it appears that the Python 3 code in admindocs has not supported this since it was added in\n​\nhttps://github.com/django/django/commit/ae0f55eb491255217d6df31296ec8102007224a6\n(\nhttps://code.djangoproject.com/ticket/27018\n).\nAs a work-around,\n__qualname__\ncan be manually defined on Feed objects:\nclass MyFeed(Feed):\n    def __init__(self):\n        super().__init__()\n        self.__qualname__ = self.__class__.__name__\nWithout knowing Django's opinion of the use of callable objects as views, I don't know the appropriate way to fix this: Should admindocs support callable objects as views, or should Feed not use a callable object as a view?",
      "issue_closed_at": "2018-04-12T12:34:31",
      "base_commit": "ee17bb8a67a9e7e688da6e6f4b3be1b3a69c09b0",
      "changes": [
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "line",
          "name": "line 2",
          "code": "from django.http import HttpResponse\nfrom django.utils.deprecation import MiddlewareMixin\n\n\nclass XViewMiddleware(MiddlewareMixin):\n    \"\"\""
        },
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "function",
          "name": "process_view",
          "class_name": "XViewMiddleware",
          "code": "def process_view(self, request, view_func, view_args, view_kwargs):\n        \"\"\"\n        If the request method is HEAD and either the IP is internal or the\n        user is a logged-in staff member, return a responsewith an x-view\n        header indicating the view function. This is used to lookup the view\n        function for an arbitrary page.\n        \"\"\"\n        assert hasattr(request, 'user'), (\n            \"The XView middleware requires authentication middleware to be \"\n            \"installed. Edit your MIDDLEWARE%s setting to insert \"\n            \"'django.contrib.auth.middleware.AuthenticationMiddleware'.\" % (\n                \"_CLASSES\" if settings.MIDDLEWARE is None else \"\"\n            )\n        )\n        if request.method == 'HEAD' and (request.META.get('REMOTE_ADDR') in settings.INTERNAL_IPS or\n                                         (request.user.is_active and request.user.is_staff)):\n            response = HttpResponse()\n            response['X-View'] = \"%s.%s\" % (view_func.__module__, view_func.__name__)\n            return response"
        },
        {
          "file": "django/contrib/admindocs/utils.py",
          "type": "line",
          "name": "line 18",
          "code": "    docutils_is_available = True\n\n\ndef trim_docstring(docstring):\n    \"\"\"\n    Uniformly trim leading/trailing whitespace from docstrings."
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "line",
          "name": "line 23",
          "code": "from django.utils.translation import gettext as _\nfrom django.views.generic import TemplateView\n\n# Exclude methods starting with these strings from documentation\nMODEL_METHODS_EXCLUDE = ('_', 'add_', 'delete', 'save', 'set_')\n"
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "function",
          "name": "get_context_data",
          "class_name": "TemplateDetailView",
          "code": "def get_context_data(self, **kwargs):\n        template = self.kwargs['template']\n        templates = []\n        try:\n            default_engine = Engine.get_default()\n        except ImproperlyConfigured:\n            # Non-trivial TEMPLATES settings aren't supported (#24125).\n            pass\n        else:\n            # This doesn't account for template loaders (#24128).\n            for index, directory in enumerate(default_engine.dirs):\n                template_file = os.path.join(directory, template)\n                if os.path.exists(template_file):\n                    with open(template_file) as f:\n                        template_contents = f.read()\n                else:\n                    template_contents = ''\n                templates.append({\n                    'file': template_file,\n                    'exists': os.path.exists(template_file),\n                    'contents': template_contents,\n                    'order': index,\n                })\n        return super().get_context_data(**{\n            **kwargs,\n            'name': template,\n            'templates': templates,\n        })"
        }
      ]
    }
  ]
}