{
  "instance_id": "django__django-11583",
  "repo": "django/django",
  "created_at": "2019-07-21T20:56:14Z",
  "problem_statement": "Auto-reloading with StatReloader very intermittently throws \"ValueError: embedded null byte\".\nDescription\n\t\nRaising this mainly so that it's tracked, as I have no idea how to reproduce it, nor why it's happening. It ultimately looks like a problem with Pathlib, which wasn't used prior to 2.2.\nStacktrace:\nTraceback (most recent call last):\n File \"manage.py\" ...\n\texecute_from_command_line(sys.argv)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/__init__.py\", line 381, in execute_from_command_line\n\tutility.execute()\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/__init__.py\", line 375, in execute\n\tself.fetch_command(subcommand).run_from_argv(self.argv)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/base.py\", line 323, in run_from_argv\n\tself.execute(*args, **cmd_options)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/commands/runserver.py\", line 60, in execute\n\tsuper().execute(*args, **options)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/base.py\", line 364, in execute\n\toutput = self.handle(*args, **options)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/commands/runserver.py\", line 95, in handle\n\tself.run(**options)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/core/management/commands/runserver.py\", line 102, in run\n\tautoreload.run_with_reloader(self.inner_run, **options)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 577, in run_with_reloader\n\tstart_django(reloader, main_func, *args, **kwargs)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 562, in start_django\n\treloader.run(django_main_thread)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 280, in run\n\tself.run_loop()\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 286, in run_loop\n\tnext(ticker)\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 326, in tick\n\tfor filepath, mtime in self.snapshot_files():\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 342, in snapshot_files\n\tfor file in self.watched_files():\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 241, in watched_files\n\tyield from iter_all_python_module_files()\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 103, in iter_all_python_module_files\n\treturn iter_modules_and_files(modules, frozenset(_error_files))\n File \"/Userz/kez/path/to/venv/lib/python3.6/site-packages/django/utils/autoreload.py\", line 132, in iter_modules_and_files\n\tresults.add(path.resolve().absolute())\n File \"/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/pathlib.py\", line 1120, in resolve\n\ts = self._flavour.resolve(self, strict=strict)\n File \"/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/pathlib.py\", line 346, in resolve\n\treturn _resolve(base, str(path)) or sep\n File \"/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/pathlib.py\", line 330, in _resolve\n\ttarget = accessor.readlink(newpath)\n File \"/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/pathlib.py\", line 441, in readlink\n\treturn os.readlink(path)\nValueError: embedded null byte\nI did print(path) before os.readlink(path) in pathlib and ended up with:\n/Users/kez\n/Users/kez/.pyenv\n/Users/kez/.pyenv/versions\n/Users/kez/.pyenv/versions/3.6.2\n/Users/kez/.pyenv/versions/3.6.2/lib\n/Users/kez/.pyenv/versions/3.6.2/lib/python3.6\n/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/asyncio\n/Users/kez/.pyenv/versions/3.6.2/lib/python3.6/asyncio/selector_events.py\n/Users\nIt always seems to be /Users which is last\nIt may have already printed /Users as part of another .resolve() multiple times (that is, the order is not deterministic, and it may have traversed beyond /Users successfully many times during startup.\nI don't know where to begin looking for the rogue null byte, nor why it only exists sometimes.\nBest guess I have is that there's a mountpoint in /Users to a samba share which may not have been connected to yet? I dunno.\nI have no idea if it's fixable without removing the use of pathlib (which tbh I think should happen anyway, because it's slow) and reverting to using os.path.join and friends. \nI have no idea if it's fixed in a later Python version, but with no easy way to reproduce ... dunno how I'd check.\nI have no idea if it's something specific to my system (pyenv, OSX 10.11, etc)\n",
  "patch": "diff --git a/django/utils/autoreload.py b/django/utils/autoreload.py\n--- a/django/utils/autoreload.py\n+++ b/django/utils/autoreload.py\n@@ -143,6 +143,10 @@ def iter_modules_and_files(modules, extra_files):\n             # The module could have been removed, don't fail loudly if this\n             # is the case.\n             continue\n+        except ValueError as e:\n+            # Network filesystems may return null bytes in file paths.\n+            logger.debug('\"%s\" raised when resolving path: \"%s\"' % (str(e), path))\n+            continue\n         results.add(resolved_path)\n     return frozenset(results)\n \n",
  "similar_bug_items": [
    {
      "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": 9112,
      "pr_title": "Fixed #27846 -- clear all cached reverse relationships on refresh_from_db()",
      "pr_body": "https://code.djangoproject.com/ticket/27846",
      "issue_id": 27846,
      "issue_title": "refresh_from_db() doesn't clear reverse OneToOneFields",
      "issue_body": "Sorry for the poor summary, it is difficult to explain in words. I have a project to demo this bug attached to this ticket, but I will try to explain the bug anyway in steps and the setup.\nSetup:\n2 models (A and B)\nB has a OneToOne to A\nBoth A and B have a field (ie TextField)\nSetup either a signal or override save() for A to update B's TextField to equal that of A's on save() or post_save for signals\nSteps:\nCreate instance of A\nGet another copy of the instance of A via A.objects.get()\nCreate instance of B using the copy of the instance of A\nDo refresh_from_db() on original instance of A\nTry to access B from A\nThe project I have provided is a slim version of this problem that demonstrates it with signals, overriden save(), and basic set and save inside the test. The basic set and save works, but the other two fail when using the above steps. Run the test suite to see.",
      "issue_closed_at": "2017-10-12T16:25:22",
      "base_commit": "df0aebc893973c78d7d2cda712ba4133dbe29b6e",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "refresh_from_db",
          "class_name": "Model",
          "code": "def refresh_from_db(self, using=None, fields=None):\n        \"\"\"\n        Reload field values from the database.\n\n        By default, the reloading happens from the database this instance was\n        loaded from, or by the read router if this instance wasn't loaded from\n        any database. The using parameter will override the default.\n\n        Fields can be used to specify which fields to reload. The fields\n        should be an iterable of field attnames. If fields is None, then\n        all non-deferred fields are reloaded.\n\n        When accessing deferred fields of an instance, the deferred loading\n        of the field will call this method.\n        \"\"\"\n        if fields is not None:\n            if len(fields) == 0:\n                return\n            if any(LOOKUP_SEP in f for f in fields):\n                raise ValueError(\n                    'Found \"%s\" in fields argument. Relations and transforms '\n                    'are not allowed in fields.' % LOOKUP_SEP)\n\n        db = using if using is not None else self._state.db\n        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n\n        # Use provided fields, if not set then reload all non-deferred fields.\n        deferred_fields = self.get_deferred_fields()\n        if fields is not None:\n            fields = list(fields)\n            db_instance_qs = db_instance_qs.only(*fields)\n        elif deferred_fields:\n            fields = [f.attname for f in self._meta.concrete_fields\n                      if f.attname not in deferred_fields]\n            db_instance_qs = db_instance_qs.only(*fields)\n\n        db_instance = db_instance_qs.get()\n        non_loaded_fields = db_instance.get_deferred_fields()\n        for field in self._meta.concrete_fields:\n            if field.attname in non_loaded_fields:\n                # This field wasn't refreshed - skip ahead.\n                continue\n            setattr(self, field.attname, getattr(db_instance, field.attname))\n            # Throw away stale foreign key references.\n            if field.is_relation and field.is_cached(self):\n                rel_instance = field.get_cached_value(self)\n                local_val = getattr(db_instance, field.attname)\n                related_val = None if rel_instance is None else getattr(rel_instance, field.target_field.attname)\n                if local_val != related_val or (local_val is None and related_val is None):\n                    field.delete_cached_value(self)\n        self._state.db = db_instance._state.db"
        }
      ]
    },
    {
      "pr_number": 8228,
      "pr_title": "Fixed #27966 -- Bumped required psycopg2 version to 2.5.4.",
      "pr_body": "",
      "issue_id": 27966,
      "issue_title": "Bump required version of pyscopg2 to 2.5.4",
      "issue_body": "​\nthis commit\nuses the cursor as context manager (line in question is marked), which were added in psycopg2 2.5 (\n​\nrelease notes\n) (see third item)\nbut\n​\nhere\ndjango checks only for 2.4.5.\n​\nthis commit here\nmade 2.4.5 a requirement and documented that in a few places.",
      "issue_closed_at": "2017-03-21T11:23:31",
      "base_commit": "7063a85579f40585f2601ba6e6887b0982e7ce43",
      "changes": [
        {
          "file": "django/db/backends/postgresql/base.py",
          "type": "function",
          "name": "psycopg2_version",
          "class_name": "DatabaseWrapper",
          "code": "def psycopg2_version(self):\n        return PSYCOPG2_VERSION"
        }
      ]
    },
    {
      "pr_number": 6540,
      "pr_title": "Fixed #26341 (again) -- Addressed multiple occurrences per line use case",
      "pr_body": "",
      "issue_id": 26341,
      "issue_title": "Weird comments in PO files (.html.py filenames)",
      "issue_body": "I just upgraded Django to 1.9.4 from 1.8.10, and sometimes, the filenames in PO files comments contain \".html.py\" extensions.\nThis is visible on django's main repository :\n​\nhttps://github.com/django/django/blob/ae4d932b1ac12651a7c57d89742c25483ee8c9f9/django/contrib/admin/locale/en/LC_MESSAGES/django.po#L282\n​\nhttps://github.com/django/django/blob/4323676ea5ab6994feb1385522665069d84f397b/django/contrib/admin/locale/en/LC_MESSAGES/django.po#L302\nIn this example, the \"contrib/admin/templates/admin/base_site.html\" file is now named \"contrib/admin/templates/admin/base_site.html.py\" (with a trailing \".py\") in the po file.\nThis seems to appear only on lines with a python file before the template html file.\nclaudep found that this could be the faulty commit :\n​\nhttps://github.com/django/django/commit/e75882332c",
      "issue_closed_at": "2016-04-30T05:07:43",
      "base_commit": "4e2ee8662753ca6a2619039b903f11c60709f398",
      "changes": [
        {
          "file": "django/core/management/commands/makemessages.py",
          "type": "function",
          "name": "postprocess_messages",
          "class_name": "BuildFile",
          "code": "def postprocess_messages(self, msgs):\n        \"\"\"\n        Postprocess messages generated by xgettext GNU gettext utility.\n\n        Transform paths as if these messages were generated from original\n        translatable files rather than from preprocessed versions.\n        \"\"\"\n        if not self.is_templatized:\n            return msgs\n\n        # Remove '.py' suffix\n        if os.name == 'nt':\n            # Preserve '.\\' prefix on Windows to respect gettext behavior\n            old_path = self.work_path\n            new_path = self.path\n        else:\n            old_path = self.work_path[2:]\n            new_path = self.path[2:]\n\n        return re.sub(\n            r'^(#: .*)(' + re.escape(old_path) + r')',\n            r'\\1' + new_path,\n            msgs,\n            flags=re.MULTILINE\n        )"
        }
      ]
    },
    {
      "pr_number": 3259,
      "pr_title": "Fixed #23537 -- Added Oracle GIS SchemaEditor.",
      "pr_body": "",
      "issue_id": 23537,
      "issue_title": "Oracle GIS backend missing SchemaEditor",
      "issue_body": "I think this results in missing indexes and metadata for GIS fields.\nThere is one test failure that will be fixed by this:\n======================================================================\nFAIL: test_add_gis_field (django.contrib.gis.tests.gis_migrations.test_operations.OperationTests)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/tim/code/django/django/contrib/gis/tests/gis_migrations/test_operations.py\", line 77, in test_add_gis_field\n    2\nAssertionError: 0 != 2",
      "issue_closed_at": "2014-09-25T19:33:18",
      "base_commit": "45bd7b3bd9008941580c100b9fc7361e3ff3ff0d",
      "changes": [
        {
          "file": "django/contrib/gis/db/backends/oracle/base.py",
          "type": "line",
          "name": "line 6",
          "code": "from django.contrib.gis.db.backends.oracle.creation import OracleCreation\nfrom django.contrib.gis.db.backends.oracle.introspection import OracleIntrospection\nfrom django.contrib.gis.db.backends.oracle.operations import OracleOperations\n\n\nclass DatabaseFeatures(BaseSpatialFeatures, OracleDatabaseFeatures):"
        },
        {
          "file": "django/contrib/gis/db/backends/oracle/base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DatabaseWrapper",
          "code": "def __init__(self, *args, **kwargs):\n        super(DatabaseWrapper, self).__init__(*args, **kwargs)\n        self.features = DatabaseFeatures(self)\n        self.ops = OracleOperations(self)\n        self.creation = OracleCreation(self)\n        self.introspection = OracleIntrospection(self)"
        },
        {
          "file": "django/contrib/gis/db/backends/oracle/operations.py",
          "type": "class",
          "name": "OracleOperations",
          "code": "class OracleOperations(DatabaseOperations, BaseSpatialOperations):\n    compiler_module = \"django.contrib.gis.db.backends.oracle.compiler\"\n\n    name = 'oracle'\n    oracle = True\n    valid_aggregates = {'Union', 'Extent'}\n\n    Adapter = OracleSpatialAdapter\n    Adaptor = Adapter  # Backwards-compatibility alias.\n\n    area = 'SDO_GEOM.SDO_AREA'\n    gml = 'SDO_UTIL.TO_GMLGEOMETRY'\n    centroid = 'SDO_GEOM.SDO_CENTROID'\n    difference = 'SDO_GEOM.SDO_DIFFERENCE'\n    distance = 'SDO_GEOM.SDO_DISTANCE'\n    extent = 'SDO_AGGR_MBR'\n    intersection = 'SDO_GEOM.SDO_INTERSECTION'\n    length = 'SDO_GEOM.SDO_LENGTH'\n    num_geom = 'SDO_UTIL.GETNUMELEM'\n    num_points = 'SDO_UTIL.GETNUMVERTICES'\n    perimeter = length\n    point_on_surface = 'SDO_GEOM.SDO_POINTONSURFACE'\n    reverse = 'SDO_UTIL.REVERSE_LINESTRING'\n    sym_difference = 'SDO_GEOM.SDO_XOR'\n    transform = 'SDO_CS.TRANSFORM'\n    union = 'SDO_GEOM.SDO_UNION'\n    unionagg = 'SDO_AGGR_UNION'\n\n    # We want to get SDO Geometries as WKT because it is much easier to\n    # instantiate GEOS proxies from WKT than SDO_GEOMETRY(...) strings.\n    # However, this adversely affects performance (i.e., Java is called\n    # to convert to WKT on every query).  If someone wishes to write a\n    # SDO_GEOMETRY(...) parser in Python, let me know =)\n    select = 'SDO_UTIL.TO_WKTGEOMETRY(%s)'\n\n    distance_functions = {\n        'distance_gt': (SDODistance('>'), dtypes),\n        'distance_gte': (SDODistance('>='), dtypes),\n        'distance_lt': (SDODistance('<'), dtypes),\n        'distance_lte': (SDODistance('<='), dtypes),\n        'dwithin': (SDODWithin(), dtypes),\n    }\n\n    geometry_functions = {\n        'contains': SDOOperation('SDO_CONTAINS'),\n        'coveredby': SDOOperation('SDO_COVEREDBY'),\n        'covers': SDOOperation('SDO_COVERS'),\n        'disjoint': SDOGeomRelate('DISJOINT'),\n        'intersects': SDOOperation('SDO_OVERLAPBDYINTERSECT'),  # TODO: Is this really the same as ST_Intersects()?\n        'equals': SDOOperation('SDO_EQUAL'),\n        'exact': SDOOperation('SDO_EQUAL'),\n        'overlaps': SDOOperation('SDO_OVERLAPS'),\n        'same_as': SDOOperation('SDO_EQUAL'),\n        'relate': (SDORelate, six.string_types),  # Oracle uses a different syntax, e.g., 'mask=inside+touch'\n        'touches': SDOOperation('SDO_TOUCH'),\n        'within': SDOOperation('SDO_INSIDE'),\n    }\n    geometry_functions.update(distance_functions)\n\n    gis_terms = set(['isnull'])\n    gis_terms.update(geometry_functions)\n\n    truncate_params = {'relate': None}\n\n    def get_db_converters(self, internal_type):\n        converters = super(OracleOperations, self).get_db_converters(internal_type)\n        geometry_fields = (\n            'PointField', 'GeometryField', 'LineStringField',\n            'PolygonField', 'MultiPointField', 'MultiLineStringField',\n            'MultiPolygonField', 'GeometryCollectionField', 'GeomField',\n            'GMLField',\n        )\n        if internal_type in geometry_fields:\n            converters.append(self.convert_textfield_value)\n        return converters\n\n    def convert_extent(self, clob):\n        if clob:\n            # Generally, Oracle returns a polygon for the extent -- however,\n            # it can return a single point if there's only one Point in the\n            # table.\n            ext_geom = Geometry(clob.read())\n            gtype = str(ext_geom.geom_type)\n            if gtype == 'Polygon':\n                # Construct the 4-tuple from the coordinates in the polygon.\n                shell = ext_geom.shell\n                ll, ur = shell[0][:2], shell[2][:2]\n            elif gtype == 'Point':\n                ll = ext_geom.coords[:2]\n                ur = ll\n            else:\n                raise Exception('Unexpected geometry type returned for extent: %s' % gtype)\n            xmin, ymin = ll\n            xmax, ymax = ur\n            return (xmin, ymin, xmax, ymax)\n        else:\n            return None\n\n    def convert_geom(self, clob, geo_field):\n        if clob:\n            return Geometry(clob.read(), geo_field.srid)\n        else:\n            return None\n\n    def geo_db_type(self, f):\n        \"\"\"\n        Returns the geometry database type for Oracle.  Unlike other spatial\n        backends, no stored procedure is necessary and it's the same for all\n        geometry types.\n        \"\"\"\n        return 'MDSYS.SDO_GEOMETRY'\n\n    def get_distance(self, f, value, lookup_type):\n        \"\"\"\n        Returns the distance parameters given the value and the lookup type.\n        On Oracle, geometry columns with a geodetic coordinate system behave\n        implicitly like a geography column, and thus meters will be used as\n        the distance parameter on them.\n        \"\"\"\n        if not value:\n            return []\n        value = value[0]\n        if isinstance(value, Distance):\n            if f.geodetic(self.connection):\n                dist_param = value.m\n            else:\n                dist_param = getattr(value, Distance.unit_attname(f.units_name(self.connection)))\n        else:\n            dist_param = value\n\n        # dwithin lookups on Oracle require a special string parameter\n        # that starts with \"distance=\".\n        if lookup_type == 'dwithin':\n            dist_param = 'distance=%s' % dist_param\n\n        return [dist_param]\n\n    def get_geom_placeholder(self, f, value):\n        \"\"\"\n        Provides a proper substitution value for Geometries that are not in the\n        SRID of the field.  Specifically, this routine will substitute in the\n        SDO_CS.TRANSFORM() function call.\n        \"\"\"\n        if value is None:\n            return 'NULL'\n\n        def transform_value(val, srid):\n            return val.srid != srid\n\n        if hasattr(value, 'expression'):\n            if transform_value(value, f.srid):\n                placeholder = '%s(%%s, %s)' % (self.transform, f.srid)\n            else:\n                placeholder = '%s'\n            # No geometry value used for F expression, substitute in\n            # the column name instead.\n            return placeholder % self.get_expression_column(value)\n        else:\n            if transform_value(value, f.srid):\n                return '%s(SDO_GEOMETRY(%%s, %s), %s)' % (self.transform, value.srid, f.srid)\n            else:\n                return 'SDO_GEOMETRY(%%s, %s)' % f.srid\n\n    def spatial_lookup_sql(self, lvalue, lookup_type, value, field, qn):\n        \"Returns the SQL WHERE clause for use in Oracle spatial SQL construction.\"\n        geo_col, db_type = lvalue\n\n        # See if an Oracle Geometry function matches the lookup type next\n        lookup_info = self.geometry_functions.get(lookup_type, False)\n        if lookup_info:\n            # Lookup types that are tuples take tuple arguments, e.g., 'relate' and\n            # 'dwithin' lookup types.\n            if isinstance(lookup_info, tuple):\n                # First element of tuple is lookup type, second element is the type\n                # of the expected argument (e.g., str, float)\n                sdo_op, arg_type = lookup_info\n                geom = value[0]\n\n                # Ensuring that a tuple _value_ was passed in from the user\n                if not isinstance(value, tuple):\n                    raise ValueError('Tuple required for `%s` lookup type.' % lookup_type)\n                if len(value) != 2:\n                    raise ValueError('2-element tuple required for %s lookup type.' % lookup_type)\n\n                # Ensuring the argument type matches what we expect.\n                if not isinstance(value[1], arg_type):\n                    raise ValueError('Argument type should be %s, got %s instead.' % (arg_type, type(value[1])))\n\n                if lookup_type == 'relate':\n                    # The SDORelate class handles construction for these queries,\n                    # and verifies the mask argument.\n                    return sdo_op(value[1]).as_sql(geo_col, self.get_geom_placeholder(field, geom))\n                else:\n                    # Otherwise, just call the `as_sql` method on the SDOOperation instance.\n                    return sdo_op.as_sql(geo_col, self.get_geom_placeholder(field, geom))\n            else:\n                # Lookup info is a SDOOperation instance, whose `as_sql` method returns\n                # the SQL necessary for the geometry function call. For example:\n                #  SDO_CONTAINS(\"geoapp_country\".\"poly\", SDO_GEOMTRY('POINT(5 23)', 4326)) = 'TRUE'\n                return lookup_info.as_sql(geo_col, self.get_geom_placeholder(field, value))\n        elif lookup_type == 'isnull':\n            # Handling 'isnull' lookup type\n            return \"%s IS %sNULL\" % (geo_col, ('' if value else 'NOT ')), []\n\n        raise TypeError(\"Got invalid lookup_type: %s\" % repr(lookup_type))\n\n    def spatial_aggregate_sql(self, agg):\n        \"\"\"\n        Returns the spatial aggregate SQL template and function for the\n        given Aggregate instance.\n        \"\"\"\n        agg_name = agg.__class__.__name__.lower()\n        if agg_name == 'union':\n            agg_name += 'agg'\n        if agg.is_extent:\n            sql_template = '%(function)s(%(field)s)'\n        else:\n            sql_template = '%(function)s(SDOAGGRTYPE(%(field)s,%(tolerance)s))'\n        sql_function = getattr(self, agg_name)\n        return self.select % sql_template, sql_function\n\n    # Routines for getting the OGC-compliant models.\n    def geometry_columns(self):\n        from django.contrib.gis.db.backends.oracle.models import OracleGeometryColumns\n        return OracleGeometryColumns\n\n    def spatial_ref_sys(self):\n        from django.contrib.gis.db.backends.oracle.models import OracleSpatialRefSys\n        return OracleSpatialRefSys\n\n    def modify_insert_params(self, placeholders, params):\n        \"\"\"Drop out insert parameters for NULL placeholder. Needed for Oracle Spatial\n        backend due to #10888\n        \"\"\"\n        # This code doesn't work for bulk insert cases.\n        assert len(placeholders) == 1\n        return [[param for pholder, param\n                 in six.moves.zip(placeholders[0], params[0]) if pholder != 'NULL'], ]"
        }
      ]
    }
  ]
}