{
  "instance_id": "django__django-16820",
  "repo": "django/django",
  "created_at": "2023-05-02T06:32:13Z",
  "problem_statement": "Squashing migrations with Meta.index_together -> indexes transition should remove deprecation warnings.\nDescription\n\t\nSquashing migrations with Meta.index_together -> Meta.indexes transition should remove deprecation warnings. As far as I'm aware, it's a 4.2 release blocker because you cannot get rid of the index_together deprecation warnings without rewriting migrations, see comment.\n",
  "patch": "diff --git a/django/db/migrations/operations/models.py b/django/db/migrations/operations/models.py\n--- a/django/db/migrations/operations/models.py\n+++ b/django/db/migrations/operations/models.py\n@@ -303,6 +303,71 @@ def reduce(self, operation, app_label):\n                         managers=self.managers,\n                     ),\n                 ]\n+        elif (\n+            isinstance(operation, IndexOperation)\n+            and self.name_lower == operation.model_name_lower\n+        ):\n+            if isinstance(operation, AddIndex):\n+                return [\n+                    CreateModel(\n+                        self.name,\n+                        fields=self.fields,\n+                        options={\n+                            **self.options,\n+                            \"indexes\": [\n+                                *self.options.get(\"indexes\", []),\n+                                operation.index,\n+                            ],\n+                        },\n+                        bases=self.bases,\n+                        managers=self.managers,\n+                    ),\n+                ]\n+            elif isinstance(operation, RemoveIndex):\n+                options_indexes = [\n+                    index\n+                    for index in self.options.get(\"indexes\", [])\n+                    if index.name != operation.name\n+                ]\n+                return [\n+                    CreateModel(\n+                        self.name,\n+                        fields=self.fields,\n+                        options={\n+                            **self.options,\n+                            \"indexes\": options_indexes,\n+                        },\n+                        bases=self.bases,\n+                        managers=self.managers,\n+                    ),\n+                ]\n+            elif isinstance(operation, RenameIndex) and operation.old_fields:\n+                options_index_together = {\n+                    fields\n+                    for fields in self.options.get(\"index_together\", [])\n+                    if fields != operation.old_fields\n+                }\n+                if options_index_together:\n+                    self.options[\"index_together\"] = options_index_together\n+                else:\n+                    self.options.pop(\"index_together\", None)\n+                return [\n+                    CreateModel(\n+                        self.name,\n+                        fields=self.fields,\n+                        options={\n+                            **self.options,\n+                            \"indexes\": [\n+                                *self.options.get(\"indexes\", []),\n+                                models.Index(\n+                                    fields=operation.old_fields, name=operation.new_name\n+                                ),\n+                            ],\n+                        },\n+                        bases=self.bases,\n+                        managers=self.managers,\n+                    ),\n+                ]\n         return super().reduce(operation, app_label)\n \n \n",
  "similar_bug_items": [
    {
      "pr_number": 6642,
      "pr_title": "Fixed #26647 -- Included the state of all applied migrations when migrating forward.",
      "pr_body": "Thanks Jasper Maes for the detailed report.\n",
      "issue_id": 26647,
      "issue_title": "Post migrate signal old content type model",
      "issue_body": "Since the post migrate signal has the apps argument and the update_contenttypes and create_permissions listeners use this apps argument to detect the ContentType model, there can be a bug when unapplying a migration and then applying it again. The listeners will use an old model with ContentType.name still present because this in the apps of the state that the migration executor returned.\nI made a\n​\ntestproject\nwere the error is replicated.\nThe tests for the first app will fail, but not for the second app. I tracked this down to the migration executor were in the crashing app the migrations are in front of the migration that removes the ContentType.name field in the full_plan variable, so the executor returns a state not including this ContentType model change.\nIf someone can push me in the right direction I am willing to make a patch for this issue.",
      "issue_closed_at": "2016-05-26T12:29:12",
      "base_commit": "30d110ef43d8a3c50ea8ec4e4fe49bd2bb859530",
      "changes": [
        {
          "file": "django/db/migrations/executor.py",
          "type": "function",
          "name": "_migrate_all_forwards",
          "class_name": "MigrationExecutor",
          "code": "def _migrate_all_forwards(self, plan, full_plan, fake, fake_initial):\n        \"\"\"\n        Take a list of 2-tuples of the form (migration instance, False) and\n        apply them in the order they occur in the full_plan.\n        \"\"\"\n        migrations_to_run = {m[0] for m in plan}\n        state = self._create_project_state()\n        applied_migrations = {\n            self.loader.graph.nodes[key] for key in self.loader.applied_migrations\n            if key in self.loader.graph.nodes\n        }\n        for migration, _ in full_plan:\n            if not migrations_to_run:\n                # We remove every migration that we applied from this set so\n                # that we can bail out once the last migration has been applied\n                # and don't always run until the very end of the migration\n                # process.\n                break\n            if migration in migrations_to_run:\n                if 'apps' not in state.__dict__:\n                    if self.progress_callback:\n                        self.progress_callback(\"render_start\")\n                    state.apps  # Render all -- performance critical\n                    if self.progress_callback:\n                        self.progress_callback(\"render_success\")\n                state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n                migrations_to_run.remove(migration)\n            elif migration in applied_migrations:\n                # Only mutate the state if the migration is actually applied\n                # to make sure the resulting state doesn't include changes\n                # from unrelated migrations.\n                migration.mutate_state(state, preserve=False)\n\n        return state"
        },
        {
          "file": "django/db/migrations/executor.py",
          "type": "function",
          "name": "_migrate_all_forwards",
          "class_name": "MigrationExecutor",
          "code": "def _migrate_all_forwards(self, plan, full_plan, fake, fake_initial):\n        \"\"\"\n        Take a list of 2-tuples of the form (migration instance, False) and\n        apply them in the order they occur in the full_plan.\n        \"\"\"\n        migrations_to_run = {m[0] for m in plan}\n        state = self._create_project_state()\n        applied_migrations = {\n            self.loader.graph.nodes[key] for key in self.loader.applied_migrations\n            if key in self.loader.graph.nodes\n        }\n        for migration, _ in full_plan:\n            if not migrations_to_run:\n                # We remove every migration that we applied from this set so\n                # that we can bail out once the last migration has been applied\n                # and don't always run until the very end of the migration\n                # process.\n                break\n            if migration in migrations_to_run:\n                if 'apps' not in state.__dict__:\n                    if self.progress_callback:\n                        self.progress_callback(\"render_start\")\n                    state.apps  # Render all -- performance critical\n                    if self.progress_callback:\n                        self.progress_callback(\"render_success\")\n                state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n                migrations_to_run.remove(migration)\n            elif migration in applied_migrations:\n                # Only mutate the state if the migration is actually applied\n                # to make sure the resulting state doesn't include changes\n                # from unrelated migrations.\n                migration.mutate_state(state, preserve=False)\n\n        return state"
        }
      ]
    },
    {
      "pr_number": 4738,
      "pr_title": "Fixed #24628 -- Fixed applied status for squashed migrations.",
      "pr_body": "",
      "issue_id": 24628,
      "issue_title": "Squash migration is not marked as applied when the migrations it replaces are",
      "issue_body": "(I am attempting a shorter and clearer description of this bug, but leaving the original description intact below - carljm).\nIn Django 1.8, consider an app\nA\nwith migrations\n1\nand\n2\nand a squashed migration\n1_squashed_2\nthat replaces both\n1\nand\n2\n. Consider a deployment of this app which has only\n1\napplied. When it receives the update including\n2\nand\n1_squashed_2\nand is migrated,\n2\nis marked as applied in the database but\n1_squashed_2\nis not.\nThis does not immediately appear to be a problem, because as long as migrations\n1\nand\n2\nexist and\n1_squashed_2\nis marked as replacing them, Django automatically considers\n1_squashed_2\nto be applied (and shows it as such in\nshowmigrations\n). But Django never actually records\n1_squashed_2\nitself as applied in the database.\nAt some point, once all deployments have migrated through\n2\n, the idea is that\n1\nand\n2\ncan be removed, and the\nreplaces\ntag removed from\n1_squashed_2\n. At this point we have a problem, because now Django considers\n1_squashed_2\nto not be applied, and tries to apply it again, causing errors because tables already exist, etc. The only solution is to manually\n--fake\nsquashed migrations on all deployments when their\nreplaces\ntag is removed, which is a pain and eliminates much of the convenience of the migration system;\n--fake\nshould not be necessary in the normal course of things.\nThe solution appears simple: anytime a migration is marked as having been applied, Django should check if it is part of a replaced set, and if that replaced set is now fully applied, the replacing migration should also be marked as applied.\nIt may be that this check should be performed by\nmigrate\neven when it hasn't applied any migrations in the current run, as that would help to correct the corruption already caused by this bug in many existing databases, but not yet noticed because the replaced migrations haven't yet been removed. (This correction should probably be mentioned to the user so it doesn't happen silently.)\nOriginal report follows:\nAfter some time in the chat with MarkusH, we decided that this should be a bug report.\nSome context: I have quite some migrations (33) of which the first already was a squash of several migrations. It did get it's 'replaced' field removed though, as the original migrations it replaced are long gone already. The goal now was to get that long list of migrations down to one again, for peace of mind.\nThis led to the following symptoms: When I created the squashed migration everything seemed fine. ./manage.py migrate did say that nothing was to be done (but then again, all the migrations to be squashed where already applied).\nBut adding another migration after that ended in tears - ./manage migrate didn't want to apply it and told me so in no uncertain terms.\n(pycess)dwt@atlan ~/Code/Projekte/pycess/pycess (git)-[master] % ./manage.py migrate\nTraceback (most recent call last):\n  File \"./manage.py\", line 10, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/core/management/__init__.py\", line 330, in execute_from_command_line\n    utility.execute()\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/core/management/__init__.py\", line 322, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/core/management/base.py\", line 347, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/core/management/base.py\", line 398, in execute\n    output = self.handle(*args, **options)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/core/management/commands/migrate.py\", line 86, in handle\n    executor = MigrationExecutor(connection, self.migration_progress_callback)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/executor.py\", line 19, in __init__\n    self.loader = MigrationLoader(self.connection)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/loader.py\", line 47, in __init__\n    self.build_graph()\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/loader.py\", line 281, in build_graph\n    _reraise_missing_dependency(migration, parent, e)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/loader.py\", line 264, in _reraise_missing_dependency\n    raise exc\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/loader.py\", line 274, in build_graph\n    self.graph.add_dependency(migration, key, parent)\n  File \"/Users/dwt/Code/Projekte/pycess/django/django/db/migrations/graph.py\", line 124, in add_dependency\n    parent\ndjango.db.migrations.graph.NodeNotFoundError: Migration process.0002_auto_20150411_1005 dependencies reference nonexistent parent node ('process', '0001_squashed_initial_2')\nHere we had quite some discussion in #django-dev with MarkusH, of which the result to me was that a) django doesn't seem to ever record that a squashed migration is applied, at least as long as it is recognizable by django as a squashed migration (i.e. it has a .replaced property). b) there seems to be no way to tell django that the replacing squashed migration really is already applied for this database.\nThe last one turned out to be achievable if you remove the old migrations and the .replaces property from the squashed migration and then apply it with --fake\nMarkusH might want to say more here that he can describe better.\nFor ease of reproduction I'm attaching the project where this occurred for me.",
      "issue_closed_at": "2015-06-02T17:18:16",
      "base_commit": "23048d186ce0041654a9f547fe3e7177efce3076",
      "changes": [
        {
          "file": "django/db/migrations/executor.py",
          "type": "function",
          "name": "migrate",
          "class_name": "MigrationExecutor",
          "code": "def migrate(self, targets, plan=None, fake=False, fake_initial=False):\n        \"\"\"\n        Migrates the database up to the given targets.\n\n        Django first needs to create all project states before a migration is\n        (un)applied and in a second step run all the database operations.\n        \"\"\"\n        if plan is None:\n            plan = self.migration_plan(targets)\n        migrations_to_run = {m[0] for m in plan}\n        # Create the forwards plan Django would follow on an empty database\n        full_plan = self.migration_plan(self.loader.graph.leaf_nodes(), clean_start=True)\n        # Holds all states right before a migration is applied\n        # if the migration is being run.\n        states = {}\n        state = ProjectState(real_apps=list(self.loader.unmigrated_apps))\n        if self.progress_callback:\n            self.progress_callback(\"render_start\")\n        # Phase 1 -- Store all project states of migrations right before they\n        # are applied. The first migration that will be applied in phase 2 will\n        # trigger the rendering of the initial project state. From this time on\n        # models will be recursively reloaded as explained in\n        # `django.db.migrations.state.get_related_models_recursive()`.\n        for migration, _ in full_plan:\n            if not migrations_to_run:\n                # We remove every migration whose state was already computed\n                # from the set below (`migrations_to_run.remove(migration)`).\n                # If no states for migrations must be computed, we can exit\n                # this loop. Migrations that occur after the latest migration\n                # that is about to be applied would only trigger unneeded\n                # mutate_state() calls.\n                break\n            do_run = migration in migrations_to_run\n            if do_run:\n                if 'apps' not in state.__dict__:\n                    state.apps  # Render all real_apps -- performance critical\n                states[migration] = state.clone()\n                migrations_to_run.remove(migration)\n            # Only preserve the state if the migration is being run later\n            state = migration.mutate_state(state, preserve=do_run)\n        if self.progress_callback:\n            self.progress_callback(\"render_success\")\n        # Phase 2 -- Run the migrations\n        for migration, backwards in plan:\n            if not backwards:\n                self.apply_migration(states[migration], migration, fake=fake, fake_initial=fake_initial)\n            else:\n                self.unapply_migration(states[migration], migration, fake=fake)"
        },
        {
          "file": "django/db/migrations/executor.py",
          "type": "function",
          "name": "unapply_migration",
          "class_name": "MigrationExecutor",
          "code": "def unapply_migration(self, state, migration, fake=False):\n        \"\"\"\n        Runs a migration backwards.\n        \"\"\"\n        if self.progress_callback:\n            self.progress_callback(\"unapply_start\", migration, fake)\n        if not fake:\n            with self.connection.schema_editor() as schema_editor:\n                state = migration.unapply(state, schema_editor)\n        # For replacement migrations, record individual statuses\n        if migration.replaces:\n            for app_label, name in migration.replaces:\n                self.recorder.record_unapplied(app_label, name)\n        else:\n            self.recorder.record_unapplied(migration.app_label, migration.name)\n        # Report progress\n        if self.progress_callback:\n            self.progress_callback(\"unapply_success\", migration, fake)\n        return state"
        },
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "build_graph",
          "class_name": "MigrationLoader",
          "code": "def build_graph(self):\n        \"\"\"\n        Builds a migration dependency graph using both the disk and database.\n        You'll need to rebuild the graph if you apply migrations. This isn't\n        usually a problem as generally migration stuff runs in a one-shot process.\n        \"\"\"\n        # Load disk data\n        self.load_disk()\n        # Load database data\n        if self.connection is None:\n            self.applied_migrations = set()\n        else:\n            recorder = MigrationRecorder(self.connection)\n            self.applied_migrations = recorder.applied_migrations()\n        # Do a first pass to separate out replacing and non-replacing migrations\n        normal = {}\n        replacing = {}\n        for key, migration in self.disk_migrations.items():\n            if migration.replaces:\n                replacing[key] = migration\n            else:\n                normal[key] = migration\n        # Calculate reverse dependencies - i.e., for each migration, what depends on it?\n        # This is just for dependency re-pointing when applying replacements,\n        # so we ignore run_before here.\n        reverse_dependencies = {}\n        for key, migration in normal.items():\n            for parent in migration.dependencies:\n                reverse_dependencies.setdefault(parent, set()).add(key)\n        # Remember the possible replacements to generate more meaningful error\n        # messages\n        reverse_replacements = {}\n        for key, migration in replacing.items():\n            for replaced in migration.replaces:\n                reverse_replacements.setdefault(replaced, set()).add(key)\n        # Carry out replacements if we can - that is, if all replaced migrations\n        # are either unapplied or missing.\n        for key, migration in replacing.items():\n            # Ensure this replacement migration is not in applied_migrations\n            self.applied_migrations.discard(key)\n            # Do the check. We can replace if all our replace targets are\n            # applied, or if all of them are unapplied.\n            applied_statuses = [(target in self.applied_migrations) for target in migration.replaces]\n            can_replace = all(applied_statuses) or (not any(applied_statuses))\n            if not can_replace:\n                continue\n            # Alright, time to replace. Step through the replaced migrations\n            # and remove, repointing dependencies if needs be.\n            for replaced in migration.replaces:\n                if replaced in normal:\n                    # We don't care if the replaced migration doesn't exist;\n                    # the usage pattern here is to delete things after a while.\n                    del normal[replaced]\n                for child_key in reverse_dependencies.get(replaced, set()):\n                    if child_key in migration.replaces:\n                        continue\n                    # List of migrations whose dependency on `replaced` needs\n                    # to be updated to a dependency on `key`.\n                    to_update = []\n                    # Child key may itself be replaced, in which case it might\n                    # not be in `normal` anymore (depending on whether we've\n                    # processed its replacement yet). If it's present, we go\n                    # ahead and update it; it may be deleted later on if it is\n                    # replaced, but there's no harm in updating it regardless.\n                    if child_key in normal:\n                        to_update.append(normal[child_key])\n                    # If the child key is replaced, we update its replacement's\n                    # dependencies too, if necessary. (We don't know if this\n                    # replacement will actually take effect or not, but either\n                    # way it's OK to update the replacing migration).\n                    if child_key in reverse_replacements:\n                        for replaces_child_key in reverse_replacements[child_key]:\n                            if replaced in replacing[replaces_child_key].dependencies:\n                                to_update.append(replacing[replaces_child_key])\n                    # Actually perform the dependency update on all migrations\n                    # that require it.\n                    for migration_needing_update in to_update:\n                        migration_needing_update.dependencies.remove(replaced)\n                        migration_needing_update.dependencies.append(key)\n            normal[key] = migration\n            # Mark the replacement as applied if all its replaced ones are\n            if all(applied_statuses):\n                self.applied_migrations.add(key)\n        # Finally, make a graph and load everything into it\n        self.graph = MigrationGraph()\n        for key, migration in normal.items():\n            self.graph.add_node(key, migration)\n\n        def _reraise_missing_dependency(migration, missing, exc):\n            \"\"\"\n            Checks if ``missing`` could have been replaced by any squash\n            migration but wasn't because the the squash migration was partially\n            applied before. In that case raise a more understandable exception.\n\n            #23556\n            \"\"\"\n            if missing in reverse_replacements:\n                candidates = reverse_replacements.get(missing, set())\n                is_replaced = any(candidate in self.graph.nodes for candidate in candidates)\n                if not is_replaced:\n                    tries = ', '.join('%s.%s' % c for c in candidates)\n                    exc_value = NodeNotFoundError(\n                        \"Migration {0} depends on nonexistent node ('{1}', '{2}'). \"\n                        \"Django tried to replace migration {1}.{2} with any of [{3}] \"\n                        \"but wasn't able to because some of the replaced migrations \"\n                        \"are already applied.\".format(\n                            migration, missing[0], missing[1], tries\n                        ),\n                        missing)\n                    exc_value.__cause__ = exc\n                    six.reraise(NodeNotFoundError, exc_value, sys.exc_info()[2])\n            raise exc\n\n        # Add all internal dependencies first to ensure __first__ dependencies\n        # find the correct root node.\n        for key, migration in normal.items():\n            for parent in migration.dependencies:\n                if parent[0] != key[0] or parent[1] == '__first__':\n                    # Ignore __first__ references to the same app (#22325)\n                    continue\n                try:\n                    self.graph.add_dependency(migration, key, parent)\n                except NodeNotFoundError as e:\n                    # Since we added \"key\" to the nodes before this implies\n                    # \"parent\" is not in there. To make the raised exception\n                    # more understandable we check if parent could have been\n                    # replaced but hasn't (eg partially applied squashed\n                    # migration)\n                    _reraise_missing_dependency(migration, parent, e)\n        for key, migration in normal.items():\n            for parent in migration.dependencies:\n                if parent[0] == key[0]:\n                    # Internal dependencies already added.\n                    continue\n                parent = self.check_key(parent, key[0])\n                if parent is not None:\n                    try:\n                        self.graph.add_dependency(migration, key, parent)\n                    except NodeNotFoundError as e:\n                        # Since we added \"key\" to the nodes before this implies\n                        # \"parent\" is not in there.\n                        _reraise_missing_dependency(migration, parent, e)\n            for child in migration.run_before:\n                child = self.check_key(child, key[0])\n                if child is not None:\n                    try:\n                        self.graph.add_dependency(migration, child, key)\n                    except NodeNotFoundError as e:\n                        # Since we added \"key\" to the nodes before this implies\n                        # \"child\" is not in there.\n                        _reraise_missing_dependency(migration, child, e)"
        }
      ]
    },
    {
      "pr_number": 9259,
      "pr_title": "Fixed #28723 -- RelatedManager.get_prefetch_queryset returns \"wrong\" cache_name",
      "pr_body": "https://code.djangoproject.com/ticket/28723",
      "issue_id": 28723,
      "issue_title": "RelatedManager.get_prefetch_queryset returns \"wrong\" cache_name",
      "issue_body": "Currently,\nRelatedManager.get_prefetch_queryset\nreturns\nself.field.related_query_name()\nas the\ncache_name\n.  In the case where no\nrelated_name\nhas been set on the\nForeignKey\n, then this does not match with the\nthrough_attr\nused by\nget_prefetcher\n.\nUsing the models in\ntests/prefetch_related/models.py\n,\nBookWithYear\n.\nobjects\n.\nprefetch_related\n(\n'bookreview_set'\n)\nwill use a\nthrough_attr\nof\n\"bookreview_set\"\n, but\n\"bookreview\"\nis what will be placed in\n_prefetched_objects_cache\n.\nI think\nrelated_manager.field.remote_field.get_accessor_name()\nshould be used instead.\n​\nhttps://github.com/django/django/pull/9259\nis a pull request which fixes this issue",
      "issue_closed_at": "2018-02-07T14:29:42",
      "base_commit": "ef718a72b3db81d35a6c1273b1565b48dd867e90",
      "changes": [
        {
          "file": "django/db/models/fields/related_descriptors.py",
          "type": "function",
          "name": "_remove_prefetched_objects",
          "class_name": "ManyRelatedManager",
          "code": "def _remove_prefetched_objects(self):\n            try:\n                self.instance._prefetched_objects_cache.pop(self.prefetch_cache_name)\n            except (AttributeError, KeyError):\n                pass"
        },
        {
          "file": "django/db/models/fields/related_descriptors.py",
          "type": "function",
          "name": "get_prefetch_queryset",
          "class_name": "ManyRelatedManager",
          "code": "def get_prefetch_queryset(self, instances, queryset=None):\n            if queryset is None:\n                queryset = super().get_queryset()\n\n            queryset._add_hints(instance=instances[0])\n            queryset = queryset.using(queryset._db or self._db)\n\n            query = {'%s__in' % self.query_field_name: instances}\n            queryset = queryset._next_is_sticky().filter(**query)\n\n            # M2M: need to annotate the query in order to get the primary model\n            # that the secondary model was actually related to. We know that\n            # there will already be a join on the join table, so we can just add\n            # the select.\n\n            # For non-autocreated 'through' models, can't assume we are\n            # dealing with PK values.\n            fk = self.through._meta.get_field(self.source_field_name)\n            join_table = fk.model._meta.db_table\n            connection = connections[queryset.db]\n            qn = connection.ops.quote_name\n            queryset = queryset.extra(select={\n                '_prefetch_related_val_%s' % f.attname:\n                '%s.%s' % (qn(join_table), qn(f.column)) for f in fk.local_related_fields})\n            return (\n                queryset,\n                lambda result: tuple(\n                    getattr(result, '_prefetch_related_val_%s' % f.attname)\n                    for f in fk.local_related_fields\n                ),\n                lambda inst: tuple(\n                    f.get_db_prep_value(getattr(inst, f.attname), connection)\n                    for f in fk.foreign_related_fields\n                ),\n                False,\n                self.prefetch_cache_name,\n                False,\n            )"
        }
      ]
    },
    {
      "pr_number": 2938,
      "pr_title": "Fixed #23071 -- Use last migration's name in dependency to other app",
      "pr_body": "Changed the autodetector to lookup the name of the other app's last\nmigration in the graph and use that as dependency instead of using\n**latest**.\n",
      "issue_id": 23071,
      "issue_title": "Can't create migration for apps that have ForeignKeys to each other",
      "issue_body": "ForeignKey's to other apps create a dependency on\n__latest__\nmigration of that app. Because of this we can't create migrations where we have ForeignKey relations that go both ways, because that would result in app1 depending on\n__latest__\nof app2 and app2 depending on\n__latest__\nof app1. Way to reproduce:\nCreate the following model in myapp1:\nclass Model1(models.Model):\n    field = models.CharField(max_length=10)\nAnd the following model in myapp2:\nclass Model2(models.Model):\n    field = models.CharField(max_length=10)\nWe run makemigations creating both initial migrations. We then add the following model in myapp1:\nclass Model3(models.Model):\n    model2 = models.ForeignKey('myapp2.Model2')\nWe run makemigrations again, this will create a myapp1 migration that depends on\n__latest__\nof myapp2.\nThen we add the following model to myapp2:\nclass Model4(models.Model):\n    model1 = models.ForeignKey('myapp1.Model1')\nWe then run makemigrations again, this will create a myapp2 migration that depends on\n__latest__\nof myapp1. Running migrate then gives us a circular dependency error:\ndjango.db.migrations.graph.CircularDependencyError: [('myapp1', u'0002_model3'), ('myapp2', u'0002_model4'), ('myapp1', u'0002_model3')]\nI think the correct way to create dependency would be to create an explicit dependency on the latest migration of the other app at the time of creating the new ForeigKey instead of using\n__latest__\n. This will mean that the second myapp1 migration will depend on myapp2's 0001_initial, and the second myapp2 migration will depend on myapp1's 0002_model3.\nI tested this and it seems to work fine.",
      "issue_closed_at": "2014-07-25T10:54:21",
      "base_commit": "b4cf7e3d1de2d9700812872b04f6fe8eb88e8bff",
      "changes": [
        {
          "file": "django/db/migrations/autodetector.py",
          "type": "function",
          "name": "_detect_changes",
          "class_name": "MigrationAutodetector",
          "code": "def _detect_changes(self, convert_apps=None, graph=None):\n        \"\"\"\n        Returns a dict of migration plans which will achieve the\n        change from from_state to to_state. The dict has app labels\n        as keys and a list of migrations as values.\n\n        The resulting migrations aren't specially named, but the names\n        do matter for dependencies inside the set.\n\n        convert_apps is the list of apps to convert to use migrations\n        (i.e. to make initial migrations for, in the usual case)\n\n        graph is an optional argument that, if provided, can help improve\n        dependency generation and avoid potential circular dependencies.\n        \"\"\"\n\n        # The first phase is generating all the operations for each app\n        # and gathering them into a big per-app list.\n        # We'll then go through that list later and order it and split\n        # into migrations to resolve dependencies caused by M2Ms and FKs.\n        self.generated_operations = {}\n\n        # Prepare some old/new state and model lists, separating\n        # proxy models and ignoring unmigrated apps.\n        self.old_apps = self.from_state.render(ignore_swappable=True)\n        self.new_apps = self.to_state.render()\n        self.old_model_keys = []\n        self.old_proxy_keys = []\n        self.new_model_keys = []\n        self.new_proxy_keys = []\n        for al, mn in sorted(self.from_state.models.keys()):\n            model = self.old_apps.get_model(al, mn)\n            if model._meta.managed and al not in self.from_state.real_apps:\n                if model._meta.proxy:\n                    self.old_proxy_keys.append((al, mn))\n                else:\n                    self.old_model_keys.append((al, mn))\n\n        for al, mn in sorted(self.to_state.models.keys()):\n            model = self.new_apps.get_model(al, mn)\n            if model._meta.managed and (\n                al not in self.from_state.real_apps or\n                (convert_apps and al in convert_apps)\n            ):\n                if model._meta.proxy:\n                    self.new_proxy_keys.append((al, mn))\n                else:\n                    self.new_model_keys.append((al, mn))\n\n        # Renames have to come first\n        self.generate_renamed_models()\n\n        # Prepare field lists, and prepare a list of the fields that used\n        # through models in the old state so we can make dependencies\n        # from the through model deletion to the field that uses it.\n        self.kept_model_keys = set(self.old_model_keys).intersection(self.new_model_keys)\n        self.through_users = {}\n        self.old_field_keys = set()\n        self.new_field_keys = set()\n        for app_label, model_name in sorted(self.kept_model_keys):\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_model_state = self.from_state.models[app_label, old_model_name]\n            new_model_state = self.to_state.models[app_label, model_name]\n            self.old_field_keys.update((app_label, model_name, x) for x, y in old_model_state.fields)\n            self.new_field_keys.update((app_label, model_name, x) for x, y in new_model_state.fields)\n\n        # Through model map generation\n        for app_label, model_name in sorted(self.old_model_keys):\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_model_state = self.from_state.models[app_label, old_model_name]\n            for field_name, field in old_model_state.fields:\n                old_field = self.old_apps.get_model(app_label, old_model_name)._meta.get_field_by_name(field_name)[0]\n                if hasattr(old_field, \"rel\") and getattr(old_field.rel, \"through\", None) and not old_field.rel.through._meta.auto_created:\n                    through_key = (\n                        old_field.rel.through._meta.app_label,\n                        old_field.rel.through._meta.object_name.lower(),\n                    )\n                    self.through_users[through_key] = (app_label, old_model_name, field_name)\n\n        # Generate non-rename model operations\n        self.generate_created_models()\n        self.generate_deleted_models()\n        self.generate_created_proxies()\n        self.generate_deleted_proxies()\n        self.generate_altered_options()\n\n        # Generate field operations\n        self.generate_added_fields()\n        self.generate_removed_fields()\n        self.generate_altered_fields()\n        self.generate_altered_unique_together()\n        self.generate_altered_index_together()\n        self.generate_altered_order_with_respect_to()\n\n        # Now, reordering to make things possible. The order we have already\n        # isn't bad, but we need to pull a few things around so FKs work nicely\n        # inside the same app\n        for app_label, ops in sorted(self.generated_operations.items()):\n            for i in range(10000):\n                found = False\n                for i, op in enumerate(ops):\n                    for dep in op._auto_deps:\n                        if dep[0] == app_label:\n                            # Alright, there's a dependency on the same app.\n                            for j, op2 in enumerate(ops):\n                                if self.check_dependency(op2, dep) and j > i:\n                                    ops = ops[:i] + ops[i + 1:j + 1] + [op] + ops[j + 1:]\n                                    found = True\n                                    break\n                        if found:\n                            break\n                    if found:\n                        break\n                if not found:\n                    break\n            else:\n                raise ValueError(\"Infinite loop caught in operation dependency resolution\")\n            self.generated_operations[app_label] = ops\n\n        # Now, we need to chop the lists of operations up into migrations with\n        # dependencies on each other.\n        # We do this by stepping up an app's list of operations until we\n        # find one that has an outgoing dependency that isn't in another app's\n        # migration yet (hasn't been chopped off its list). We then chop off the\n        # operations before it into a migration and move onto the next app.\n        # If we loop back around without doing anything, there's a circular\n        # dependency (which _should_ be impossible as the operations are all\n        # split at this point so they can't depend and be depended on)\n\n        self.migrations = {}\n        num_ops = sum(len(x) for x in self.generated_operations.values())\n        chop_mode = False\n        while num_ops:\n            # On every iteration, we step through all the apps and see if there\n            # is a completed set of operations.\n            # If we find that a subset of the operations are complete we can\n            # try to chop it off from the rest and continue, but we only\n            # do this if we've already been through the list once before\n            # without any chopping and nothing has changed.\n            for app_label in sorted(self.generated_operations.keys()):\n                chopped = []\n                dependencies = set()\n                for operation in list(self.generated_operations[app_label]):\n                    deps_satisfied = True\n                    operation_dependencies = set()\n                    for dep in operation._auto_deps:\n                        if dep[0] == \"__setting__\":\n                            operation_dependencies.add((dep[0], dep[1]))\n                        elif dep[0] != app_label:\n                            # External app dependency. See if it's not yet\n                            # satisfied.\n                            for other_operation in self.generated_operations.get(dep[0], []):\n                                if self.check_dependency(other_operation, dep):\n                                    deps_satisfied = False\n                                    break\n                            if not deps_satisfied:\n                                break\n                            else:\n                                if self.migrations.get(dep[0], None):\n                                    operation_dependencies.add((dep[0], self.migrations[dep[0]][-1].name))\n                                else:\n                                    # If we can't find the other app, we add a first/last dependency,\n                                    # but only if we've already been through once and checked everything\n                                    if chop_mode:\n                                        # If the app already exists, we add __latest__, as we don't know which\n                                        # migration contains the target field.\n                                        # If it's not yet migrated or has no migrations, we use __first__\n                                        if graph and not graph.root_nodes(dep[0]):\n                                            operation_dependencies.add((dep[0], \"__first__\"))\n                                        else:\n                                            operation_dependencies.add((dep[0], \"__latest__\"))\n                                    else:\n                                        deps_satisfied = False\n                    if deps_satisfied:\n                        chopped.append(operation)\n                        dependencies.update(operation_dependencies)\n                        self.generated_operations[app_label] = self.generated_operations[app_label][1:]\n                    else:\n                        break\n                # Make a migration! Well, only if there's stuff to put in it\n                if dependencies or chopped:\n                    if not self.generated_operations[app_label] or chop_mode:\n                        subclass = type(str(\"Migration\"), (Migration,), {\"operations\": [], \"dependencies\": []})\n                        instance = subclass(\"auto_%i\" % (len(self.migrations.get(app_label, [])) + 1), app_label)\n                        instance.dependencies = list(dependencies)\n                        instance.operations = chopped\n                        self.migrations.setdefault(app_label, []).append(instance)\n                        chop_mode = False\n                    else:\n                        self.generated_operations[app_label] = chopped + self.generated_operations[app_label]\n            new_num_ops = sum(len(x) for x in self.generated_operations.values())\n            if new_num_ops == num_ops:\n                if not chop_mode:\n                    chop_mode = True\n                else:\n                    raise ValueError(\"Cannot resolve operation dependencies\")\n            num_ops = new_num_ops\n\n        # OK, add in internal dependencies among the migrations\n        for app_label, migrations in self.migrations.items():\n            for m1, m2 in zip(migrations, migrations[1:]):\n                m2.dependencies.append((app_label, m1.name))\n\n        # De-dupe dependencies\n        for app_label, migrations in self.migrations.items():\n            for migration in migrations:\n                migration.dependencies = list(set(migration.dependencies))\n\n        # Optimize migrations\n        for app_label, migrations in self.migrations.items():\n            for migration in migrations:\n                migration.operations = MigrationOptimizer().optimize(migration.operations, app_label=app_label)\n\n        return self.migrations"
        },
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "get_migration_by_prefix",
          "class_name": "MigrationLoader",
          "code": "def get_migration_by_prefix(self, app_label, name_prefix):\n        \"Returns the migration(s) which match the given app label and name _prefix_\"\n        # Do the search\n        results = []\n        for l, n in self.disk_migrations:\n            if l == app_label and n.startswith(name_prefix):\n                results.append((l, n))\n        if len(results) > 1:\n            raise AmbiguityError(\"There is more than one migration for '%s' with the prefix '%s'\" % (app_label, name_prefix))\n        elif len(results) == 0:\n            raise KeyError(\"There no migrations for '%s' with the prefix '%s'\" % (app_label, name_prefix))\n        else:\n            return self.disk_migrations[results[0]]"
        }
      ]
    },
    {
      "pr_number": 14727,
      "pr_title": "Fixed #29063 -- Fixed migrate crash when specifying a name of partially applied squashed migrations.",
      "pr_body": "ticket-29063\r\n\r\nSupersedes #14643, which was merely an attempt to spruce up the `NodeNotFoundError` to be more informative.\r\n\r\nThe original ticket does not say with specificity how to reproduce the issue. Mariusz demonstrated on #14643 that several related scenarios already have sufficient messaging. The scenario in this regression test--attempting to migrate to an incompletely applied squashed migration, and moreover, specifying it by name--is one we can fix at the source.\r\n\r\nGiven that, I think we can close the ticket with this change. (While an understandable suggestion, I don't think we need to be raising warnings, as the original report mused.)",
      "issue_id": 29063,
      "issue_title": "Naming an incompletely applied squashed migration as a migration target fails with bare NodeNotFoundError",
      "issue_body": "In Line 205-208 in django/db/migrations/loader.py replacement migrations (created with squash) are checked if they can be applied. If any of the to be replaced migrations isn't already applied the replacement migration is not added to the nodes list.\nThis leads to the fact that if some of the migrations are removed or not completely applied before the squash is added and there is a dependency on the replacement migration, the user gets a 'NodeNotFoundError' where the replacement migration that is not being applied because of line 206 is the missing one.\nThis is very confusing to the user, raising a warning in line 208 would inform the user that the squashed migration can not be applied because not all the 'child' migrations are applied.\nHad to debug into that to figure that out.",
      "issue_closed_at": "2021-08-04T05:23:36",
      "base_commit": "7afca03c4058c25f2e1cec8d0b07c2157c93e831",
      "changes": [
        {
          "file": "django/core/management/commands/migrate.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        database = options['database']\n        if not options['skip_checks']:\n            self.check(databases=[database])\n\n        self.verbosity = options['verbosity']\n        self.interactive = options['interactive']\n\n        # Import the 'management' module within each installed app, to register\n        # dispatcher events.\n        for app_config in apps.get_app_configs():\n            if module_has_submodule(app_config.module, \"management\"):\n                import_module('.management', app_config.name)\n\n        # Get the database we're operating from\n        connection = connections[database]\n\n        # Hook for backends needing any database preparation\n        connection.prepare_database()\n        # Work out which apps have migrations and which do not\n        executor = MigrationExecutor(connection, self.migration_progress_callback)\n\n        # Raise an error if any migrations are applied before their dependencies.\n        executor.loader.check_consistent_history(connection)\n\n        # Before anything else, see if there's conflicting apps and drop out\n        # hard if there are any\n        conflicts = executor.loader.detect_conflicts()\n        if conflicts:\n            name_str = \"; \".join(\n                \"%s in %s\" % (\", \".join(names), app)\n                for app, names in conflicts.items()\n            )\n            raise CommandError(\n                \"Conflicting migrations detected; multiple leaf nodes in the \"\n                \"migration graph: (%s).\\nTo fix them run \"\n                \"'python manage.py makemigrations --merge'\" % name_str\n            )\n\n        # If they supplied command line arguments, work out what they mean.\n        run_syncdb = options['run_syncdb']\n        target_app_labels_only = True\n        if options['app_label']:\n            # Validate app_label.\n            app_label = options['app_label']\n            try:\n                apps.get_app_config(app_label)\n            except LookupError as err:\n                raise CommandError(str(err))\n            if run_syncdb:\n                if app_label in executor.loader.migrated_apps:\n                    raise CommandError(\"Can't use run_syncdb with app '%s' as it has migrations.\" % app_label)\n            elif app_label not in executor.loader.migrated_apps:\n                raise CommandError(\"App '%s' does not have migrations.\" % app_label)\n\n        if options['app_label'] and options['migration_name']:\n            migration_name = options['migration_name']\n            if migration_name == \"zero\":\n                targets = [(app_label, None)]\n            else:\n                try:\n                    migration = executor.loader.get_migration_by_prefix(app_label, migration_name)\n                except AmbiguityError:\n                    raise CommandError(\n                        \"More than one migration matches '%s' in app '%s'. \"\n                        \"Please be more specific.\" %\n                        (migration_name, app_label)\n                    )\n                except KeyError:\n                    raise CommandError(\"Cannot find a migration matching '%s' from app '%s'.\" % (\n                        migration_name, app_label))\n                targets = [(app_label, migration.name)]\n            target_app_labels_only = False\n        elif options['app_label']:\n            targets = [key for key in executor.loader.graph.leaf_nodes() if key[0] == app_label]\n        else:\n            targets = executor.loader.graph.leaf_nodes()\n\n        plan = executor.migration_plan(targets)\n        exit_dry = plan and options['check_unapplied']\n\n        if options['plan']:\n            self.stdout.write('Planned operations:', self.style.MIGRATE_LABEL)\n            if not plan:\n                self.stdout.write('  No planned migration operations.')\n            for migration, backwards in plan:\n                self.stdout.write(str(migration), self.style.MIGRATE_HEADING)\n                for operation in migration.operations:\n                    message, is_error = self.describe_operation(operation, backwards)\n                    style = self.style.WARNING if is_error else None\n                    self.stdout.write('    ' + message, style)\n            if exit_dry:\n                sys.exit(1)\n            return\n        if exit_dry:\n            sys.exit(1)\n\n        # At this point, ignore run_syncdb if there aren't any apps to sync.\n        run_syncdb = options['run_syncdb'] and executor.loader.unmigrated_apps\n        # Print some useful info\n        if self.verbosity >= 1:\n            self.stdout.write(self.style.MIGRATE_HEADING(\"Operations to perform:\"))\n            if run_syncdb:\n                if options['app_label']:\n                    self.stdout.write(\n                        self.style.MIGRATE_LABEL(\"  Synchronize unmigrated app: %s\" % app_label)\n                    )\n                else:\n                    self.stdout.write(\n                        self.style.MIGRATE_LABEL(\"  Synchronize unmigrated apps: \") +\n                        (\", \".join(sorted(executor.loader.unmigrated_apps)))\n                    )\n            if target_app_labels_only:\n                self.stdout.write(\n                    self.style.MIGRATE_LABEL(\"  Apply all migrations: \") +\n                    (\", \".join(sorted({a for a, n in targets})) or \"(none)\")\n                )\n            else:\n                if targets[0][1] is None:\n                    self.stdout.write(\n                        self.style.MIGRATE_LABEL('  Unapply all migrations: ') +\n                        str(targets[0][0])\n                    )\n                else:\n                    self.stdout.write(self.style.MIGRATE_LABEL(\n                        \"  Target specific migration: \") + \"%s, from %s\"\n                        % (targets[0][1], targets[0][0])\n                    )\n\n        pre_migrate_state = executor._create_project_state(with_applied_migrations=True)\n        pre_migrate_apps = pre_migrate_state.apps\n        emit_pre_migrate_signal(\n            self.verbosity, self.interactive, connection.alias, stdout=self.stdout, apps=pre_migrate_apps, plan=plan,\n        )\n\n        # Run the syncdb phase.\n        if run_syncdb:\n            if self.verbosity >= 1:\n                self.stdout.write(self.style.MIGRATE_HEADING(\"Synchronizing apps without migrations:\"))\n            if options['app_label']:\n                self.sync_apps(connection, [app_label])\n            else:\n                self.sync_apps(connection, executor.loader.unmigrated_apps)\n\n        # Migrate!\n        if self.verbosity >= 1:\n            self.stdout.write(self.style.MIGRATE_HEADING(\"Running migrations:\"))\n        if not plan:\n            if self.verbosity >= 1:\n                self.stdout.write(\"  No migrations to apply.\")\n                # If there's changes that aren't in migrations yet, tell them how to fix it.\n                autodetector = MigrationAutodetector(\n                    executor.loader.project_state(),\n                    ProjectState.from_apps(apps),\n                )\n                changes = autodetector.changes(graph=executor.loader.graph)\n                if changes:\n                    self.stdout.write(self.style.NOTICE(\n                        \"  Your models in app(s): %s have changes that are not \"\n                        \"yet reflected in a migration, and so won't be \"\n                        \"applied.\" % \", \".join(repr(app) for app in sorted(changes))\n                    ))\n                    self.stdout.write(self.style.NOTICE(\n                        \"  Run 'manage.py makemigrations' to make new \"\n                        \"migrations, and then re-run 'manage.py migrate' to \"\n                        \"apply them.\"\n                    ))\n            fake = False\n            fake_initial = False\n        else:\n            fake = options['fake']\n            fake_initial = options['fake_initial']\n        post_migrate_state = executor.migrate(\n            targets, plan=plan, state=pre_migrate_state.clone(), fake=fake,\n            fake_initial=fake_initial,\n        )\n        # post_migrate signals have access to all models. Ensure that all models\n        # are reloaded in case any are delayed.\n        post_migrate_state.clear_delayed_apps_cache()\n        post_migrate_apps = post_migrate_state.apps\n\n        # Re-render models of real apps to include relationships now that\n        # we've got a final state. This wouldn't be necessary if real apps\n        # models were rendered with relationships in the first place.\n        with post_migrate_apps.bulk_update():\n            model_keys = []\n            for model_state in post_migrate_apps.real_models:\n                model_key = model_state.app_label, model_state.name_lower\n                model_keys.append(model_key)\n                post_migrate_apps.unregister_model(*model_key)\n        post_migrate_apps.render_multiple([\n            ModelState.from_model(apps.get_model(*model)) for model in model_keys\n        ])\n\n        # Send the post_migrate signal, so individual apps can do whatever they need\n        # to do at this point.\n        emit_post_migrate_signal(\n            self.verbosity, self.interactive, connection.alias, stdout=self.stdout, apps=post_migrate_apps, plan=plan,\n        )"
        },
        {
          "file": "django/core/management/commands/sqlmigrate.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        # Get the database we're operating from\n        connection = connections[options['database']]\n\n        # Load up an loader to get all the migration data, but don't replace\n        # migrations.\n        loader = MigrationLoader(connection, replace_migrations=False)\n\n        # Resolve command-line arguments into a migration\n        app_label, migration_name = options['app_label'], options['migration_name']\n        # Validate app_label\n        try:\n            apps.get_app_config(app_label)\n        except LookupError as err:\n            raise CommandError(str(err))\n        if app_label not in loader.migrated_apps:\n            raise CommandError(\"App '%s' does not have migrations\" % app_label)\n        try:\n            migration = loader.get_migration_by_prefix(app_label, migration_name)\n        except AmbiguityError:\n            raise CommandError(\"More than one migration matches '%s' in app '%s'. Please be more specific.\" % (\n                migration_name, app_label))\n        except KeyError:\n            raise CommandError(\"Cannot find a migration matching '%s' from app '%s'. Is it in INSTALLED_APPS?\" % (\n                migration_name, app_label))\n        target = (app_label, migration.name)\n\n        # Show begin/end around output for atomic migrations, if the database\n        # supports transactional DDL.\n        self.output_transaction = migration.atomic and connection.features.can_rollback_ddl\n\n        # Make a plan that represents just the requested migrations and show SQL\n        # for it\n        plan = [(loader.graph.nodes[target], options['backwards'])]\n        sql_statements = loader.collect_sql(plan)\n        if not sql_statements and options['verbosity'] >= 1:\n            self.stderr.write('No operations found.')\n        return '\\n'.join(sql_statements)"
        },
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "get_migration_by_prefix",
          "class_name": "MigrationLoader",
          "code": "def get_migration_by_prefix(self, app_label, name_prefix):\n        \"\"\"\n        Return the migration(s) which match the given app label and name_prefix.\n        \"\"\"\n        # Do the search\n        results = []\n        for migration_app_label, migration_name in self.disk_migrations:\n            if migration_app_label == app_label and migration_name.startswith(name_prefix):\n                results.append((migration_app_label, migration_name))\n        if len(results) > 1:\n            raise AmbiguityError(\n                \"There is more than one migration for '%s' with the prefix '%s'\" % (app_label, name_prefix)\n            )\n        elif not results:\n            raise KeyError(\"There no migrations for '%s' with the prefix '%s'\" % (app_label, name_prefix))\n        else:\n            return self.disk_migrations[results[0]]"
        }
      ]
    }
  ]
}