{
    "Selected_candidate": {
        "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)"
            }
        ]
    },
    "Justification": "Candidate B is notably relevant because it also involves the Django Admin, specifically concerning inline model administration. Both the CURRENT bug and Candidate B interact with the admin interface's functionalities, particularly around permissions and field handling in form submissions. The issues observed in both reports regarding unexpected behaviors in the admin view highlight potential parallels in debugging approaches, especially as they both stem from admin class setups. Fixing the addition permission method inconsistency could provide insights into similar inconsistencies encountered with verbose_name_plural behavior in the CURRENT bug report."
}