{
  "instance_id": "django__django-11630",
  "repo": "django/django",
  "created_at": "2019-08-05T11:22:41Z",
  "problem_statement": "Django throws error when different apps with different models have the same name table name.\nDescription\n\t\nError message:\ntable_name: (models.E028) db_table 'table_name' is used by multiple models: base.ModelName, app2.ModelName.\nWe have a Base app that points to a central database and that has its own tables. We then have multiple Apps that talk to their own databases. Some share the same table names.\nWe have used this setup for a while, but after upgrading to Django 2.2 we're getting an error saying we're not allowed 2 apps, with 2 different models to have the same table names. \nIs this correct behavior? We've had to roll back to Django 2.0 for now.\n",
  "patch": "diff --git a/django/core/checks/model_checks.py b/django/core/checks/model_checks.py\n--- a/django/core/checks/model_checks.py\n+++ b/django/core/checks/model_checks.py\n@@ -4,7 +4,8 @@\n from itertools import chain\n \n from django.apps import apps\n-from django.core.checks import Error, Tags, register\n+from django.conf import settings\n+from django.core.checks import Error, Tags, Warning, register\n \n \n @register(Tags.models)\n@@ -35,14 +36,25 @@ def check_all_models(app_configs=None, **kwargs):\n             indexes[model_index.name].append(model._meta.label)\n         for model_constraint in model._meta.constraints:\n             constraints[model_constraint.name].append(model._meta.label)\n+    if settings.DATABASE_ROUTERS:\n+        error_class, error_id = Warning, 'models.W035'\n+        error_hint = (\n+            'You have configured settings.DATABASE_ROUTERS. Verify that %s '\n+            'are correctly routed to separate databases.'\n+        )\n+    else:\n+        error_class, error_id = Error, 'models.E028'\n+        error_hint = None\n     for db_table, model_labels in db_table_models.items():\n         if len(model_labels) != 1:\n+            model_labels_str = ', '.join(model_labels)\n             errors.append(\n-                Error(\n+                error_class(\n                     \"db_table '%s' is used by multiple models: %s.\"\n-                    % (db_table, ', '.join(db_table_models[db_table])),\n+                    % (db_table, model_labels_str),\n                     obj=db_table,\n-                    id='models.E028',\n+                    hint=(error_hint % model_labels_str) if error_hint else None,\n+                    id=error_id,\n                 )\n             )\n     for index_name, model_labels in indexes.items():\n",
  "similar_bug_items": [
    {
      "pr_number": 10199,
      "pr_title": "Fixed #26352 -- Made system check allow ManyToManyField to target the same model if through_fields differs.",
      "pr_body": "See https://code.djangoproject.com/ticket/26352",
      "issue_id": 26352,
      "issue_title": "models.E003 check incorrectly prevents duplicate ManyToMany through-self that differ by through_fields",
      "issue_body": "Let's say we're building a Twitter-style friendship model, where a user can follow other users, and can have users that follow them.\nA sensible way to model that might look like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\nclass\nFollowship\n(\nmodels\n.\nModel\n):\nuser\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\ntarget\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\nHere we have an intermediary table called \"Followship\", which relates a user to the target user that they are following.\nWe also have two helpful convenience ManyToMany fields defined on the user:\nuser.friends.all()\nreturns all other users that the user is following, while\nuser.followers.all()\nreturns the users that are following our current user.\nThe above models should work fine... but they don't, because Django's model checking framework throws the following error:\nusers.User: (models.E003) The model has two many-to-many relations through the intermediate model 'users.Followship'.\nI don't think this warning is warranted in this case, because the\nthrough_fields\narguments provided to the friends/followers ManyToManyFields mean that the relationships defined here make sense and the models should work correctly.\nI've tested this theory by disabling the warning entirely using an over-ride class method on user, like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\n@classmethod\ndef\n_check_m2m_through_same_relationship\n(\ncls\n):\n# Disable models.E003 check for this model\nreturn\n[]\nThis does the trick: the check is suppressed, and the models work as expected.\nI think the model check framework is being overly strict here. I think it should be modified to enable this pattern, provided the through_fields= parameters on the duplicate ManyToMany fields are present and differentiate the fields correctly.",
      "issue_closed_at": "2018-08-22T11:28:35",
      "base_commit": "f2d5dafec93e6b3100f004c559ebe21e2b783ae7",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "_check_m2m_through_same_relationship",
          "class_name": "Model",
          "code": "def _check_m2m_through_same_relationship(cls):\n        \"\"\" Check if no relationship model is used by more than one m2m field.\n        \"\"\"\n\n        errors = []\n        seen_intermediary_signatures = []\n\n        fields = cls._meta.local_many_to_many\n\n        # Skip when the target model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.model, ModelBase))\n\n        # Skip when the relationship model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.through, ModelBase))\n\n        for f in fields:\n            signature = (f.remote_field.model, cls, f.remote_field.through)\n            if signature in seen_intermediary_signatures:\n                errors.append(\n                    checks.Error(\n                        \"The model has two many-to-many relations through \"\n                        \"the intermediate model '%s'.\" % f.remote_field.through._meta.label,\n                        obj=cls,\n                        id='models.E003',\n                    )\n                )\n            else:\n                seen_intermediary_signatures.append(signature)\n        return errors"
        }
      ]
    },
    {
      "pr_number": 9421,
      "pr_title": "Fixed #28884 -- Fixed crash in renames of fields of tables referenced by m2ms on SQlite.",
      "pr_body": "https://code.djangoproject.com/ticket/28884\r\n\r\nIntrospected database constraints instead of relying on `_meta.related_objects` to determine whether or not a table or a column is referenced on rename operations.\r\n\r\nThis has the side effect of ignoring both ``db_constraint=False`` and virtual fields such as ``GenericRelation`` which are not backend by database level constraints and thus shouldn't prevent the rename operations from being performed in a transaction.\r\n\r\nThis also highlighted false negatives in the existing schema and migrations tests where `_meta.related_objects` was not appropriately populated.",
      "issue_id": 28884,
      "issue_title": "RenameField crashes with AttributeError when renaming a ManyToManyField (sqlite3)",
      "issue_body": "Original issue:\n​\nhttps://groups.google.com/forum/#!topic/django-users/O7s658gIHTE\nWith Django 2.0, a\nRenameField\non a model which has a reverse many to many relationship raises an exception:\nAttributeError: 'ManyToManyRel' object has no attribute 'field_name'\n.\nThis is a regression in Django 2.0: the same migration with Django 1.11 terminates successfully\nBelow the code and the commands to reproduce the problem:\n# myapp/models.py\n\nfrom django.db import models\n\n\nclass ModelA(models.Model):\n    new_name = models.IntegerField()\n\n\nclass ModelB(models.Model):\n    model_as = models.ManyToManyField('ModelA')\n\n# myapp/migrations/0001_initial.py\n\nfrom django.db import migrations, models\n\n\nclass Migration(migrations.Migration):\n\n    initial = True\n\n    dependencies = [\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name='ModelA',\n            fields=[\n                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n                ('old_name', models.IntegerField()),\n            ],\n        ),\n        migrations.CreateModel(\n            name='ModelB',\n            fields=[\n                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n                ('model_as', models.ManyToManyField(to='myapp.ModelA')),\n            ],\n        ),\n    ]\n\n# myapp/migrations/0002_auto_20171204_1012.py\n\nfrom django.db import migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        ('myapp', '0001_initial'),\n    ]\n\n    operations = [\n        migrations.RenameField(\n            model_name='modela',\n            old_name='old_name',\n            new_name='new_name',\n        ),\n    ]\n\n$ ./manage.py migrate\nOperations to perform:\n  Apply all migrations: admin, auth, contenttypes, myapp, sessions\nRunning migrations:\n  Applying contenttypes.0001_initial... OK\n  Applying auth.0001_initial... OK\n  Applying admin.0001_initial... OK\n  Applying admin.0002_logentry_remove_auto_add... OK\n  Applying contenttypes.0002_remove_content_type_name... OK\n  Applying auth.0002_alter_permission_name_max_length... OK\n  Applying auth.0003_alter_user_email_max_length... OK\n  Applying auth.0004_alter_user_username_opts... OK\n  Applying auth.0005_alter_user_last_login_null... OK\n  Applying auth.0006_require_contenttypes_0002... OK\n  Applying auth.0007_alter_validators_add_error_messages... OK\n  Applying auth.0008_alter_user_username_max_length... OK\n  Applying auth.0009_alter_user_last_name_max_length... OK\n  Applying myapp.0001_initial... OK\n  Applying myapp.0002_auto_20171204_1012...Traceback (most recent call last):\n  File \"./manage.py\", line 15, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/core/management/__init__.py\", line 371, in execute_from_command_line\n    utility.execute()\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/core/management/__init__.py\", line 365, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/core/management/base.py\", line 288, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/core/management/base.py\", line 335, in execute\n    output = self.handle(*args, **options)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/core/management/commands/migrate.py\", line 200, in handle\n    fake_initial=fake_initial,\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/migrations/executor.py\", line 117, in migrate\n    state = self._migrate_all_forwards(state, plan, full_plan, fake=fake, fake_initial=fake_initial)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/migrations/executor.py\", line 147, in _migrate_all_forwards\n    state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/migrations/executor.py\", line 244, in apply_migration\n    state = migration.apply(state, schema_editor)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/migrations/migration.py\", line 122, in apply\n    operation.database_forwards(self.app_label, schema_editor, old_state, project_state)\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/migrations/operations/fields.py\", line 304, in database_forwards\n    to_model._meta.get_field(self.new_name),\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/backends/sqlite3/schema.py\", line 81, in alter_field\n    any(r.field_name == old_field.name for r in model._meta.related_objects)):\n  File \"/home/edg/src/example/env/lib/python3.6/site-packages/django/db/backends/sqlite3/schema.py\", line 81, in <genexpr>\n    any(r.field_name == old_field.name for r in model._meta.related_objects)):\nAttributeError: 'ManyToManyRel' object has no attribute 'field_name'\n\n$ pip freeze\nDjango==2.0\npytz==2017.3",
      "issue_closed_at": "2017-12-22T15:09:50",
      "base_commit": "f3a98224e6dd5f8846008512f281e452dc3b1909",
      "changes": [
        {
          "file": "django/db/backends/sqlite3/introspection.py",
          "type": "function",
          "name": "_table_info",
          "class_name": "DatabaseIntrospection",
          "code": "def _table_info(self, cursor, name):\n        cursor.execute('PRAGMA table_info(%s)' % self.connection.ops.quote_name(name))\n        # cid, name, type, notnull, default_value, pk\n        return [{\n            'name': field[1],\n            'type': field[2],\n            'size': get_field_size(field[2]),\n            'null_ok': not field[3],\n            'default': field[4],\n            'pk': field[5],  # undocumented\n        } for field in cursor.fetchall()]"
        },
        {
          "file": "django/db/backends/sqlite3/introspection.py",
          "type": "function",
          "name": "get_constraints",
          "class_name": "DatabaseIntrospection",
          "code": "def get_constraints(self, cursor, table_name):\n        \"\"\"\n        Retrieve any constraints or keys (unique, pk, fk, check, index) across\n        one or more columns.\n        \"\"\"\n        constraints = {}\n        # Get the index info\n        cursor.execute(\"PRAGMA index_list(%s)\" % self.connection.ops.quote_name(table_name))\n        for row in cursor.fetchall():\n            # Sqlite3 3.8.9+ has 5 columns, however older versions only give 3\n            # columns. Discard last 2 columns if there.\n            number, index, unique = row[:3]\n            # Get the index info for that index\n            cursor.execute('PRAGMA index_info(%s)' % self.connection.ops.quote_name(index))\n            for index_rank, column_rank, column in cursor.fetchall():\n                if index not in constraints:\n                    constraints[index] = {\n                        \"columns\": [],\n                        \"primary_key\": False,\n                        \"unique\": bool(unique),\n                        \"foreign_key\": False,\n                        \"check\": False,\n                        \"index\": True,\n                    }\n                constraints[index]['columns'].append(column)\n            # Add type and column orders for indexes\n            if constraints[index]['index'] and not constraints[index]['unique']:\n                # SQLite doesn't support any index type other than b-tree\n                constraints[index]['type'] = Index.suffix\n                cursor.execute(\n                    \"SELECT sql FROM sqlite_master \"\n                    \"WHERE type='index' AND name=%s\" % self.connection.ops.quote_name(index)\n                )\n                orders = []\n                # There would be only 1 row to loop over\n                for sql, in cursor.fetchall():\n                    order_info = sql.split('(')[-1].split(')')[0].split(',')\n                    orders = ['DESC' if info.endswith('DESC') else 'ASC' for info in order_info]\n                constraints[index]['orders'] = orders\n        # Get the PK\n        pk_column = self.get_primary_key_column(cursor, table_name)\n        if pk_column:\n            # SQLite doesn't actually give a name to the PK constraint,\n            # so we invent one. This is fine, as the SQLite backend never\n            # deletes PK constraints by name, as you can't delete constraints\n            # in SQLite; we remake the table with a new PK instead.\n            constraints[\"__primary__\"] = {\n                \"columns\": [pk_column],\n                \"primary_key\": True,\n                \"unique\": False,  # It's not actually a unique constraint.\n                \"foreign_key\": False,\n                \"check\": False,\n                \"index\": False,\n            }\n        # Get foreign keys\n        cursor.execute('PRAGMA foreign_key_list(%s)' % self.connection.ops.quote_name(table_name))\n        for row in cursor.fetchall():\n            # Remaining on_update/on_delete/match values are of no interest here\n            id_, seq, table, from_, to = row[:5]\n            constraints['fk_%d' % id_] = {\n                'columns': [from_],\n                'primary_key': False,\n                'unique': False,\n                'foreign_key': (table, to),\n                'check': False,\n                'index': False,\n            }\n        return constraints"
        },
        {
          "file": "django/db/backends/sqlite3/schema.py",
          "type": "function",
          "name": "quote_value",
          "class_name": "DatabaseSchemaEditor",
          "code": "def quote_value(self, value):\n        # The backend \"mostly works\" without this function and there are use\n        # cases for compiling Python without the sqlite3 libraries (e.g.\n        # security hardening).\n        try:\n            import sqlite3\n            value = sqlite3.adapt(value)\n        except ImportError:\n            pass\n        except sqlite3.ProgrammingError:\n            pass\n        # Manual emulation of SQLite parameter quoting\n        if isinstance(value, bool):\n            return str(int(value))\n        elif isinstance(value, (Decimal, float, int)):\n            return str(value)\n        elif isinstance(value, str):\n            return \"'%s'\" % value.replace(\"\\'\", \"\\'\\'\")\n        elif value is None:\n            return \"NULL\"\n        elif isinstance(value, (bytes, bytearray, memoryview)):\n            # Bytes are only allowed for BLOB fields, encoded as string\n            # literals containing hexadecimal data and preceded by a single \"X\"\n            # character.\n            return \"X'%s'\" % value.hex()\n        else:\n            raise ValueError(\"Cannot quote parameter value %r of type %s\" % (value, type(value)))"
        },
        {
          "file": "django/db/backends/sqlite3/schema.py",
          "type": "function",
          "name": "alter_db_table",
          "class_name": "DatabaseSchemaEditor",
          "code": "def alter_db_table(self, model, old_db_table, new_db_table, disable_constraints=True):\n        if model._meta.related_objects and disable_constraints:\n            if self.connection.in_atomic_block:\n                raise NotSupportedError((\n                    'Renaming the %r table while in a transaction is not '\n                    'supported on SQLite because it would break referential '\n                    'integrity. Try adding `atomic = False` to the Migration class.'\n                ) % old_db_table)\n            self.connection.enable_constraint_checking()\n            super().alter_db_table(model, old_db_table, new_db_table)\n            self.connection.disable_constraint_checking()\n        else:\n            super().alter_db_table(model, old_db_table, new_db_table)"
        },
        {
          "file": "django/db/backends/sqlite3/schema.py",
          "type": "function",
          "name": "alter_field",
          "class_name": "DatabaseSchemaEditor",
          "code": "def alter_field(self, model, old_field, new_field, strict=False):\n        old_field_name = old_field.name\n        if (new_field.name != old_field_name and\n                any(r.field_name == old_field.name for r in model._meta.related_objects)):\n            if self.connection.in_atomic_block:\n                raise NotSupportedError((\n                    'Renaming the %r.%r column while in a transaction is not '\n                    'supported on SQLite because it would break referential '\n                    'integrity. Try adding `atomic = False` to the Migration class.'\n                ) % (model._meta.db_table, old_field_name))\n            with atomic(self.connection.alias):\n                super().alter_field(model, old_field, new_field, strict=strict)\n                # Follow SQLite's documented procedure for performing changes\n                # that don't affect the on-disk content.\n                # https://sqlite.org/lang_altertable.html#otheralter\n                with self.connection.cursor() as cursor:\n                    schema_version = cursor.execute('PRAGMA schema_version').fetchone()[0]\n                    cursor.execute('PRAGMA writable_schema = 1')\n                    table_name = model._meta.db_table\n                    references_template = ' REFERENCES \"%s\" (\"%%s\") ' % table_name\n                    old_column_name = old_field.get_attname_column()[1]\n                    new_column_name = new_field.get_attname_column()[1]\n                    search = references_template % old_column_name\n                    replacement = references_template % new_column_name\n                    cursor.execute('UPDATE sqlite_master SET sql = replace(sql, %s, %s)', (search, replacement))\n                    cursor.execute('PRAGMA schema_version = %d' % (schema_version + 1))\n                    cursor.execute('PRAGMA writable_schema = 0')\n                    # The integrity check will raise an exception and rollback\n                    # the transaction if the sqlite_master updates corrupt the\n                    # database.\n                    cursor.execute('PRAGMA integrity_check')\n            # Perform a VACUUM to refresh the database representation from\n            # the sqlite_master table.\n            with self.connection.cursor() as cursor:\n                cursor.execute('VACUUM')\n        else:\n            super().alter_field(model, old_field, new_field, strict=strict)"
        }
      ]
    },
    {
      "pr_number": 9412,
      "pr_title": "Fixed #28876 -- Fixed class-based model indexes and constraints names for models with quoted db_table.",
      "pr_body": "https://code.djangoproject.com/ticket/28876",
      "issue_id": 28876,
      "issue_title": "badly formed Table Index and constraint's name when  `Meta.db_table` is set with quoted string",
      "issue_body": "2 cases where the error occur .\nalways because it was set a\nMeta.db_table\nquoted custom name\nThe same models.Model was tested twice, with index been set with\nfield.db_index\nand\nMeta.indexes\nIt was tested with\nDjango 1.11\nPostgreSQL\nModel\nclass\nPublisher\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nverbose_name\n=\n'publisher name'\n,\nmax_length\n=\n50\n,\nnull\n=\nFalse\n)\nclass\nMeta\n:\ndb_table\n=\n'\"tbl_publisher_litle_name\"'\nget_latest_by\n=\n\"name\"\nordering\n=\n[\n'name'\n,\n]\nverbose_name\n=\n'Publiser'\nverbose_name_plural\n=\n'Publishers'\nindexes\n=\n[\nmodels\n.\nIndex\n(\nfields\n=\n[\n'name'\n,\n]),\n]\nMigration traceback\nOperations\nto\nperform:\nApply\nall\nmigrations:\nadmin,\nauth,\ncontenttypes,\nsessions,\ntestapp\nRunning\nmigrations:\nApplying\ncontenttypes.0001_initial...\nOK\nApplying\nauth.0001_initial...\nOK\nApplying\nadmin.0001_initial...\nOK\nApplying\nadmin.0002_logentry_remove_auto_add...\nOK\nApplying\ncontenttypes.0002_remove_content_type_name...\nOK\nApplying\nauth.0002_alter_permission_name_max_length...\nOK\nApplying\nauth.0003_alter_user_email_max_length...\nOK\nApplying\nauth.0004_alter_user_username_opts...\nOK\nApplying\nauth.0005_alter_user_last_login_null...\nOK\nApplying\nauth.0006_require_contenttypes_0002...\nOK\nApplying\nauth.0007_alter_validators_add_error_messages...\nOK\nApplying\nauth.0008_alter_user_username_max_length...\nOK\nApplying\nsessions.0001_initial...\nOK\nApplying\ntestapp.0001_initial...Traceback\n(\nmost\nrecent\ncall\nlast\n)\n:\nFile\n\"manage.py\"\n,\nline\n22\n,\nin\n<module>\nexecute_from_command_line\n(\nsys.argv\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/__init__.py\"\n,\nline\n363\n,\nin\nexecute_from_command_line\nutility.execute\n()\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/__init__.py\"\n,\nline\n355\n,\nin\nexecute\nself.fetch_command\n(\nsubcommand\n)\n.run_from_argv\n(\nself.argv\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/base.py\"\n,\nline\n283\n,\nin\nrun_from_argv\nself.execute\n(\n*args,\n**cmd_options\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/base.py\"\n,\nline\n330\n,\nin\nexecute\noutput\n=\nself.handle\n(\n*args,\n**options\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/commands/migrate.py\"\n,\nline\n204\n,\nin\nhandle\nfake_initial\n=\nfake_initial,\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n115\n,\nin\nmigrate\nstate\n=\nself._migrate_all_forwards\n(\nstate,\nplan,\nfull_plan,\nfake\n=\nfake,\nfake_initial\n=\nfake_initial\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n145\n,\nin\n_migrate_all_forwards\nstate\n=\nself.apply_migration\n(\nstate,\nmigration,\nfake\n=\nfake,\nfake_initial\n=\nfake_initial\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n244\n,\nin\napply_migration\nstate\n=\nmigration.apply\n(\nstate,\nschema_editor\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/migration.py\"\n,\nline\n129\n,\nin\napply\noperation.database_forwards\n(\nself.app_label,\nschema_editor,\nold_state,\nproject_state\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/operations/models.py\"\n,\nline\n785\n,\nin\ndatabase_forwards\nschema_editor.add_index\n(\nmodel,\nself.index\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/base/schema.py\"\n,\nline\n330\n,\nin\nadd_index\nself.execute\n(\nindex.create_sql\n(\nmodel,\nself\n))\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/base/schema.py\"\n,\nline\n119\n,\nin\nexecute\ncursor.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n80\n,\nin\nexecute\nreturn\nsuper\n(\nCursorDebugWrapper,\nself\n)\n.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n65\n,\nin\nexecute\nreturn\nself.cursor.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/utils.py\"\n,\nline\n94\n,\nin\n__exit__\nsix.reraise\n(\ndj_exc_type,\ndj_exc_value,\ntraceback\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n65\n,\nin\nexecute\nreturn\nself.cursor.execute\n(\nsql,\nparams\n)\ndjango.db.utils.ProgrammingError:\nzero-length\ndelimited\nidentifier\nat\nor\nnear\n\"\"\"\"\nLINE\n1\n:\nCREATE\nINDEX\n\"\"\ntbl_publis_name_b0b929_idx\n\" ON \"\ntbl_publisher...\n^\n(\ndj111\n)\n20171202\n.Sat16:15:27cadu>/Volumes/p10G/prj/dj_datadictionary_testproject/testproject>\nAnd again, when db_index is set on field ...\nthe migration is OK, but the error continues.\nModel with no Meta.indexes  but field.index=True\nclass\nPublisher\n(\nmodels\n.\nModel\n):\n\"\"\"\nBook's Author - author is a Book's model supplement.\n\"\"\"\nname\n=\nmodels\n.\nCharField\n(\nverbose_name\n=\n'publisher name'\n,\nmax_length\n=\n50\n,\nnull\n=\nFalse\n,\ndb_index\n=\nTrue\n)\nclass\nMeta\n:\ndb_table\n=\n'\"tbl_publisher_litle_name\"'\nget_latest_by\n=\n\"name\"\nordering\n=\n[\n'name'\n,\n]\nverbose_name\n=\n'Publiser'\nverbose_name_plural\n=\n'Publishers'\n# indexes = [\n#     models.Index(fields=['name', ]),\n# ]\ndef\n__unicode__\n(\nself\n):\nreturn\nself\n.\nname\nMigration\nmigrations\n.\nCreateModel\n(\nname\n=\n'Publisher'\n,\nfields\n=\n[\n(\n'id'\n,\nmodels\n.\nAutoField\n(\nauto_created\n=\nTrue\n,\nprimary_key\n=\nTrue\n,\nserialize\n=\nFalse\n,\nverbose_name\n=\n'ID'\n)),\n(\n'name'\n,\nmodels\n.\nCharField\n(\ndb_index\n=\nTrue\n,\nmax_length\n=\n50\n,\nverbose_name\n=\n'publisher name'\n)),\n],\noptions\n=\n{\n'get_latest_by'\n:\n'name'\n,\n'ordering'\n:\n[\n'name'\n],\n'verbose_name_plural'\n:\n'Publishers'\n,\n'db_table'\n:\n'\"tbl_publisher_litle_name\"'\n,\n'verbose_name'\n:\n'Publiser'\n,\n},\n),\nmigrations\n.\nAddField\n(\nmodel_name\n=\n'book'\n,\nname\n=\n'publisher'\n,\nfield\n=\nmodels\n.\nForeignKey\n(\nblank\n=\nTrue\n,\nnull\n=\nTrue\n,\non_delete\n=\ndjango\n.\ndb\n.\nmodels\n.\ndeletion\n.\nCASCADE\n,\nto\n=\n'testapp.Publisher'\n),\n),\n]\nMigration \"2\" traceback\n$python\nmanage.py\nmakemigrations\nMigrations\nfor\n'testapp'\n:\ntestproject/testapp/migrations/0001_initial.py\n-\nCreate\nmodel\nAuthor\n-\nCreate\nmodel\nBook\n-\nCreate\nmodel\nBookOtherTitle\n-\nCreate\nmodel\nPublisher\n-\nAdd\nfield\npublisher\nto\nbook\n(\ndj111\n)\n20171202\n.Sat16:45:19cadu>/Volumes/p10G/prj/dj_datadictionary_testproject/testproject>\ncadu.\n[\n532\n]\n$python\nmanage.py\nmigrate\nOperations\nto\nperform:\nApply\nall\nmigrations:\nadmin,\nauth,\ncontenttypes,\nsessions,\ntestapp\nRunning\nmigrations:\nApplying\ncontenttypes.0001_initial...\nOK\nApplying\nauth.0001_initial...\nOK\nApplying\nadmin.0001_initial...\nOK\nApplying\nadmin.0002_logentry_remove_auto_add...\nOK\nApplying\ncontenttypes.0002_remove_content_type_name...\nOK\nApplying\nauth.0002_alter_permission_name_max_length...\nOK\nApplying\nauth.0003_alter_user_email_max_length...\nOK\nApplying\nauth.0004_alter_user_username_opts...\nOK\nApplying\nauth.0005_alter_user_last_login_null...\nOK\nApplying\nauth.0006_require_contenttypes_0002...\nOK\nApplying\nauth.0007_alter_validators_add_error_messages...\nOK\nApplying\nauth.0008_alter_user_username_max_length...\nOK\nApplying\nsessions.0001_initial...\nOK\nApplying\ntestapp.0001_initial...Traceback\n(\nmost\nrecent\ncall\nlast\n)\n:\nFile\n\"manage.py\"\n,\nline\n22\n,\nin\n<module>\nexecute_from_command_line\n(\nsys.argv\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/__init__.py\"\n,\nline\n363\n,\nin\nexecute_from_command_line\nutility.execute\n()\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/__init__.py\"\n,\nline\n355\n,\nin\nexecute\nself.fetch_command\n(\nsubcommand\n)\n.run_from_argv\n(\nself.argv\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/base.py\"\n,\nline\n283\n,\nin\nrun_from_argv\nself.execute\n(\n*args,\n**cmd_options\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/base.py\"\n,\nline\n330\n,\nin\nexecute\noutput\n=\nself.handle\n(\n*args,\n**options\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/core/management/commands/migrate.py\"\n,\nline\n204\n,\nin\nhandle\nfake_initial\n=\nfake_initial,\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n115\n,\nin\nmigrate\nstate\n=\nself._migrate_all_forwards\n(\nstate,\nplan,\nfull_plan,\nfake\n=\nfake,\nfake_initial\n=\nfake_initial\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n145\n,\nin\n_migrate_all_forwards\nstate\n=\nself.apply_migration\n(\nstate,\nmigration,\nfake\n=\nfake,\nfake_initial\n=\nfake_initial\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/migrations/executor.py\"\n,\nline\n244\n,\nin\napply_migration\nstate\n=\nmigration.apply\n(\nstate,\nschema_editor\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/base/schema.py\"\n,\nline\n92\n,\nin\n__exit__\nself.execute\n(\nsql\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/base/schema.py\"\n,\nline\n119\n,\nin\nexecute\ncursor.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n80\n,\nin\nexecute\nreturn\nsuper\n(\nCursorDebugWrapper,\nself\n)\n.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n65\n,\nin\nexecute\nreturn\nself.cursor.execute\n(\nsql,\nparams\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/utils.py\"\n,\nline\n94\n,\nin\n__exit__\nsix.reraise\n(\ndj_exc_type,\ndj_exc_value,\ntraceback\n)\nFile\n\"/Users/cadu/Envs/dj111/lib/python2.7/site-packages/django/db/backends/utils.py\"\n,\nline\n65\n,\nin\nexecute\nreturn\nself.cursor.execute\n(\nsql,\nparams\n)\ndjango.db.utils.ProgrammingError:\nsyntax\nerror\nat\nor\nnear\n\"table_bo\"\nLINE\n1\n:\n...TRAINT\n\"table_book_double_qu_book_id_2cd88caf_fk_\"\ntable_bo\n\" ...\n^",
      "issue_closed_at": "2017-12-05T16:01:30",
      "base_commit": "f2ec89691236b59f54f9cf41bccd79a5853af36c",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_rename_field_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _rename_field_sql(self, table, old_field, new_field, new_type):\n        return self.sql_rename_column % {\n            \"table\": self.quote_name(table),\n            \"old_column\": self.quote_name(old_field.column),\n            \"new_column\": self.quote_name(new_field.column),\n            \"type\": new_type,\n        }"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "create_fk_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def create_fk_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))"
        },
        {
          "file": "django/db/models/indexes.py",
          "type": "line",
          "name": "line 1",
          "code": "import hashlib\n\nfrom django.utils.encoding import force_bytes\n\n__all__ = ['Index']"
        },
        {
          "file": "django/db/models/indexes.py",
          "type": "function",
          "name": "set_name_with_model",
          "class_name": "Index",
          "code": "def set_name_with_model(self, model):\n        \"\"\"\n        Generate a unique name for the index.\n\n        The name is divided into 3 parts - table name (12 chars), field name\n        (8 chars) and unique hash + suffix (10 chars). Each part is made to\n        fit its size by truncating the excess length.\n        \"\"\"\n        table_name = model._meta.db_table\n        column_names = [model._meta.get_field(field_name).column for field_name, order in self.fields_orders]\n        column_names_with_order = [\n            (('-%s' if order else '%s') % column_name)\n            for column_name, (field_name, order) in zip(column_names, self.fields_orders)\n        ]\n        # The length of the parts of the name is based on the default max\n        # length of 30 characters.\n        hash_data = [table_name] + column_names_with_order + [self.suffix]\n        self.name = '%s_%s_%s' % (\n            table_name[:11],\n            column_names[0][:7],\n            '%s_%s' % (self._hash_generator(*hash_data), self.suffix),\n        )\n        assert len(self.name) <= self.max_name_length, (\n            'Index too long for multiple database support. Is self.suffix '\n            'longer than 3 characters?'\n        )\n        self.check_name()"
        }
      ]
    },
    {
      "pr_number": 8213,
      "pr_title": "Refs #27935 -- Fixed BrinIndex.max_name_length if a project's default database isn't PostgreSQL.",
      "pr_body": "",
      "issue_id": 27935,
      "issue_title": "BrinIndex crashes if name  > 30 characters",
      "issue_body": "BrinIndex.suffix ( django.contrib.postgres.indexes) longer 3 char, but Index.set_name_with_model method allowed 3 char in suffix\nAssertionError: Index too long for multiple database support. Is self.suffix longer than 3 characters?",
      "issue_closed_at": "2017-03-20T10:30:05",
      "base_commit": "6cb0a3ac2836c044987f7a0cca0d1e99d52e002e",
      "changes": [
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "line",
          "name": "line 1",
          "code": "from django.db import connection\nfrom django.db.models import Index\nfrom django.utils.functional import cached_property\n\n__all__ = ['BrinIndex', 'GinIndex']\n\n\nclass BrinIndex(Index):\n    suffix = 'brin'\n\n    def __init__(self, fields=[], name=None, pages_per_range=None):\n        if pages_per_range is not None and pages_per_range <= 0:"
        },
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "function",
          "name": "get_sql_create_template_values",
          "class_name": "BrinIndex",
          "code": "def get_sql_create_template_values(self, model, schema_editor, using):\n        parameters = super().get_sql_create_template_values(model, schema_editor, using=' USING brin')\n        if self.pages_per_range is not None:\n            parameters['extra'] = ' WITH (pages_per_range={})'.format(\n                schema_editor.quote_value(self.pages_per_range)) + parameters['extra']\n        return parameters"
        }
      ]
    },
    {
      "pr_number": 7682,
      "pr_title": "Fixed #27176 -- Raised an exception for reentrant calls to apps.populate",
      "pr_body": "Ticket: https://code.djangoproject.com/ticket/27176\r\n\r\n",
      "issue_id": 27176,
      "issue_title": "django.setup() should raise an exception instead of hanging on re-entrant calls",
      "issue_body": "#25864\nand\n#26152\ndiscuss issues with calling\ndjango.setup()\nin a file that is imported during the app-loading process. This is actually about\napps.populate()\n-- I wrote\ndjango.setup()\nin the title because it's the public API people are familiar with.\nActually there are three cases.\nIf\ndjango.setup()\nhas already completed successfully, calling it again returns immediately.\nif self.ready:\n            return\nIf\ndjango.setup()\nis running in another thread, calling it again returns as soon as that other thread has completed the process.\n# populate() might be called by two threads in parallel on servers\n        # that create threads before initializing the WSGI callable.\n        with self._lock:\n            if self.ready:\n                return\nIf\ndjango.setup()\nis running in the same thread, calling it must crash. Indeed,\ndjango.setup()\nisn't allowed to return until app-loading has completed. The first call cannot complete without going through the second one, which cannot return, so there's a deadlock. The code tries to detect this condition:\n# app_config should be pristine, otherwise the code below won't\n            # guarantee that the order matches the order in INSTALLED_APPS.\n            if self.app_configs:\n                raise RuntimeError(\"populate() isn't reentrant\")\n#26152\nsuggest that the code hangs instead of raising an exception. Indeed, the second call must block on\nwith self._lock:\n, since\nself._lock\nisn't reentrant. Making it a\nRLock\ninstead of a\nLock\nshould allow the second thread to proceed and to hit that exception.\nAlso the error condition is incorrect: it won't trigger if the reentrant call happens while importing the first app. (We're reasoning in the single threaded case, which is easy.) Instead\npopulate()\nshould set a private attribute before it starts importing apps and raise the exception if that attribute is set. Alternatively it could check if the\nRLock\nis being taken for the second time, but no public API for this appears to exist.",
      "issue_closed_at": "2017-02-25T14:56:31",
      "base_commit": "b427f0d674362d22c063852754914d9315cbc2fa",
      "changes": [
        {
          "file": "django/apps/registry.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Apps",
          "code": "def __init__(self, installed_apps=()):\n        # installed_apps is set to None when creating the master registry\n        # because it cannot be populated at that point. Other registries must\n        # provide a list of installed apps and are populated immediately.\n        if installed_apps is None and hasattr(sys.modules[__name__], 'apps'):\n            raise RuntimeError(\"You must supply an installed_apps argument.\")\n\n        # Mapping of app labels => model names => model classes. Every time a\n        # model is imported, ModelBase.__new__ calls apps.register_model which\n        # creates an entry in all_models. All imported models are registered,\n        # regardless of whether they're defined in an installed application\n        # and whether the registry has been populated. Since it isn't possible\n        # to reimport a module safely (it could reexecute initialization code)\n        # all_models is never overridden or reset.\n        self.all_models = defaultdict(OrderedDict)\n\n        # Mapping of labels to AppConfig instances for installed apps.\n        self.app_configs = OrderedDict()\n\n        # Stack of app_configs. Used to store the current state in\n        # set_available_apps and set_installed_apps.\n        self.stored_app_configs = []\n\n        # Whether the registry is populated.\n        self.apps_ready = self.models_ready = self.ready = False\n\n        # Lock for thread-safe population.\n        self._lock = threading.Lock()\n\n        # Maps (\"app_label\", \"modelname\") tuples to lists of functions to be\n        # called when the corresponding model is ready. Used by this class's\n        # `lazy_model_operation()` and `do_pending_operations()` methods.\n        self._pending_operations = defaultdict(list)\n\n        # Populate apps and models, unless it's the master registry.\n        if installed_apps is not None:\n            self.populate(installed_apps)"
        },
        {
          "file": "django/apps/registry.py",
          "type": "function",
          "name": "populate",
          "class_name": "Apps",
          "code": "def populate(self, installed_apps=None):\n        \"\"\"\n        Loads application configurations and models.\n\n        This method imports each application module and then each model module.\n\n        It is thread safe and idempotent, but not reentrant.\n        \"\"\"\n        if self.ready:\n            return\n\n        # populate() might be called by two threads in parallel on servers\n        # that create threads before initializing the WSGI callable.\n        with self._lock:\n            if self.ready:\n                return\n\n            # app_config should be pristine, otherwise the code below won't\n            # guarantee that the order matches the order in INSTALLED_APPS.\n            if self.app_configs:\n                raise RuntimeError(\"populate() isn't reentrant\")\n\n            # Phase 1: initialize app configs and import app modules.\n            for entry in installed_apps:\n                if isinstance(entry, AppConfig):\n                    app_config = entry\n                else:\n                    app_config = AppConfig.create(entry)\n                if app_config.label in self.app_configs:\n                    raise ImproperlyConfigured(\n                        \"Application labels aren't unique, \"\n                        \"duplicates: %s\" % app_config.label)\n\n                self.app_configs[app_config.label] = app_config\n                app_config.apps = self\n\n            # Check for duplicate app names.\n            counts = Counter(\n                app_config.name for app_config in self.app_configs.values())\n            duplicates = [\n                name for name, count in counts.most_common() if count > 1]\n            if duplicates:\n                raise ImproperlyConfigured(\n                    \"Application names aren't unique, \"\n                    \"duplicates: %s\" % \", \".join(duplicates))\n\n            self.apps_ready = True\n\n            # Phase 2: import models modules.\n            for app_config in self.app_configs.values():\n                app_config.import_models()\n\n            self.clear_cache()\n\n            self.models_ready = True\n\n            # Phase 3: run ready() methods of app configs.\n            for app_config in self.get_app_configs():\n                app_config.ready()\n\n            self.ready = True"
        },
        {
          "file": "django/apps/registry.py",
          "type": "function",
          "name": "set_installed_apps",
          "class_name": "Apps",
          "code": "def set_installed_apps(self, installed):\n        \"\"\"\n        Enables a different set of installed apps for get_app_config[s].\n\n        installed must be an iterable in the same format as INSTALLED_APPS.\n\n        set_installed_apps() must be balanced with unset_installed_apps(),\n        even if it exits with an exception.\n\n        Primarily used as a receiver of the setting_changed signal in tests.\n\n        This method may trigger new imports, which may add new models to the\n        registry of all imported models. They will stay in the registry even\n        after unset_installed_apps(). Since it isn't possible to replay\n        imports safely (eg. that could lead to registering listeners twice),\n        models are registered when they're imported and never removed.\n        \"\"\"\n        if not self.ready:\n            raise AppRegistryNotReady(\"App registry isn't ready yet.\")\n        self.stored_app_configs.append(self.app_configs)\n        self.app_configs = OrderedDict()\n        self.apps_ready = self.models_ready = self.ready = False\n        self.clear_cache()\n        self.populate(installed)"
        }
      ]
    }
  ]
}