{
  "instance_id": "django__django-14580",
  "repo": "django/django",
  "created_at": "2021-07-01T07:38:03Z",
  "problem_statement": "Missing import statement in generated migration (NameError: name 'models' is not defined)\nDescription\n\t\nI found a bug in Django's latest release: 3.2.4. \nGiven the following contents of models.py:\nfrom django.db import models\nclass MyField(models.TextField):\n\tpass\nclass MyBaseModel(models.Model):\n\tclass Meta:\n\t\tabstract = True\nclass MyMixin:\n\tpass\nclass MyModel(MyMixin, MyBaseModel):\n\tname = MyField(primary_key=True)\nThe makemigrations command will generate the following migration file:\n# Generated by Django 3.2.4 on 2021-06-30 19:13\nimport app.models\nfrom django.db import migrations\nclass Migration(migrations.Migration):\n\tinitial = True\n\tdependencies = [\n\t]\n\toperations = [\n\t\tmigrations.CreateModel(\n\t\t\tname='MyModel',\n\t\t\tfields=[\n\t\t\t\t('name', app.models.MyField(primary_key=True, serialize=False)),\n\t\t\t],\n\t\t\toptions={\n\t\t\t\t'abstract': False,\n\t\t\t},\n\t\t\tbases=(app.models.MyMixin, models.Model),\n\t\t),\n\t]\nWhich will then fail with the following error:\n File \"/home/jj/django_example/app/migrations/0001_initial.py\", line 7, in <module>\n\tclass Migration(migrations.Migration):\n File \"/home/jj/django_example/app/migrations/0001_initial.py\", line 23, in Migration\n\tbases=(app.models.MyMixin, models.Model),\nNameError: name 'models' is not defined\nExpected behavior: Django generates a migration file that is valid Python.\nActual behavior: Django generates a migration file that is missing an import statement.\nI think this is a bug of the module django.db.migrations.writer, but I'm not sure. I will be happy to assist with debugging.\nThanks for your attention,\nJaap Joris\n",
  "patch": "diff --git a/django/db/migrations/serializer.py b/django/db/migrations/serializer.py\n--- a/django/db/migrations/serializer.py\n+++ b/django/db/migrations/serializer.py\n@@ -273,7 +273,7 @@ def _format(self):\n class TypeSerializer(BaseSerializer):\n     def serialize(self):\n         special_cases = [\n-            (models.Model, \"models.Model\", []),\n+            (models.Model, \"models.Model\", ['from django.db import models']),\n             (type(None), 'type(None)', []),\n         ]\n         for case, string, imports in special_cases:\n",
  "similar_bug_items": [
    {
      "pr_number": 6178,
      "pr_title": "Fixed #26186 -- Documented how app relative relationships of abstract models behave.",
      "pr_body": "This partially reverts commit bc7d201bdbaeac14a49f51a9ef292d6312b4c45e.\n\nRefs #25858.\n",
      "issue_id": 26186,
      "issue_title": "When extending from abstract model, ForeignKey points to wrong application",
      "issue_body": "This might be related to\nhttps://code.djangoproject.com/ticket/25858\nI have following code:\napp_A/abstract.py:\nclass AbstractModel1(Model):\n    class Meta:\n        abstract = True\n\n\nclass AbstractModel2(Model):\n    model1 = ForeignKey('Model1')\n\n    class Meta:\n          abstract = True\napp_A/models.py:\nfrom .abstract import AbstractModel1, AbstractModel12\n\nclass Model1(AbstractModel1):\n    pass\n\nclass Model2(AbstractModel12):\n    pass\napp_B/models.py:\nfrom app_A.abstract import AbstractModel1, AbstractModel2\n\nclass Model1(AbstractModel1):\n    pass\n\nclass Model2(AbstractModel12):\n    pass\nin Django 1.8, the\napp_B.Model2.model1\nwould point to\napp_B.Model1\n, but in Django 1.9.2 it points to\napp_A.Model1\nso my code no longer works.\nTest case can be found:\n​\nhttps://github.com/skyjur/django-ticketing/tree/master/ticket_26186",
      "issue_closed_at": "2016-02-29T21:07:53",
      "base_commit": "eac1423f9ebcf432dc5be95d605d124a05ab2686",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "line",
          "name": "line 35",
          "code": "RECURSIVE_RELATIONSHIP_CONSTANT = 'self'\n\n\ndef resolve_relation(scope_model, relation, resolve_recursive_relationship=True):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model."
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_relation",
          "class_name": null,
          "code": "def resolve_relation(scope_model, relation, resolve_recursive_relationship=True):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model.\n\n    The relation argument can be:\n      * RECURSIVE_RELATIONSHIP_CONSTANT, i.e. the string \"self\", in which case\n        the model argument will be returned.\n      * A bare model name without an app_label, in which case scope_model's\n        app_label will be prepended.\n      * An \"app_label.ModelName\" string.\n      * A model class, which will be returned unchanged.\n    \"\"\"\n    # Check for recursive relations\n    if relation == RECURSIVE_RELATIONSHIP_CONSTANT:\n        if resolve_recursive_relationship:\n            relation = scope_model\n    # Look for an \"app.Model\" relation\n    elif isinstance(relation, six.string_types) and '.' not in relation:\n        relation = \"%s.%s\" % (scope_model._meta.app_label, relation)\n\n    return relation"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_related_class",
          "class_name": "RelatedField",
          "code": "def resolve_related_class(model, related, field):\n                field.remote_field.model = related\n                field.do_related_class(related, model)"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_through_model",
          "class_name": "ManyToManyField",
          "code": "def resolve_through_model(_, model, field):\n                    field.remote_field.through = model"
        }
      ]
    },
    {
      "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": 14434,
      "pr_title": "Fixed #32777 -- Passed table reference as a string to DatabaseSchemaEditor._index_columns().",
      "pr_body": "Ticket: https://code.djangoproject.com/ticket/32777\r\n\r\nThis makes `_create_unique_sql` behave like `_create_index_sql`.",
      "issue_id": 32777,
      "issue_title": "Statement created by _create_unique_sql makes references_column always false",
      "issue_body": "This is due to an instance of\nTable\nis passed as an argument to\nColumns\nwhen a string is expected.",
      "issue_closed_at": "2021-05-24T00:10:46",
      "base_commit": "5e04e84d67da8163f365e9f5fcd169e2630e2873",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "create_unique_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def create_unique_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "create_unique_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def create_unique_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))"
        }
      ]
    },
    {
      "pr_number": 7171,
      "pr_title": "Fixed #25109 -- Stopped silencing explicitly specified migration modules import errors.",
      "pr_body": "",
      "issue_id": 25109,
      "issue_title": "MigrationLoader.load_disk hides ImportError for invalid MIGRATION_MODULES",
      "issue_body": "With an invalid module name in\nMIGRATION_MODULES\n, you will get a rather confusing\nInvalidBasesError\n:\nInvalidBasesError: Cannot resolve bases for [<ModelState: 'djangocms_text_ckeditor.Text'>]\nThis can happen if you are inheriting models from an app with migrations (e.g. contrib.auth)\n in an app with no migrations; see https://docs.djangoproject.com/en/1.8/topics/migrations/#dependencies for more\nThis is caused by ignoring the\nImportError\nin\n​\nhttps://github.com/django/django/blob/d72f8862cb1a39934952e708c3c869be1399846e/django/db/migrations/loader.py#L70-L78\n, which is meant to handle non-existent migrations.\nThe following patch might fix it:\ndiff --git i/django/db/migrations/loader.py w/django/db/migrations/loader.py\nindex a8f4be4..e872294 100644\n--- i/django/db/migrations/loader.py\n+++ w/django/db/migrations/loader.py\n@@ -72,7 +72,9 @@ def load_disk(self):\n             except ImportError as e:\n                 # I hate doing this, but I don't want to squash other import errors.\n                 # Might be better to try a directory check directly.\n-                if \"No module named\" in str(e) and MIGRATIONS_MODULE_NAME in str(e):\n+                if (\"No module named\" in str(e)\n+                        and MIGRATIONS_MODULE_NAME in str(e)\n+                        and app_config.label not in settings.MIGRATION_MODULES):\n                     self.unmigrated_apps.add(app_config.label)\n                     continue\n                 raise\nBut then Django's tests itself fail, because they use this \"hack\":\nsettings.MIGRATION_MODULES = {\n        # these 'tests.migrations' modules don't actually exist, but this lets\n        # us skip creating migrations for the test models.\n        'auth': 'django.contrib.auth.tests.migrations',\n        'contenttypes': 'contenttypes_tests.migrations',\n    }\n([Source](\n​\nhttps://github.com/django/django/blob/d72f8862cb1a39934952e708c3c869be1399846e/tests/runtests.py#L142-147\n))\nI've seen this issue multiple times in the context of Django CMS (and its plugins), because in the progress of migrating to Django's migrations they were using modules like\ndjangocms_text_ckeditor.migrations_django\nwhich then were renamed.",
      "issue_closed_at": "2016-08-30T19:05:26",
      "base_commit": "a72411e140a886bdadbc666f9921c32b7aaed754",
      "changes": [
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "__init__",
          "class_name": "MigrationLoader",
          "code": "def __init__(self, connection, load=True, ignore_no_migrations=False):\n        self.connection = connection\n        self.disk_migrations = None\n        self.applied_migrations = None\n        self.ignore_no_migrations = ignore_no_migrations\n        if load:\n            self.build_graph()"
        },
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "load_disk",
          "class_name": "MigrationLoader",
          "code": "def load_disk(self):\n        \"\"\"\n        Loads the migrations from all INSTALLED_APPS from disk.\n        \"\"\"\n        self.disk_migrations = {}\n        self.unmigrated_apps = set()\n        self.migrated_apps = set()\n        for app_config in apps.get_app_configs():\n            # Get the migrations module directory\n            module_name = self.migrations_module(app_config.label)\n            if module_name is None:\n                self.unmigrated_apps.add(app_config.label)\n                continue\n            was_loaded = module_name in sys.modules\n            try:\n                module = import_module(module_name)\n            except ImportError as e:\n                # I hate doing this, but I don't want to squash other import errors.\n                # Might be better to try a directory check directly.\n                if \"No module named\" in str(e) and MIGRATIONS_MODULE_NAME in str(e):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                raise\n            else:\n                # PY3 will happily import empty dirs as namespaces.\n                if not hasattr(module, '__file__'):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                # Module is not a package (e.g. migrations.py).\n                if not hasattr(module, '__path__'):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                # Force a reload if it's already loaded (tests need this)\n                if was_loaded:\n                    six.moves.reload_module(module)\n            self.migrated_apps.add(app_config.label)\n            directory = os.path.dirname(module.__file__)\n            # Scan for .py files\n            migration_names = set()\n            for name in os.listdir(directory):\n                if name.endswith(\".py\"):\n                    import_name = name.rsplit(\".\", 1)[0]\n                    if import_name[0] not in \"_.~\":\n                        migration_names.add(import_name)\n            # Load them\n            for migration_name in migration_names:\n                migration_module = import_module(\"%s.%s\" % (module_name, migration_name))\n                if not hasattr(migration_module, \"Migration\"):\n                    raise BadMigrationError(\n                        \"Migration %s in app %s has no Migration class\" % (migration_name, app_config.label)\n                    )\n                self.disk_migrations[app_config.label, migration_name] = migration_module.Migration(\n                    migration_name,\n                    app_config.label,\n                )"
        },
        {
          "file": "django/db/migrations/loader.py",
          "type": "function",
          "name": "load_disk",
          "class_name": "MigrationLoader",
          "code": "def load_disk(self):\n        \"\"\"\n        Loads the migrations from all INSTALLED_APPS from disk.\n        \"\"\"\n        self.disk_migrations = {}\n        self.unmigrated_apps = set()\n        self.migrated_apps = set()\n        for app_config in apps.get_app_configs():\n            # Get the migrations module directory\n            module_name = self.migrations_module(app_config.label)\n            if module_name is None:\n                self.unmigrated_apps.add(app_config.label)\n                continue\n            was_loaded = module_name in sys.modules\n            try:\n                module = import_module(module_name)\n            except ImportError as e:\n                # I hate doing this, but I don't want to squash other import errors.\n                # Might be better to try a directory check directly.\n                if \"No module named\" in str(e) and MIGRATIONS_MODULE_NAME in str(e):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                raise\n            else:\n                # PY3 will happily import empty dirs as namespaces.\n                if not hasattr(module, '__file__'):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                # Module is not a package (e.g. migrations.py).\n                if not hasattr(module, '__path__'):\n                    self.unmigrated_apps.add(app_config.label)\n                    continue\n                # Force a reload if it's already loaded (tests need this)\n                if was_loaded:\n                    six.moves.reload_module(module)\n            self.migrated_apps.add(app_config.label)\n            directory = os.path.dirname(module.__file__)\n            # Scan for .py files\n            migration_names = set()\n            for name in os.listdir(directory):\n                if name.endswith(\".py\"):\n                    import_name = name.rsplit(\".\", 1)[0]\n                    if import_name[0] not in \"_.~\":\n                        migration_names.add(import_name)\n            # Load them\n            for migration_name in migration_names:\n                migration_module = import_module(\"%s.%s\" % (module_name, migration_name))\n                if not hasattr(migration_module, \"Migration\"):\n                    raise BadMigrationError(\n                        \"Migration %s in app %s has no Migration class\" % (migration_name, app_config.label)\n                    )\n                self.disk_migrations[app_config.label, migration_name] = migration_module.Migration(\n                    migration_name,\n                    app_config.label,\n                )"
        },
        {
          "file": "django/db/migrations/questioner.py",
          "type": "function",
          "name": "ask_initial",
          "class_name": "MigrationQuestioner",
          "code": "def ask_initial(self, app_label):\n        \"Should we create an initial migration for the app?\"\n        # If it was specified on the command line, definitely true\n        if app_label in self.specified_apps:\n            return True\n        # Otherwise, we look to see if it has a migrations module\n        # without any Python files in it, apart from __init__.py.\n        # Apps from the new app template will have these; the python\n        # file check will ensure we skip South ones.\n        try:\n            app_config = apps.get_app_config(app_label)\n        except LookupError:         # It's a fake app.\n            return self.defaults.get(\"ask_initial\", False)\n        migrations_import_path = MigrationLoader.migrations_module(app_config.label)\n        if migrations_import_path is None:\n            # It's an application with migrations disabled.\n            return self.defaults.get(\"ask_initial\", False)\n        try:\n            migrations_module = importlib.import_module(migrations_import_path)\n        except ImportError:\n            return self.defaults.get(\"ask_initial\", False)\n        else:\n            if hasattr(migrations_module, \"__file__\"):\n                filenames = os.listdir(os.path.dirname(migrations_module.__file__))\n            elif hasattr(migrations_module, \"__path__\"):\n                if len(migrations_module.__path__) > 1:\n                    return False\n                filenames = os.listdir(list(migrations_module.__path__)[0])\n            return not any(x.endswith(\".py\") for x in filenames if x != \"__init__.py\")"
        },
        {
          "file": "django/db/migrations/writer.py",
          "type": "function",
          "name": "as_string",
          "class_name": "MigrationWriter",
          "code": "def as_string(self):\n        \"\"\"\n        Returns a string of the file contents.\n        \"\"\"\n        items = {\n            \"replaces_str\": \"\",\n            \"initial_str\": \"\",\n        }\n\n        imports = set()\n\n        # Deconstruct operations\n        operations = []\n        for operation in self.migration.operations:\n            operation_string, operation_imports = OperationWriter(operation).serialize()\n            imports.update(operation_imports)\n            operations.append(operation_string)\n        items[\"operations\"] = \"\\n\".join(operations) + \"\\n\" if operations else \"\"\n\n        # Format dependencies and write out swappable dependencies right\n        dependencies = []\n        for dependency in self.migration.dependencies:\n            if dependency[0] == \"__setting__\":\n                dependencies.append(\"        migrations.swappable_dependency(settings.%s),\" % dependency[1])\n                imports.add(\"from django.conf import settings\")\n            else:\n                # No need to output bytestrings for dependencies\n                dependency = tuple(force_text(s) for s in dependency)\n                dependencies.append(\"        %s,\" % self.serialize(dependency)[0])\n        items[\"dependencies\"] = \"\\n\".join(dependencies) + \"\\n\" if dependencies else \"\"\n\n        # Format imports nicely, swapping imports of functions from migration files\n        # for comments\n        migration_imports = set()\n        for line in list(imports):\n            if re.match(\"^import (.*)\\.\\d+[^\\s]*$\", line):\n                migration_imports.add(line.split(\"import\")[1].strip())\n                imports.remove(line)\n                self.needs_manual_porting = True\n\n        # django.db.migrations is always used, but models import may not be.\n        # If models import exists, merge it with migrations import.\n        if \"from django.db import models\" in imports:\n            imports.discard(\"from django.db import models\")\n            imports.add(\"from django.db import migrations, models\")\n        else:\n            imports.add(\"from django.db import migrations\")\n\n        # Sort imports by the package / module to be imported (the part after\n        # \"from\" in \"from ... import ...\" or after \"import\" in \"import ...\").\n        sorted_imports = sorted(imports, key=lambda i: i.split()[1])\n        items[\"imports\"] = \"\\n\".join(sorted_imports) + \"\\n\" if imports else \"\"\n        if migration_imports:\n            items[\"imports\"] += (\n                \"\\n\\n# Functions from the following migrations need manual \"\n                \"copying.\\n# Move them and any dependencies into this file, \"\n                \"then update the\\n# RunPython operations to refer to the local \"\n                \"versions:\\n# %s\"\n            ) % \"\\n# \".join(sorted(migration_imports))\n        # If there's a replaces, make a string for it\n        if self.migration.replaces:\n            items['replaces_str'] = \"\\n    replaces = %s\\n\" % self.serialize(self.migration.replaces)[0]\n        # Hinting that goes into comment\n        items.update(\n            version=get_version(),\n            timestamp=now().strftime(\"%Y-%m-%d %H:%M\"),\n        )\n\n        if self.migration.initial:\n            items['initial_str'] = \"\\n    initial = True\\n\"\n\n        return (MIGRATION_TEMPLATE % items).encode(\"utf8\")"
        }
      ]
    },
    {
      "pr_number": 9468,
      "pr_title": "Fixed #28932 -- Prevented Oracle from truncating trailing zeros in the fractional part of DecimalField.",
      "pr_body": "https://code.djangoproject.com/ticket/28932",
      "issue_id": 28932,
      "issue_title": "DecimalFields truncates trailing zeros on Oracle.",
      "issue_body": "Oracle automatically removes trailing zeros from\nNUMBER\nfields, we should probably restore decimal field converter on Oracle, removed in\n​\nhttps://github.com/django/django/commit/7c1f3901bcdabb1340a621e7df9b24f3acd0d6f3\n.",
      "issue_closed_at": "2017-12-20T10:56:17",
      "base_commit": "9c9ef5835200a1bc6410d2769b9699baba9f9a8b",
      "changes": [
        {
          "file": "django/db/backends/oracle/base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FormatStylePlaceholderCursor",
          "code": "def __init__(self, connection):\n        self.cursor = connection.cursor()\n        self.cursor.outputtypehandler = self._output_type_handler\n        # The default for cx_Oracle < 5.3 is 50.\n        self.cursor.arraysize = 100"
        },
        {
          "file": "django/db/backends/oracle/base.py",
          "type": "function",
          "name": "_output_type_handler",
          "class_name": "FormatStylePlaceholderCursor",
          "code": "def _output_type_handler(cursor, name, defaultType, length, precision, scale):\n        \"\"\"\n        Called for each db column fetched from cursors. Return numbers as the\n        appropriate Python type.\n        \"\"\"\n        if defaultType == Database.NUMBER:\n            if scale == -127:\n                if precision == 0:\n                    # NUMBER column: decimal-precision floating point.\n                    # This will normally be an integer from a sequence,\n                    # but it could be a decimal value.\n                    outconverter = FormatStylePlaceholderCursor._output_number_converter\n                else:\n                    # FLOAT column: binary-precision floating point.\n                    # This comes from FloatField columns.\n                    outconverter = float\n            elif precision > 0:\n                # NUMBER(p,s) column: decimal-precision fixed point.\n                # This comes from IntegerField and DecimalField columns.\n                outconverter = int if scale == 0 else decimal.Decimal\n            else:\n                # No type information. This normally comes from a\n                # mathematical expression in the SELECT list. Guess int\n                # or Decimal based on whether it has a decimal point.\n                outconverter = FormatStylePlaceholderCursor._output_number_converter\n            return cursor.var(\n                Database.STRING,\n                size=255,\n                arraysize=cursor.arraysize,\n                outconverter=outconverter,\n            )"
        }
      ]
    }
  ]
}