{
  "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": "",
      "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": "",
      "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\u2019 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": "",
      "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": "",
      "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": "\u2026mported\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": "",
      "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"
        }
      ]
    }
  ]
}