{
  "Selected_candidate": {
    "pr_number": 6173,
    "pr_title": "Fixed #26249 -- Fixed collectstatic crash for files in STATIC_ROOT referenced by absolute URL.",
    "pr_body": "collectstatic crashed when:\n- a hashing static file storage backend was used\n- a static file referenced another static file located directly in\n  STATIC_ROOT (not a subdirectory) with an absolute URL (which must\n  start with STATIC_URL, which cannot be empty)\n\nIt seems to me that the current implementation reimplements relative\npath joining and doesn't handle edge cases correctly. I suspect it\nassumes that STATIC_URL is of the form r'/[^/]+/'.\n\nThrowing out the current code in favor of the posixpath module makes the\nlogic much easier to follow. Handling absolute paths correctly also\nbecomes easier.\n\nFinally I reworked comments to make the code hopefully easier to\nunderstand for future developers.\n",
    "issue_id": 26249,
    "issue_title": "ManifestStaticFilesStorage crashes on absolute URLs",
    "issue_body": "To reproduce, enable ManifestStaticFilesStorage:\nSTATIC_ROOT= '...'\nSTATIC_URL = '/static/'\nSTATICFILES_DIRS = ['...']\nSTATICFILES_STORAGE = 'django.contrib.staticfiles.storage.ManifestStaticFilesStorage'\nCreate\na.css\nin one of\nSTATICFILES_DIRS\nwith this content:\n@font-face{font-family:A;src:url(/static/a.woff) format(\"woff\");}\nCreate a\na.woff\nfile next to\na.css\n.\nThen collectstatic crashes with this stack trace:\nPost-processing 'a.css' failed!\n\nTraceback (most recent call last):\n  File \"/Users/myk/.virtualenvs/project/bin/django-admin\", line 11, in <module>\n    sys.exit(execute_from_command_line())\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/core/management/__init__.py\", line 353, in execute_from_command_line\n    utility.execute()\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/core/management/__init__.py\", line 345, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/core/management/base.py\", line 348, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/core/management/base.py\", line 399, in execute\n    output = self.handle(*args, **options)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py\", line 176, in handle\n    collected = self.collect()\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py\", line 128, in collect\n    raise processed\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/storage.py\", line 245, in post_process\n    content = pattern.sub(converter, content)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/storage.py\", line 184, in converter\n    hashed_url = self.url(unquote(joined_result), force=True)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/storage.py\", line 131, in url\n    hashed_name = self.stored_name(clean_name)\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/storage.py\", line 280, in stored_name\n    cache_name = self.clean_name(self.hashed_name(name))\n  File \"/Users/myk/.virtualenvs/project/lib/python3.5/site-packages/django/contrib/staticfiles/storage.py\", line 94, in hashed_name\n    (clean_name, self))\nValueError: The file 'a.css/a.woff' could not be found with <django.contrib.staticfiles.storage.ManifestStaticFilesStorage object at 0x105a34240>.\nUsing\nurl(./a.woff)\ninstead of\nurl(/static/a.woff)\navoids the issue.\nIt looks like\nHashedFilesMixin.url_converter\ndoesn't handle that case gracefully, so we end up with\na.css/a.woff\ninstead of just\na.woff\n. When a URL starts with\nSTATIC_URL\n, perhaps it should strip it?\n(I ended up with the\n/static/\nprefix because I set webpack's\noutput.publicPath\nto\n/static/\n. I don't remember why I need it, but it doesn't make the bug less valid.)\n(EDIT: I need to set\noutput.publicPath\nto\n/static/\nbecause webpack's code splitting doesn't work without this -- chunks get loaded from /current/url/chunk.js instead of /static/chuck.js, which obviously doesn't work. I don't want to include the hostname in the URL because I want to use the same build in staging and production.).",
    "issue_closed_at": "2016-02-23T12:38:56",
    "base_commit": "c62807968d7930bfd34afc2036c67921b943592f",
    "changes": [
      {
        "file": "django/contrib/staticfiles/storage.py",
        "type": "function",
        "name": "__init__",
        "class_name": "CachedFilesMixin",
        "code": "def __init__(self, *args, **kwargs):\n        super(CachedFilesMixin, self).__init__(*args, **kwargs)\n        try:\n            self.hashed_files = _MappingCache(caches['staticfiles'])\n        except InvalidCacheBackendError:\n            # Use the default backend\n            self.hashed_files = _MappingCache(default_cache)"
      },
      {
        "file": "django/contrib/staticfiles/storage.py",
        "type": "function",
        "name": "hashed_name",
        "class_name": "HashedFilesMixin",
        "code": "def hashed_name(self, name, content=None):\n        parsed_name = urlsplit(unquote(name))\n        clean_name = parsed_name.path.strip()\n        opened = False\n        if content is None:\n            if not self.exists(clean_name):\n                raise ValueError(\"The file '%s' could not be found with %r.\" %\n                                 (clean_name, self))\n            try:\n                content = self.open(clean_name)\n            except IOError:\n                # Handle directory paths and fragments\n                return name\n            opened = True\n        try:\n            file_hash = self.file_hash(clean_name, content)\n        finally:\n            if opened:\n                content.close()\n        path, filename = os.path.split(clean_name)\n        root, ext = os.path.splitext(filename)\n        if file_hash is not None:\n            file_hash = \".%s\" % file_hash\n        hashed_name = os.path.join(path, \"%s%s%s\" %\n                                   (root, file_hash, ext))\n        unparsed_name = list(parsed_name)\n        unparsed_name[2] = hashed_name\n        # Special casing for a @font-face hack, like url(myfont.eot?#iefix\")\n        # http://www.fontspring.com/blog/the-new-bulletproof-font-face-syntax\n        if '?#' in name and not unparsed_name[3]:\n            unparsed_name[2] += '?'\n        return urlunsplit(unparsed_name)"
      },
      {
        "file": "django/contrib/staticfiles/storage.py",
        "type": "function",
        "name": "url",
        "class_name": "HashedFilesMixin",
        "code": "def url(self, name, force=False):\n        \"\"\"\n        Returns the real URL in DEBUG mode.\n        \"\"\"\n        if settings.DEBUG and not force:\n            hashed_name, fragment = name, ''\n        else:\n            clean_name, fragment = urldefrag(name)\n            if urlsplit(clean_name).path.endswith('/'):  # don't hash paths\n                hashed_name = name\n            else:\n                hashed_name = self.stored_name(clean_name)\n\n        final_url = super(HashedFilesMixin, self).url(hashed_name)\n\n        # Special casing for a @font-face hack, like url(myfont.eot?#iefix\")\n        # http://www.fontspring.com/blog/the-new-bulletproof-font-face-syntax\n        query_fragment = '?#' in name  # [sic!]\n        if fragment or query_fragment:\n            urlparts = list(urlsplit(final_url))\n            if fragment and not urlparts[4]:\n                urlparts[4] = fragment\n            if query_fragment and not urlparts[3]:\n                urlparts[2] += '?'\n            final_url = urlunsplit(urlparts)\n\n        return unquote(final_url)"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful report as it directly involves the `collectstatic` functionality, a core aspect of handling static files in Django. Both the CURRENT bug and this candidate deal with processing static URLs and dynamic behavior concerning the STATIC_URL configuration. The structural similarity is evident in the stack traces regarding URL processing, and the candidate includes a specific fix related to the handling of absolute paths, which would be relevant to the CURRENT bug's need for handling SCRIPT_NAME correctly. Furthermore, since they share similar components within Django's staticfiles module, insights from Candidate A can guide adjustments to the CURRENT bug regarding dynamic SCRIPT_NAME usage.",
  "instance_id": "django__django-11564",
  "repo": "django/django",
  "created_at": "2019-07-12T21:06:28Z",
  "problem_statement": "Add support for SCRIPT_NAME in STATIC_URL and MEDIA_URL\nDescription\n\t \n\t\t(last modified by Rostyslav Bryzgunov)\n\t \nBy default, {% static '...' %} tag just appends STATIC_URL in the path. When running on sub-path, using SCRIPT_NAME WSGI param, it results in incorrect static URL - it doesn't prepend SCRIPT_NAME prefix.\nThis problem can be solved with prepending SCRIPT_NAME to STATIC_URL in settings.py but that doesn't work when SCRIPT_NAME is a dynamic value.\nThis can be easily added into default Django static tag and django.contrib.staticfiles tag as following:\ndef render(self, context):\n\turl = self.url(context)\n\t# Updating url here with request.META['SCRIPT_NAME'] \n\tif self.varname is None:\n\t\treturn url\n\tcontext[self.varname] = url\n\t\treturn ''\nOn more research I found that FileSystemStorage and StaticFilesStorage ignores SCRIPT_NAME as well. \nWe might have to do a lot of changes but I think it's worth the efforts.\n",
  "patch": "diff --git a/django/conf/__init__.py b/django/conf/__init__.py\n--- a/django/conf/__init__.py\n+++ b/django/conf/__init__.py\n@@ -15,7 +15,8 @@\n \n import django\n from django.conf import global_settings\n-from django.core.exceptions import ImproperlyConfigured\n+from django.core.exceptions import ImproperlyConfigured, ValidationError\n+from django.core.validators import URLValidator\n from django.utils.deprecation import RemovedInDjango40Warning\n from django.utils.functional import LazyObject, empty\n \n@@ -109,6 +110,26 @@ def configure(self, default_settings=global_settings, **options):\n             setattr(holder, name, value)\n         self._wrapped = holder\n \n+    @staticmethod\n+    def _add_script_prefix(value):\n+        \"\"\"\n+        Add SCRIPT_NAME prefix to relative paths.\n+\n+        Useful when the app is being served at a subpath and manually prefixing\n+        subpath to STATIC_URL and MEDIA_URL in settings is inconvenient.\n+        \"\"\"\n+        # Don't apply prefix to valid URLs.\n+        try:\n+            URLValidator()(value)\n+            return value\n+        except (ValidationError, AttributeError):\n+            pass\n+        # Don't apply prefix to absolute paths.\n+        if value.startswith('/'):\n+            return value\n+        from django.urls import get_script_prefix\n+        return '%s%s' % (get_script_prefix(), value)\n+\n     @property\n     def configured(self):\n         \"\"\"Return True if the settings have already been configured.\"\"\"\n@@ -128,6 +149,14 @@ def PASSWORD_RESET_TIMEOUT_DAYS(self):\n             )\n         return self.__getattr__('PASSWORD_RESET_TIMEOUT_DAYS')\n \n+    @property\n+    def STATIC_URL(self):\n+        return self._add_script_prefix(self.__getattr__('STATIC_URL'))\n+\n+    @property\n+    def MEDIA_URL(self):\n+        return self._add_script_prefix(self.__getattr__('MEDIA_URL'))\n+\n \n class Settings:\n     def __init__(self, settings_module):\n"
}