{
  "instance_id": "django__django-12125",
  "repo": "django/django",
  "created_at": "2019-11-22T12:55:45Z",
  "problem_statement": "makemigrations produces incorrect path for inner classes\nDescription\n\t\nWhen you define a subclass from django.db.models.Field as an inner class of some other class, and use this field inside a django.db.models.Model class, then when you run manage.py makemigrations, a migrations file is created which refers to the inner class as if it were a top-level class of the module it is in.\nTo reproduce, create the following as your model:\nclass Outer(object):\n\tclass Inner(models.CharField):\n\t\tpass\nclass A(models.Model):\n\tfield = Outer.Inner(max_length=20)\nAfter running manage.py makemigrations, the generated migrations file contains the following:\nmigrations.CreateModel(\n\tname='A',\n\tfields=[\n\t\t('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n\t\t('field', test1.models.Inner(max_length=20)),\n\t],\n),\nNote the test1.models.Inner, which should have been test1.models.Outer.Inner.\nThe real life case involved an EnumField from django-enumfields, defined as an inner class of a Django Model class, similar to this:\nimport enum\nfrom enumfields import Enum, EnumField\nclass Thing(models.Model):\n\t@enum.unique\n\tclass State(Enum):\n\t\ton = 'on'\n\t\toff = 'off'\n\tstate = EnumField(enum=State)\nThis results in the following migrations code:\nmigrations.CreateModel(\n\tname='Thing',\n\tfields=[\n\t\t('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n\t\t('state', enumfields.fields.EnumField(enum=test1.models.State, max_length=10)),\n\t],\n),\nThis refers to test1.models.State, instead of to test1.models.Thing.State.\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@@ -269,7 +269,7 @@ def serialize(self):\n             if module == builtins.__name__:\n                 return self.value.__name__, set()\n             else:\n-                return \"%s.%s\" % (module, self.value.__name__), {\"import %s\" % module}\n+                return \"%s.%s\" % (module, self.value.__qualname__), {\"import %s\" % module}\n \n \n class UUIDSerializer(BaseSerializer):\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": 10260,
      "pr_title": "Fixed #29613 -- Fixed --keepdb on PostgreSQL if the database exists and the user can't create databases.",
      "pr_body": "Ticket [29613](https://code.djangoproject.com/ticket/29613).",
      "issue_id": 29613,
      "issue_title": "Allow --keepdb to work on PostgreSQL if the database exists but the user can't create databases",
      "issue_body": "The popular Web Faction hosting service uses a shared database server. Users can create databases using the web UI or XML RPC calls, but not using the SQL CREATE syntax.\nRunning tests throws a ProgrammingError, with the message 'permission denied to create database', even if the test database has been previously created manually.\nThe error code for this error is '42501', which appears to correspond to errorcodes.INSUFFICIENT_PRIVILEGE.\ndjango/db/backends/postgresql/creation.py only handles the error errorcodes.DUPLICATE_DATABASE in _execute_create_test_db(), line 35. Because the error code does not match the program exits with a log message. But it would be fine to proceed with the error code '42501' also, making use of the --keepdb mechanism.\nThis appears to be a regression, as I did not experience this issue either using postgresql_psycopg2 driver or using Django 1.11",
      "issue_closed_at": "2018-08-03T03:32:30",
      "base_commit": "d8e2be459f97f1773c7edf7d37de180139146176",
      "changes": [
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "line",
          "name": "line 3",
          "code": "from psycopg2 import errorcodes\n\nfrom django.db.backends.base.creation import BaseDatabaseCreation\n\n\nclass DatabaseCreation(BaseDatabaseCreation):"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "sql_table_creation_suffix",
          "class_name": "DatabaseCreation",
          "code": "def sql_table_creation_suffix(self):\n        test_settings = self.connection.settings_dict['TEST']\n        assert test_settings['COLLATION'] is None, (\n            \"PostgreSQL does not support collation setting at database creation time.\"\n        )\n        return self._get_database_create_suffix(\n            encoding=test_settings['CHARSET'],\n            template=test_settings.get('TEMPLATE'),\n        )"
        }
      ]
    },
    {
      "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": 9396,
      "pr_title": "[1.11.x] Fixed #28856 -- Fixed an issue with caching of coerced gfk pointing to mti model.",
      "pr_body": "https://code.djangoproject.com/ticket/28856\r\n\r\nSee #9395 for the patch against master.",
      "issue_id": 28856,
      "issue_title": "GenericForeignKey attributes create new instances on every access",
      "issue_body": "Given these models:\nclass OtherSuper(models.Model):\n    pass\n\n\nclass OtherSub(OtherSuper):\n    pass\n\n\nclass Ref(models.Model):\n    obj_type = models.ForeignKey(ContentType, on_delete=models.PROTECT)\n    obj_id = models.CharField(max_length=255)\n    obj = GenericForeignKey('obj_type', 'obj_id')\nI get this behavior:\nIn [1]: ref = Ref.objects.create(obj=OtherSub.objects.create())\n\nIn [2]: id(ref.obj) == id(ref.obj)\nOut[2]: True\n\nIn [3]: ref.refresh_from_db()\n\nIn [4]: id(ref.obj) == id(ref.obj)\nOut[4]: False\nEach time\nref.obj\nis accessed, a new instance is created for its value. This is a problem, since doing something like\nref.obj.field = 1; ref.obj.save()\nwon't actually update the field in the database. This only happens when the referenced object is an instance of a model that subclasses another model. (So, it wouldn't happen if referencing\nOtherSuper\nin the models above.) The\nrefresh_from_db()\ncall is also necessary to reproduce this in a simple test like the above; it happens with any instance created from an existing DB record.\nI've written a regression test against stable/1.10.x . I'll attach a patch.\nI discovered this because I have code that does the above (changes a field on the related model and calls save). I call this a regression because it works correctly on 1.9.\nI'm not sure what the underly bug is; I looked at the diff in\ncontenttypes\nbetween 1.9 and 1.10, and there are more than a few changes. Hopefully someone who understands the\nGenericForeignKey\nimplementation can figure this out.",
      "issue_closed_at": "2017-12-08T13:15:18",
      "base_commit": "3545e844885608932a692d952c12cd863e2320b5",
      "changes": [
        {
          "file": "django/contrib/contenttypes/fields.py",
          "type": "function",
          "name": "__get__",
          "class_name": "GenericForeignKey",
          "code": "def __get__(self, instance, cls=None):\n        if instance is None:\n            return self\n\n        # Don't use getattr(instance, self.ct_field) here because that might\n        # reload the same ContentType over and over (#5570). Instead, get the\n        # content type ID here, and later when the actual instance is needed,\n        # use ContentType.objects.get_for_id(), which has a global cache.\n        f = self.model._meta.get_field(self.ct_field)\n        ct_id = getattr(instance, f.get_attname(), None)\n        pk_val = getattr(instance, self.fk_field)\n\n        try:\n            rel_obj = getattr(instance, self.cache_attr)\n        except AttributeError:\n            rel_obj = None\n        else:\n            if rel_obj and (ct_id != self.get_content_type(obj=rel_obj, using=instance._state.db).id or\n                            rel_obj._meta.pk.to_python(pk_val) != rel_obj._get_pk_val()):\n                rel_obj = None\n\n        if rel_obj is not None:\n            return rel_obj\n\n        if ct_id is not None:\n            ct = self.get_content_type(id=ct_id, using=instance._state.db)\n            try:\n                rel_obj = ct.get_object_for_this_type(pk=pk_val)\n            except ObjectDoesNotExist:\n                pass\n        setattr(instance, self.cache_attr, rel_obj)\n        return rel_obj"
        }
      ]
    },
    {
      "pr_number": 7036,
      "pr_title": "Fixed #27024 -- Prevented logging error with empty string as geometry widget value",
      "pr_body": "Thanks Gavin Wahl for the report.\n",
      "issue_id": 27024,
      "issue_title": "BaseGeometryWidget logs a false positive: Error creating geometry from value ''",
      "issue_body": "When using a form with a PointField (or presumably anything using BaseGeometryWidget), if you\nDon't fill out the map field\nCause a ValidationError for another field\nSubmit the form\nThis error-level message will be logged:\nError creating geometry from value ''\nThere should not be any error logged because there is no error. Maybe\nBaseGeometryWidget.deserialize\nneeds to check for\nEMPTY_VALUES\n?",
      "issue_closed_at": "2016-08-08T09:25:25",
      "base_commit": "2a11d2d7a7d5c6609c85dbc631fad6b8a8645a64",
      "changes": [
        {
          "file": "django/contrib/gis/admin/widgets.py",
          "type": "function",
          "name": "render",
          "class_name": "OpenLayersWidget",
          "code": "def render(self, name, value, attrs=None):\n        # Update the template parameters with any attributes passed in.\n        if attrs:\n            self.params.update(attrs)\n            self.params['editable'] = self.params['modifiable']\n        else:\n            self.params['editable'] = True\n\n        # Defaulting the WKT value to a blank string -- this\n        # will be tested in the JavaScript and the appropriate\n        # interface will be constructed.\n        self.params['wkt'] = ''\n\n        # If a string reaches here (via a validation error on another\n        # field) then just reconstruct the Geometry.\n        if isinstance(value, six.string_types):\n            try:\n                value = GEOSGeometry(value)\n            except (GEOSException, ValueError) as err:\n                logger.error(\"Error creating geometry from value '%s' (%s)\", value, err)\n                value = None\n\n        if (value and value.geom_type.upper() != self.geom_type and\n                self.geom_type != 'GEOMETRY'):\n            value = None\n\n        # Constructing the dictionary of the map options.\n        self.params['map_options'] = self.map_options()\n\n        # Constructing the JavaScript module name using the name of\n        # the GeometryField (passed in via the `attrs` keyword).\n        # Use the 'name' attr for the field name (rather than 'field')\n        self.params['name'] = name\n        # note: we must switch out dashes for underscores since js\n        # functions are created using the module variable\n        js_safe_name = self.params['name'].replace('-', '_')\n        self.params['module'] = 'geodjango_%s' % js_safe_name\n\n        if value:\n            # Transforming the geometry to the projection used on the\n            # OpenLayers map.\n            srid = self.params['srid']\n            if value.srid != srid:\n                try:\n                    ogr = value.ogr\n                    ogr.transform(srid)\n                    wkt = ogr.wkt\n                except GDALException as err:\n                    logger.error(\n                        \"Error transforming geometry from srid '%s' to srid '%s' (%s)\",\n                        value.srid, srid, err\n                    )\n                    wkt = ''\n            else:\n                wkt = value.wkt\n\n            # Setting the parameter WKT with that of the transformed\n            # geometry.\n            self.params['wkt'] = wkt\n\n        self.params.update(geo_context)\n        return loader.render_to_string(self.template, self.params)"
        },
        {
          "file": "django/contrib/gis/forms/widgets.py",
          "type": "function",
          "name": "deserialize",
          "class_name": "BaseGeometryWidget",
          "code": "def deserialize(self, value):\n        try:\n            return GEOSGeometry(value, self.map_srid)\n        except (GEOSException, ValueError) as err:\n            logger.error(\"Error creating geometry from value '%s' (%s)\", value, err)\n        return None"
        }
      ]
    }
  ]
}