{
  "instance_id": "pytest-dev__pytest-6116",
  "repo": "pytest-dev/pytest",
  "created_at": "2019-11-01T20:05:53Z",
  "problem_statement": "pytest --collect-only needs a one char shortcut command\nI find myself needing to run `--collect-only` very often and that cli argument is a very long to type one. \r\n\r\nI do think that it would be great to allocate a character for it, not sure which one yet. Please use up/down thumbs to vote if you would find it useful or not and eventually proposing which char should be used. \r\n\r\nClearly this is a change very easy to implement but first I want to see if others would find it useful or not.\npytest --collect-only needs a one char shortcut command\nI find myself needing to run `--collect-only` very often and that cli argument is a very long to type one. \r\n\r\nI do think that it would be great to allocate a character for it, not sure which one yet. Please use up/down thumbs to vote if you would find it useful or not and eventually proposing which char should be used. \r\n\r\nClearly this is a change very easy to implement but first I want to see if others would find it useful or not.\n",
  "patch": "diff --git a/src/_pytest/main.py b/src/_pytest/main.py\n--- a/src/_pytest/main.py\n+++ b/src/_pytest/main.py\n@@ -109,6 +109,7 @@ def pytest_addoption(parser):\n     group.addoption(\n         \"--collectonly\",\n         \"--collect-only\",\n+        \"--co\",\n         action=\"store_true\",\n         help=\"only collect tests, don't execute them.\",\n     ),\n",
  "similar_bug_items": [
    {
      "pr_number": 3110,
      "pr_title": "Fix progress report when tests fail during teardown",
      "pr_body": "Fix #3088\r\n",
      "issue_id": 3088,
      "issue_title": "Invalid percentages of Test Suite progress",
      "issue_body": "Progress of Test Suite shows more than 100% if Teardown fails:\r\n```\r\n$ python3 -m pytest -v ./test.py &> res.txt; cat res.txt                                                                                                                                        \r\n============================= test session starts ==============================                                                                                                                                   \r\nplatform linux -- Python 3.6.4, pytest-3.3.1, py-1.5.0, pluggy-0.6.0 -- /usr/bin/python3\r\ncachedir: .cache\r\nrootdir: /home/pavel, inifile:\r\nplugins: allure-adaptor-1.7.8\r\ncollecting ... collected 1 item\r\n\r\ntest.py::test_foo1 PASSED                                                [100%]\r\ntest.py::test_foo1 ERROR                                                 [200%]\r\n\r\n==================================== ERRORS ====================================\r\n```\r\ntest.py:\r\n```\r\nimport pytest                                                                                                                                                                                                      \r\n\r\n@pytest.fixture\r\ndef fail_teardown():\r\n    print(\"SETUP\")\r\n    yield fail_teardown\r\n    print(\"TEARDOWN\")\r\n    assert 1 == 2\r\n\r\ndef test_foo1(fail_teardown):\r\n    print(\"TEST\")\r\n```\r\nSystem:\r\n```\r\n$ pip list 2>/dev/null                                                                                                                                                                          \r\nappdirs (1.4.3)                                                                                                                                                                                                    \r\nasn1crypto (0.24.0)                                                                                                                                                                                                \r\nattrs (17.4.0)\r\nbcrypt (3.1.4)\r\ncffi (1.11.2)\r\ncryptography (2.1.4)\r\nidna (2.6)\r\nisc (2.0)\r\njson2html (1.1.1)\r\nlxml (4.1.1)                                                                                                                                                                                                       \r\nmsgpack-python (0.4.8)                                                                                                                                                                                             \r\nnamedlist (1.7)                                                                                                                                                                                                    \r\nnetifaces (0.10.6)                                                                                                                                                                                                 \r\npackaging (16.8)                                                                                                                                                                                                   \r\nparamiko (2.4.0)                                                                                                                                                                                                   \r\npip (9.0.1)                                                                                                                                                                                                        \r\npluggy (0.6.0)                                                                                                                                                                                                     \r\nply (3.10)                                                                                                                                                                                                         \r\npy (1.5.0)                                                                                                                                                                                                         \r\npyasn1 (0.4.2)                                                                                                                                                                                                     \r\npycparser (2.18)                                                                                                                                                                                                   \r\nPyNaCl (1.2.0)                                                                                                                                                                                                     \r\npyparsing (2.2.0)                                                                                                                                                                                                  \r\npyserial (3.4)                                                                                                                                                                                                     \r\npytest (3.3.1)                                                                                                                                                                                                     \r\npytest-allure-adaptor (1.7.8)                                                                                                                                                                                      \r\nscp (0.10.2)                                                                                                                                                                                                       \r\nsetuptools (38.2.5)                                                                                                                                                                                                \r\nsix (1.11.0)                                                                                                                                                                                                       \r\nspeedtest-cli (1.0.7)                                                                                                                                                                                              \r\nws4py (0.3.4)                                                                                                                                                                                            \r\n$ uname -a                                                                                                                                                                                      \r\nLinux afitester 4.14.10-1-ARCH #1 SMP PREEMPT Fri Dec 29 20:17:35 UTC 2017 x86_64 GNU/Linux\r\n```",
      "issue_closed_at": "2018-01-12T10:27:27Z",
      "base_commit": "b0032ba2b3258ada67bb1fa705c18af1741778fd",
      "changes": [
        {
          "file": "_pytest/terminal.py",
          "type": "function",
          "name": "__init__",
          "class_name": "TerminalReporter",
          "code": "def __init__(self, config, file=None):\n        import _pytest.config\n        self.config = config\n        self.verbosity = self.config.option.verbose\n        self.showheader = self.verbosity >= 0\n        self.showfspath = self.verbosity >= 0\n        self.showlongtestinfo = self.verbosity > 0\n        self._numcollected = 0\n        self._session = None\n\n        self.stats = {}\n        self.startdir = py.path.local()\n        if file is None:\n            file = sys.stdout\n        self._tw = _pytest.config.create_terminal_writer(config, file)\n        # self.writer will be deprecated in pytest-3.4\n        self.writer = self._tw\n        self._screen_width = self._tw.fullwidth\n        self.currentfspath = None\n        self.reportchars = getreportopt(config)\n        self.hasmarkup = self._tw.hasmarkup\n        self.isatty = file.isatty()\n        self._progress_items_reported = 0\n        self._show_progress_info = (self.config.getoption('capture') != 'no' and\n                                    self.config.getini('console_output_style') == 'progress')"
        },
        {
          "file": "_pytest/terminal.py",
          "type": "function",
          "name": "write_ensure_prefix",
          "class_name": "TerminalReporter",
          "code": "def write_ensure_prefix(self, prefix, extra=\"\", **kwargs):\n        if self.currentfspath != prefix:\n            self._tw.line()\n            self.currentfspath = prefix\n            self._tw.write(prefix)\n        if extra:\n            self._tw.write(extra, **kwargs)\n            self.currentfspath = -2\n            self._write_progress_information_filling_space()"
        },
        {
          "file": "_pytest/terminal.py",
          "type": "function",
          "name": "pytest_runtest_logreport",
          "class_name": "TerminalReporter",
          "code": "def pytest_runtest_logreport(self, report):\n        rep = report\n        res = self.config.hook.pytest_report_teststatus(report=rep)\n        cat, letter, word = res\n        if isinstance(word, tuple):\n            word, markup = word\n        else:\n            markup = None\n        self.stats.setdefault(cat, []).append(rep)\n        self._tests_ran = True\n        if not letter and not word:\n            # probably passed setup/teardown\n            return\n        running_xdist = hasattr(rep, 'node')\n        self._progress_items_reported += 1\n        if self.verbosity <= 0:\n            if not running_xdist and self.showfspath:\n                self.write_fspath_result(rep.nodeid, letter)\n            else:\n                self._tw.write(letter)\n            self._write_progress_if_past_edge()\n        else:\n            if markup is None:\n                if rep.passed:\n                    markup = {'green': True}\n                elif rep.failed:\n                    markup = {'red': True}\n                elif rep.skipped:\n                    markup = {'yellow': True}\n                else:\n                    markup = {}\n            line = self._locationline(rep.nodeid, *rep.location)\n            if not running_xdist:\n                self.write_ensure_prefix(line, word, **markup)\n            else:\n                self.ensure_newline()\n                self._tw.write(\"[%s]\" % rep.node.gateway.id)\n                if self._show_progress_info:\n                    self._tw.write(self._get_progress_information_message() + \" \", cyan=True)\n                else:\n                    self._tw.write(' ')\n                self._tw.write(word, **markup)\n                self._tw.write(\" \" + line)\n                self.currentfspath = -2"
        },
        {
          "file": "_pytest/terminal.py",
          "type": "function",
          "name": "pytest_runtest_logreport",
          "class_name": "TerminalReporter",
          "code": "def pytest_runtest_logreport(self, report):\n        rep = report\n        res = self.config.hook.pytest_report_teststatus(report=rep)\n        cat, letter, word = res\n        if isinstance(word, tuple):\n            word, markup = word\n        else:\n            markup = None\n        self.stats.setdefault(cat, []).append(rep)\n        self._tests_ran = True\n        if not letter and not word:\n            # probably passed setup/teardown\n            return\n        running_xdist = hasattr(rep, 'node')\n        self._progress_items_reported += 1\n        if self.verbosity <= 0:\n            if not running_xdist and self.showfspath:\n                self.write_fspath_result(rep.nodeid, letter)\n            else:\n                self._tw.write(letter)\n            self._write_progress_if_past_edge()\n        else:\n            if markup is None:\n                if rep.passed:\n                    markup = {'green': True}\n                elif rep.failed:\n                    markup = {'red': True}\n                elif rep.skipped:\n                    markup = {'yellow': True}\n                else:\n                    markup = {}\n            line = self._locationline(rep.nodeid, *rep.location)\n            if not running_xdist:\n                self.write_ensure_prefix(line, word, **markup)\n            else:\n                self.ensure_newline()\n                self._tw.write(\"[%s]\" % rep.node.gateway.id)\n                if self._show_progress_info:\n                    self._tw.write(self._get_progress_information_message() + \" \", cyan=True)\n                else:\n                    self._tw.write(' ')\n                self._tw.write(word, **markup)\n                self._tw.write(\" \" + line)\n                self.currentfspath = -2"
        },
        {
          "file": "_pytest/terminal.py",
          "type": "function",
          "name": "pytest_runtest_logreport",
          "class_name": "TerminalReporter",
          "code": "def pytest_runtest_logreport(self, report):\n        rep = report\n        res = self.config.hook.pytest_report_teststatus(report=rep)\n        cat, letter, word = res\n        if isinstance(word, tuple):\n            word, markup = word\n        else:\n            markup = None\n        self.stats.setdefault(cat, []).append(rep)\n        self._tests_ran = True\n        if not letter and not word:\n            # probably passed setup/teardown\n            return\n        running_xdist = hasattr(rep, 'node')\n        self._progress_items_reported += 1\n        if self.verbosity <= 0:\n            if not running_xdist and self.showfspath:\n                self.write_fspath_result(rep.nodeid, letter)\n            else:\n                self._tw.write(letter)\n            self._write_progress_if_past_edge()\n        else:\n            if markup is None:\n                if rep.passed:\n                    markup = {'green': True}\n                elif rep.failed:\n                    markup = {'red': True}\n                elif rep.skipped:\n                    markup = {'yellow': True}\n                else:\n                    markup = {}\n            line = self._locationline(rep.nodeid, *rep.location)\n            if not running_xdist:\n                self.write_ensure_prefix(line, word, **markup)\n            else:\n                self.ensure_newline()\n                self._tw.write(\"[%s]\" % rep.node.gateway.id)\n                if self._show_progress_info:\n                    self._tw.write(self._get_progress_information_message() + \" \", cyan=True)\n                else:\n                    self._tw.write(' ')\n                self._tw.write(word, **markup)\n                self._tw.write(\" \" + line)\n                self.currentfspath = -2"
        }
      ]
    },
    {
      "pr_number": 5559,
      "pr_title": "Merge master into features",
      "pr_body": "",
      "issue_id": 5547,
      "issue_title": "pytest stepwise doesn't work with xfail strict failures",
      "issue_body": "```\r\ngraingert@onomastic:~/projects/foo$ cat tests/test_foo.py \r\nimport pytest\r\n\r\n\r\n@pytest.mark.xfail(reason=\"pass\")\r\ndef test_a():\r\n    pass\r\n\r\n\r\n@pytest.mark.xfail(reason=\"pass\")\r\ndef test_b():\r\n    pass\r\ngraingert@onomastic:~/projects/foo$ cat tests/pytest.ini \r\n[pytest]\r\naddopts = --strict\r\nxfail_strict=true\r\ngraingert@onomastic:~/projects/foo$ pytest --sw tests/\r\n================================ test session starts ================================\r\nplatform linux -- Python 3.7.3, pytest-5.0.0, py-1.8.0, pluggy-0.12.0\r\nrootdir: /home/graingert/projects/foo/tests, inifile: pytest.ini\r\ncollected 2 items                                                                   \r\nstepwise: no previously failed tests, not skipping.\r\n\r\ntests/test_foo.py FF                                                          [100%]\r\n\r\n===================================== FAILURES ======================================\r\n______________________________________ test_a _______________________________________\r\n[XPASS(strict)] pass\r\n______________________________________ test_b _______________________________________\r\n[XPASS(strict)] pass\r\n============================= 2 failed in 0.01 seconds ==============================\r\n```",
      "issue_closed_at": "2019-07-04T23:50:34Z",
      "base_commit": "60a358fa2dc82a571c68d1be2d25703b51351538",
      "changes": [
        {
          "file": "doc/en/example/nonpython/conftest.py",
          "type": "line",
          "name": "line 3",
          "code": "\n\ndef pytest_collect_file(parent, path):\n    if path.ext == \".yml\" and path.basename.startswith(\"test\"):\n        return YamlFile(path, parent)\n\n"
        },
        {
          "file": "src/_pytest/_code/code.py",
          "type": "function",
          "name": "match",
          "class_name": "ExceptionInfo",
          "code": "def match(self, regexp):\n        \"\"\"\n        Check whether the regular expression 'regexp' is found in the string\n        representation of the exception using ``re.search``. If it matches\n        then True is returned (so that it is possible to write\n        ``assert excinfo.match()``). If it doesn't match an AssertionError is\n        raised.\n        \"\"\"\n        __tracebackhide__ = True\n        if not re.search(regexp, str(self.value)):\n            assert 0, \"Pattern '{!s}' not found in '{!s}'\".format(regexp, self.value)\n        return True"
        },
        {
          "file": "src/_pytest/stepwise.py",
          "type": "function",
          "name": "pytest_collection_modifyitems",
          "class_name": "StepwisePlugin",
          "code": "def pytest_collection_modifyitems(self, session, config, items):\n        if not self.active:\n            return\n        if not self.lastfailed:\n            self.report_status = \"no previously failed tests, not skipping.\"\n            return\n\n        already_passed = []\n        found = False\n\n        # Make a list of all tests that have been run before the last failing one.\n        for item in items:\n            if item.nodeid == self.lastfailed:\n                found = True\n                break\n            else:\n                already_passed.append(item)\n\n        # If the previously failed test was not found among the test items,\n        # do not skip any tests.\n        if not found:\n            self.report_status = \"previously failed test not found, not skipping.\"\n            already_passed = []\n        else:\n            self.report_status = \"skipping {} already passed items.\".format(\n                len(already_passed)\n            )\n\n        for item in already_passed:\n            items.remove(item)\n\n        config.hook.pytest_deselected(items=already_passed)"
        }
      ]
    },
    {
      "pr_number": 1787,
      "pr_title": "Rewrite asserts in test-modules loaded very early in the startup",
      "pr_body": "I investigated this some more and found the problem:\n\nThe assertion-rewrite hook is installed very early on, during the construction of the `Config` object. After installing the hook and still during `Config`'s construction, plugins and `conftest` files are also imported and are rewritten if `AssertionRewriteHook._should_rewrite` returns `True`. \n\nNow the problem with the example posted in #1784 is that it imports `test_foocompare` at the top-level of the `conftest.py` file which is loaded very early, when the `session` object has not even been created yet (much less set on the hook), so it doesn't try to rewrite that module.\n\nLooking at the code I noticed that the code block that should handle this, the `for pat in self.fnpaths` loop, doesn't really depend on `self.session` at all except for that weird comment about a cycle. Changing the code to always check if the filename matches one of the test patterns seems to do the trick.\n\nFix #1784\n",
      "issue_id": 1784,
      "issue_title": "pytest_assertrepr_compare not being used?",
      "issue_body": "Running regendoc in #1780, @The-Compiler noticed that an example from the docs regarding `pytest_assertrepr_compare` was not working as it should:\n\n``` python\n# conftest.py\nfrom test_foocompare import Foo\ndef pytest_assertrepr_compare(op, left, right):\n   if isinstance(left, Foo) and isinstance(right, Foo) and op == \"==\":\n       return ['Comparing Foo instances:',\n               '   vals: %s != %s' % (left.val, right.val)]\n\n# test_foocompare\nclass Foo:\n    def __init__(self, val):\n        self.val = val\n\n    def __eq__(self, other):\n        return self.val == other.val\n\n\ndef test_compare():\n    f1 = Foo(1)\n    f2 = Foo(2)\n    assert f1 == f2\n```\n\n```\n$ py.test\n\n================================== FAILURES ===================================\n________________________________ test_compare _________________________________\n\n    def test_compare():\n        f1 = Foo(1)\n        f2 = Foo(2)\n>       assert f1 == f2\nE       AssertionError\n\ntest_foocompare.py:12: AssertionError\n```\n",
      "issue_closed_at": "2016-08-05T09:46:10Z",
      "base_commit": "d5be6cba13d2c120be08d9dd9e886148475de9d7",
      "changes": [
        {
          "file": "_pytest/assertion/__init__.py",
          "type": "function",
          "name": "pytest_namespace",
          "class_name": null,
          "code": "def pytest_namespace():\n    return {'register_assert_rewrite': register_assert_rewrite}"
        },
        {
          "file": "_pytest/assertion/rewrite.py",
          "type": "line",
          "name": "line 11",
          "code": "import struct\nimport sys\nimport types\n\nimport py\nfrom _pytest.assertion import util"
        },
        {
          "file": "_pytest/assertion/rewrite.py",
          "type": "function",
          "name": "_should_rewrite",
          "class_name": "AssertionRewritingHook",
          "code": "def _should_rewrite(self, name, fn_pypath, state):\n        # always rewrite conftest files\n        fn = str(fn_pypath)\n        if fn_pypath.basename == 'conftest.py':\n            state.trace(\"rewriting conftest file: %r\" % (fn,))\n            return True\n        elif self.session is not None:\n            if self.session.isinitpath(fn):\n                state.trace(\"matched test file (was specified on cmdline): %r\" %\n                            (fn,))\n                return True\n            else:\n                # modules not passed explicitly on the command line are only\n                # rewritten if they match the naming convention for test files\n                session = self.session  # avoid a cycle here\n                self.session = None\n                try:\n                    for pat in self.fnpats:\n                        if fn_pypath.fnmatch(pat):\n                            state.trace(\"matched test file %r\" % (fn,))\n                            return True\n                finally:\n                    self.session = session\n                    del session\n        else:\n            for marked in self._must_rewrite:\n                if marked.startswith(name):\n                    return True\n        return False"
        }
      ]
    },
    {
      "pr_number": 3893,
      "pr_title": "travis: run tests on macOS",
      "pr_body": "Closes #3892 \r\nCloses #3888\r\n\r\nThanks for submitting a PR, your contribution is really appreciated!\r\n\r\nHere's a quick checklist that should be present in PRs (you can delete this text from the final description, this is\r\njust a guideline):\r\n\r\n- [ ] Create a new changelog file in the `changelog` folder, with a name like `<ISSUE NUMBER>.<TYPE>.rst`. See [changelog/README.rst](/changelog/README.rst) for details.\r\n- [x] Target the `master` branch for bug fixes, documentation updates and trivial changes.\r\n- [ ] Target the `features` branch for new features and removals/deprecations.\r\n- [ ] Include documentation when adding new features.\r\n- [ ] Include new tests or update existing tests when applicable.\r\n\r\nUnless your change is trivial or a small documentation fix (e.g.,  a typo or reword of a small section) please:\r\n\r\n- [ ] ~Add yourself to `AUTHORS` in alphabetical order;~ trivial change - no code\r\n",
      "issue_id": 3888,
      "issue_title": "doctests regression with v3.7.3 on OS-X",
      "issue_body": "- [x] Include a detailed description of the bug or suggestion\r\n\r\nAfter upgrading to v3.7.3, all doctests started failing with the exception:\r\n\r\n```\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:206: TypeError: cannot unpack non-iterable NoneType object\r\n```\r\n\r\nEverything is OK with v3.7.2.\r\n\r\nSlightly longer traceback (For the record, looking into the stacktrace, I see some references to \"darwin\"-specific functions):\r\n\r\n```\r\n___________________________________________________________________ [doctest] xxx ___________________________________________________________________\r\n\r\nself = <CallInfo when='call' exception: cannot unpack non-iterable NoneType object>, func = <function call_runtest_hook.<locals>.<lambda> at 0x1112f49d8>, when = 'call'\r\ntreat_keyboard_interrupt_as_exception = False\r\n\r\n    def __init__(self, func, when, treat_keyboard_interrupt_as_exception=False):\r\n        #: context of invocation: one of \"setup\", \"call\",\r\n        #: \"teardown\", \"memocollect\"\r\n        self.when = when\r\n        self.start = time()\r\n        try:\r\n>           self.result = func()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:201:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\n>   lambda: ihook(item=item, **kwds),\r\n    when=when,\r\n    treat_keyboard_interrupt_as_exception=item.config.getvalue(\"usepdb\"),\r\n        )\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:183:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <_HookCaller 'pytest_runtest_call'>, args = (), kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, notincall = set()\r\n\r\n    def __call__(self, *args, **kwargs):\r\n        if args:\r\n            raise TypeError(\"hook calling supports only keyword arguments\")\r\n        assert not self.is_historic()\r\n        if self.argnames:\r\n            notincall = set(self.argnames) - set(['__multicall__']) - set(\r\n                kwargs.keys())\r\n            if notincall:\r\n                warnings.warn(\r\n                    \"Argument(s) {} which are declared in the hookspec \"\r\n                    \"can not be found in this hook call\"\r\n                    .format(tuple(notincall)),\r\n                    stacklevel=2,\r\n                )\r\n>       return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/hooks.py:258:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <_pytest.config.PytestPluginManager object at 0x1034317b8>, hook = <_HookCaller 'pytest_runtest_call'>\r\nmethods = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\nkwargs = {'item': <DoctestItem 'somemodule.somefunction'>}\r\n\r\n    def _hookexec(self, hook, methods, kwargs):\r\n        # called from all hookcaller instances.\r\n        # enable_tracing will set its own wrapping function at self._inner_hookexec\r\n>       return self._inner_hookexec(hook, methods, kwargs)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/manager.py:67:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook = <_HookCaller 'pytest_runtest_call'>\r\nmethods = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\nkwargs = {'item': <DoctestItem 'somemodule.somefunction'>}\r\n\r\n    self._inner_hookexec = lambda hook, methods, kwargs: \\\r\n        hook.multicall(\r\n            methods, kwargs,\r\n>           firstresult=hook.spec_opts.get('firstresult'),\r\n        )\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/manager.py:61:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook_impls = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\ncaller_kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, firstresult = False\r\n\r\n    def _multicall(hook_impls, caller_kwargs, firstresult=False):\r\n        \"\"\"Execute a call into multiple python functions/methods and return the\r\n        result(s).\r\n\r\n        ``caller_kwargs`` comes from _HookCaller.__call__().\r\n        \"\"\"\r\n        __tracebackhide__ = True\r\n        results = []\r\n        excinfo = None\r\n        try:  # run impl and wrapper setup functions in a loop\r\n            teardowns = []\r\n            try:\r\n                for hook_impl in reversed(hook_impls):\r\n                    try:\r\n                        args = [caller_kwargs[argname] for argname in hook_impl.argnames]\r\n                    except KeyError:\r\n                        for argname in hook_impl.argnames:\r\n                            if argname not in caller_kwargs:\r\n                                raise HookCallError(\r\n                                    \"hook call must provide argument %r\" % (argname,))\r\n\r\n                    if hook_impl.hookwrapper:\r\n                        try:\r\n                            gen = hook_impl.function(*args)\r\n                            next(gen)   # first yield\r\n                            teardowns.append(gen)\r\n                        except StopIteration:\r\n                            _raise_wrapfail(gen, \"did not yield\")\r\n                    else:\r\n                        res = hook_impl.function(*args)\r\n                        if res is not None:\r\n                            results.append(res)\r\n                            if firstresult:  # halt further impl calls\r\n                                break\r\n            except BaseException:\r\n                excinfo = sys.exc_info()\r\n        finally:\r\n            if firstresult:  # first result hooks return a single value\r\n                outcome = _Result(results[0] if results else None, excinfo)\r\n            else:\r\n                outcome = _Result(results, excinfo)\r\n\r\n            # run all wrapper post-yield blocks\r\n            for gen in reversed(teardowns):\r\n                try:\r\n                    gen.send(outcome)\r\n                    _raise_wrapfail(gen, \"has second yield\")\r\n                except StopIteration:\r\n                    pass\r\n\r\n>           return outcome.get_result()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:201:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <pluggy.callers._Result object at 0x11116efd0>\r\n\r\n    def get_result(self):\r\n        \"\"\"Get the result(s) for this hook call.\r\n\r\n            If the hook was marked as a ``firstresult`` only a single value\r\n            will be returned otherwise a list of results.\r\n            \"\"\"\r\n        __tracebackhide__ = True\r\n        if self._excinfo is None:\r\n            return self._result\r\n        else:\r\n            ex = self._excinfo\r\n            if _py3:\r\n>               raise ex[1].with_traceback(ex[2])\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:76:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook_impls = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\ncaller_kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, firstresult = False\r\n\r\n    def _multicall(hook_impls, caller_kwargs, firstresult=False):\r\n        \"\"\"Execute a call into multiple python functions/methods and return the\r\n        result(s).\r\n\r\n        ``caller_kwargs`` comes from _HookCaller.__call__().\r\n        \"\"\"\r\n        __tracebackhide__ = True\r\n        results = []\r\n        excinfo = None\r\n        try:  # run impl and wrapper setup functions in a loop\r\n            teardowns = []\r\n            try:\r\n                for hook_impl in reversed(hook_impls):\r\n                    try:\r\n                        args = [caller_kwargs[argname] for argname in hook_impl.argnames]\r\n                    except KeyError:\r\n                        for argname in hook_impl.argnames:\r\n                            if argname not in caller_kwargs:\r\n                                raise HookCallError(\r\n                                    \"hook call must provide argument %r\" % (argname,))\r\n\r\n                    if hook_impl.hookwrapper:\r\n                        try:\r\n                            gen = hook_impl.function(*args)\r\n                            next(gen)   # first yield\r\n                            teardowns.append(gen)\r\n                        except StopIteration:\r\n                            _raise_wrapfail(gen, \"did not yield\")\r\n                    else:\r\n>                       res = hook_impl.function(*args)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:180:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nitem = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def pytest_runtest_call(item):\r\n        _update_current_test_var(item, \"call\")\r\n        sys.last_type, sys.last_value, sys.last_traceback = (None, None, None)\r\n        try:\r\n>           item.runtest()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:111:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def runtest(self):\r\n        _check_all_skipped(self.dtest)\r\n>       self._disable_output_capturing_for_darwin()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:192:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def _disable_output_capturing_for_darwin(self):\r\n        \"\"\"\r\n            Disable output capturing. Otherwise, stdout is lost to doctest (#985)\r\n            \"\"\"\r\n        if platform.system() != \"Darwin\":\r\n            return\r\n        capman = self.config.pluginmanager.getplugin(\"capturemanager\")\r\n        if capman:\r\n>           out, err = capman.suspend_global_capture(in_=True)\r\nE           TypeError: cannot unpack non-iterable NoneType object\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:206: TypeError\r\n```\r\n\r\n- [x] `pip list` of the virtual environment you are using\r\n\r\n```\r\n...\r\nmypy                     0.620\r\npytest                   3.7.3\r\npytest-cov               2.5.1\r\npytest-django            3.4.2\r\ntox                      3.2.1\r\n...\r\n```\r\n\r\n- [x] pytest and operating system versions\r\n\r\n`pytest v3.7.3` and \r\n\r\n```\r\n$ sw_vers\r\nProductName:\tMac OS X\r\nProductVersion:\t10.13.6\r\nBuildVersion:\t17G65\r\n```\r\n\r\n- [ ] Minimal example if possible\r\n",
      "issue_closed_at": "2018-08-28T20:06:18Z",
      "base_commit": "2a059b1c1bff392252c02413c89913dd672af168",
      "changes": [
        {
          "file": "src/_pytest/doctest.py",
          "type": "function",
          "name": "_disable_output_capturing_for_darwin",
          "class_name": "DoctestItem",
          "code": "def _disable_output_capturing_for_darwin(self):\n        \"\"\"\n        Disable output capturing. Otherwise, stdout is lost to doctest (#985)\n        \"\"\"\n        if platform.system() != \"Darwin\":\n            return\n        capman = self.config.pluginmanager.getplugin(\"capturemanager\")\n        if capman:\n            out, err = capman.suspend_global_capture(in_=True)\n            sys.stdout.write(out)\n            sys.stderr.write(err)"
        }
      ]
    },
    {
      "pr_number": 5205,
      "pr_title": "Introduce record_testsuite_property fixture",
      "pr_body": "This exposes the functionality introduced in fa6acdc as a session-scoped fixture.\r\n\r\nPlugins that want to remain compatible with the `xunit2`\r\nstandard should use this fixture instead of `record_property`.\r\n\r\nFix #5202\r\n\r\n@Zac-HD Hypothesis might want to use this fixture to produce `xunit2`-compatible reports according to #5202.\r\n\r\ncc @danilomendesdias",
      "issue_id": 5202,
      "issue_title": "Invalid XML schema for <properties> tags in JUnit reports ",
      "issue_body": "The problem:\r\n\r\nJUnit breaks when it reads an XML generated by pytest if plugins make use of  `record-property`. This behavior happens with newer versions of  hypothesis (https://github.com/HypothesisWorks/hypothesis/issues/1935).\r\n\r\n```\r\n[xUnit] [ERROR] - The result file '/somewhere/tests/pytests.xml' for the metric 'JUnit' is not valid. The result file has been skipped.\r\n```\r\n\r\nIn fact, as already mentioned in https://github.com/pytest-dev/pytest/issues/1126#issuecomment-484581283,  `record-property` is adding `<properties>` inside `<testcase>` which seems to be wrong (it should be inside `<testsuite>`). See: https://github.com/windyroad/JUnit-Schema/blob/master/JUnit.xsd .\r\n\r\nIt happens with all junit families.\r\n\r\nReproducing:\r\n\r\n```\r\n$ pip list\r\nPackage        Version \r\n-------------- --------\r\napipkg         1.5     \r\natomicwrites   1.3.0   \r\nattrs          19.1.0  \r\ncertifi        2019.3.9\r\nexecnet        1.6.0   \r\nhypothesis     4.18.3  \r\nmore-itertools 4.3.0   \r\npip            19.1    \r\npluggy         0.9.0   \r\npy             1.8.0   \r\npytest         4.4.1   \r\npytest-forked  1.0.2   \r\npytest-xdist   1.28.0  \r\nsetuptools     41.0.1  \r\nsix            1.12.0  \r\nwheel          0.33.1 \r\n```\r\n\r\n`test_xml_generation.py`\r\n```\r\nfrom hypothesis import given, strategies\r\n\r\n\r\n@given(x=strategies.integers(1, 10,))\r\ndef test_xml_generation(x):\r\n    assert 1 <= x <= 10\r\n```\r\n\r\n```\r\n$ pytest --junitxml=report.xml\r\n```\r\n\r\n`report.xml`\r\n```\r\n<?xml version=\"1.0\" encoding=\"utf-8\"?>\r\n<testsuite errors=\"0\" failures=\"0\" name=\"pytest\" skipped=\"0\" tests=\"1\" time=\"0.211\">\r\n    <testcase classname=\"test_xml_generation\" file=\"test_xml_generation.py\" line=\"3\" name=\"test_xml_generation\"\r\n              time=\"0.074\">\r\n        <properties>\r\n            <property name=\"hypothesis-stats\"\r\n                      value=\"[&apos;test_xml_generation.py::test_xml_generation:&apos;, &apos;&apos;, &apos;  - 100 passing examples, 0 failing examples, 0 invalid examples&apos;, &apos;  - Typical runtimes: &lt; 1ms&apos;, &apos;  - Fraction of time spent in data generation: ~ 49%&apos;, &apos;  - Stopped because settings.max_examples=100&apos;, &apos;&apos;]\"/>\r\n        </properties>\r\n    </testcase>\r\n</testsuite>\r\n```\r\n\r\nI was trying to create a PR to fix this, but when I saw https://github.com/pytest-dev/pytest/blob/7dcd9bf5add337686ec6f2ee81b24e8424319dba/src/_pytest/junitxml.py code I realized that what is needed to do could have more implications that I though. I think that nobody uses this feature with JUnit (as it breaks) and removing that is something to think about.\r\n\r\n",
      "issue_closed_at": "2019-05-11T16:27:27Z",
      "base_commit": "3a4a815c41badd1a6bac958aa18ddeb0c16cd202",
      "changes": [
        {
          "file": "src/_pytest/junitxml.py",
          "type": "function",
          "name": "add_attr_noop",
          "class_name": null,
          "code": "def add_attr_noop(name, value):\n        pass"
        },
        {
          "file": "src/_pytest/junitxml.py",
          "type": "function",
          "name": "__init__",
          "class_name": "LogXML",
          "code": "def __init__(\n        self,\n        logfile,\n        prefix,\n        suite_name=\"pytest\",\n        logging=\"no\",\n        report_duration=\"total\",\n        family=\"xunit1\",\n    ):\n        logfile = os.path.expanduser(os.path.expandvars(logfile))\n        self.logfile = os.path.normpath(os.path.abspath(logfile))\n        self.prefix = prefix\n        self.suite_name = suite_name\n        self.logging = logging\n        self.report_duration = report_duration\n        self.family = family\n        self.stats = dict.fromkeys([\"error\", \"passed\", \"failure\", \"skipped\"], 0)\n        self.node_reporters = {}  # nodeid -> _NodeReporter\n        self.node_reporters_ordered = []\n        self.global_properties = []\n        # List of reports that failed on call but teardown is pending.\n        self.open_reports = []\n        self.cnt_double_fail_tests = 0\n\n        # Replaces convenience family with real family\n        if self.family == \"legacy\":\n            self.family = \"xunit1\""
        },
        {
          "file": "src/_pytest/junitxml.py",
          "type": "function",
          "name": "pytest_terminal_summary",
          "class_name": "LogXML",
          "code": "def pytest_terminal_summary(self, terminalreporter):\n        terminalreporter.write_sep(\"-\", \"generated xml file: %s\" % (self.logfile))"
        }
      ]
    }
  ]
}