{
    "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."
}