{
    "Selected_candidate": {
        "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)"
            }
        ]
    },
    "Justification": "Candidate E addresses a significant regression related to database schema management in Django, which ties closely to the CURRENT bug's focus on the MigrationRecorder behavior under the defined db_router rules. Both bugs highlight issues concerning how Django interacts with database connections and schema, particularly in terms of handling tables and constraints. Also, the similarity in the migration processes indicates that insights from the fixes proposed for Candidate E could shed light on potential solutions for ensuring that the MigrationRecorder abides by the rules set out in the db_router. Therefore, it provides a relevant context that may help in addressing the CURRENT bug."
}