{
  "Selected_candidate": {
    "pr_number": 813,
    "pr_title": "Collect functools.partial objects",
    "pr_body": "Fix for issue #811.\n\nI feel the current implementation is hackish and might break in some corner-cases.\n\nPerhaps it would be better to don't support tests created by `functools.partial` objects at all, and issue a warning during collection and add a note to the documentation?\n",
    "issue_id": 811,
    "issue_title": "Test functions generated by functools.partial raise an internal error during collection",
    "issue_body": "As discussed in #740, the following test functions fail during collection:\n\n``` python\nimport functools\ndef do_it(x):\n    assert x == 2\ntest_foo = functools.partial(do_it, x=1)  \n```\n\n```\n=================================== ERRORS ====================================\n________________________ ERROR collecting test_foo.py _________________________\n...\\lib\\inspect.py:752: in getargs\n    raise TypeError('{!r} is not a code object'.format(co))\nE   TypeError: <functools.partial object at 0x0000000003D37278> is not a code object\n=============================== warning summary ===============================\nWC2 X:\\test_foo.py cannot collect 'test_foo' because it is not a function.\n===================== 1 warnings, 1 error in 0.02 seconds =====================\n```\n",
    "issue_closed_at": "2015-07-17T00:05:43Z",
    "base_commit": "330de0a93db404921c7a4c80305b43fd87bc3119",
    "changes": [
      {
        "file": "_pytest/python.py",
        "type": "line",
        "name": "line 1",
        "code": "\"\"\" Python test discovery, setup and run of test functions. \"\"\"\nimport fnmatch\nimport py\nimport inspect\nimport sys"
      },
      {
        "file": "_pytest/python.py",
        "type": "line",
        "name": "line 18",
        "code": "# used to work around a python2 exception info leak\nexc_clear = getattr(sys, 'exc_clear', lambda: None)\n\ndef getfslineno(obj):\n    # xxx let decorators etc specify a sane ordering\n    while hasattr(obj, \"__wrapped__\"):\n        obj = obj.__wrapped__\n    if hasattr(obj, 'place_as'):\n        obj = obj.place_as\n    fslineno = py.code.getfslineno(obj)"
      },
      {
        "file": "_pytest/python.py",
        "type": "function",
        "name": "setup",
        "class_name": "Function",
        "code": "def setup(self):\n        # check if parametrization happend with an empty list\n        try:\n            self.callspec._emptyparamspecified\n        except AttributeError:\n            pass\n        else:\n            fs, lineno = self._getfslineno()\n            pytest.skip(\"got empty parameter set, function %s at %s:%d\" %(\n                self.function.__name__, fs, lineno))\n        super(Function, self).setup()\n        fillfixtures(self)"
      },
      {
        "file": "_pytest/python.py",
        "type": "function",
        "name": "formatrepr",
        "class_name": "FixtureLookupError",
        "code": "def formatrepr(self):\n        tblines = []\n        addline = tblines.append\n        stack = [self.request._pyfuncitem.obj]\n        stack.extend(map(lambda x: x.func, self.fixturestack))\n        msg = self.msg\n        if msg is not None:\n            stack = stack[:-1] # the last fixture raise an error, let's present\n                               # it at the requesting side\n        for function in stack:\n            fspath, lineno = getfslineno(function)\n            try:\n                lines, _ = inspect.getsourcelines(function)\n            except IOError:\n                error_msg = \"file %s, line %s: source code not available\"\n                addline(error_msg % (fspath, lineno+1))\n            else:\n                addline(\"file %s, line %s\" % (fspath, lineno+1))\n                for i, line in enumerate(lines):\n                    line = line.rstrip()\n                    addline(\"  \" + line)\n                    if line.lstrip().startswith('def'):\n                        break\n\n        if msg is None:\n            fm = self.request._fixturemanager\n            available = []\n            for name, fixturedef in fm._arg2fixturedefs.items():\n                parentid = self.request._pyfuncitem.parent.nodeid\n                faclist = list(fm._matchfactories(fixturedef, parentid))\n                if faclist:\n                    available.append(name)\n            msg = \"fixture %r not found\" % (self.argname,)\n            msg += \"\\n available fixtures: %s\" %(\", \".join(available),)\n            msg += \"\\n use 'py.test --fixtures [testpath]' for help on them.\"\n\n        return FixtureLookupErrorRepr(fspath, lineno, tblines, msg, self.argname)"
      },
      {
        "file": "_pytest/python.py",
        "type": "function",
        "name": "getfuncargnames",
        "class_name": null,
        "code": "def getfuncargnames(function, startindex=None):\n    # XXX merge with main.py's varnames\n    #assert not inspect.isclass(function)\n    realfunction = function\n    while hasattr(realfunction, \"__wrapped__\"):\n        realfunction = realfunction.__wrapped__\n    if startindex is None:\n        startindex = inspect.ismethod(function) and 1 or 0\n    if realfunction != function:\n        startindex += num_mock_patch_args(function)\n        function = realfunction\n    argnames = inspect.getargs(py.code.getrawcode(function))[0]\n    defaults = getattr(function, 'func_defaults',\n                       getattr(function, '__defaults__', None)) or ()\n    numdefaults = len(defaults)\n    if numdefaults:\n        return tuple(argnames[startindex:-numdefaults])\n    return tuple(argnames[startindex:])"
      }
    ]
  },
  "Justification": "Candidate E is the most relevant choice as it addresses a test function issue, similar to the CURRENT bug's focus on improving assertion error messages within test functions. Both reports involve scenarios where Python's testing framework does not behave as expected during assertions. While the CURRENT bug deals with improving reported failures from assertions using iterables, Candidate E focuses on error handling related to test function collection, which may share analogous challenges in reporting and capturing issues during testing. Therefore, insights from Candidate E regarding the handling of test functions could help improve the assertion messages in the CURRENT bug report.",
  "instance_id": "pytest-dev__pytest-5103",
  "repo": "pytest-dev/pytest",
  "created_at": "2019-04-13T16:17:45Z",
  "problem_statement": "Unroll the iterable for all/any calls to get better reports\nSometime I need to assert some predicate on all of an iterable, and for that the builtin functions `all`/`any` are great - but the failure messages aren't useful at all!\r\nFor example - the same test written in three ways:\r\n\r\n- A generator expression\r\n```sh                                                                                                                                                                                                                         \r\n    def test_all_even():\r\n        even_stevens = list(range(1,100,2))\r\n>       assert all(is_even(number) for number in even_stevens)\r\nE       assert False\r\nE        +  where False = all(<generator object test_all_even.<locals>.<genexpr> at 0x101f82ed0>)\r\n```\r\n- A list comprehension\r\n```sh\r\n    def test_all_even():\r\n        even_stevens = list(range(1,100,2))\r\n>       assert all([is_even(number) for number in even_stevens])\r\nE       assert False\r\nE        +  where False = all([False, False, False, False, False, False, ...])\r\n```\r\n- A for loop\r\n```sh\r\n    def test_all_even():\r\n        even_stevens = list(range(1,100,2))\r\n        for number in even_stevens:\r\n>           assert is_even(number)\r\nE           assert False\r\nE            +  where False = is_even(1)\r\n\r\ntest_all_any.py:7: AssertionError\r\n```\r\nThe only one that gives a meaningful report is the for loop - but it's way more wordy, and `all` asserts don't translate to a for loop nicely (I'll have to write a `break` or a helper function - yuck)\r\nI propose the assertion re-writer \"unrolls\" the iterator to the third form, and then uses the already existing reports.\r\n\r\n- [x] Include a detailed description of the bug or suggestion\r\n- [x] `pip list` of the virtual environment you are using\r\n```\r\nPackage        Version\r\n-------------- -------\r\natomicwrites   1.3.0  \r\nattrs          19.1.0 \r\nmore-itertools 7.0.0  \r\npip            19.0.3 \r\npluggy         0.9.0  \r\npy             1.8.0  \r\npytest         4.4.0  \r\nsetuptools     40.8.0 \r\nsix            1.12.0 \r\n```\r\n- [x] pytest and operating system versions\r\n`platform darwin -- Python 3.7.3, pytest-4.4.0, py-1.8.0, pluggy-0.9.0`\r\n- [x] Minimal example if possible\r\n\n",
  "patch": "diff --git a/src/_pytest/assertion/rewrite.py b/src/_pytest/assertion/rewrite.py\n--- a/src/_pytest/assertion/rewrite.py\n+++ b/src/_pytest/assertion/rewrite.py\n@@ -964,6 +964,8 @@ def visit_Call_35(self, call):\n         \"\"\"\n         visit `ast.Call` nodes on Python3.5 and after\n         \"\"\"\n+        if isinstance(call.func, ast.Name) and call.func.id == \"all\":\n+            return self._visit_all(call)\n         new_func, func_expl = self.visit(call.func)\n         arg_expls = []\n         new_args = []\n@@ -987,6 +989,27 @@ def visit_Call_35(self, call):\n         outer_expl = \"%s\\n{%s = %s\\n}\" % (res_expl, res_expl, expl)\n         return res, outer_expl\n \n+    def _visit_all(self, call):\n+        \"\"\"Special rewrite for the builtin all function, see #5062\"\"\"\n+        if not isinstance(call.args[0], (ast.GeneratorExp, ast.ListComp)):\n+            return\n+        gen_exp = call.args[0]\n+        assertion_module = ast.Module(\n+            body=[ast.Assert(test=gen_exp.elt, lineno=1, msg=\"\", col_offset=1)]\n+        )\n+        AssertionRewriter(module_path=None, config=None).run(assertion_module)\n+        for_loop = ast.For(\n+            iter=gen_exp.generators[0].iter,\n+            target=gen_exp.generators[0].target,\n+            body=assertion_module.body,\n+            orelse=[],\n+        )\n+        self.statements.append(for_loop)\n+        return (\n+            ast.Num(n=1),\n+            \"\",\n+        )  # Return an empty expression, all the asserts are in the for_loop\n+\n     def visit_Starred(self, starred):\n         # From Python 3.5, a Starred node can appear in a function call\n         res, expl = self.visit(starred.value)\n@@ -997,6 +1020,8 @@ def visit_Call_legacy(self, call):\n         \"\"\"\n         visit `ast.Call nodes on 3.4 and below`\n         \"\"\"\n+        if isinstance(call.func, ast.Name) and call.func.id == \"all\":\n+            return self._visit_all(call)\n         new_func, func_expl = self.visit(call.func)\n         arg_expls = []\n         new_args = []\n"
}