{
  "id": "django__django-14155",
  "question": "ResolverMatch.__repr__() doesn't handle functools.partial() nicely.\nDescription\n\t \n\t\t(last modified by Nick Pope)\n\t \nWhen a partial function is passed as the view, the __repr__ shows the func argument as functools.partial which isn't very helpful, especially as it doesn't reveal the underlying function or arguments provided.\nBecause a partial function also has arguments provided up front, we need to handle those specially so that they are accessible in __repr__.\nISTM that we can simply unwrap functools.partial objects in ResolverMatch.__init__().\n",
  "answer": "{\"task_id\": \"2d1e30dd-2897-4150-bbc2-0fb97cf91723\", \"workflow_run_id\": \"d4ecc5bf-e4f2-4136-8fd7-b72e90561a6e\", \"data\": {\"id\": \"d4ecc5bf-e4f2-4136-8fd7-b72e90561a6e\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/urls/resolvers.py b/django/urls/resolvers.py\\\\nindex abcdef..654321 100644\\\\n--- a/django/urls/resolvers.py\\\\n+++ b/django/urls/resolvers.py\\\\n@@ -52,6 +52,14 @@ class ResolverMatch:\\\\n     def __init__(self, func, args, kwargs, url_name=None, app_names=None, namespaces=None, route=None):\\\\n         from django.core.handlers.wsgi import WSGIHandler\\\\n\\\\n+        self.original_func = func\\\\n+        # Handle functools.partial\\\\n+        if isinstance(func, functools.partial):\\\\n+            self.partial_args = func.args\\\\n+            self.partial_kwargs = func.keywords\\\\n+            self.original_func = func\\\\n+            func = func.func\\\\n+\\\\n         self.func = func\\\\n         self.args = args\\\\n         self.kwargs = kwargs\\\\n@@ -92,8 +100,19 @@ class ResolverMatch:\\\\n \\\\n     def __repr__(self):\\\\n         func_repr = self.func\\\\n+        # Enhanced functools.partial representation\\\\n+        if isinstance(self.original_func, functools.partial):\\\\n+            args_str = \\\\\\\", \\\\\\\".join(repr(a) for a in self.partial_args)\\\\n+            kwargs_str = \\\\\\\", \\\\\\\".join(f\\\\\\\"{k}={v!r}\\\\\\\" for k, v in self.partial_kwargs.items())\\\\n+            if args_str and kwargs_str:\\\\n+                func_repr = f\\\\\\\"functools.partial({func_repr}, {args_str}, {kwargs_str})\\\\\\\"\\\\n+            elif args_str:\\\\n+                func_repr = f\\\\\\\"functools.partial({func_repr}, {args_str})\\\\\\\"\\\\n+            elif kwargs_str:\\\\n+                func_repr = f\\\\\\\"functools.partial({func_repr}, {kwargs_str})\\\\\\\"\\\\n         args_repr = self.args if self.args else \\\\\\\"[]\\\\\\\"\\\\n         kwargs_repr = self.kwargs if self.kwargs else \\\\\\\"{}\\\\\\\"\\\\n-        return f\\\\\\\"<ResolverMatch func={func_repr} args={args_repr} kwargs={kwargs_repr}>\\\\\\\"\\\\n+        return f\\\\\\\"<ResolverMatch func={func_repr} args={args_repr} kwargs={kwargs_repr}>\\\\\\\"\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.97,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.95\\n      },\\n      \\\"reasoning\\\": \\\"This patch provides a comprehensive and detailed representation of functools.partial objects, including both positional and keyword arguments. It maintains the original function reference and enhances the __repr__ method to display the partial function's details clearly. The patch is well-aligned with best practices and has a low risk of introducing new bugs.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/urls/resolvers.py b/django/urls/resolvers.py\\\\nindex abcdef..123456 100644\\\\n--- a/django/urls/resolvers.py\\\\n+++ b/django/urls/resolvers.py\\\\n@@ -52,6 +52,12 @@ class ResolverMatch:\\\\n     def __init__(self, func, args, kwargs, url_name=None, app_names=None, namespaces=None, route=None):\\\\n         from django.core.handlers.wsgi import WSGIHandler\\\\n\\\\n+        # Unwrap functools.partial objects\\\\n+        if isinstance(func, functools.partial):\\\\n+            self.partial_args = func.args\\\\n+            self.partial_kwargs = func.keywords\\\\n+            func = func.func\\\\n+\\\\n         self.func = func\\\\n         self.args = args\\\\n         self.kwargs = kwargs\\\\n@@ -92,8 +98,16 @@ class ResolverMatch:\\\\n \\\\n     def __repr__(self):\\\\n         func_repr = self.func\\\\n+        # Display functools.partial nicely\\\\n+        if hasattr(self, \\\\\\\"partial_args\\\\\\\") or hasattr(self, \\\\\\\"partial_kwargs\\\\\\\"):\\\\n+            func_repr = f\\\\\\\"functools.partial({func_repr}, {self.partial_args}, {self.partial_kwargs})\\\\\\\"\\\\n         args_repr = self.args if self.args else \\\\\\\"[]\\\\\\\"\\\\n         kwargs_repr = self.kwargs if self.kwargs else \\\\\\\"{}\\\\\\\"\\\\n-        return f\\\\\\\"<ResolverMatch func={func_repr} args={args_repr} kwargs={kwargs_repr}>\\\\\\\"\\\\n+        return f\\\\\\\"<ResolverMatch func={func_repr} args={args_repr} kwargs={kwargs_repr}>\\\\\\\"\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch unwraps functools.partial objects and modifies the __repr__ method to include partial arguments. However, it lacks the detailed formatting of arguments and keyword arguments provided by the second patch. While it improves the representation of partial functions, it is less comprehensive and slightly more prone to errors in complex scenarios.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is ranked higher due to its comprehensive handling of functools.partial objects, providing a detailed and clear representation in the __repr__ method. It adheres to best practices and minimizes the risk of introducing new bugs. Patch 1, while functional, is less detailed and slightly riskier in complex scenarios.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    from django.urls.resolvers import ResolverMatch\\\\n    from functools import partial\\\\n    # Create a partial function\\\\n    def sample_function(a, b):\\\\n        return a + b\\\\n    partial_func = partial(sample_function, b=2)\\\\n    # Create a ResolverMatch instance with the partial function\\\\n    resolver_match = ResolverMatch(func=partial_func, args=(3,), kwargs={})\\\\n    # Check if the original issue occurs in __repr__\\\\n    assert str(resolver_match) == '<ResolverMatch func=functools.partial(...)>', 'Original issue not reproduced'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    from django.urls.resolvers import ResolverMatch\\\\n    from functools import partial\\\\n    # Case where partial function has both args and kwargs\\\\n    def function_with_args_kwargs(a, b, c=3):\\\\n        return a + b + c\\\\n    partial_func_args_kwargs = partial(function_with_args_kwargs, b=2)\\\\n    resolver_match_args_kwargs = ResolverMatch(func=partial_func_args_kwargs, args=(3,), kwargs={})\\\\n    assert 'functools.partial' in str(resolver_match_args_kwargs), 'Edge case with args and kwargs failed'\\\\n    # Case where partial function has only args\\\\n    def function_with_args_only(a, b):\\\\n        return a + b\\\\n    partial_func_args_only = partial(function_with_args_only, b=2)\\\\n    resolver_match_args_only = ResolverMatch(func=partial_func_args_only, args=(3,), kwargs={})\\\\n    assert 'functools.partial' in str(resolver_match_args_only), 'Edge case with args only failed'\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch1():\\\\n    from django.urls.resolvers import ResolverMatch\\\\n    from functools import partial\\\\n    # Create a partial function\\\\n    def sample_function(a, b):\\\\n        return a + b\\\\n    partial_func = partial(sample_function, b=2)\\\\n    # Create a ResolverMatch instance with the partial function\\\\n    resolver_match = ResolverMatch(func=partial_func, args=(3,), kwargs={})\\\\n    # Check if the patch 1 correctly unwraps partial objects in __init__ and enhances __repr__\\\\n    assert str(resolver_match) == '<ResolverMatch func=sample_function args=(3,) kwargs={}>'\\\\n\\\",\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch2():\\\\n    from django.urls.resolvers import ResolverMatch\\\\n    from functools import partial\\\\n    # Create a partial function\\\\n    def sample_function(a, b, c=3):\\\\n        return a + b + c\\\\n    partial_func = partial(sample_function, b=2)\\\\n    # Create a ResolverMatch instance with the partial function\\\\n    resolver_match = ResolverMatch(func=partial_func, args=(3,), kwargs={})\\\\n    # Check if the patch 2 correctly represents partial arguments in __repr__\\\\n    assert str(resolver_match) == '<ResolverMatch func=functools.partial(sample_function, 3, b=2) args=(3,) kwargs={}>'\\\\n\\\",\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated comprehensive test cases for reproducing the original issue, testing edge cases, and validating the patches for ResolverMatch.__repr__() handling of functools.partial() objects.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 238.820562, \"total_tokens\": 13586, \"total_steps\": 9, \"created_at\": 1753304975, \"finished_at\": 1753305213}}"
}