{
  "instance_id": "django__django-15851",
  "repo": "django/django",
  "created_at": "2022-07-18T01:36:33Z",
  "problem_statement": "dbshell additional parameters should be passed before dbname on PostgreSQL.\nDescription\n\t\npsql expects all options to proceed the database name, if provided. So, if doing something like `./manage.py dbshell -- -c \"select * from some_table;\" one will get this:\n$ ./manage.py dbshell -- -c \"select * from some_table;\"\npsql: warning: extra command-line argument \"-c\" ignored\npsql: warning: extra command-line argument \"select * from some_table;\" ignored\npsql (10.21)\nType \"help\" for help.\nsome_database=>\nIt appears the args list just need to be constructed in the proper order, leaving the database name for the end of the args list.\n",
  "patch": "diff --git a/django/db/backends/postgresql/client.py b/django/db/backends/postgresql/client.py\n--- a/django/db/backends/postgresql/client.py\n+++ b/django/db/backends/postgresql/client.py\n@@ -32,9 +32,9 @@ def settings_to_cmd_args_env(cls, settings_dict, parameters):\n             args += [\"-h\", host]\n         if port:\n             args += [\"-p\", str(port)]\n+        args.extend(parameters)\n         if dbname:\n             args += [dbname]\n-        args.extend(parameters)\n \n         env = {}\n         if passwd:\n",
  "similar_bug_items": [
    {
      "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": 9345,
      "pr_title": "Fixed #28792 -- Fixed index name truncation of namespaced tables.",
      "pr_body": "Refs #27458, #27843.",
      "issue_id": 28792,
      "issue_title": "Index names can be incorrectly truncated when using a namespaced table name",
      "issue_body": "When using a namespaced\n_meta.db_table\n(e.g. Oracle's\n'schema\".\"table'\n) it's possible that\n_create_index_name\nreturns an index name truncating the namespace resulting in an invalid identifier or one that isn't namespaced anymore and thus created in the user's namespace.\nFor example, given the following model:\nclass\nFoo\n(\nmodels\n.\nModel\n):\nfield\n=\nmodels\n.\nIntegerField\n(\nindex\n=\nTrue\n)\nclass\nMeta\n:\ndb_table\n=\n'long_name\".\"table_name'\nThe resulting index name will be\n'long_name\"_field_d21c9e0a'\nwhich is invalid SQL even when quoted to\n'\"long_name\"_field_d21c9e0a\"'\n.\nMarking as a release blocker because this is a regression which I believe was introduced by\n#27458\nand wasn't addressed by\n#27843\n. I confirm that this uses to work on Django 1.10 but was broken on 1.11 as I stumbled upon the issue when upgrading a Django 1.8 LTS codebase to 1.11 LTS on the 1.10 -> 1.11 step.\nThe tests added by\n#27458\njust happened to work because the index name truncation cut the string the in a way that both double quotes are stripped. It should be possible to tweak the table names to trigger the errors but I felt like directly testing\n_create_index_name\nwas more appropriate.",
      "issue_closed_at": "2017-11-14T20:53:46",
      "base_commit": "532a4f22ad94db320cb0fd66f4c7ee57d17ac65a",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "line",
          "name": "line 5",
          "code": "from django.db.backends.ddl_references import (\n    Columns, ForeignKeyName, IndexName, Statement, Table,\n)\nfrom django.db.backends.utils import strip_quotes\nfrom django.db.models import Index\nfrom django.db.transaction import TransactionManagementError, atomic\nfrom django.utils import timezone"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_create_index_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _create_index_name(self, table_name, column_names, suffix=\"\"):\n        \"\"\"\n        Generate a unique name for an index/unique constraint.\n\n        The name is divided into 3 parts: the table name, the column names,\n        and a unique digest and suffix.\n        \"\"\"\n        table_name = strip_quotes(table_name)\n        hash_data = [table_name] + list(column_names)\n        hash_suffix_part = '%s%s' % (self._digest(*hash_data), suffix)\n        max_length = self.connection.ops.max_name_length() or 200\n        # If everything fits into max_length, use that name.\n        index_name = '%s_%s_%s' % (table_name, '_'.join(column_names), hash_suffix_part)\n        if len(index_name) <= max_length:\n            return index_name\n        # Shorten a long suffix.\n        if len(hash_suffix_part) > max_length / 3:\n            hash_suffix_part = hash_suffix_part[:max_length // 3]\n        other_length = (max_length - len(hash_suffix_part)) // 2 - 1\n        index_name = '%s_%s_%s' % (\n            table_name[:other_length],\n            '_'.join(column_names)[:other_length],\n            hash_suffix_part,\n        )\n        # Prepend D if needed to prevent the name from starting with an\n        # underscore or a number (not permitted on Oracle).\n        if index_name[0] == \"_\" or index_name[0].isdigit():\n            index_name = \"D%s\" % index_name[:-1]\n        return index_name"
        },
        {
          "file": "django/db/backends/utils.py",
          "type": "line",
          "name": "line 3",
          "code": "import functools\nimport hashlib\nimport logging\nimport re\nfrom time import time\n\nfrom django.conf import settings"
        },
        {
          "file": "django/db/backends/utils.py",
          "type": "function",
          "name": "rev_typecast_decimal",
          "class_name": null,
          "code": "def rev_typecast_decimal(d):\n    if d is None:\n        return None\n    return str(d)"
        }
      ]
    },
    {
      "pr_number": 14315,
      "pr_title": "Fixed #32687 -- Restored passing process’ environment to underlying tool in dbshell on PostgreSQL.",
      "pr_body": "[ticket-32687](https://code.djangoproject.com/ticket/32687#ticket)\r\nIntroduced by https://github.com/django/django/commit/bbe6fbb8768e8fb1aecb96d51c049d7ceaf802d3\r\n\r\npostgresql client returns empty dict instead of None for env\r\nas a result os.environ is not used and empty env passed\r\nto subprocess.",
      "issue_id": 32687,
      "issue_title": "database client runshell doesn't respect os.environ values in some cases",
      "issue_body": "postgresql client returns empty dict instead of None for env\nas a result os.environ is not used and empty env passed\nto subprocess.\nBug introduced in\n​\nhttps://github.com/django/django/commit/bbe6fbb8768e8fb1aecb96d51c049d7ceaf802d3#diff-e98866ed4d445fbc94bb60bedffd5d8cf07af55dca6e8ffa4945931486efc3eeR23-R26\nPR\n​\nhttps://github.com/django/django/pull/14315",
      "issue_closed_at": "2021-04-27T05:01:49",
      "base_commit": "187118203197801c6cb72dc8b06b714b23b6dd3d",
      "changes": [
        {
          "file": "django/db/backends/base/client.py",
          "type": "function",
          "name": "settings_to_cmd_args_env",
          "class_name": "BaseDatabaseClient",
          "code": "def settings_to_cmd_args_env(cls, settings_dict, parameters):\n        raise NotImplementedError(\n            'subclasses of BaseDatabaseClient must provide a '\n            'settings_to_cmd_args_env() method or override a runshell().'\n        )"
        },
        {
          "file": "django/db/backends/postgresql/client.py",
          "type": "function",
          "name": "settings_to_cmd_args_env",
          "class_name": "DatabaseClient",
          "code": "def settings_to_cmd_args_env(cls, settings_dict, parameters):\n        args = [cls.executable_name]\n        options = settings_dict.get('OPTIONS', {})\n\n        host = settings_dict.get('HOST')\n        port = settings_dict.get('PORT')\n        dbname = settings_dict.get('NAME')\n        user = settings_dict.get('USER')\n        passwd = settings_dict.get('PASSWORD')\n        passfile = options.get('passfile')\n        service = options.get('service')\n        sslmode = options.get('sslmode')\n        sslrootcert = options.get('sslrootcert')\n        sslcert = options.get('sslcert')\n        sslkey = options.get('sslkey')\n\n        if not dbname and not service:\n            # Connect to the default 'postgres' db.\n            dbname = 'postgres'\n        if user:\n            args += ['-U', user]\n        if host:\n            args += ['-h', host]\n        if port:\n            args += ['-p', str(port)]\n        if dbname:\n            args += [dbname]\n        args.extend(parameters)\n\n        env = {}\n        if passwd:\n            env['PGPASSWORD'] = str(passwd)\n        if service:\n            env['PGSERVICE'] = str(service)\n        if sslmode:\n            env['PGSSLMODE'] = str(sslmode)\n        if sslrootcert:\n            env['PGSSLROOTCERT'] = str(sslrootcert)\n        if sslcert:\n            env['PGSSLCERT'] = str(sslcert)\n        if sslkey:\n            env['PGSSLKEY'] = str(sslkey)\n        if passfile:\n            env['PGPASSFILE'] = str(passfile)\n        return args, env"
        }
      ]
    },
    {
      "pr_number": 8754,
      "pr_title": "Fixed #28391 -- Fixed Cast() with CharField and max_length on MySQL.",
      "pr_body": "https://code.djangoproject.com/ticket/28391",
      "issue_id": 28391,
      "issue_title": "Cast doesn't take into account CharField's max_length on MySQL.",
      "issue_body": "Correct behavior, e.g. (based on\ndb_functions/test_cast.py\n):\n>>> Author.objects.create(name='Bob', age=1111)\n>>> numbers = Author.objects.annotate(cast_string=Cast('age', models.CharField(max_length=3)))\n>>> numbers.get().cast_string\n111\non MySQL it returns\n1111\n(related with\n#28371\n).",
      "issue_closed_at": "2017-07-17T14:12:39",
      "base_commit": "feeafdad02e2874e2e2f879a825d3527f6b193ad",
      "changes": [
        {
          "file": "django/db/backends/sqlite3/features.py",
          "type": "class",
          "name": "DatabaseFeatures",
          "code": "class DatabaseFeatures(BaseDatabaseFeatures):\n    # SQLite cannot handle us only partially reading from a cursor's result set\n    # and then writing the same rows to the database in another cursor. This\n    # setting ensures we always read result sets fully into memory all in one\n    # go.\n    can_use_chunked_reads = False\n    test_db_allows_multiple_connections = False\n    supports_unspecified_pk = True\n    supports_timezones = False\n    max_query_params = 999\n    supports_mixed_date_datetime_comparisons = False\n    has_bulk_insert = True\n    supports_column_check_constraints = False\n    autocommits_when_autocommit_is_off = True\n    can_introspect_decimal_field = False\n    can_introspect_positive_integer_field = True\n    can_introspect_small_integer_field = True\n    supports_transactions = True\n    atomic_transactions = False\n    can_rollback_ddl = True\n    supports_paramstyle_pyformat = False\n    supports_sequence_reset = False\n    can_clone_databases = True\n    supports_temporal_subtraction = True\n    ignores_table_name_case = True\n\n    @cached_property\n    def uses_savepoints(self):\n        return Database.sqlite_version_info >= (3, 6, 8)\n\n    @cached_property\n    def supports_index_column_ordering(self):\n        return Database.sqlite_version_info >= (3, 3, 0)\n\n    @cached_property\n    def can_release_savepoints(self):\n        return self.uses_savepoints\n\n    @cached_property\n    def can_share_in_memory_db(self):\n        return (\n            Database.__name__ == 'sqlite3.dbapi2' and\n            Database.sqlite_version_info >= (3, 7, 13)\n        )\n\n    @cached_property\n    def supports_stddev(self):\n        \"\"\"\n        Confirm support for STDDEV and related stats functions.\n\n        SQLite supports STDDEV as an extension package; so\n        connection.ops.check_expression_support() can't unilaterally\n        rule out support for STDDEV. Manually check whether the call works.\n        \"\"\"\n        with self.connection.cursor() as cursor:\n            cursor.execute('CREATE TABLE STDDEV_TEST (X INT)')\n            try:\n                cursor.execute('SELECT STDDEV(*) FROM STDDEV_TEST')\n                has_support = True\n            except utils.DatabaseError:\n                has_support = False\n            cursor.execute('DROP TABLE STDDEV_TEST')\n        return has_support"
        },
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "clean",
          "class_name": "Field",
          "code": "def clean(self, value, model_instance):\n        \"\"\"\n        Convert the value's type and run validation. Validation errors\n        from to_python() and validate() are propagated. Return the correct\n        value if no error is raised.\n        \"\"\"\n        value = self.to_python(value)\n        self.validate(value, model_instance)\n        self.run_validators(value)\n        return value"
        },
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "db_type",
          "class_name": "Field",
          "code": "def db_type(self, connection):\n        \"\"\"\n        Return the database column data type for this field, for the provided\n        connection.\n        \"\"\"\n        # The default implementation of this method looks at the\n        # backend-specific data_types dictionary, looking up the field by its\n        # \"internal type\".\n        #\n        # A Field class can implement the get_internal_type() method to specify\n        # which *preexisting* Django Field class it's most similar to -- i.e.,\n        # a custom field might be represented by a TEXT column type, which is\n        # the same as the TextField Django field type, which means the custom\n        # field's get_internal_type() returns 'TextField'.\n        #\n        # But the limitation of the get_internal_type() / data_types approach\n        # is that it cannot handle database column types that aren't already\n        # mapped to one of the built-in Django field types. In this case, you\n        # can implement db_type() instead of get_internal_type() to specify\n        # exactly which wacky database column type you want to use.\n        data = DictWrapper(self.__dict__, connection.ops.quote_name, \"qn_\")\n        try:\n            return connection.data_types[self.get_internal_type()] % data\n        except KeyError:\n            return None"
        },
        {
          "file": "django/db/models/functions/base.py",
          "type": "class",
          "name": "Cast",
          "code": "class Cast(Func):\n    \"\"\"Coerce an expression to a new field type.\"\"\"\n    function = 'CAST'\n    template = '%(function)s(%(expressions)s AS %(db_type)s)'\n\n    mysql_types = {\n        fields.CharField: 'char',\n        fields.IntegerField: 'signed integer',\n        fields.BigIntegerField: 'signed integer',\n        fields.SmallIntegerField: 'signed integer',\n        fields.FloatField: 'signed',\n        fields.PositiveIntegerField: 'unsigned integer',\n        fields.PositiveSmallIntegerField: 'unsigned integer',\n    }\n\n    def __init__(self, expression, output_field):\n        super().__init__(expression, output_field=output_field)\n\n    def as_sql(self, compiler, connection, **extra_context):\n        if 'db_type' not in extra_context:\n            extra_context['db_type'] = self.output_field.db_type(connection)\n        return super().as_sql(compiler, connection, **extra_context)\n\n    def as_mysql(self, compiler, connection):\n        extra_context = {}\n        output_field_class = type(self.output_field)\n        if output_field_class in self.mysql_types:\n            extra_context['db_type'] = self.mysql_types[output_field_class]\n        return self.as_sql(compiler, connection, **extra_context)\n\n    def as_postgresql(self, compiler, connection):\n        # CAST would be valid too, but the :: shortcut syntax is more readable.\n        return self.as_sql(compiler, connection, template='%(expressions)s::%(db_type)s')"
        },
        {
          "file": "django/db/models/functions/base.py",
          "type": "function",
          "name": "as_mysql",
          "class_name": "Length",
          "code": "def as_mysql(self, compiler, connection):\n        return super().as_sql(compiler, connection, function='CHAR_LENGTH')"
        }
      ]
    },
    {
      "pr_number": 11169,
      "pr_title": "Fixed #30318 --  Added check for importability of arguments of custom error handler views.",
      "pr_body": "…mported\r\n\r\nThanks to Jon on Stack Overflow for reporting the issue.",
      "issue_id": 30318,
      "issue_title": "Add new system check message when custom error handler 'path.to.view' cannot be imported",
      "issue_body": "#29642\nadded checks for the signatures of custom error handlers.\nWhen the 'path.to.view' cannot be imported, it raises ModuleNotFoundError or ViewDoesNotExist, as seen in this Stack Overflow question:\n​\nhttps://stackoverflow.com/q/55481810/113962\nI suggest we catch the exception, and add another check code, e.g.\n* **urls.E008**: The custom ``handlerXXX`` view ``'path.to.view'`` cannot be imported.",
      "issue_closed_at": "2019-04-25T04:38:58",
      "base_commit": "fc9566d42daf28cdaa25a5db1b5ade253ceb064f",
      "changes": [
        {
          "file": "django/urls/resolvers.py",
          "type": "line",
          "name": "line 15",
          "code": "from django.conf import settings\nfrom django.core.checks import Error, Warning\nfrom django.core.checks.urls import check_resolver\nfrom django.core.exceptions import ImproperlyConfigured\nfrom django.utils.datastructures import MultiValueDict\nfrom django.utils.functional import cached_property\nfrom django.utils.http import RFC3986_SUBDELIMS, escape_leading_slashes"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_check_custom_error_handlers",
          "class_name": "URLResolver",
          "code": "def _check_custom_error_handlers(self):\n        messages = []\n        # All handlers take (request, exception) arguments except handler500\n        # which takes (request).\n        for status_code, num_parameters in [(400, 2), (403, 2), (404, 2), (500, 1)]:\n            handler, param_dict = self.resolve_error_handler(status_code)\n            signature = inspect.signature(handler)\n            args = [None] * num_parameters\n            try:\n                signature.bind(*args)\n            except TypeError:\n                msg = (\n                    \"The custom handler{status_code} view '{path}' does not \"\n                    \"take the correct number of arguments ({args}).\"\n                ).format(\n                    status_code=status_code,\n                    path=handler.__module__ + '.' + handler.__qualname__,\n                    args='request, exception' if num_parameters == 2 else 'request',\n                )\n                messages.append(Error(msg, id='urls.E007'))\n        return messages"
        }
      ]
    }
  ]
}