{
  "id": "django__django-11564",
  "question": "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",
  "answer": "{\"task_id\": \"a4b52d0f-7741-488a-b98b-e5ef1116f063\", \"workflow_run_id\": \"92a46919-c2b9-4f1c-b9b8-68bad78c749a\", \"data\": {\"id\": \"92a46919-c2b9-4f1c-b9b8-68bad78c749a\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/templatetags/static.py b/django/templatetags/static.py\\\\nindex 47a6b74..97c3789 100644\\\\n--- a/django/templatetags/static.py\\\\n+++ b/django/templatetags/static.py\\\\n@@ -54,8 +54,13 @@ class StaticNode(Node):\\\\n     def render(self, context):\\\\n         url = self.url(context)\\\\n         # Update url with SCRIPT_NAME from request if available\\\\n+        request = context.get('request')\\\\n+        if request and 'SCRIPT_NAME' in request.META:\\\\n+            script_name = request.META['SCRIPT_NAME']\\\\n+            url = script_name.rstrip('/') + '/' + url.lstrip('/')\\\\n         if self.varname is None:\\\\n             return url\\\\n         context[self.varname] = url\\\\n         return ''\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly addresses the issue by modifying the StaticNode class to prepend SCRIPT_NAME from the request context. It is a straightforward and low-risk change that aligns well with Django's existing architecture.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/contrib/staticfiles/storage.py b/django/contrib/staticfiles/storage.py\\\\nindex d5f2a9c..f6ab36a 100644\\\\n--- a/django/contrib/staticfiles/storage.py\\\\n+++ b/django/contrib/staticfiles/storage.py\\\\n@@ -123,7 +123,12 @@ class StaticFilesStorage(FileSystemStorage):\\\\n     def url(self, name, **kwargs):\\\\n         provided_url = super().url(name, **kwargs)\\\\n         # Prepend SCRIPT_NAME to the URL if it's provided in kwargs\\\\n-        return provided_url\\\\n+        script_name = kwargs.get('SCRIPT_NAME')\\\\n+        if script_name:\\\\n+            return script_name.rstrip('/') + '/' + provided_url.lstrip('/')\\\\n+        else:\\\\n+            return provided_url\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the StaticFilesStorage class to accept SCRIPT_NAME as a keyword argument, which is a flexible approach. However, it requires the caller to provide SCRIPT_NAME, which might not always be feasible.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with considerations\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/core/files/storage.py b/django/core/files/storage.py\\\\nindex ab6c47f..76e93fe 100644\\\\n--- a/django/core/files/storage.py\\\\n+++ b/django/core/files/storage.py\\\\n@@ -84,7 +84,12 @@ class FileSystemStorage(Storage):\\\\n     def url(self, name, **kwargs):\\\\n         url = self.base_url + filepath_to_uri(name)\\\\n         # Incorporate SCRIPT_NAME from kwargs into the URL\\\\n-        return url\\\\n+        script_name = kwargs.get('SCRIPT_NAME')\\\\n+        if script_name:\\\\n+            return script_name.rstrip('/') + '/' + url.lstrip('/')\\\\n+        else:\\\\n+            return url\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"Similar to Patch 2, this patch modifies the FileSystemStorage class to handle SCRIPT_NAME. It shares the same limitation of requiring SCRIPT_NAME to be passed explicitly.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with considerations\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/django/contrib/staticfiles/storage.py b/django/contrib/staticfiles/storage.py\\\\nindex d5f2a9c..87d4191 100644\\\\n--- a/django/contrib/staticfiles/storage.py\\\\n+++ b/django/contrib/staticfiles/storage.py\\\\n@@ -123,7 +123,15 @@ class StaticFilesStorage(FileSystemStorage):\\\\n     def url(self, name, **kwargs):\\\\n         provided_url = super().url(name, **kwargs)\\\\n+        # Try to get SCRIPT_NAME from thread local storage if not in kwargs\\\\n+        from threading import local\\\\n+        _thread_locals = local()\\\\n+\\\\n+        def get_script_name():\\\\n+            return getattr(_thread_locals, 'SCRIPT_NAME', '')\\\\n+        script_name = kwargs.get('SCRIPT_NAME') or get_script_name()\\\\n+        \\\\n         # Prepend SCRIPT_NAME to the URL if it's available\\\\n-        return provided_url\\\\n+        if script_name:\\\\n+            return script_name.rstrip('/') + '/' + provided_url.lstrip('/')\\\\n+        return provided_url\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.7,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.3,\\n        \\\"best_practices\\\": 0.7,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch attempts to retrieve SCRIPT_NAME from thread-local storage, which introduces complexity and potential threading issues. It is less aligned with Django's typical design patterns and increases the risk of bugs.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective and low-risk solution, directly addressing the issue within the StaticNode class. Patches 2 and 3 offer a flexible approach by allowing SCRIPT_NAME to be passed as a parameter, but they require additional handling by the caller. Patch 4 introduces unnecessary complexity and potential threading issues, making it the least favorable option.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Simulate a request with SCRIPT_NAME in context and test if STATIC_URL is correctly prepended with SCRIPT_NAME\\\\n    context = {'request': {'META': {'SCRIPT_NAME': '/subpath'}}}\\\\n    static_node = StaticNode('path/to/static/file.png', None)\\\\n    output = static_node.render(context)\\\\n    assert output == '/subpath/static/path/to/static/file.png', 'Static URL not correctly prepended with SCRIPT_NAME'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Test if SCRIPT_NAME is correctly handled when it's empty or not provided\\\\n    context_empty = {'request': {'META': {'SCRIPT_NAME': ''}}}\\\\n    static_node_empty = StaticNode('path/to/static/file.png', None)\\\\n    output_empty = static_node_empty.render(context_empty)\\\\n    assert output_empty == '/static/path/to/static/file.png', 'Static URL not correctly generated with empty SCRIPT_NAME'\\\\n    context_none = {}\\\\n    static_node_none = StaticNode('path/to/static/file.png', None)\\\\n    output_none = static_node_none.render(context_none)\\\\n    assert output_none == '/static/path/to/static/file.png', 'Static URL not correctly generated without SCRIPT_NAME'\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_1():\\\\n    # Validate Patch 1 by checking if STATIC_URL is correctly prepended with SCRIPT_NAME in the render method of StaticNode\\\\n    context = {'request': {'META': {'SCRIPT_NAME': '/subpath'}}}\\\\n    static_node = StaticNode('path/to/static/file.png', None)\\\\n    output = static_node.render(context)\\\\n    assert output == '/subpath/static/path/to/static/file.png', 'Patch 1 failed to prepend SCRIPT_NAME to STATIC_URL'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_2():\\\\n    # Validate Patch 2 by verifying that SCRIPT_NAME is correctly appended to the URL in StaticFilesStorage\\\\n    static_files_storage = StaticFilesStorage()\\\\n    url_with_script_name = static_files_storage.url('path/to/static/file.png', SCRIPT_NAME='/subpath')\\\\n    assert url_with_script_name == '/subpath/static/path/to/static/file.png', 'Patch 2 failed to append SCRIPT_NAME to URL'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_3():\\\\n    # Validate Patch 3 by ensuring that SCRIPT_NAME is properly integrated into the URL in FileSystemStorage\\\\n    file_system_storage = FileSystemStorage()\\\\n    url_with_script_name = file_system_storage.url('path/to/static/file.png', SCRIPT_NAME='/subpath')\\\\n    assert url_with_script_name == '/subpath/static/path/to/static/file.png', 'Patch 3 failed to include SCRIPT_NAME in the URL'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_4\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_4():\\\\n    # Validate Patch 4 by checking if SCRIPT_NAME is correctly retrieved from thread local storage when not provided in StaticFilesStorage\\\\n    from threading import local\\\\n    _thread_locals = local()\\\\n    setattr(_thread_locals, 'SCRIPT_NAME', '/subpath')\\\\n    static_files_storage = StaticFilesStorage()\\\\n    url_with_script_name = static_files_storage.url('path/to/static/file.png')\\\\n    assert url_with_script_name == '/subpath/static/path/to/static/file.png', 'Patch 4 failed to get SCRIPT_NAME from thread local storage'\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original issue, test edge cases, and validate the effectiveness of the proposed patches for incorporating SCRIPT_NAME into STATIC_URL and MEDIA_URL in Django.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 530.018826, \"total_tokens\": 28401, \"total_steps\": 9, \"created_at\": 1753284495, \"finished_at\": 1753285025}}"
}