{
  "instance_id": "django__django-16139",
  "repo": "django/django",
  "created_at": "2022-09-30T08:51:16Z",
  "problem_statement": "Accessing UserAdmin via to_field leads to link to PasswordResetForm being broken (404)\nDescription\n\t \n\t\t(last modified by Simon Kern)\n\t \nAccessing the UserAdmin via another model's Admin that has a reference to User (with to_field set, e.g., to_field=\"uuid\") leads to the UserAdmin being accessed via an url that looks similar to this one:\n.../user/22222222-3333-4444-5555-666677778888/change/?_to_field=uuid\nHowever the underlying form looks like this: \nCode highlighting:\nclass UserChangeForm(forms.ModelForm):\n\tpassword = ReadOnlyPasswordHashField(\n\t\tlabel=_(\"Password\"),\n\t\thelp_text=_(\n\t\t\t\"Raw passwords are not stored, so there is no way to see this \"\n\t\t\t\"user’s password, but you can change the password using \"\n\t\t\t'<a href=\"{}\">this form</a>.'\n\t\t),\n\t)\n\t...\n\t...\n\tdef __init__(self, *args, **kwargs):\n\t\tsuper().__init__(*args, **kwargs)\n\t\tpassword = self.fields.get(\"password\")\n\t\tif password:\n\t\t\tpassword.help_text = password.help_text.format(\"../password/\")\n\t...\n\t...\nThis results in the link to the PasswordResetForm being wrong and thus ending up in a 404. If we drop the assumption that UserAdmin is always accessed via its pk, then we're good to go. It's as simple as replacing password.help_text = password.help_text.format(\"../password/\") with password.help_text = password.help_text.format(f\"../../{self.instance.pk}/password/\")\nI've opened a pull request on GitHub for this Ticket, please see:\n​PR\n",
  "patch": "diff --git a/django/contrib/auth/forms.py b/django/contrib/auth/forms.py\n--- a/django/contrib/auth/forms.py\n+++ b/django/contrib/auth/forms.py\n@@ -163,7 +163,9 @@ def __init__(self, *args, **kwargs):\n         super().__init__(*args, **kwargs)\n         password = self.fields.get(\"password\")\n         if password:\n-            password.help_text = password.help_text.format(\"../password/\")\n+            password.help_text = password.help_text.format(\n+                f\"../../{self.instance.pk}/password/\"\n+            )\n         user_permissions = self.fields.get(\"user_permissions\")\n         if user_permissions:\n             user_permissions.queryset = user_permissions.queryset.select_related(\n",
  "similar_bug_items": [
    {
      "pr_number": 5484,
      "pr_title": "Fixed #25596 -- Fixed regression in password change view with custom user model.",
      "pr_body": "https://code.djangoproject.com/ticket/25596\n",
      "issue_id": 25596,
      "issue_title": "Can't change user's password in admin when using custom User model",
      "issue_body": "Django 1.9b1\nI'm using custom User model which is defined as:\nAUTH_USER_MODEL = 'users.User'\n\nINSTALLED_APPS = [\n    'django.contrib.admin',\n    'django.contrib.auth',\n    ...\n    'apps.users',\n]\nWhen I tried to change user's password (using /admin/users/user/ID/password/) I've got an error:\nTraceback:\nFile \"/src/django/django/core/handlers/base.py\" in get_response\n  149.                     response = self.process_exception_by_middleware(e, request)\n\nFile \"/src/django/django/core/handlers/base.py\" in get_response\n  147.                     response = wrapped_callback(request, *callback_args, **callback_kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in _wrapped_view\n  149.                     response = view_func(request, *args, **kwargs)\n\nFile \"/src/django/django/views/decorators/cache.py\" in _wrapped_view_func\n  57.         response = view_func(request, *args, **kwargs)\n\nFile \"/src/django/django/contrib/admin/sites.py\" in inner\n  244.             return view(request, *args, **kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in _wrapper\n  67.             return bound_func(*args, **kwargs)\n\nFile \"/src/django/django/views/decorators/debug.py\" in sensitive_post_parameters_wrapper\n  76.             return view(request, *args, **kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in bound_func\n  63.                 return func.__get__(self, type(self))(*args2, **kwargs2)\n\nFile \"/src/django/django/contrib/auth/admin.py\" in user_change_password\n  155.                         args=(user.pk,),\n\nFile \"/src/django/django/core/urlresolvers.py\" in reverse\n  600.     return force_text(iri_to_uri(resolver._reverse_with_prefix(view, prefix, *args, **kwargs)))\n\nFile \"/src/django/django/core/urlresolvers.py\" in _reverse_with_prefix\n  508.                              (lookup_view_s, args, kwargs, len(patterns), patterns))\n\nException Type: NoReverseMatch at /panel/users/user/8/password/\nException Value: Reverse for 'auth_user_change' with arguments '(8,)' and keyword arguments '{}' not found. 0 pattern(s) tried: []\ndjango/auth/admin.py:151\nreverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\nThere should not be fixed \"auth_\" prefix, but something like user._meta.app_name(?)",
      "issue_closed_at": "2015-10-27T07:38:10",
      "base_commit": "1f07da3e29c7c3d47968e1c4531dd9bf902575b7",
      "changes": [
        {
          "file": "django/contrib/auth/admin.py",
          "type": "function",
          "name": "user_change_password",
          "class_name": "UserAdmin",
          "code": "def user_change_password(self, request, id, form_url=''):\n        if not self.has_change_permission(request):\n            raise PermissionDenied\n        user = self.get_object(request, unquote(id))\n        if user is None:\n            raise Http404(_('%(name)s object with primary key %(key)r does not exist.') % {\n                'name': force_text(self.model._meta.verbose_name),\n                'key': escape(id),\n            })\n        if request.method == 'POST':\n            form = self.change_password_form(user, request.POST)\n            if form.is_valid():\n                form.save()\n                change_message = self.construct_change_message(request, form, None)\n                self.log_change(request, user, change_message)\n                msg = ugettext('Password changed successfully.')\n                messages.success(request, msg)\n                update_session_auth_hash(request, form.user)\n                return HttpResponseRedirect(\n                    reverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\n                )\n        else:\n            form = self.change_password_form(user)\n\n        fieldsets = [(None, {'fields': list(form.base_fields)})]\n        adminForm = admin.helpers.AdminForm(form, fieldsets, {})\n\n        context = {\n            'title': _('Change password: %s') % escape(user.get_username()),\n            'adminForm': adminForm,\n            'form_url': form_url,\n            'form': form,\n            'is_popup': (IS_POPUP_VAR in request.POST or\n                         IS_POPUP_VAR in request.GET),\n            'add': True,\n            'change': False,\n            'has_delete_permission': False,\n            'has_change_permission': True,\n            'has_absolute_url': False,\n            'opts': self.model._meta,\n            'original': user,\n            'save_as': False,\n            'show_save': True,\n        }\n        context.update(admin.site.each_context(request))\n\n        request.current_app = self.admin_site.name\n\n        return TemplateResponse(request,\n            self.change_user_password_template or\n            'admin/auth/user/change_password.html',\n            context)"
        },
        {
          "file": "django/contrib/auth/admin.py",
          "type": "function",
          "name": "user_change_password",
          "class_name": "UserAdmin",
          "code": "def user_change_password(self, request, id, form_url=''):\n        if not self.has_change_permission(request):\n            raise PermissionDenied\n        user = self.get_object(request, unquote(id))\n        if user is None:\n            raise Http404(_('%(name)s object with primary key %(key)r does not exist.') % {\n                'name': force_text(self.model._meta.verbose_name),\n                'key': escape(id),\n            })\n        if request.method == 'POST':\n            form = self.change_password_form(user, request.POST)\n            if form.is_valid():\n                form.save()\n                change_message = self.construct_change_message(request, form, None)\n                self.log_change(request, user, change_message)\n                msg = ugettext('Password changed successfully.')\n                messages.success(request, msg)\n                update_session_auth_hash(request, form.user)\n                return HttpResponseRedirect(\n                    reverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\n                )\n        else:\n            form = self.change_password_form(user)\n\n        fieldsets = [(None, {'fields': list(form.base_fields)})]\n        adminForm = admin.helpers.AdminForm(form, fieldsets, {})\n\n        context = {\n            'title': _('Change password: %s') % escape(user.get_username()),\n            'adminForm': adminForm,\n            'form_url': form_url,\n            'form': form,\n            'is_popup': (IS_POPUP_VAR in request.POST or\n                         IS_POPUP_VAR in request.GET),\n            'add': True,\n            'change': False,\n            'has_delete_permission': False,\n            'has_change_permission': True,\n            'has_absolute_url': False,\n            'opts': self.model._meta,\n            'original': user,\n            'save_as': False,\n            'show_save': True,\n        }\n        context.update(admin.site.each_context(request))\n\n        request.current_app = self.admin_site.name\n\n        return TemplateResponse(request,\n            self.change_user_password_template or\n            'admin/auth/user/change_password.html',\n            context)"
        }
      ]
    },
    {
      "pr_number": 13982,
      "pr_title": "Fixed #32425 -- Fixed adding nullable field with default on MySQL.",
      "pr_body": "ticket link: https://code.djangoproject.com/ticket/32425#ticket\r\n\r\nHi, team. I wanna fix some picky issue. \r\nI suggest below code for solving this problem. (same code creates different schemas)\r\n\r\nbut It has duplicate code with base/schema.py _alter_column_default_sql.\r\n\r\nWe can do like this. but it need to one more SQL. \r\nFor mysql users, performance is not good, but better than schema changes.\r\n\r\nmysql/schema.py\r\n\r\n```\r\ndef _alter_column_default_sql(self, model, old_field, new_field, drop=False):\r\n      super()._alter_column_default_sql(model, old_field, new_field, drop)\r\n\r\n      if drop and new_field.null:\r\n          new_db_params = new_field.db_parameters(connection=self.connection)\r\n          new_default = self.effective_default(new_field)\r\n          default = self._column_default_sql(new_field)\r\n          params = [new_default]\r\n          sql = self.sql_alter_column_default_null\r\n          return (\r\n              sql % {\r\n                  'column': self.quote_name(new_field.column),\r\n                  'type': new_db_params['type'],\r\n                  'default': default,\r\n              },\r\n              params,\r\n          )\r\n\r\n```",
      "issue_id": 32425,
      "issue_title": "MySQL Schema is different about the same class definitions. (depends on create table vs alter table)",
      "issue_body": "Hi, My name is Jordan. When I delete the column, I found some picky points in Django.\nMySQL Schema is different about the same class definitions.\nMySQL Schemas are different about the nullable column default value.\nFor example,\n1) when it was made by 'create table'\nclass PhoneBook(models.Model):\n    name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nThe above code creates the migrations file as shown below.\noperations = [\n        migrations.CreateModel(\n            name='PhoneBook',\n            fields=[\n                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n                ('name', models.CharField(blank=True, default='jordan', max_length=32, null=True)),\n                ('phone_number', models.CharField(blank=True, max_length=32, null=True)),\n            ],\n        ),\n]\nWhen Migrate, SQL is executed as shown below.\nCREATE TABLE `main_phonebook` (\n\t`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,\n\t`name` varchar(32) NULL,\n\t`phone_number` varchar(32) NULL\n)ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n2) when it was made by 'alter table'\nFor the situation where the ‘alter table’ is applied, let's define the model class first as below and then add the name column.\nclass PhoneBook2(models.Model):\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nand I added 'name' column.\nclass PhoneBook2(models.Model):\n    name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\noperations = [\n        migrations.AddField(\n            model_name='phonebook2',\n            name='name',\n            field=models.CharField(blank=True, default='jordan', max_length=32, null=True),\n        ),\n    ]\nthis operations make this SQL.\nALTER TABLE `main_phonebook2` ADD COLUMN `name` varchar(32) DEFAULT %s NULL ['jordan']\nALTER TABLE `main_phonebook2` ALTER COLUMN `name` DROP DEFAULT []\nand table schema is like below.\nCREATE TABLE `main_phonebook2` (\n  `id` int(11) NOT NULL AUTO_INCREMENT,\n  `phone_number` varchar(32) DEFAULT NULL,\n  `name` varchar(32),\n  PRIMARY KEY (`id`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\nWe can see that the same code creates different schemas.\n+ when i try to remove the column\nclass PhoneBook(models.Model):\n    # name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\n\n\nclass PhoneBook2(models.Model):\n    # name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nhappened like this.\nIn [2]: p=PhoneBook(phone_number='010-1234-1234')                         \n\nIn [3]: p.save()                                                          \nINSERT INTO `main_phonebook` (`phone_number`) VALUES (%s) ['010-1234-1234']\n\nIn [4]: p2=PhoneBook2(phone_number='010-1234-1234')                       \n\nIn [5]: p2.save()                                                         \nINSERT INTO `main_phonebook2` (`phone_number`) VALUES (%s) ['010-1234-1234']\n\n!!!!!!!!error!!!!!!!!!!!!!!!\nIntegrityError: (1364, \"Field 'name' doesn't have a default value\")\nI made PR for fixing this.\n​\nhttps://github.com/django/django/pull/13982",
      "issue_closed_at": "2021-02-09T01:25:39",
      "base_commit": "b99d6c9cbc8eecf480892599201eef0d14b20d71",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_alter_column_default_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _alter_column_default_sql(self, model, old_field, new_field, drop=False):\n        \"\"\"\n        Hook to specialize column default alteration.\n\n        Return a (sql, params) fragment to add or drop (depending on the drop\n        argument) a default to new_field's column.\n        \"\"\"\n        new_default = self.effective_default(new_field)\n        default = self._column_default_sql(new_field)\n        params = [new_default]\n\n        if drop:\n            params = []\n        elif self.connection.features.requires_literal_defaults:\n            # Some databases (Oracle) can't take defaults as a parameter\n            # If this is the case, the SchemaEditor for that database should\n            # implement prepare_default().\n            default = self.prepare_default(new_default)\n            params = []\n\n        new_db_params = new_field.db_parameters(connection=self.connection)\n        sql = self.sql_alter_column_no_default if drop else self.sql_alter_column_default\n        return (\n            sql % {\n                'column': self.quote_name(new_field.column),\n                'type': new_db_params['type'],\n                'default': default,\n            },\n            params,\n        )"
        },
        {
          "file": "django/db/backends/mysql/schema.py",
          "type": "class",
          "name": "DatabaseSchemaEditor",
          "code": "class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):\n\n    sql_rename_table = \"RENAME TABLE %(old_table)s TO %(new_table)s\"\n\n    sql_alter_column_null = \"MODIFY %(column)s %(type)s NULL\"\n    sql_alter_column_not_null = \"MODIFY %(column)s %(type)s NOT NULL\"\n    sql_alter_column_type = \"MODIFY %(column)s %(type)s\"\n    sql_alter_column_collate = \"MODIFY %(column)s %(type)s%(collation)s\"\n\n    # No 'CASCADE' which works as a no-op in MySQL but is undocumented\n    sql_delete_column = \"ALTER TABLE %(table)s DROP COLUMN %(column)s\"\n\n    sql_delete_unique = \"ALTER TABLE %(table)s DROP INDEX %(name)s\"\n    sql_create_column_inline_fk = (\n        ', ADD CONSTRAINT %(name)s FOREIGN KEY (%(column)s) '\n        'REFERENCES %(to_table)s(%(to_column)s)'\n    )\n    sql_delete_fk = \"ALTER TABLE %(table)s DROP FOREIGN KEY %(name)s\"\n\n    sql_delete_index = \"DROP INDEX %(name)s ON %(table)s\"\n\n    sql_create_pk = \"ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)\"\n    sql_delete_pk = \"ALTER TABLE %(table)s DROP PRIMARY KEY\"\n\n    sql_create_index = 'CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s'\n\n    @property\n    def sql_delete_check(self):\n        if self.connection.mysql_is_mariadb:\n            # The name of the column check constraint is the same as the field\n            # name on MariaDB. Adding IF EXISTS clause prevents migrations\n            # crash. Constraint is removed during a \"MODIFY\" column statement.\n            return 'ALTER TABLE %(table)s DROP CONSTRAINT IF EXISTS %(name)s'\n        return 'ALTER TABLE %(table)s DROP CHECK %(name)s'\n\n    @property\n    def sql_rename_column(self):\n        # MariaDB >= 10.5.2 and MySQL >= 8.0.4 support an\n        # \"ALTER TABLE ... RENAME COLUMN\" statement.\n        if self.connection.mysql_is_mariadb:\n            if self.connection.mysql_version >= (10, 5, 2):\n                return super().sql_rename_column\n        elif self.connection.mysql_version >= (8, 0, 4):\n            return super().sql_rename_column\n        return 'ALTER TABLE %(table)s CHANGE %(old_column)s %(new_column)s %(type)s'\n\n    def quote_value(self, value):\n        self.connection.ensure_connection()\n        if isinstance(value, str):\n            value = value.replace('%', '%%')\n        # MySQLdb escapes to string, PyMySQL to bytes.\n        quoted = self.connection.connection.escape(value, self.connection.connection.encoders)\n        if isinstance(value, str) and isinstance(quoted, bytes):\n            quoted = quoted.decode()\n        return quoted\n\n    def _is_limited_data_type(self, field):\n        db_type = field.db_type(self.connection)\n        return db_type is not None and db_type.lower() in self.connection._limited_data_types\n\n    def skip_default(self, field):\n        if not self._supports_limited_data_type_defaults:\n            return self._is_limited_data_type(field)\n        return False\n\n    @property\n    def _supports_limited_data_type_defaults(self):\n        # MariaDB >= 10.2.1 and MySQL >= 8.0.13 supports defaults for BLOB\n        # and TEXT.\n        if self.connection.mysql_is_mariadb:\n            return self.connection.mysql_version >= (10, 2, 1)\n        return self.connection.mysql_version >= (8, 0, 13)\n\n    def _column_default_sql(self, field):\n        if (\n            not self.connection.mysql_is_mariadb and\n            self._supports_limited_data_type_defaults and\n            self._is_limited_data_type(field)\n        ):\n            # MySQL supports defaults for BLOB and TEXT columns only if the\n            # default value is written as an expression i.e. in parentheses.\n            return '(%s)'\n        return super()._column_default_sql(field)\n\n    def add_field(self, model, field):\n        super().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])\n\n    def _field_should_be_indexed(self, model, field):\n        create_index = super()._field_should_be_indexed(model, field)\n        storage = self.connection.introspection.get_storage_engine(\n            self.connection.cursor(), model._meta.db_table\n        )\n        # No need to create an index for ForeignKey fields except if\n        # db_constraint=False because the index from that constraint won't be\n        # created.\n        if (storage == \"InnoDB\" and\n                create_index and\n                field.get_internal_type() == 'ForeignKey' and\n                field.db_constraint):\n            return False\n        return not self._is_limited_data_type(field) and create_index\n\n    def _delete_composed_index(self, model, fields, *args):\n        \"\"\"\n        MySQL can remove an implicit FK index on a field when that field is\n        covered by another index like a unique_together. \"covered\" here means\n        that the more complex index starts like the simpler one.\n        http://bugs.mysql.com/bug.php?id=37910 / Django ticket #24757\n        We check here before removing the [unique|index]_together if we have to\n        recreate a FK index.\n        \"\"\"\n        first_field = model._meta.get_field(fields[0])\n        if first_field.get_internal_type() == 'ForeignKey':\n            constraint_names = self._constraint_names(model, [first_field.column], index=True)\n            if not constraint_names:\n                self.execute(\n                    self._create_index_sql(model, fields=[first_field], suffix='')\n                )\n        return super()._delete_composed_index(model, fields, *args)\n\n    def _set_field_new_type_null_status(self, field, new_type):\n        \"\"\"\n        Keep the null property of the old field. If it has changed, it will be\n        handled separately.\n        \"\"\"\n        if field.null:\n            new_type += \" NULL\"\n        else:\n            new_type += \" NOT NULL\"\n        return new_type\n\n    def _alter_column_type_sql(self, model, old_field, new_field, new_type):\n        new_type = self._set_field_new_type_null_status(old_field, new_type)\n        return super()._alter_column_type_sql(model, old_field, new_field, new_type)\n\n    def _rename_field_sql(self, table, old_field, new_field, new_type):\n        new_type = self._set_field_new_type_null_status(old_field, new_type)\n        return super()._rename_field_sql(table, old_field, new_field, new_type)"
        },
        {
          "file": "django/db/backends/oracle/schema.py",
          "type": "class",
          "name": "DatabaseSchemaEditor",
          "code": "class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):\n\n    sql_create_column = \"ALTER TABLE %(table)s ADD %(column)s %(definition)s\"\n    sql_alter_column_type = \"MODIFY %(column)s %(type)s\"\n    sql_alter_column_null = \"MODIFY %(column)s NULL\"\n    sql_alter_column_not_null = \"MODIFY %(column)s NOT NULL\"\n    sql_alter_column_default = \"MODIFY %(column)s DEFAULT %(default)s\"\n    sql_alter_column_no_default = \"MODIFY %(column)s DEFAULT NULL\"\n    sql_alter_column_collate = \"MODIFY %(column)s %(type)s%(collation)s\"\n\n    sql_delete_column = \"ALTER TABLE %(table)s DROP COLUMN %(column)s\"\n    sql_create_column_inline_fk = 'CONSTRAINT %(name)s REFERENCES %(to_table)s(%(to_column)s)%(deferrable)s'\n    sql_delete_table = \"DROP TABLE %(table)s CASCADE CONSTRAINTS\"\n    sql_create_index = \"CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s\"\n\n    def quote_value(self, value):\n        if isinstance(value, (datetime.date, datetime.time, datetime.datetime)):\n            return \"'%s'\" % value\n        elif isinstance(value, str):\n            return \"'%s'\" % value.replace(\"\\'\", \"\\'\\'\").replace('%', '%%')\n        elif isinstance(value, (bytes, bytearray, memoryview)):\n            return \"'%s'\" % value.hex()\n        elif isinstance(value, bool):\n            return \"1\" if value else \"0\"\n        else:\n            return str(value)\n\n    def remove_field(self, model, field):\n        # If the column is an identity column, drop the identity before\n        # removing the field.\n        if self._is_identity_column(model._meta.db_table, field.column):\n            self._drop_identity(model._meta.db_table, field.column)\n        super().remove_field(model, field)\n\n    def delete_model(self, model):\n        # Run superclass action\n        super().delete_model(model)\n        # Clean up manually created sequence.\n        self.execute(\"\"\"\n            DECLARE\n                i INTEGER;\n            BEGIN\n                SELECT COUNT(1) INTO i FROM USER_SEQUENCES\n                    WHERE SEQUENCE_NAME = '%(sq_name)s';\n                IF i = 1 THEN\n                    EXECUTE IMMEDIATE 'DROP SEQUENCE \"%(sq_name)s\"';\n                END IF;\n            END;\n        /\"\"\" % {'sq_name': self.connection.ops._get_no_autofield_sequence_name(model._meta.db_table)})\n\n    def alter_field(self, model, old_field, new_field, strict=False):\n        try:\n            super().alter_field(model, old_field, new_field, strict)\n        except DatabaseError as e:\n            description = str(e)\n            # If we're changing type to an unsupported type we need a\n            # SQLite-ish workaround\n            if 'ORA-22858' in description or 'ORA-22859' in description:\n                self._alter_field_type_workaround(model, old_field, new_field)\n            # If an identity column is changing to a non-numeric type, drop the\n            # identity first.\n            elif 'ORA-30675' in description:\n                self._drop_identity(model._meta.db_table, old_field.column)\n                self.alter_field(model, old_field, new_field, strict)\n            # If a primary key column is changing to an identity column, drop\n            # the primary key first.\n            elif 'ORA-30673' in description and old_field.primary_key:\n                self._delete_primary_key(model, strict=True)\n                self._alter_field_type_workaround(model, old_field, new_field)\n            else:\n                raise\n\n    def _alter_field_type_workaround(self, model, old_field, new_field):\n        \"\"\"\n        Oracle refuses to change from some type to other type.\n        What we need to do instead is:\n        - Add a nullable version of the desired field with a temporary name. If\n          the new column is an auto field, then the temporary column can't be\n          nullable.\n        - Update the table to transfer values from old to new\n        - Drop old column\n        - Rename the new column and possibly drop the nullable property\n        \"\"\"\n        # Make a new field that's like the new one but with a temporary\n        # column name.\n        new_temp_field = copy.deepcopy(new_field)\n        new_temp_field.null = (new_field.get_internal_type() not in ('AutoField', 'BigAutoField', 'SmallAutoField'))\n        new_temp_field.column = self._generate_temp_name(new_field.column)\n        # Add it\n        self.add_field(model, new_temp_field)\n        # Explicit data type conversion\n        # https://docs.oracle.com/en/database/oracle/oracle-database/18/sqlrf\n        # /Data-Type-Comparison-Rules.html#GUID-D0C5A47E-6F93-4C2D-9E49-4F2B86B359DD\n        new_value = self.quote_name(old_field.column)\n        old_type = old_field.db_type(self.connection)\n        if re.match('^N?CLOB', old_type):\n            new_value = \"TO_CHAR(%s)\" % new_value\n            old_type = 'VARCHAR2'\n        if re.match('^N?VARCHAR2', old_type):\n            new_internal_type = new_field.get_internal_type()\n            if new_internal_type == 'DateField':\n                new_value = \"TO_DATE(%s, 'YYYY-MM-DD')\" % new_value\n            elif new_internal_type == 'DateTimeField':\n                new_value = \"TO_TIMESTAMP(%s, 'YYYY-MM-DD HH24:MI:SS.FF')\" % new_value\n            elif new_internal_type == 'TimeField':\n                # TimeField are stored as TIMESTAMP with a 1900-01-01 date part.\n                new_value = \"TO_TIMESTAMP(CONCAT('1900-01-01 ', %s), 'YYYY-MM-DD HH24:MI:SS.FF')\" % new_value\n        # Transfer values across\n        self.execute(\"UPDATE %s set %s=%s\" % (\n            self.quote_name(model._meta.db_table),\n            self.quote_name(new_temp_field.column),\n            new_value,\n        ))\n        # Drop the old field\n        self.remove_field(model, old_field)\n        # Rename and possibly make the new field NOT NULL\n        super().alter_field(model, new_temp_field, new_field)\n\n    def _alter_column_type_sql(self, model, old_field, new_field, new_type):\n        auto_field_types = {'AutoField', 'BigAutoField', 'SmallAutoField'}\n        # Drop the identity if migrating away from AutoField.\n        if (\n            old_field.get_internal_type() in auto_field_types and\n            new_field.get_internal_type() not in auto_field_types and\n            self._is_identity_column(model._meta.db_table, new_field.column)\n        ):\n            self._drop_identity(model._meta.db_table, new_field.column)\n        return super()._alter_column_type_sql(model, old_field, new_field, new_type)\n\n    def normalize_name(self, name):\n        \"\"\"\n        Get the properly shortened and uppercased identifier as returned by\n        quote_name() but without the quotes.\n        \"\"\"\n        nn = self.quote_name(name)\n        if nn[0] == '\"' and nn[-1] == '\"':\n            nn = nn[1:-1]\n        return nn\n\n    def _generate_temp_name(self, for_name):\n        \"\"\"Generate temporary names for workarounds that need temp columns.\"\"\"\n        suffix = hex(hash(for_name)).upper()[1:]\n        return self.normalize_name(for_name + \"_\" + suffix)\n\n    def prepare_default(self, value):\n        return self.quote_value(value)\n\n    def _field_should_be_indexed(self, model, field):\n        create_index = super()._field_should_be_indexed(model, field)\n        db_type = field.db_type(self.connection)\n        if db_type is not None and db_type.lower() in self.connection._limited_data_types:\n            return False\n        return create_index\n\n    def _unique_should_be_added(self, old_field, new_field):\n        return (\n            super()._unique_should_be_added(old_field, new_field) and\n            not self._field_became_primary_key(old_field, new_field)\n        )\n\n    def _is_identity_column(self, table_name, column_name):\n        with self.connection.cursor() as cursor:\n            cursor.execute(\"\"\"\n                SELECT\n                    CASE WHEN identity_column = 'YES' THEN 1 ELSE 0 END\n                FROM user_tab_cols\n                WHERE table_name = %s AND\n                      column_name = %s\n            \"\"\", [self.normalize_name(table_name), self.normalize_name(column_name)])\n            row = cursor.fetchone()\n            return row[0] if row else False\n\n    def _drop_identity(self, table_name, column_name):\n        self.execute('ALTER TABLE %(table)s MODIFY %(column)s DROP IDENTITY' % {\n            'table': self.quote_name(table_name),\n            'column': self.quote_name(column_name),\n        })\n\n    def _get_default_collation(self, table_name):\n        with self.connection.cursor() as cursor:\n            cursor.execute(\"\"\"\n                SELECT default_collation FROM user_tables WHERE table_name = %s\n            \"\"\", [self.normalize_name(table_name)])\n            return cursor.fetchone()[0]\n\n    def _alter_column_collation_sql(self, model, new_field, new_type, new_collation):\n        if new_collation is None:\n            new_collation = self._get_default_collation(model._meta.db_table)\n        return super()._alter_column_collation_sql(model, new_field, new_type, new_collation)"
        }
      ]
    },
    {
      "pr_number": 15034,
      "pr_title": "Fixed #33234 -- Fixed autodetector crash for proxy models inheriting from non-model class.",
      "pr_body": "ticket-33234\r\n\r\nRegression in aa4acc164d1247c0de515c959f7b09648b57dc42.\r\n\r\nThanks Kevin Marsh for the report.",
      "issue_id": 33234,
      "issue_title": "Migrations autodetector breaking with proxy models inheriting from mixin",
      "issue_body": "So I've run into an issue with the new automigrations feature in Django 4.0b1 (and 4.0a1)  relating to proxy models inheriting from a non-Model class.\nFor examples given these three models:\nfrom\ndjango.db\nimport\nmodels\nclass\nMyMixin\n:\n...\nclass\nMyBaseModel\n(\nmodels\n.\nModel\n):\n...\nclass\nMyProxyModel\n(\nMyMixin\n,\nMyBaseModel\n):\nclass\nMeta\n:\nproxy\n=\nTrue\nRunning\nmigrate\n/\nmakemigrations\nwill result in\nMyProxyModel\ncrashing when it tries running\n_find_concrete_model_from_proxy\n:\nFile \"/some/annonymized/path/django/core/management/commands/makemigrations.py\", line 172, in handle\n    changes = autodetector.changes(\n  File \"/some/annonymized/path/django/db/migrations/autodetector.py\", line 43, in changes\n    changes = self._detect_changes(convert_apps, graph)\n  File \"/some/annonymized/path/django/db/migrations/autodetector.py\", line 156, in _detect_changes\n    self.to_state.resolve_fields_and_relations()\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 449, in resolve_fields_and_relations\n    concretes, proxies = self._get_concrete_models_mapping_and_proxy_models()\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 470, in _get_concrete_models_mapping_and_proxy_models\n    concrete_models_mapping[model_key] = self._find_concrete_model_from_proxy(\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 479, in _find_concrete_model_from_proxy\n    base_key = make_model_tuple(base)\n  File \"/some/annonymized/path/django/db/models/utils.py\", line 18, in make_model_tuple\n    model_tuple = model._meta.app_label, model._meta.model_name\nAttributeError: type object 'MyMixin' has no attribute '_meta'\nChanging\nMyMixin\nto inherit from\nmodels.Model\nprevents this crash but might not always be possible depending on how the user is using that mixin. Or if we've decided that all inheritance for proxy models must be from\nmodels.Model\nthen that should be flagged up more explicitly to developers (eg. checking for it in\nModelBase.__new__\nor something).",
      "issue_closed_at": "2021-11-02T14:10:58",
      "base_commit": "073b7b5915fdfb89a144e81173176ee13ff92a25",
      "changes": [
        {
          "file": "django/db/migrations/state.py",
          "type": "function",
          "name": "_get_concrete_models_mapping_and_proxy_models",
          "class_name": "ProjectState",
          "code": "def _get_concrete_models_mapping_and_proxy_models(self):\n        concrete_models_mapping = {}\n        proxy_models = {}\n        # Split models to proxy and concrete models.\n        for model_key, model_state in self.models.items():\n            if model_state.options.get('proxy'):\n                proxy_models[model_key] = model_state\n                # Find a concrete model for the proxy.\n                concrete_models_mapping[model_key] = self._find_concrete_model_from_proxy(\n                    proxy_models, model_state,\n                )\n            else:\n                concrete_models_mapping[model_key] = model_key\n        return concrete_models_mapping, proxy_models"
        }
      ]
    },
    {
      "pr_number": 3989,
      "pr_title": "Fixed #24220 - Allowed lazy objects for success_url",
      "pr_body": "See https://code.djangoproject.com/ticket/24220 for more info\n",
      "issue_id": 24220,
      "issue_title": "get_success_url raises an exception when success_url is lazy object",
      "issue_body": "In\n#24133\nwas added deprecation warning when\nself.success_url\ncontains old %-style Python formatting strings.\nHowever,\nsuccess_url\ncould be a lazy object like\nreverse_lazy\nand trying to call\nre.search\non lazy object results in\nTypeError\n:\n../../../../.envs/lingui-next/lib/python3.4/site-packages/django/views/generic/edit.py:166: in get_success_url\n    if re.search(r'%\\([^\\)]+\\)', self.success_url):\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\npattern = '%\\\\([^\\\\)]+\\\\)'\nstring = <django.utils.functional.lazy.<locals>.__proxy__ object at 0x10678ca90>\nflags = 0\n\n    def search(pattern, string, flags=0):\n        \"\"\"Scan through string looking for a match to the pattern, returning\n        a match object, or None if no match was found.\"\"\"\n>       return _compile(pattern, flags).search(string)\nE       TypeError: expected string or buffer\n\n../../../../.envs/lingui-next/lib/python3.4/re.py:166: TypeError",
      "issue_closed_at": "2015-01-26T11:40:17",
      "base_commit": "ea0ea7859a224225950a4df7c23eb3a7d823ddcd",
      "changes": [
        {
          "file": "django/views/generic/edit.py",
          "type": "function",
          "name": "get_success_url",
          "class_name": "DeletionMixin",
          "code": "def get_success_url(self):\n        if self.success_url:\n            if PERCENT_PLACEHOLDER_REGEX.search(self.success_url):\n                warnings.warn(\n                    \"%()s placeholder style in success_url is deprecated. \"\n                    \"Please replace them by the {} Python format syntax.\",\n                    RemovedInDjango20Warning, stacklevel=2\n                )\n                return self.success_url % self.object.__dict__\n            else:\n                return self.success_url.format(**self.object.__dict__)\n        else:\n            raise ImproperlyConfigured(\n                \"No URL to redirect to. Provide a success_url.\")"
        },
        {
          "file": "django/views/generic/edit.py",
          "type": "function",
          "name": "post",
          "class_name": "DeletionMixin",
          "code": "def post(self, request, *args, **kwargs):\n        return self.delete(request, *args, **kwargs)"
        }
      ]
    },
    {
      "pr_number": 5139,
      "pr_title": "Fixed #19263 -- Fixed crash when filtering using __in and an empty QuerySet.",
      "pr_body": "https://code.djangoproject.com/ticket/19263\n",
      "issue_id": 19263,
      "issue_title": "Filtering __in a sliced queryset with a 0 limit raises an error",
      "issue_body": "I've noticed that after upgrading to Django 1.4,\n__in\nqueries really don't like empty sets. Simple queries still work, like\nUser.objects.filter(groups__in=[])\n, but most failures I've seen are with Paginators. I think this is the minimum set to cause a DatabaseError, create any app, add a models.py with:\nfrom\ndjango.db\nimport\nmodels\nclass\nAuthor\n(\nmodels\n.\nModel\n):\npass\nclass\nBook\n(\nmodels\n.\nModel\n):\nauthor\n=\nmodels\n.\nForeignKey\n(\nAuthor\n)\ndef\ncrash\n():\nfrom\ndjango.core.paginator\nimport\nPaginator\npages\n=\nPaginator\n(\nAuthor\n.\nobjects\n.\nall\n(),\n25\n)\npage\n=\npages\n.\npage\n(\n1\n)\nbooks\n=\nBook\n.\nobjects\n.\nfilter\n(\nauthor__in\n=\npage\n.\nobject_list\n)\nprint\nbooks\ncalling crash() will cause this stack trace:\nC:\\Workspace\\someproject\\src\\someproject\\test.py in <module>()\n      6\n      7 books = Book.objects.filter(author__in=page.object_list)\n----> 8 print books\n      9\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in __repr__(self)\n     70\n     71     def __repr__(self):\n---> 72         data = list(self[:REPR_OUTPUT_SIZE + 1])\n     73         if len(data) > REPR_OUTPUT_SIZE:\n     74             data[-1] = \"...(remaining elements truncated)...\"\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in __len__(self)\n     85                 self._result_cache = list(self.iterator())\n     86         elif self._iter:\n---> 87             self._result_cache.extend(self._iter)\n     88         if self._prefetch_related_lookups and not self._prefetch_done:\n     89             self._prefetch_related_objects()\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in iterator(self)\n    289             klass_info = get_klass_info(model, max_depth=max_depth,\n    290                                         requested=requested, only_load=only_load)\n--> 291         for row in compiler.results_iter():\n    292             if fill_cache:\n    293                 obj, _ = get_cached_row(row, index_start, db, klass_info,\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\sql\\compiler.pyc in results_iter(self)\n    761         if self.query.select_for_update and transaction.is_managed(self.using):\n    762             transaction.set_dirty(self.using)\n--> 763         for rows in self.execute_sql(MULTI):\n    764             for row in rows:\n    765                 if resolve_columns:\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\sql\\compiler.pyc in execute_sql(self, result_type)\n    816\n    817         cursor = self.connection.cursor()\n--> 818         cursor.execute(sql, params)\n    819\n    820         if not result_type:\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\backends\\util.pyc in execute(self, sql, params)\n     38         start = time()\n     39         try:\n---> 40             return self.cursor.execute(sql, params)\n     41         finally:\n     42             stop = time()\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\backends\\postgresql_psycopg2\\base.pyc in execute(self, query, args)\n     50     def execute(self, query, args=None):\n     51         try:\n---> 52             return self.cursor.execute(query, args)\n     53         except Database.IntegrityError, e:\n     54             raise utils.IntegrityError, utils.IntegrityError(*tuple(e)), sys.exc_info()[2]\n\nDatabaseError: syntax error at or near \")\"\nLINE 1: ...ugtest_book\" WHERE \"bugtest_book\".\"author_id\" IN () LIMIT 21\nThe SQL statement created is:\nSELECT \"bugtest_book\".\"id\", \"bugtest_book\".\"author_id\" FROM \"bugtest_book\" WHERE \"bugtest_book\".\"author_id\" IN () LIMIT 21",
      "issue_closed_at": "2015-09-04T07:00:51",
      "base_commit": "7c0850028f25eebaa9b521b5d02afac084ff2c6f",
      "changes": [
        {
          "file": "django/db/models/sql/compiler.py",
          "type": "function",
          "name": "as_nested_sql",
          "class_name": "SQLCompiler",
          "code": "def as_nested_sql(self):\n        \"\"\"\n        Perform the same functionality as the as_sql() method, returning an\n        SQL string and parameters. However, the alias prefixes are bumped\n        beforehand (in a copy -- the current query isn't changed), and any\n        ordering is removed if the query is unsliced.\n\n        Used when nesting this query inside another.\n        \"\"\"\n        obj = self.query.clone()\n        if obj.low_mark == 0 and obj.high_mark is None and not self.query.distinct_fields:\n            # If there is no slicing in use, then we can safely drop all ordering\n            obj.clear_ordering(True)\n        return obj.get_compiler(connection=self.connection).as_sql(subquery=True)"
        }
      ]
    }
  ]
}