{
  "instance_id": "django__django-16400",
  "repo": "django/django",
  "created_at": "2022-12-23T17:17:00Z",
  "problem_statement": "migrate management command does not respect database parameter when adding Permissions.\nDescription\n\t \n\t\t(last modified by Vasanth)\n\t \nWhen invoking migrate with a database parameter, the migration runs successfully. However, there seems to be a DB read request that runs after the migration. This call does not respect the db param and invokes the db router .\nWhen naming the db as a parameter, all DB calls in the context of the migrate command are expected to use the database specified.\nI came across this as I am currently using a thread-local variable to get the active DB with a custom DB router for a multi-tenant service .\nMinimal example \nSetup the custom middleware and custom DB Router as show below. Then run any DB migration. We see that \"read {}\" is being printed before the exception message.\nIdeally none of this code must be called as the DB was specified during management command.\nfrom threading import local\nfrom django.conf import settings\nlocal_state = local()\nclass InvalidTenantException(Exception):\n\tpass\nclass TenantSubdomainMiddleware:\n\tdef __init__(self, get_response):\n\t\tself.get_response = get_response\n\tdef __call__(self, request):\n\t\t## Get Subdomain\n\t\thost = request.get_host().split(\":\")[0]\n\t\tlocal_state.subdomain = (\n\t\t\t# We assume single level of subdomain : app.service.com \n\t\t\t# HOST_IP : used to for local dev. \n\t\t\thost if host in settings.HOST_IP else host.split(\".\")[0]\n\t\t)\n\t\tresponse = self.get_response(request)\n\t\treturn response\nclass TenantDatabaseRouter:\n\tdef _default_db(self):\n\t\tsubdomain = getattr(local_state, \"subdomain\", None)\n\t\tif subdomain is not None and subdomain in settings.TENANT_MAP:\n\t\t\tdb_name = settings.TENANT_MAP[local_state.subdomain]\n\t\t\treturn db_name\n\t\telse:\n\t\t\traise InvalidTenantException()\n\tdef db_for_read(self, model, **hints):\n\t\tprint(\"read\", hints)\n\t\treturn self._default_db()\n\tdef db_for_write(self, model, **hints):\n\t\tprint(\"write\", hints)\n\t\treturn self._default_db()\n\tdef allow_relation(self, obj1, obj2, **hints):\n\t\treturn None\n\tdef allow_migrate(self, db, app_label, model_name=None, **hints):\n\t\treturn None\n## settings.py\nMIDDLEWARE = [\n\t\"utils.tenant_db_router.TenantSubdomainMiddleware\",\n\t\"django.middleware.security.SecurityMiddleware\",\n\t...\n]\nTENANT_MAP = {\"localhost\":\"default\", \"tenant_1\":\"default\"}\nDATABASE_ROUTERS = [\"utils.tenant_db_router.TenantDatabaseRouter\"]\n",
  "patch": "diff --git a/django/contrib/auth/management/__init__.py b/django/contrib/auth/management/__init__.py\n--- a/django/contrib/auth/management/__init__.py\n+++ b/django/contrib/auth/management/__init__.py\n@@ -95,11 +95,16 @@ def create_permissions(\n         .values_list(\"content_type\", \"codename\")\n     )\n \n-    perms = [\n-        Permission(codename=codename, name=name, content_type=ct)\n-        for ct, (codename, name) in searched_perms\n-        if (ct.pk, codename) not in all_perms\n-    ]\n+    perms = []\n+    for ct, (codename, name) in searched_perms:\n+        if (ct.pk, codename) not in all_perms:\n+            permission = Permission()\n+            permission._state.db = using\n+            permission.codename = codename\n+            permission.name = name\n+            permission.content_type = ct\n+            perms.append(permission)\n+\n     Permission.objects.using(using).bulk_create(perms)\n     if verbosity >= 2:\n         for perm in perms:\n",
  "similar_bug_items": [
    {
      "pr_number": 9112,
      "pr_title": "Fixed #27846 -- clear all cached reverse relationships on refresh_from_db()",
      "pr_body": "https://code.djangoproject.com/ticket/27846",
      "issue_id": 27846,
      "issue_title": "refresh_from_db() doesn't clear reverse OneToOneFields",
      "issue_body": "Sorry for the poor summary, it is difficult to explain in words. I have a project to demo this bug attached to this ticket, but I will try to explain the bug anyway in steps and the setup.\nSetup:\n2 models (A and B)\nB has a OneToOne to A\nBoth A and B have a field (ie TextField)\nSetup either a signal or override save() for A to update B's TextField to equal that of A's on save() or post_save for signals\nSteps:\nCreate instance of A\nGet another copy of the instance of A via A.objects.get()\nCreate instance of B using the copy of the instance of A\nDo refresh_from_db() on original instance of A\nTry to access B from A\nThe project I have provided is a slim version of this problem that demonstrates it with signals, overriden save(), and basic set and save inside the test. The basic set and save works, but the other two fail when using the above steps. Run the test suite to see.",
      "issue_closed_at": "2017-10-12T16:25:22",
      "base_commit": "df0aebc893973c78d7d2cda712ba4133dbe29b6e",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "refresh_from_db",
          "class_name": "Model",
          "code": "def refresh_from_db(self, using=None, fields=None):\n        \"\"\"\n        Reload field values from the database.\n\n        By default, the reloading happens from the database this instance was\n        loaded from, or by the read router if this instance wasn't loaded from\n        any database. The using parameter will override the default.\n\n        Fields can be used to specify which fields to reload. The fields\n        should be an iterable of field attnames. If fields is None, then\n        all non-deferred fields are reloaded.\n\n        When accessing deferred fields of an instance, the deferred loading\n        of the field will call this method.\n        \"\"\"\n        if fields is not None:\n            if len(fields) == 0:\n                return\n            if any(LOOKUP_SEP in f for f in fields):\n                raise ValueError(\n                    'Found \"%s\" in fields argument. Relations and transforms '\n                    'are not allowed in fields.' % LOOKUP_SEP)\n\n        db = using if using is not None else self._state.db\n        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n\n        # Use provided fields, if not set then reload all non-deferred fields.\n        deferred_fields = self.get_deferred_fields()\n        if fields is not None:\n            fields = list(fields)\n            db_instance_qs = db_instance_qs.only(*fields)\n        elif deferred_fields:\n            fields = [f.attname for f in self._meta.concrete_fields\n                      if f.attname not in deferred_fields]\n            db_instance_qs = db_instance_qs.only(*fields)\n\n        db_instance = db_instance_qs.get()\n        non_loaded_fields = db_instance.get_deferred_fields()\n        for field in self._meta.concrete_fields:\n            if field.attname in non_loaded_fields:\n                # This field wasn't refreshed - skip ahead.\n                continue\n            setattr(self, field.attname, getattr(db_instance, field.attname))\n            # Throw away stale foreign key references.\n            if field.is_relation and field.is_cached(self):\n                rel_instance = field.get_cached_value(self)\n                local_val = getattr(db_instance, field.attname)\n                related_val = None if rel_instance is None else getattr(rel_instance, field.target_field.attname)\n                if local_val != related_val or (local_val is None and related_val is None):\n                    field.delete_cached_value(self)\n        self._state.db = db_instance._state.db"
        }
      ]
    },
    {
      "pr_number": 9512,
      "pr_title": "Fixed #28918 -- Fixed Model.refresh_from_db() for instances hidden by the default manager.",
      "pr_body": "https://code.djangoproject.com/ticket/28918",
      "issue_id": 28918,
      "issue_title": "Model.refresh_from_db() should use _base_manager instead of the default_manager",
      "issue_body": "When using\nrefresh_from_db\n, currently the\n_default_manager\nis used.\nBased upon the documentation at\n​\nhttps://docs.djangoproject.com/en/2.0/topics/db/managers/#default-managers\n, it seems that there's a requirement for\n_base_manager\nto not filter out results, while for\n_default_manager\nthere's just a preference for this to be the case.\nI think the stronger \"guarantee\" provided by\n_base_manager\nis more appropriate in this case.\nThe following patch both adds a test and fixes the issue.\ndiff --git a/django/db/models/base.py b/django/db/models/base.py\nindex 27ca63fd22..0813bedcb0 100644\n--- a/django/db/models/base.py\n+++ b/django/db/models/base.py\n@@ -591,7 +591,7 @@ class Model(metaclass=ModelBase):\n                     'are not allowed in fields.' % LOOKUP_SEP)\n\n         db = using if using is not None else self._state.db\n-        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n+        db_instance_qs = self.__class__._base_manager.using(db).filter(pk=self.pk)\n\n         # Use provided fields, if not set then reload all non-deferred fields.\n         deferred_fields = self.get_deferred_fields()\ndiff --git a/tests/custom_managers/tests.py b/tests/custom_managers/tests.py\nindex ee2ac1d552..61b1a07933 100644\n--- a/tests/custom_managers/tests.py\n+++ b/tests/custom_managers/tests.py\n@@ -643,3 +643,13 @@ class CustomManagersRegressTestCase(TestCase):\n         qs_custom = Person.custom_init_queryset_manager.all()\n         qs_default = Person.objects.all()\n         self.assertQuerysetEqual(qs_custom, qs_default)\n+\n+    def test_refresh_from_db_when_default_manager_filters(self):\n+        \"\"\"\n+        obj.refresh_from_db() should also fetch object if default manager\n+        hides it.\n+        \"\"\"\n+        book = Book._base_manager.create(is_published=False)\n+        Book._base_manager.filter(pk=book.pk).update(title='Hi')\n+        book.refresh_from_db()\n+        self.assertEqual(book.title, 'Hi')",
      "issue_closed_at": "2017-12-30T17:01:08",
      "base_commit": "dcdd219ee1e062dc6189f382e0298e0adf5d5ddf",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "refresh_from_db",
          "class_name": "Model",
          "code": "def refresh_from_db(self, using=None, fields=None):\n        \"\"\"\n        Reload field values from the database.\n\n        By default, the reloading happens from the database this instance was\n        loaded from, or by the read router if this instance wasn't loaded from\n        any database. The using parameter will override the default.\n\n        Fields can be used to specify which fields to reload. The fields\n        should be an iterable of field attnames. If fields is None, then\n        all non-deferred fields are reloaded.\n\n        When accessing deferred fields of an instance, the deferred loading\n        of the field will call this method.\n        \"\"\"\n        if fields is not None:\n            if not fields:\n                return\n            if any(LOOKUP_SEP in f for f in fields):\n                raise ValueError(\n                    'Found \"%s\" in fields argument. Relations and transforms '\n                    'are not allowed in fields.' % LOOKUP_SEP)\n\n        db = using if using is not None else self._state.db\n        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n\n        # Use provided fields, if not set then reload all non-deferred fields.\n        deferred_fields = self.get_deferred_fields()\n        if fields is not None:\n            fields = list(fields)\n            db_instance_qs = db_instance_qs.only(*fields)\n        elif deferred_fields:\n            fields = [f.attname for f in self._meta.concrete_fields\n                      if f.attname not in deferred_fields]\n            db_instance_qs = db_instance_qs.only(*fields)\n\n        db_instance = db_instance_qs.get()\n        non_loaded_fields = db_instance.get_deferred_fields()\n        for field in self._meta.concrete_fields:\n            if field.attname in non_loaded_fields:\n                # This field wasn't refreshed - skip ahead.\n                continue\n            setattr(self, field.attname, getattr(db_instance, field.attname))\n            # Throw away stale foreign key references.\n            if field.is_relation and field.is_cached(self):\n                rel_instance = field.get_cached_value(self)\n                local_val = getattr(db_instance, field.attname)\n                related_val = None if rel_instance is None else getattr(rel_instance, field.target_field.attname)\n                if local_val != related_val or (local_val is None and related_val is None):\n                    field.delete_cached_value(self)\n\n        # Clear cached relations.\n        for field in self._meta.related_objects:\n            if field.is_cached(self):\n                field.delete_cached_value(self)\n\n        self._state.db = db_instance._state.db"
        }
      ]
    },
    {
      "pr_number": 8899,
      "pr_title": "[1.11.x] Fixed #28451 -- Fixed regression in Oracle sequence naming",
      "pr_body": "Backport of 99c7a18f4edf84d3b2bbddd3dfc7c3287bc8fa23 from master,\r\nwith additional migration script & release notes.",
      "issue_id": 28451,
      "issue_title": "Change in Oracle sequence name truncation causes regression when updating existing database",
      "issue_body": "Summary\n: a change introduced in 1.11 in how the Oracle backend truncates sequence names introduces persistent \"ORA-02289: sequence does not exist\" errors after upgrading to 1.11.\nExplanation\n(as far best as I can tell): In the Oracle backend, sequence names are truncated to 30 characters.\nIn 1.10.7 (and 1.9.13 & 1.8.17) the method to do that in\ndjango.db.backends.oracle.operations\nis as follows:\ndef _get_sequence_name(self, table):\n    name_length = self.max_name_length() - 3\n    return '%s_SQ' % truncate_name(table, name_length).upper()\nIn 1.11.3 it's:\ndef _get_sequence_name(self, table):\n    name_length = self.max_name_length() - 3\n    sequence_name = '%s_SQ' % strip_quotes(table)\n    return truncate_name(sequence_name, name_length).upper()\nNote the subtle change - in 1.10.7 the return value always ends with '_SQ'; in 1.11.3 the '_SQ' is part of what gets truncated. (For context, truncate_name basically takes a hash of the string and appends the last few digits to the name of the table - so you end up with e.g. 'PATIENTS_PATIENTDOCTORRELA8026', to fit in the 30-character limit).\nThe consequence of this is that after upgrading an Oracle-backed app to 1.11, inserts start failing (because\nlast_insert_id\nis looking for the sequence name 'PATIENTS_PATIENTDOCTORR36D1', whereas the actual sequence name is 'PATIENTS_PATIENTDOCTORRC0BD_SQ' - because that's what was generated in the prior version; or, to be precise, whenever the table was created, several versions before that).\nAs far as I can tell (though I can't be sure) this change was an inadvertent side effect of\n69b7d4b1\n, which was the fix for\n#27458\n. I say 'inadvertent' because it doesn't appear to be the focus of the change, and the tests don't appear to be taking that into account. In general, most tests wouldn't pick up the problem because it only manifests if you have existing sequences - for a fresh database, it's fine (since the sequences will be created with the new naming scheme and everything is hunky-dory).\n(NB: the same thing appears to have happened for triggers, though this particular database isn't using triggers so I didn't hit that particular error).\nAs a quick test, patching Django to use the pre-1.11 version of\n_get_sequence_name\nworked correctly, so I'm fairly confident that's the issue (there was another change in it, the strip_quotes bit, so if we go that way for the ultimate fix we'll probably want to keep that rather than just reverting).\nI'd be glad to work on a patch but to be honest I'm not clear what direction to take... as I see it, the options are:\n1) Revert the behavior - make\n_get_sequence_name\nreturn '%s_SQ' like it did pre-1.11 (but with the strip_quotes fix). This has a bad backcompat issue in that it'll introduce essentially the same problem for sequences created with 1.11.x... so that doesn't really seem like a good idea.\n2) Create a helper to rename sequences when a change like this is introduced. In theory this is an implementation detail and Django should be able to tweak the way the truncated names are generated, as long as there's a transition path (though I say that as someone who doesn't use custom sequences much - others may have a different opinion).\nSo I can imagine a utility of some sort to cross-check sequence names (for autonumber fields and such) with what Django expects, and either interactively or automatically rename them. Perhaps call it out in the release notes?\nFor completeness, this feels related to\n#23577\nbut feels a bit different, and IMO is more dire because it's less obvious and less likely to get noticed right away.",
      "issue_closed_at": "2017-08-22T15:09:28",
      "base_commit": "60f81118f412268f317abbcc7509e315a714315d",
      "changes": [
        {
          "file": "django/db/backends/oracle/operations.py",
          "type": "function",
          "name": "combine_expression",
          "class_name": "DatabaseOperations",
          "code": "def combine_expression(self, connector, sub_expressions):\n        lhs, rhs = sub_expressions\n        if connector == '%%':\n            return 'MOD(%s)' % ','.join(sub_expressions)\n        elif connector == '&':\n            return 'BITAND(%s)' % ','.join(sub_expressions)\n        elif connector == '|':\n            return 'BITAND(-%(lhs)s-1,%(rhs)s)+%(lhs)s' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '<<':\n            return '(%(lhs)s * POWER(2, %(rhs)s))' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '>>':\n            return 'FLOOR(%(lhs)s / POWER(2, %(rhs)s))' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '^':\n            return 'POWER(%s)' % ','.join(sub_expressions)\n        return super(DatabaseOperations, self).combine_expression(connector, sub_expressions)"
        }
      ]
    },
    {
      "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": 14372,
      "pr_title": "Fixed #32718 -- Relaxed file name validation in FileField.",
      "pr_body": "See [comment](https://code.djangoproject.com/ticket/32718#comment:26), ticket-32718.\r\n\r\n- ~~If `filename` passed to the `FileField.generate_filename()` is an absolute path, it will be converted to the `os.path.basename(filename)`.~~\r\n- Validate `filename` returned by `FileField.upload_to()` not a `filename` passed to the `FileField.generate_filename()` (`upload_to()` may completely ignored passed `filename`).\r\n- Allow relative paths (without dot segments) in the generated file name.\r\n\r\nThanks Jakub Kleň for the report.\r\n\r\nRegression in 0b79eb36915d178aef5c6a7bbce71b1e76d376d3.\r\n\r\n- [x] Discuss support for absolute paths.\r\n- [x] Release notes.",
      "issue_id": 32718,
      "issue_title": "Saving a FileField raises SuspiciousFileOperation in some scenarios.",
      "issue_body": "I came across this issue today when I was updating Django from 3.2.0 -> 3.2.1.\nIt's directly caused by:\n​\nhttps://docs.djangoproject.com/en/3.2/releases/3.2.1/#cve-2021-31542-potential-directory-traversal-via-uploaded-files\nStarting from 3.2.1, Django requires that only the basename is passed to\nFieldFile.save\nmethod, because otherwise it raises a new exception:\nSuspiciousFileOperation: File name ... includes path elements\nThe issue is that in\nFileField.pre_save\n, a full path is passed to\nFieldFile.save\n, causing the exception to be raised.\nCorrect me if I'm wrong, but file-like objects always contain the full path to the file in the\nname\nattribute (the built-in Django\nFile\nclass even uses it to reopen the file if it was closed), and so it seems to be a bug in Django itself.\nSteps to reproduce:\nmodel_instance.file_attribute = File(open(path, 'rb'))\nmodel_instance.save()\nI also created a PR with the fix:\n​\nhttps://github.com/django/django/pull/14354",
      "issue_closed_at": "2021-05-13T01:53:57",
      "base_commit": "b81c7562fc33f50166d5120138d6398dc42b13c3",
      "changes": [
        {
          "file": "django/core/files/utils.py",
          "type": "line",
          "name": "line 1",
          "code": "import os\n\nfrom django.core.exceptions import SuspiciousFileOperation\n\n\ndef validate_file_name(name):\n    if name != os.path.basename(name):\n        raise SuspiciousFileOperation(\"File name '%s' includes path elements\" % name)\n\n    # Remove potentially dangerous names\n    if name in {'', '.', '..'}:\n        raise SuspiciousFileOperation(\"Could not derive file name from '%s'\" % name)\n\n    return name\n\n"
        },
        {
          "file": "django/db/models/fields/files.py",
          "type": "function",
          "name": "generate_filename",
          "class_name": "FileField",
          "code": "def generate_filename(self, instance, filename):\n        \"\"\"\n        Apply (if callable) or prepend (if a string) upload_to to the filename,\n        then delegate further processing of the name to the storage backend.\n        Until the storage layer, all file paths are expected to be Unix style\n        (with forward slashes).\n        \"\"\"\n        filename = validate_file_name(filename)\n        if callable(self.upload_to):\n            filename = self.upload_to(instance, filename)\n        else:\n            dirname = datetime.datetime.now().strftime(str(self.upload_to))\n            filename = posixpath.join(dirname, filename)\n        return self.storage.generate_filename(filename)"
        }
      ]
    }
  ]
}