{
    "Selected_candidate": {
        "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        })"
            }
        ]
    },
    "Justification": "Candidate E is the most helpful because it deals with issues related to callable objects and how they are managed in Django's admin documentation view, directly paralleling the changes sought in the CURRENT bug report regarding the access and handling of model classes in the custom admin view. Both reports share a context of modifying behavior in the admin interface, impacting how views and models are accessed and utilized. This alignment suggests that insights from Candidate E could provide useful patterns or fixes relevant to the CURRENT issue."
}