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