{
  "instance_id": "django__django-10924",
  "repo": "django/django",
  "created_at": "2019-02-03T11:30:12Z",
  "problem_statement": "Allow FilePathField path to accept a callable.\nDescription\n\t\nI have a special case where I want to create a model containing the path to some local files on the server/dev machine. Seeing as the place where these files are stored is different on different machines I have the following:\nimport os\nfrom django.conf import settings\nfrom django.db import models\nclass LocalFiles(models.Model):\n\tname = models.CharField(max_length=255)\n\tfile = models.FilePathField(path=os.path.join(settings.LOCAL_FILE_DIR, 'example_dir'))\nNow when running manage.py makemigrations it will resolve the path based on the machine it is being run on. Eg: /home/<username>/server_files/example_dir\nI had to manually change the migration to include the os.path.join() part to not break this when running the migration on production/other machine.\n",
  "patch": "diff --git a/django/db/models/fields/__init__.py b/django/db/models/fields/__init__.py\n--- a/django/db/models/fields/__init__.py\n+++ b/django/db/models/fields/__init__.py\n@@ -1709,7 +1709,7 @@ def get_prep_value(self, value):\n \n     def formfield(self, **kwargs):\n         return super().formfield(**{\n-            'path': self.path,\n+            'path': self.path() if callable(self.path) else self.path,\n             'match': self.match,\n             'recursive': self.recursive,\n             'form_class': forms.FilePathField,\n",
  "similar_bug_items": [
    {
      "pr_number": 7472,
      "pr_title": "Fixed #27436 -- Display absolute path in makemigrations if a relative path doesn't exist.",
      "pr_body": "For example on Windows it's impossible to obtain a relative path\r\nbetween files located on separate drives (C: and D:, for example).",
      "issue_id": 27436,
      "issue_title": "migrations.test_commands.MakeMigrationsTests fail on Windows when run on a different drive than C:",
      "issue_body": "Some of the test cases in MakeMigrationsTests fail on Windows due to inability to find a relative path between files located on separate drives.\nThe tests create a temporary migrations directory using\ntempfile.mkdtemp\n, which on Windows creates a directory under C:\\Temp (or C:\\Users\\<username>\\AppData\\Local\\Temp).\nIf one clones Django source code to a drive other than C:, then the makemigrations command tests fail with an error like the following (Windows 7, Python 3.5.1):\nERROR: test_makemigrations_non_interactive_not_null_alteration (migrations.test_commands.MakeMigrationsTests)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"D:\\dev\\django\\tests\\migrations\\test_commands.py\", line 834, in test_makemigrations_non_interactive_not_null_alteration\n    call_command(\"makemigrations\", \"migrations\", interactive=False, stdout=out)\n  File \"d:\\dev\\django\\django\\core\\management\\__init__.py\", line 130, in call_command\n    return command.execute(*args, **defaults)\n  File \"d:\\dev\\django\\django\\core\\management\\base.py\", line 330, in execute\n    output = self.handle(*args, **options)\n  File \"d:\\dev\\django\\django\\core\\management\\commands\\makemigrations.py\", line 193, in handle\n    self.write_migration_files(changes)\n  File \"d:\\dev\\django\\django\\core\\management\\commands\\makemigrations.py\", line 211, in write_migration_files\n    migration_string = os.path.relpath(writer.path)\n  File \"D:\\Miniconda3\\lib\\ntpath.py\", line 574, in relpath\n    path_drive, start_drive))\nValueError: path is on mount 'C:', start on mount 'D:'\nSince the value returned from\nrelpath\nis used only for display and not for any I/O, I'd suggest catching this\nValueError\nand using an absolute path in that case.",
      "issue_closed_at": "2016-11-08T17:06:24",
      "base_commit": "dacef9137f43fff88b527d1c02f6fe6a81e975aa",
      "changes": [
        {
          "file": "django/core/management/commands/makemigrations.py",
          "type": "function",
          "name": "write_migration_files",
          "class_name": "Command",
          "code": "def write_migration_files(self, changes):\n        \"\"\"\n        Takes a changes dict and writes them out as migration files.\n        \"\"\"\n        directory_created = {}\n        for app_label, app_migrations in changes.items():\n            if self.verbosity >= 1:\n                self.stdout.write(self.style.MIGRATE_HEADING(\"Migrations for '%s':\" % app_label) + \"\\n\")\n            for migration in app_migrations:\n                # Describe the migration\n                writer = MigrationWriter(migration)\n                if self.verbosity >= 1:\n                    # Display a relative path if it's below the current working\n                    # directory, or an absolute path otherwise.\n                    migration_string = os.path.relpath(writer.path)\n                    if migration_string.startswith('..'):\n                        migration_string = writer.path\n                    self.stdout.write(\"  %s:\\n\" % (self.style.MIGRATE_LABEL(migration_string),))\n                    for operation in migration.operations:\n                        self.stdout.write(\"    - %s\\n\" % operation.describe())\n                if not self.dry_run:\n                    # Write the migrations file to the disk.\n                    migrations_directory = os.path.dirname(writer.path)\n                    if not directory_created.get(app_label):\n                        if not os.path.isdir(migrations_directory):\n                            os.mkdir(migrations_directory)\n                        init_path = os.path.join(migrations_directory, \"__init__.py\")\n                        if not os.path.isfile(init_path):\n                            open(init_path, \"w\").close()\n                        # We just do this once per app\n                        directory_created[app_label] = True\n                    migration_string = writer.as_string()\n                    with io.open(writer.path, \"w\", encoding='utf-8') as fh:\n                        fh.write(migration_string)\n                elif self.verbosity == 3:\n                    # Alternatively, makemigrations --dry-run --verbosity 3\n                    # will output the migrations to stdout rather than saving\n                    # the file to the disk.\n                    self.stdout.write(self.style.MIGRATE_HEADING(\n                        \"Full migrations file '%s':\" % writer.filename) + \"\\n\"\n                    )\n                    self.stdout.write(\"%s\\n\" % writer.as_string())"
        }
      ]
    },
    {
      "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": 8003,
      "pr_title": "Fixed #27800 -- Fixed QuerySet.annotate(Length(...)).distinct() crash.",
      "pr_body": "This pull request fixes a recently introduced `.distinct()` bug by reverting 3b2db6ec12ce7d4b32f60dd7713e5f23cac498b7.\r\n\r\nAlso, I've added a unit test to prevent future regressions.\r\n\r\nBasically the issue is that `.annotate()` introduces fields that are not attached to any particular model and therefore Django cannot assume that `field.model` would always work without throwing an exception.\r\n\r\nEarlier tests did not catch the error because by coincidence the annotation `Func`tions forwarded the field on which they were operating as their `output_field` (possibly another bug?). The newly added test calls `Length()` `Func`tion which returns an IntegerFIeld() as the `output_field` and thus is not attached to any model fields.",
      "issue_id": 27800,
      "issue_title": "Crash with QuerySet.annotate(Length(...)).distinct()",
      "issue_body": "After\n3b2db6ec12ce7d4b32f60dd7713e5f23cac498b7\n, this query for\ntest/annotations/\ncrashes with\nAttributeError: 'IntegerField' object has no attribute 'model'\n.\nEmployee.objects.annotate(\n    name_len=Length('first_name'),\n).distinct('name_len').values_list('name_len', flat=True)\n​\nPR",
      "issue_closed_at": "2017-01-31T17:46:01",
      "base_commit": "ed3215ad53c0b22c58883f97286a878c577e25d6",
      "changes": [
        {
          "file": "django/db/models/sql/query.py",
          "type": "function",
          "name": "names_to_path",
          "class_name": "Query",
          "code": "def names_to_path(self, names, opts, allow_many=True, fail_on_missing=False):\n        \"\"\"\n        Walks the list of names and turns them into PathInfo tuples. Note that\n        a single name in 'names' can generate multiple PathInfos (m2m for\n        example).\n\n        'names' is the path of names to travel, 'opts' is the model Options we\n        start the name resolving from, 'allow_many' is as for setup_joins().\n        If fail_on_missing is set to True, then a name that can't be resolved\n        will generate a FieldError.\n\n        Returns a list of PathInfo tuples. In addition returns the final field\n        (the last used join field), and target (which is a field guaranteed to\n        contain the same value as the final field). Finally, the method returns\n        those names that weren't found (which are likely transforms and the\n        final lookup).\n        \"\"\"\n        path, names_with_path = [], []\n        for pos, name in enumerate(names):\n            cur_names_with_path = (name, [])\n            if name == 'pk':\n                name = opts.pk.name\n\n            field = None\n            try:\n                field = opts.get_field(name)\n            except FieldDoesNotExist:\n                if name in self.annotation_select:\n                    field = self.annotation_select[name].output_field\n\n            if field is not None:\n                # Fields that contain one-to-many relations with a generic\n                # model (like a GenericForeignKey) cannot generate reverse\n                # relations and therefore cannot be used for reverse querying.\n                if field.is_relation and not field.related_model:\n                    raise FieldError(\n                        \"Field %r does not generate an automatic reverse \"\n                        \"relation and therefore cannot be used for reverse \"\n                        \"querying. If it is a GenericForeignKey, consider \"\n                        \"adding a GenericRelation.\" % name\n                    )\n                model = field.model._meta.concrete_model\n            else:\n                # We didn't find the current field, so move position back\n                # one step.\n                pos -= 1\n                if pos == -1 or fail_on_missing:\n                    field_names = list(get_field_names_from_opts(opts))\n                    available = sorted(field_names + list(self.annotation_select))\n                    raise FieldError(\"Cannot resolve keyword '%s' into field. \"\n                                     \"Choices are: %s\" % (name, \", \".join(available)))\n                break\n            # Check if we need any joins for concrete inheritance cases (the\n            # field lives in parent, but we are currently in one of its\n            # children)\n            if model is not opts.model:\n                path_to_parent = opts.get_path_to_parent(model)\n                if path_to_parent:\n                    path.extend(path_to_parent)\n                    cur_names_with_path[1].extend(path_to_parent)\n                    opts = path_to_parent[-1].to_opts\n            if hasattr(field, 'get_path_info'):\n                pathinfos = field.get_path_info()\n                if not allow_many:\n                    for inner_pos, p in enumerate(pathinfos):\n                        if p.m2m:\n                            cur_names_with_path[1].extend(pathinfos[0:inner_pos + 1])\n                            names_with_path.append(cur_names_with_path)\n                            raise MultiJoin(pos + 1, names_with_path)\n                last = pathinfos[-1]\n                path.extend(pathinfos)\n                final_field = last.join_field\n                opts = last.to_opts\n                targets = last.target_fields\n                cur_names_with_path[1].extend(pathinfos)\n                names_with_path.append(cur_names_with_path)\n            else:\n                # Local non-relational field.\n                final_field = field\n                targets = (field,)\n                if fail_on_missing and pos + 1 != len(names):\n                    raise FieldError(\n                        \"Cannot resolve keyword %r into field. Join on '%s'\"\n                        \" not permitted.\" % (names[pos + 1], name))\n                break\n        return path, final_field, targets, names[pos + 1:]"
        }
      ]
    },
    {
      "pr_number": 4356,
      "pr_title": "Fixed #24515 -- Fixed DjangoTranslation plural handling",
      "pr_body": "",
      "issue_id": 24515,
      "issue_title": "Plural handling broken",
      "issue_body": "Translation of plural strings is currently not following the plural equation from the po files and enforce the English default equation.\ntests/i18n/tests.py\nTest case:diff --git a/tests/i18n/tests.py b/tests/i18n/tests.py\nindex ad23861..65aedc0 100644\na\nb\nfrom django.utils.translation import (\n30\n30\nget_language, get_language_from_request, get_language_info, gettext,\n31\n31\ngettext_lazy, ngettext_lazy, npgettext, npgettext_lazy, pgettext,\n32\n32\npgettext_lazy, string_concat, to_locale, trans_real, ugettext,\n33\nugettext_lazy, ungettext\n_lazy,\n33\nugettext_lazy, ungettext\n, ungettext\n_lazy,\n34\n34\n)\n35\n35\n36\n36\nfrom .forms import CompanyForm, I18nForm, SelectDateForm\n…\n…\ndef patch_formats(lang, **settings):\n57\n57\n58\n58\nclass TranslationTests(TestCase):\n59\n59\n60\n@translation.override('fr')\n61\ndef test_plural(self):\n62\n\"\"\"\n63\nTest plurals with ungettext. French differs from English in that 0 is singular.\n64\n\"\"\"\n65\nself.assertEqual(ungettext(\"%d year\", \"%d years\", 0) % 0, \"0 année\")\n66\nself.assertEqual(ungettext(\"%d year\", \"%d years\", 2) % 2, \"2 années\")\n67\nself.assertEqual(ungettext(\"%(size)d byte\", \"%(size)d bytes\", 0) % {'size': 0}, \"0 octet\")\n68\nself.assertEqual(ungettext(\"%(size)d byte\", \"%(size)d bytes\", 2) % {'size': 2}, \"2 octets\")\n69\n60\n70\ndef test_override(self):\n61\n71\nactivate('de')\n62\n72\ntry:",
      "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": 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]]"
        }
      ]
    }
  ]
}