{
  "instance_id": "django__django-11815",
  "repo": "django/django",
  "created_at": "2019-09-24T21:45:36Z",
  "problem_statement": "Migrations uses value of enum object instead of its name.\nDescription\n\t \n\t\t(last modified by oasl)\n\t \nWhen using Enum object as a default value for a CharField, the generated migration file uses the value of the Enum object instead of the its name. This causes a problem when using Django translation on the value of the Enum object. \nThe problem is that, when the Enum object value get translated to the users language, the old migration files raise an error stating that the Enum does not have the corresponding value. (because the Enum value is translated to another language)\nExample:\nLet say we have this code in models.py:\nfrom enum import Enum\nfrom django.utils.translation import gettext_lazy as _\nfrom django.db import models\nclass Status(Enum):\n\tGOOD = _('Good') # 'Good' will be translated\n\tBAD = _('Bad') # 'Bad' will be translated\n\tdef __str__(self):\n\t\treturn self.name\nclass Item(models.Model):\n\tstatus = models.CharField(default=Status.GOOD, max_length=128)\nIn the generated migration file, the code will be:\n...\n('status', models.CharField(default=Status('Good'), max_length=128))\n...\nAfter the translation, 'Good' will be translated to another word and it will not be part of the Status Enum class any more, so the migration file will raise the error on the previous line:\nValueError: 'Good' is not a valid Status\nShouldn't the code generated by the migration uses the name of the Status Enum 'GOOD', not the value of it, since it is changeable?\nIt should be:\n('status', models.CharField(default=Status['GOOD'], max_length=128))\nThis will be correct regardless of the translated word\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@@ -120,9 +120,10 @@ class EnumSerializer(BaseSerializer):\n     def serialize(self):\n         enum_class = self.value.__class__\n         module = enum_class.__module__\n-        v_string, v_imports = serializer_factory(self.value.value).serialize()\n-        imports = {'import %s' % module, *v_imports}\n-        return \"%s.%s(%s)\" % (module, enum_class.__name__, v_string), imports\n+        return (\n+            '%s.%s[%r]' % (module, enum_class.__name__, self.value.name),\n+            {'import %s' % module},\n+        )\n \n \n class FloatSerializer(BaseSimpleSerializer):\n",
  "similar_bug_items": [
    {
      "pr_number": 4356,
      "pr_title": "Fixed #24515 -- Fixed DjangoTranslation plural handling",
      "pr_body": "",
      "issue_id": 24515,
      "issue_title": "Plural handling broken",
      "issue_body": "",
      "issue_closed_at": "2015-03-21T04:28:18",
      "base_commit": "aea02ddfb7c610db9a7cb291b113d6e561d8eba9",
      "changes": [
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DjangoTranslation",
          "code": "def __init__(self, language):\n        \"\"\"Create a GNUTranslations() using many locale directories\"\"\"\n        gettext_module.GNUTranslations.__init__(self)\n\n        self.__language = language\n        self.__to_language = to_language(language)\n        self.__locale = to_locale(language)\n        self.plural = lambda n: int(n != 1)\n\n        self._init_translation_catalog()\n        self._add_installed_apps_translations()\n        self._add_local_translations()\n        self._add_fallback()"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "_new_gnu_trans",
          "class_name": "DjangoTranslation",
          "code": "def _new_gnu_trans(self, localedir, use_null_fallback=True):\n        \"\"\"\n        Returns a mergeable gettext.GNUTranslations instance.\n\n        A convenience wrapper. By default gettext uses 'fallback=False'.\n        Using param `use_null_fallback` to avoid confusion with any other\n        references to 'fallback'.\n        \"\"\"\n        translation = gettext_module.translation(\n            domain='django',\n            localedir=localedir,\n            languages=[self.__locale],\n            codeset='utf-8',\n            fallback=use_null_fallback)\n        if not hasattr(translation, '_catalog'):\n            # provides merge support for NullTranslations()\n            translation._catalog = {}\n            translation._info = {}\n        return translation"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "_init_translation_catalog",
          "class_name": "DjangoTranslation",
          "code": "def _init_translation_catalog(self):\n        \"\"\"Creates a base catalog using global django translations.\"\"\"\n        settingsfile = upath(sys.modules[settings.__module__].__file__)\n        localedir = os.path.join(os.path.dirname(settingsfile), 'locale')\n        use_null_fallback = True\n        if self.__language == settings.LANGUAGE_CODE:\n            # default lang should be present and parseable, if not\n            # gettext will raise an IOError (refs #18192).\n            use_null_fallback = False\n        translation = self._new_gnu_trans(localedir, use_null_fallback)\n        self._info = translation._info.copy()\n        self._catalog = translation._catalog.copy()"
        }
      ]
    },
    {
      "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": "",
      "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": 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": "",
      "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": 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": "",
      "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": 11615,
      "pr_title": "Fixed #29019 -- Added ManyToManyField support to REQUIRED_FIELDS.",
      "pr_body": "[ticket 29019](https://code.djangoproject.com/ticket/29019)",
      "issue_id": 29019,
      "issue_title": "createsuperuser crashes if a ManyToManyField is in REQUIRED_FIELDS",
      "issue_body": "",
      "issue_closed_at": "2019-08-26T08:17:38",
      "base_commit": "5dac63bb844d0a976e1dd1591a323c5ba9674a97",
      "changes": [
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "add_arguments",
          "class_name": "Command",
          "code": "def add_arguments(self, parser):\n        parser.add_argument(\n            '--%s' % self.UserModel.USERNAME_FIELD,\n            help='Specifies the login for the superuser.',\n        )\n        parser.add_argument(\n            '--noinput', '--no-input', action='store_false', dest='interactive',\n            help=(\n                'Tells Django to NOT prompt the user for input of any kind. '\n                'You must use --%s with --noinput, along with an option for '\n                'any other required field. Superusers created with --noinput will '\n                'not be able to log in until they\\'re given a valid password.' %\n                self.UserModel.USERNAME_FIELD\n            ),\n        )\n        parser.add_argument(\n            '--database',\n            default=DEFAULT_DB_ALIAS,\n            help='Specifies the database to use. Default is \"default\".',\n        )\n        for field in self.UserModel.REQUIRED_FIELDS:\n            parser.add_argument(\n                '--%s' % field,\n                help='Specifies the %s for the superuser.' % field,\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        username = options[self.UserModel.USERNAME_FIELD]\n        database = options['database']\n        user_data = {}\n        verbose_field_name = self.username_field.verbose_name\n        try:\n            self.UserModel._meta.get_field(PASSWORD_FIELD)\n        except exceptions.FieldDoesNotExist:\n            pass\n        else:\n            # If not provided, create the user with an unusable password.\n            user_data[PASSWORD_FIELD] = None\n        try:\n            if options['interactive']:\n                # Same as user_data but with foreign keys as fake model\n                # instances instead of raw IDs.\n                fake_user_data = {}\n                if hasattr(self.stdin, 'isatty') and not self.stdin.isatty():\n                    raise NotRunningInTTYException\n                default_username = get_default_username()\n                if username:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        self.stderr.write(error_msg)\n                        username = None\n                elif username == '':\n                    raise CommandError('%s cannot be blank.' % capfirst(verbose_field_name))\n                # Prompt for username.\n                while username is None:\n                    message = self._get_input_message(self.username_field, default_username)\n                    username = self.get_input_data(self.username_field, message, default_username)\n                    if username:\n                        error_msg = self._validate_username(username, verbose_field_name, database)\n                        if error_msg:\n                            self.stderr.write(error_msg)\n                            username = None\n                            continue\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                fake_user_data[self.UserModel.USERNAME_FIELD] = (\n                    self.username_field.remote_field.model(username)\n                    if self.username_field.remote_field else username\n                )\n                # Prompt for required fields.\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = options[field_name]\n                    while user_data[field_name] is None:\n                        message = self._get_input_message(field)\n                        input_value = self.get_input_data(field, message)\n                        user_data[field_name] = input_value\n                        fake_user_data[field_name] = input_value\n\n                        # Wrap any foreign keys in fake model instances\n                        if field.remote_field:\n                            fake_user_data[field_name] = field.remote_field.model(input_value)\n\n                # Prompt for a password if the model has one.\n                while PASSWORD_FIELD in user_data and user_data[PASSWORD_FIELD] is None:\n                    password = getpass.getpass()\n                    password2 = getpass.getpass('Password (again): ')\n                    if password != password2:\n                        self.stderr.write(\"Error: Your passwords didn't match.\")\n                        # Don't validate passwords that don't match.\n                        continue\n                    if password.strip() == '':\n                        self.stderr.write(\"Error: Blank passwords aren't allowed.\")\n                        # Don't validate blank passwords.\n                        continue\n                    try:\n                        validate_password(password2, self.UserModel(**fake_user_data))\n                    except exceptions.ValidationError as err:\n                        self.stderr.write('\\n'.join(err.messages))\n                        response = input('Bypass password validation and create user anyway? [y/N]: ')\n                        if response.lower() != 'y':\n                            continue\n                    user_data[PASSWORD_FIELD] = password\n            else:\n                # Non-interactive mode.\n                # Use password from environment variable, if provided.\n                if PASSWORD_FIELD in user_data and 'DJANGO_SUPERUSER_PASSWORD' in os.environ:\n                    user_data[PASSWORD_FIELD] = os.environ['DJANGO_SUPERUSER_PASSWORD']\n                # Use username from environment variable, if not provided in\n                # options.\n                if username is None:\n                    username = os.environ.get('DJANGO_SUPERUSER_' + self.UserModel.USERNAME_FIELD.upper())\n                if username is None:\n                    raise CommandError('You must use --%s with --noinput.' % self.UserModel.USERNAME_FIELD)\n                else:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        raise CommandError(error_msg)\n\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    env_var = 'DJANGO_SUPERUSER_' + field_name.upper()\n                    value = options[field_name] or os.environ.get(env_var)\n                    if not value:\n                        raise CommandError('You must use --%s with --noinput.' % field_name)\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = field.clean(value, None)\n\n            self.UserModel._default_manager.db_manager(database).create_superuser(**user_data)\n            if options['verbosity'] >= 1:\n                self.stdout.write(\"Superuser created successfully.\")\n        except KeyboardInterrupt:\n            self.stderr.write('\\nOperation cancelled.')\n            sys.exit(1)\n        except exceptions.ValidationError as e:\n            raise CommandError('; '.join(e.messages))\n        except NotRunningInTTYException:\n            self.stdout.write(\n                'Superuser creation skipped due to not running in a TTY. '\n                'You can run `manage.py createsuperuser` in your project '\n                'to create one manually.'\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, *args, **options):\n        username = options[self.UserModel.USERNAME_FIELD]\n        database = options['database']\n        user_data = {}\n        verbose_field_name = self.username_field.verbose_name\n        try:\n            self.UserModel._meta.get_field(PASSWORD_FIELD)\n        except exceptions.FieldDoesNotExist:\n            pass\n        else:\n            # If not provided, create the user with an unusable password.\n            user_data[PASSWORD_FIELD] = None\n        try:\n            if options['interactive']:\n                # Same as user_data but with foreign keys as fake model\n                # instances instead of raw IDs.\n                fake_user_data = {}\n                if hasattr(self.stdin, 'isatty') and not self.stdin.isatty():\n                    raise NotRunningInTTYException\n                default_username = get_default_username()\n                if username:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        self.stderr.write(error_msg)\n                        username = None\n                elif username == '':\n                    raise CommandError('%s cannot be blank.' % capfirst(verbose_field_name))\n                # Prompt for username.\n                while username is None:\n                    message = self._get_input_message(self.username_field, default_username)\n                    username = self.get_input_data(self.username_field, message, default_username)\n                    if username:\n                        error_msg = self._validate_username(username, verbose_field_name, database)\n                        if error_msg:\n                            self.stderr.write(error_msg)\n                            username = None\n                            continue\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                fake_user_data[self.UserModel.USERNAME_FIELD] = (\n                    self.username_field.remote_field.model(username)\n                    if self.username_field.remote_field else username\n                )\n                # Prompt for required fields.\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = options[field_name]\n                    while user_data[field_name] is None:\n                        message = self._get_input_message(field)\n                        input_value = self.get_input_data(field, message)\n                        user_data[field_name] = input_value\n                        fake_user_data[field_name] = input_value\n\n                        # Wrap any foreign keys in fake model instances\n                        if field.remote_field:\n                            fake_user_data[field_name] = field.remote_field.model(input_value)\n\n                # Prompt for a password if the model has one.\n                while PASSWORD_FIELD in user_data and user_data[PASSWORD_FIELD] is None:\n                    password = getpass.getpass()\n                    password2 = getpass.getpass('Password (again): ')\n                    if password != password2:\n                        self.stderr.write(\"Error: Your passwords didn't match.\")\n                        # Don't validate passwords that don't match.\n                        continue\n                    if password.strip() == '':\n                        self.stderr.write(\"Error: Blank passwords aren't allowed.\")\n                        # Don't validate blank passwords.\n                        continue\n                    try:\n                        validate_password(password2, self.UserModel(**fake_user_data))\n                    except exceptions.ValidationError as err:\n                        self.stderr.write('\\n'.join(err.messages))\n                        response = input('Bypass password validation and create user anyway? [y/N]: ')\n                        if response.lower() != 'y':\n                            continue\n                    user_data[PASSWORD_FIELD] = password\n            else:\n                # Non-interactive mode.\n                # Use password from environment variable, if provided.\n                if PASSWORD_FIELD in user_data and 'DJANGO_SUPERUSER_PASSWORD' in os.environ:\n                    user_data[PASSWORD_FIELD] = os.environ['DJANGO_SUPERUSER_PASSWORD']\n                # Use username from environment variable, if not provided in\n                # options.\n                if username is None:\n                    username = os.environ.get('DJANGO_SUPERUSER_' + self.UserModel.USERNAME_FIELD.upper())\n                if username is None:\n                    raise CommandError('You must use --%s with --noinput.' % self.UserModel.USERNAME_FIELD)\n                else:\n                    error_msg = self._validate_username(username, verbose_field_name, database)\n                    if error_msg:\n                        raise CommandError(error_msg)\n\n                user_data[self.UserModel.USERNAME_FIELD] = username\n                for field_name in self.UserModel.REQUIRED_FIELDS:\n                    env_var = 'DJANGO_SUPERUSER_' + field_name.upper()\n                    value = options[field_name] or os.environ.get(env_var)\n                    if not value:\n                        raise CommandError('You must use --%s with --noinput.' % field_name)\n                    field = self.UserModel._meta.get_field(field_name)\n                    user_data[field_name] = field.clean(value, None)\n\n            self.UserModel._default_manager.db_manager(database).create_superuser(**user_data)\n            if options['verbosity'] >= 1:\n                self.stdout.write(\"Superuser created successfully.\")\n        except KeyboardInterrupt:\n            self.stderr.write('\\nOperation cancelled.')\n            sys.exit(1)\n        except exceptions.ValidationError as e:\n            raise CommandError('; '.join(e.messages))\n        except NotRunningInTTYException:\n            self.stdout.write(\n                'Superuser creation skipped due to not running in a TTY. '\n                'You can run `manage.py createsuperuser` in your project '\n                'to create one manually.'\n            )"
        },
        {
          "file": "django/contrib/auth/management/commands/createsuperuser.py",
          "type": "function",
          "name": "_get_input_message",
          "class_name": "Command",
          "code": "def _get_input_message(self, field, default=None):\n        return '%s%s%s: ' % (\n            capfirst(field.verbose_name),\n            \" (leave blank to use '%s')\" % default if default else '',\n            ' (%s.%s)' % (\n                field.remote_field.model._meta.object_name,\n                field.remote_field.field_name,\n            ) if field.remote_field else '',\n        )"
        }
      ]
    }
  ]
}