{
  "instance_id": "django__django-13230",
  "repo": "django/django",
  "created_at": "2020-07-23T14:59:50Z",
  "problem_statement": "Add support for item_comments to syndication framework\nDescription\n\t\nAdd comments argument to feed.add_item() in syndication.views so that item_comments can be defined directly without having to take the detour via item_extra_kwargs .\nAdditionally, comments is already explicitly mentioned in the feedparser, but not implemented in the view.\n",
  "patch": "diff --git a/django/contrib/syndication/views.py b/django/contrib/syndication/views.py\n--- a/django/contrib/syndication/views.py\n+++ b/django/contrib/syndication/views.py\n@@ -212,6 +212,7 @@ def get_feed(self, obj, request):\n                 author_name=author_name,\n                 author_email=author_email,\n                 author_link=author_link,\n+                comments=self._get_dynamic_attr('item_comments', item),\n                 categories=self._get_dynamic_attr('item_categories', item),\n                 item_copyright=self._get_dynamic_attr('item_copyright', item),\n                 **self.item_extra_kwargs(item)\n",
  "similar_bug_items": [
    {
      "pr_number": 12332,
      "pr_title": "Fixed #30439 - Added support for different plurals for the same language.",
      "pr_body": "ticket-30439",
      "issue_id": 30439,
      "issue_title": "Translations issues on Django upgrade due to unexpected changes in plural forms",
      "issue_body": "When locales have different plural forms, the ngettext can be easily broken because it uses plural equation from the first loaded gettext catalog for all of them.\nFor example in Czech locale, there are different plural equations being used even inside Django with either 3 or 4 plural forms. When the one with 4 forms is loaded first, Django is looking for non existing plural in catalogs.\nReproducer in Django 2.2.1:\n>>> from django.utils import translation\n>>> translation.activate('cs')\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 1)\n'Heslo je příliš krátké. Musí mít délku aspoň %(min_length)d znak.'\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 10)\n'This password is too short. It must contain at least %(min_length)d characters.'\nThe second invocation is trying to find 4th plural in 3 plurals catalog.",
      "issue_closed_at": "2020-03-10T09:56:53",
      "base_commit": "591e2270dc8c685625be25dbed908a9a3897ba1d",
      "changes": [
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "reset_cache",
          "class_name": null,
          "code": "def reset_cache(**kwargs):\n    \"\"\"\n    Reset global state when LANGUAGES setting has been changed, as some\n    languages should no longer be accepted.\n    \"\"\"\n    if kwargs['setting'] in ('LANGUAGES', 'LANGUAGE_CODE'):\n        check_for_language.cache_clear()\n        get_languages.cache_clear()\n        get_supported_language_variant.cache_clear()"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DjangoTranslation",
          "code": "def __init__(self, language, domain=None, localedirs=None):\n        \"\"\"Create a GNUTranslations() using many locale directories\"\"\"\n        gettext_module.GNUTranslations.__init__(self)\n        if domain is not None:\n            self.domain = domain\n\n        self.__language = language\n        self.__to_language = to_language(language)\n        self.__locale = to_locale(language)\n        self._catalog = None\n        # If a language doesn't have a catalog, use the Germanic default for\n        # pluralization: anything except one is pluralized.\n        self.plural = lambda n: int(n != 1)\n\n        if self.domain == 'django':\n            if localedirs is not None:\n                # A module-level cache is used for caching 'django' translations\n                warnings.warn(\"localedirs is ignored when domain is 'django'.\", RuntimeWarning)\n                localedirs = None\n            self._init_translation_catalog()\n\n        if localedirs:\n            for localedir in localedirs:\n                translation = self._new_gnu_trans(localedir)\n                self.merge(translation)\n        else:\n            self._add_installed_apps_translations()\n\n        self._add_local_translations()\n        if self.__language == settings.LANGUAGE_CODE and self.domain == 'django' and self._catalog is None:\n            # default lang should have at least one translation file available.\n            raise OSError('No translation files found for default language %s.' % settings.LANGUAGE_CODE)\n        self._add_fallback(localedirs)\n        if self._catalog is None:\n            # No catalogs found for this language, set an empty catalog.\n            self._catalog = {}"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "merge",
          "class_name": "DjangoTranslation",
          "code": "def merge(self, other):\n        \"\"\"Merge another translation into this catalog.\"\"\"\n        if not getattr(other, '_catalog', None):\n            return  # NullTranslations() has no _catalog\n        if self._catalog is None:\n            # Take plural and _info from first catalog found (generally Django's).\n            self.plural = other.plural\n            self._info = other._info.copy()\n            self._catalog = other._catalog.copy()\n        else:\n            self._catalog.update(other._catalog)\n        if other._fallback:\n            self.add_fallback(other._fallback)"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "to_language",
          "class_name": "DjangoTranslation",
          "code": "def to_language(self):\n        \"\"\"Return the translation language name.\"\"\"\n        return self.__to_language"
        }
      ]
    },
    {
      "pr_number": 11615,
      "pr_title": "Fixed #29019 -- Added ManyToManyField support to REQUIRED_FIELDS.",
      "pr_body": "[ticket 29019](https://code.djangoproject.com/ticket/29019)",
      "issue_id": 29019,
      "issue_title": "createsuperuser crashes if a ManyToManyField is in REQUIRED_FIELDS",
      "issue_body": "I've defined a custom user model with a ManyToMany field.\nWhen running\nmanage.py createsuperuser\nI receive the following error after entering the user's email address:\nTraceback (most recent call last):\n  File \"./manage.py\", line 22, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/core/management/__init__.py\", line 371, in execute_from_command_line\n    utility.execute()\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/core/management/__init__.py\", line 365, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/core/management/base.py\", line 288, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/contrib/auth/management/commands/createsuperuser.py\", line 59, in execute\n    return super().execute(*args, **options)\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/core/management/base.py\", line 335, in execute\n    output = self.handle(*args, **options)\n  File \"/Users/jkirsop/Development/artemis/venv/lib/python3.6/site-packages/django/contrib/auth/management/commands/createsuperuser.py\", line 133, in handle\n    ) if field.remote_field else '',\nAttributeError: 'ManyToManyRel' object has no attribute 'field_name'\nModels\nMy custom user model is defined as such (relevant pieces only included):\nclass OrgUser(AbstractBaseUser, PermissionsMixin):\n\temail = models.EmailField(\n\t\tverbose_name='email address',\n\t\tmax_length=255,\n\t\tunique=True,\n\t)\n\torgs = models.ManyToManyField(Organisation)\n\tUSERNAME_FIELD = 'email'\n\tREQUIRED_FIELDS = ['orgs']\n        objects = OrgUserManager()\nand Organisations\nclass Organisation(models.Model):\n\tname = models.CharField(max_length=60)\n\n\tdef __str__(self):\n\t\treturn self.name\nIt seems that if I remove the need for the Orgs to be a\nREQUIRED_FIELD\nthe issue goes away. However, it's central to my project and needs to be defined on every user.\nHappy to update the ticket with any other code snippets if required.",
      "issue_closed_at": "2019-08-26T08:17:38",
      "base_commit": "5dac63bb844d0a976e1dd1591a323c5ba9674a97",
      "changes": [
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "add_arguments",
          "class_name": "Command",
          "code": "def add_arguments(self, parser):\n        parser.add_argument(\n            '--%s' % self.UserModel.USERNAME_FIELD,\n            help='Specifies the login for the superuser.',\n        )\n        parser.add_argument(\n            '--noinput', '--no-input', action='store_false', dest='interactive',\n            help=(\n                'Tells Django to NOT prompt the user for input of any kind. '\n                'You must use --%s with --noinput, along with an option for '\n                'any other required field. Superusers created with --noinput will '\n                'not be able to log in until they\\'re given a valid password.' %\n                self.UserModel.USERNAME_FIELD\n            ),\n        )\n        parser.add_argument(\n            '--database',\n            default=DEFAULT_DB_ALIAS,\n            help='Specifies the database to use. Default is \"default\".',\n        )\n        for field in self.UserModel.REQUIRED_FIELDS:\n            parser.add_argument(\n                '--%s' % field,\n                help='Specifies the %s for the superuser.' % field,\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        username = options[self.UserModel.USERNAME_FIELD]\n        database = options['database']\n        user_data = {}\n        verbose_field_name = self.username_field.verbose_name\n        try:\n            self.UserModel._meta.get_field(PASSWORD_FIELD)\n        except exceptions.FieldDoesNotExist:\n            pass\n        else:\n            # If not provided, create the user with an unusable password.\n            user_data[PASSWORD_FIELD] = None\n        try:\n            if options['interactive']:\n                # Same as user_data but with foreign keys as fake model\n                # instances instead of raw IDs.\n                fake_user_data = {}\n                if hasattr(self.stdin, 'isatty') and not self.stdin.isatty():\n                    raise NotRunningInTTYException\n                default_username = get_default_username()\n                if username:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        self.stderr.write(error_msg)\n                        username = None\n                elif username == '':\n                    raise CommandError('%s cannot be blank.' % capfirst(verbose_field_name))\n                # Prompt for username.\n                while username is None:\n                    message = self._get_input_message(self.username_field, default_username)\n                    username = self.get_input_data(self.username_field, message, default_username)\n                    if username:\n                        error_msg = self._validate_username(username, verbose_field_name, database)\n                        if error_msg:\n                            self.stderr.write(error_msg)\n                            username = None\n                            continue\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                fake_user_data[self.UserModel.USERNAME_FIELD] = (\n                    self.username_field.remote_field.model(username)\n                    if self.username_field.remote_field else username\n                )\n                # Prompt for required fields.\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = options[field_name]\n                    while user_data[field_name] is None:\n                        message = self._get_input_message(field)\n                        input_value = self.get_input_data(field, message)\n                        user_data[field_name] = input_value\n                        fake_user_data[field_name] = input_value\n\n                        # Wrap any foreign keys in fake model instances\n                        if field.remote_field:\n                            fake_user_data[field_name] = field.remote_field.model(input_value)\n\n                # Prompt for a password if the model has one.\n                while PASSWORD_FIELD in user_data and user_data[PASSWORD_FIELD] is None:\n                    password = getpass.getpass()\n                    password2 = getpass.getpass('Password (again): ')\n                    if password != password2:\n                        self.stderr.write(\"Error: Your passwords didn't match.\")\n                        # Don't validate passwords that don't match.\n                        continue\n                    if password.strip() == '':\n                        self.stderr.write(\"Error: Blank passwords aren't allowed.\")\n                        # Don't validate blank passwords.\n                        continue\n                    try:\n                        validate_password(password2, self.UserModel(**fake_user_data))\n                    except exceptions.ValidationError as err:\n                        self.stderr.write('\\n'.join(err.messages))\n                        response = input('Bypass password validation and create user anyway? [y/N]: ')\n                        if response.lower() != 'y':\n                            continue\n                    user_data[PASSWORD_FIELD] = password\n            else:\n                # Non-interactive mode.\n                # Use password from environment variable, if provided.\n                if PASSWORD_FIELD in user_data and 'DJANGO_SUPERUSER_PASSWORD' in os.environ:\n                    user_data[PASSWORD_FIELD] = os.environ['DJANGO_SUPERUSER_PASSWORD']\n                # Use username from environment variable, if not provided in\n                # options.\n                if username is None:\n                    username = os.environ.get('DJANGO_SUPERUSER_' + self.UserModel.USERNAME_FIELD.upper())\n                if username is None:\n                    raise CommandError('You must use --%s with --noinput.' % self.UserModel.USERNAME_FIELD)\n                else:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        raise CommandError(error_msg)\n\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    env_var = 'DJANGO_SUPERUSER_' + field_name.upper()\n                    value = options[field_name] or os.environ.get(env_var)\n                    if not value:\n                        raise CommandError('You must use --%s with --noinput.' % field_name)\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = field.clean(value, None)\n\n            self.UserModel._default_manager.db_manager(database).create_superuser(**user_data)\n            if options['verbosity'] >= 1:\n                self.stdout.write(\"Superuser created successfully.\")\n        except KeyboardInterrupt:\n            self.stderr.write('\\nOperation cancelled.')\n            sys.exit(1)\n        except exceptions.ValidationError as e:\n            raise CommandError('; '.join(e.messages))\n        except NotRunningInTTYException:\n            self.stdout.write(\n                'Superuser creation skipped due to not running in a TTY. '\n                'You can run `manage.py createsuperuser` in your project '\n                'to create one manually.'\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        username = options[self.UserModel.USERNAME_FIELD]\n        database = options['database']\n        user_data = {}\n        verbose_field_name = self.username_field.verbose_name\n        try:\n            self.UserModel._meta.get_field(PASSWORD_FIELD)\n        except exceptions.FieldDoesNotExist:\n            pass\n        else:\n            # If not provided, create the user with an unusable password.\n            user_data[PASSWORD_FIELD] = None\n        try:\n            if options['interactive']:\n                # Same as user_data but with foreign keys as fake model\n                # instances instead of raw IDs.\n                fake_user_data = {}\n                if hasattr(self.stdin, 'isatty') and not self.stdin.isatty():\n                    raise NotRunningInTTYException\n                default_username = get_default_username()\n                if username:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        self.stderr.write(error_msg)\n                        username = None\n                elif username == '':\n                    raise CommandError('%s cannot be blank.' % capfirst(verbose_field_name))\n                # Prompt for username.\n                while username is None:\n                    message = self._get_input_message(self.username_field, default_username)\n                    username = self.get_input_data(self.username_field, message, default_username)\n                    if username:\n                        error_msg = self._validate_username(username, verbose_field_name, database)\n                        if error_msg:\n                            self.stderr.write(error_msg)\n                            username = None\n                            continue\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                fake_user_data[self.UserModel.USERNAME_FIELD] = (\n                    self.username_field.remote_field.model(username)\n                    if self.username_field.remote_field else username\n                )\n                # Prompt for required fields.\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = options[field_name]\n                    while user_data[field_name] is None:\n                        message = self._get_input_message(field)\n                        input_value = self.get_input_data(field, message)\n                        user_data[field_name] = input_value\n                        fake_user_data[field_name] = input_value\n\n                        # Wrap any foreign keys in fake model instances\n                        if field.remote_field:\n                            fake_user_data[field_name] = field.remote_field.model(input_value)\n\n                # Prompt for a password if the model has one.\n                while PASSWORD_FIELD in user_data and user_data[PASSWORD_FIELD] is None:\n                    password = getpass.getpass()\n                    password2 = getpass.getpass('Password (again): ')\n                    if password != password2:\n                        self.stderr.write(\"Error: Your passwords didn't match.\")\n                        # Don't validate passwords that don't match.\n                        continue\n                    if password.strip() == '':\n                        self.stderr.write(\"Error: Blank passwords aren't allowed.\")\n                        # Don't validate blank passwords.\n                        continue\n                    try:\n                        validate_password(password2, self.UserModel(**fake_user_data))\n                    except exceptions.ValidationError as err:\n                        self.stderr.write('\\n'.join(err.messages))\n                        response = input('Bypass password validation and create user anyway? [y/N]: ')\n                        if response.lower() != 'y':\n                            continue\n                    user_data[PASSWORD_FIELD] = password\n            else:\n                # Non-interactive mode.\n                # Use password from environment variable, if provided.\n                if PASSWORD_FIELD in user_data and 'DJANGO_SUPERUSER_PASSWORD' in os.environ:\n                    user_data[PASSWORD_FIELD] = os.environ['DJANGO_SUPERUSER_PASSWORD']\n                # Use username from environment variable, if not provided in\n                # options.\n                if username is None:\n                    username = os.environ.get('DJANGO_SUPERUSER_' + self.UserModel.USERNAME_FIELD.upper())\n                if username is None:\n                    raise CommandError('You must use --%s with --noinput.' % self.UserModel.USERNAME_FIELD)\n                else:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        raise CommandError(error_msg)\n\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    env_var = 'DJANGO_SUPERUSER_' + field_name.upper()\n                    value = options[field_name] or os.environ.get(env_var)\n                    if not value:\n                        raise CommandError('You must use --%s with --noinput.' % field_name)\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = field.clean(value, None)\n\n            self.UserModel._default_manager.db_manager(database).create_superuser(**user_data)\n            if options['verbosity'] >= 1:\n                self.stdout.write(\"Superuser created successfully.\")\n        except KeyboardInterrupt:\n            self.stderr.write('\\nOperation cancelled.')\n            sys.exit(1)\n        except exceptions.ValidationError as e:\n            raise CommandError('; '.join(e.messages))\n        except NotRunningInTTYException:\n            self.stdout.write(\n                'Superuser creation skipped due to not running in a TTY. '\n                'You can run `manage.py createsuperuser` in your project '\n                'to create one manually.'\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "_get_input_message",
          "class_name": "Command",
          "code": "def _get_input_message(self, field, default=None):\n        return '%s%s%s: ' % (\n            capfirst(field.verbose_name),\n            \" (leave blank to use '%s')\" % default if default else '',\n            ' (%s.%s)' % (\n                field.remote_field.model._meta.object_name,\n                field.remote_field.field_name,\n            ) if field.remote_field else '',\n        )"
        }
      ]
    },
    {
      "pr_number": 9920,
      "pr_title": "Fixed #28462 -- Decreased memory usage with ModelAdmin.list_editable.",
      "pr_body": "New PR related to https://github.com/django/django/pull/9820#issuecomment-386203216\r\n\r\nBased on the bug https://code.djangoproject.com/ticket/28462",
      "issue_id": 28462,
      "issue_title": "ModelAdmin.list_editable unusably slow and memory intensive with large datasets",
      "issue_body": "Since 1.10\nlist_editable\non\nModelAdmin\nis unusable for Models with a large-ish dataset.\nThe problem is caused by a recent change to how the FormSet is generated for the admin. Previously it was generated from the ChangeList result list, but it has been changed to use the admin's\nget_queryset\nwhich will return more than the current \"pageful\" of results (potentially the entire dataset) causing Django to generate a form for each instance. This results in Django consuming all available RAM and in some cases the python instance crashing. My personal laptop became unresponsive and I had to force power off.\nSee\n​\nhttps://github.com/django/django/commit/917cc288a38f3c114a5440f0749b7e5e1086eb36#commitcomment-23412084",
      "issue_closed_at": "2018-06-01T10:00:25",
      "base_commit": "e1ebd22558342bb0088a3da3571863a20413fa2a",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "line",
          "name": "line 1",
          "code": "import copy\nimport json\nimport operator\nfrom collections import OrderedDict\nfrom functools import partial, reduce, update_wrapper\nfrom urllib.parse import quote as urlquote"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "add_view",
          "class_name": "ModelAdmin",
          "code": "def add_view(self, request, form_url='', extra_context=None):\n        return self.changeform_view(request, None, form_url, extra_context)"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "changelist_view",
          "class_name": "ModelAdmin",
          "code": "def changelist_view(self, request, extra_context=None):\n        \"\"\"\n        The 'change list' admin view for this model.\n        \"\"\"\n        from django.contrib.admin.views.main import ERROR_FLAG\n        opts = self.model._meta\n        app_label = opts.app_label\n        if not self.has_view_permission(request) and not self.has_change_permission(request):\n            raise PermissionDenied\n\n        try:\n            cl = self.get_changelist_instance(request)\n        except IncorrectLookupParameters:\n            # Wacky lookup parameters were given, so redirect to the main\n            # changelist page, without parameters, and pass an 'invalid=1'\n            # parameter via the query string. If wacky parameters were given\n            # and the 'invalid=1' parameter was already in the query string,\n            # something is screwed up with the database, so display an error\n            # page.\n            if ERROR_FLAG in request.GET:\n                return SimpleTemplateResponse('admin/invalid_setup.html', {\n                    'title': _('Database error'),\n                })\n            return HttpResponseRedirect(request.path + '?' + ERROR_FLAG + '=1')\n\n        # If the request was POSTed, this might be a bulk action or a bulk\n        # edit. Try to look up an action or confirmation first, but if this\n        # isn't an action the POST will fall through to the bulk edit check,\n        # below.\n        action_failed = False\n        selected = request.POST.getlist(helpers.ACTION_CHECKBOX_NAME)\n\n        actions = self.get_actions(request)\n        # Actions with no confirmation\n        if (actions and request.method == 'POST' and\n                'index' in request.POST and '_save' not in request.POST):\n            if not self.has_change_permission(request):\n                raise PermissionDenied\n            if selected:\n                response = self.response_action(request, queryset=cl.get_queryset(request))\n                if response:\n                    return response\n                else:\n                    action_failed = True\n            else:\n                msg = _(\"Items must be selected in order to perform \"\n                        \"actions on them. No items have been changed.\")\n                self.message_user(request, msg, messages.WARNING)\n                action_failed = True\n\n        # Actions with confirmation\n        if (actions and request.method == 'POST' and\n                helpers.ACTION_CHECKBOX_NAME in request.POST and\n                'index' not in request.POST and '_save' not in request.POST):\n            if not self.has_change_permission(request):\n                raise PermissionDenied\n            if selected:\n                response = self.response_action(request, queryset=cl.get_queryset(request))\n                if response:\n                    return response\n                else:\n                    action_failed = True\n\n        if action_failed:\n            # Redirect back to the changelist page to avoid resubmitting the\n            # form if the user refreshes the browser or uses the \"No, take\n            # me back\" button on the action confirmation page.\n            return HttpResponseRedirect(request.get_full_path())\n\n        # If we're allowing changelist editing, we need to construct a formset\n        # for the changelist given all the fields to be edited. Then we'll\n        # use the formset to validate/process POSTed data.\n        formset = cl.formset = None\n\n        # Handle POSTed bulk-edit data.\n        if request.method == 'POST' and cl.list_editable and '_save' in request.POST:\n            if not self.has_change_permission(request):\n                raise PermissionDenied\n            FormSet = self.get_changelist_formset(request)\n            formset = cl.formset = FormSet(request.POST, request.FILES, queryset=self.get_queryset(request))\n            if formset.is_valid():\n                changecount = 0\n                for form in formset.forms:\n                    if form.has_changed():\n                        obj = self.save_form(request, form, change=True)\n                        self.save_model(request, obj, form, change=True)\n                        self.save_related(request, form, formsets=[], change=True)\n                        change_msg = self.construct_change_message(request, form, None)\n                        self.log_change(request, obj, change_msg)\n                        changecount += 1\n\n                if changecount:\n                    msg = ngettext(\n                        \"%(count)s %(name)s was changed successfully.\",\n                        \"%(count)s %(name)s were changed successfully.\",\n                        changecount\n                    ) % {\n                        'count': changecount,\n                        'name': model_ngettext(opts, changecount),\n                    }\n                    self.message_user(request, msg, messages.SUCCESS)\n\n                return HttpResponseRedirect(request.get_full_path())\n\n        # Handle GET -- construct a formset for display.\n        elif cl.list_editable and self.has_change_permission(request):\n            FormSet = self.get_changelist_formset(request)\n            formset = cl.formset = FormSet(queryset=cl.result_list)\n\n        # Build the list of media to be used by the formset.\n        if formset:\n            media = self.media + formset.media\n        else:\n            media = self.media\n\n        # Build the action form and populate it with available actions.\n        if actions:\n            action_form = self.action_form(auto_id=None)\n            action_form.fields['action'].choices = self.get_action_choices(request)\n            media += action_form.media\n        else:\n            action_form = None\n\n        selection_note_all = ngettext(\n            '%(total_count)s selected',\n            'All %(total_count)s selected',\n            cl.result_count\n        )\n\n        context = {\n            **self.admin_site.each_context(request),\n            'module_name': str(opts.verbose_name_plural),\n            'selection_note': _('0 of %(cnt)s selected') % {'cnt': len(cl.result_list)},\n            'selection_note_all': selection_note_all % {'total_count': cl.result_count},\n            'title': cl.title,\n            'is_popup': cl.is_popup,\n            'to_field': cl.to_field,\n            'cl': cl,\n            'media': media,\n            'has_add_permission': self.has_add_permission(request),\n            'opts': cl.opts,\n            'action_form': action_form,\n            'actions_on_top': self.actions_on_top,\n            'actions_on_bottom': self.actions_on_bottom,\n            'actions_selection_counter': self.actions_selection_counter,\n            'preserved_filters': self.get_preserved_filters(request),\n            **(extra_context or {}),\n        }\n\n        request.current_app = self.admin_site.name\n\n        return TemplateResponse(request, self.change_list_template or [\n            'admin/%s/%s/change_list.html' % (app_label, opts.model_name),\n            'admin/%s/change_list.html' % app_label,\n            'admin/change_list.html'\n        ], context)"
        }
      ]
    },
    {
      "pr_number": 10260,
      "pr_title": "Fixed #29613 -- Fixed --keepdb on PostgreSQL if the database exists and the user can't create databases.",
      "pr_body": "Ticket [29613](https://code.djangoproject.com/ticket/29613).",
      "issue_id": 29613,
      "issue_title": "Allow --keepdb to work on PostgreSQL if the database exists but the user can't create databases",
      "issue_body": "The popular Web Faction hosting service uses a shared database server. Users can create databases using the web UI or XML RPC calls, but not using the SQL CREATE syntax.\nRunning tests throws a ProgrammingError, with the message 'permission denied to create database', even if the test database has been previously created manually.\nThe error code for this error is '42501', which appears to correspond to errorcodes.INSUFFICIENT_PRIVILEGE.\ndjango/db/backends/postgresql/creation.py only handles the error errorcodes.DUPLICATE_DATABASE in _execute_create_test_db(), line 35. Because the error code does not match the program exits with a log message. But it would be fine to proceed with the error code '42501' also, making use of the --keepdb mechanism.\nThis appears to be a regression, as I did not experience this issue either using postgresql_psycopg2 driver or using Django 1.11",
      "issue_closed_at": "2018-08-03T03:32:30",
      "base_commit": "d8e2be459f97f1773c7edf7d37de180139146176",
      "changes": [
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "line",
          "name": "line 3",
          "code": "from psycopg2 import errorcodes\n\nfrom django.db.backends.base.creation import BaseDatabaseCreation\n\n\nclass DatabaseCreation(BaseDatabaseCreation):"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "sql_table_creation_suffix",
          "class_name": "DatabaseCreation",
          "code": "def sql_table_creation_suffix(self):\n        test_settings = self.connection.settings_dict['TEST']\n        assert test_settings['COLLATION'] is None, (\n            \"PostgreSQL does not support collation setting at database creation time.\"\n        )\n        return self._get_database_create_suffix(\n            encoding=test_settings['CHARSET'],\n            template=test_settings.get('TEMPLATE'),\n        )"
        }
      ]
    },
    {
      "pr_number": 10985,
      "pr_title": "Fixed #30181 -- Made cache.get() with default work correctly on PyLibMCCache if None is cached.",
      "pr_body": "This PR fixes https://code.djangoproject.com/ticket/30181",
      "issue_id": 30181,
      "issue_title": "Fix cache.get() with default on PyLibMCCache if None is cached",
      "issue_body": "If\nNone\nis saved to the cache, the\ndefault\nkeyword argument in\ncache.get()\nshadows it if using any Memcached backend. This is not consistent with LocMemCache.\nThe issue can be very easily fixed within\nPyLibMCCache\n, as it supports retrieving a default value if key is not present.\nFor\nMemcachedCache\n, unfortunatelly, this is not possible because of\npython-memcached\nlimitations, although I've opened up an issue and PR to the upstream package maintainer that would add functionality that would allow this in\nMemcachedCache\nas well:\n​\nhttps://github.com/linsomniac/python-memcached/issues/159\nI will fix this issue for\nPyLibMCCache\nfor now, and once (if) the upstream PR for\npython-memcached\ngets approved, merged and released, I'll also submit a PR that would fix this issue for\nMemcachedCache\n(although that would require a minimum version of\npython-memcached\n, so that one would probably be a little more complex).",
      "issue_closed_at": "2019-02-14T18:58:15",
      "base_commit": "741ce81a426e4d0cd7581d98d3b8e3290f513f09",
      "changes": [
        {
          "file": "django/core/cache/backends/memcached.py",
          "type": "function",
          "name": "add",
          "class_name": "BaseMemcachedCache",
          "code": "def add(self, key, value, timeout=DEFAULT_TIMEOUT, version=None):\n        key = self.make_key(key, version=version)\n        return self._cache.add(key, value, self.get_backend_timeout(timeout))"
        },
        {
          "file": "django/core/cache/backends/memcached.py",
          "type": "function",
          "name": "touch",
          "class_name": "PyLibMCCache",
          "code": "def touch(self, key, timeout=DEFAULT_TIMEOUT, version=None):\n        key = self.make_key(key, version=version)\n        if timeout == 0:\n            return self._cache.delete(key)\n        return self._cache.touch(key, self.get_backend_timeout(timeout))"
        }
      ]
    }
  ]
}