{
  "Selected_candidate": {
    "pr_number": 1405,
    "pr_title": "Display collect progress only when in a terminal",
    "pr_body": "Fix #1397\n",
    "issue_id": 1397,
    "issue_title": "\"collecting\" output with --color=yes in Continuous Integration output",
    "issue_body": "Hi,\n\nWe like to use `--color=yes` when running py.test in continuous integration services like Jenkins (with the [Ansi Color Plugin](https://wiki.jenkins-ci.org/display/JENKINS/AnsiColor+Plugin)). This enables us to see the same nice colors in Jenkins that we see on the terminal.\n\nThe only problem is that the \"collecting\" display progress actually piles up in the terminal because py.test output is being redirected and captured by the server. This can easily be reproduced by redirecting to a file instead:\n\n``` python\nimport pytest\n@pytest.mark.parametrize('i', range(10))\ndef test_foo(i):\n    if i == 5:\n        assert 0\n```\n\n```\npy.test --color=yes test_foo.py > out\n```\n\nContents of `out`:\n\n```\n\u001b[1m============================= test session starts =============================\u001b[0m\nplatform win32 -- Python 2.7.11, pytest-2.8.5, py-1.4.31, pluggy-0.3.1\nrootdir: x:\\jobs_done10, inifile: \n\u001b[1m\ncollecting 0 items\u001b[0m\u001b[1m\ncollecting 10 items\u001b[0m\u001b[1m\ncollected 10 items \n\u001b[0m\nfoo.py .....F....\n\n================================== FAILURES ===================================\n\u001b[1m\u001b[31m_________________________________ test_foo[5] _________________________________\u001b[0m\n\ni = 5\n\n\u001b[1m    @pytest.mark.parametrize('i', range(10))\u001b[0m\n\u001b[1m    def test_foo(i):\u001b[0m\n\u001b[1m        if i == 5:\u001b[0m\n\u001b[1m>           assert 0\u001b[0m\n\u001b[1m\u001b[31mE           assert 0\u001b[0m\n\nfoo.py:6: AssertionError\n\u001b[1m\u001b[31m===================== 1 failed, 9 passed in 0.02 seconds ======================\u001b[0m\n```\n\nThe color codes are correct, but the \"collecting\" messages are a problem because they occupy many lines of output in test suites with hundreds of tests.\n\nIs there any way to prevent those \"collecting\" messages from appearing? `-q` gets rid of them, but I want the header to appear, specially in CI.\n",
    "issue_closed_at": "2016-02-21T07:17:35Z",
    "base_commit": "3874d53ee121415c5a964a3f706b97c8010796aa",
    "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\n        self.stats = {}\n        self.startdir = py.path.local()\n        if file is None:\n            file = sys.stdout\n        self._tw = self.writer = _pytest.config.create_terminal_writer(config,\n                                                                       file)\n        self.currentfspath = None\n        self.reportchars = getreportopt(config)\n        self.hasmarkup = self._tw.hasmarkup"
      },
      {
        "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        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        if self.verbosity <= 0:\n            if not hasattr(rep, 'node') and self.showfspath:\n                self.write_fspath_result(rep.nodeid, letter)\n            else:\n                self._tw.write(letter)\n        else:\n            if isinstance(word, tuple):\n                word, markup = word\n            else:\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            line = self._locationline(rep.nodeid, *rep.location)\n            if not hasattr(rep, 'node'):\n                self.write_ensure_prefix(line, word, **markup)\n                #self._tw.write(word, **markup)\n            else:\n                self.ensure_newline()\n                if hasattr(rep, 'node'):\n                    self._tw.write(\"[%s] \" % rep.node.gateway.id)\n                self._tw.write(word, **markup)\n                self._tw.write(\" \" + line)\n                self.currentfspath = -2"
      },
      {
        "file": "_pytest/terminal.py",
        "type": "function",
        "name": "pytest_collectreport",
        "class_name": "TerminalReporter",
        "code": "def pytest_collectreport(self, report):\n        if report.failed:\n            self.stats.setdefault(\"error\", []).append(report)\n        elif report.skipped:\n            self.stats.setdefault(\"skipped\", []).append(report)\n        items = [x for x in report.result if isinstance(x, pytest.Item)]\n        self._numcollected += len(items)\n        if self.hasmarkup:\n            #self.write_fspath_result(report.nodeid, 'E')\n            self.report_collect()"
      },
      {
        "file": "_pytest/terminal.py",
        "type": "function",
        "name": "report_collect",
        "class_name": "TerminalReporter",
        "code": "def report_collect(self, final=False):\n        if self.config.option.verbose < 0:\n            return\n\n        errors = len(self.stats.get('error', []))\n        skipped = len(self.stats.get('skipped', []))\n        if final:\n            line = \"collected \"\n        else:\n            line = \"collecting \"\n        line += str(self._numcollected) + \" items\"\n        if errors:\n            line += \" / %d errors\" % errors\n        if skipped:\n            line += \" / %d skipped\" % skipped\n        if self.hasmarkup:\n            if final:\n                line += \" \\n\"\n            self.rewrite(line, bold=True)\n        else:\n            self.write_line(line)"
      }
    ]
  },
  "Justification": "Candidate D is the most relevant because it also deals with the output behavior of pytest, specifically in continuous integration environments, which could tie into the current bug's handling of how modules are skipped based on the Python version. Both involve nuances of pytest's output handling—one about collecting tests with colored output, and the current bug report relates to skipping tests based on environment conditions. The familiarity with pytest's reporting and subsequent interface changes makes Candidate D a practical guide for debug strategies in handling pytest's behavior.",
  "instance_id": "pytest-dev__pytest-8906",
  "repo": "pytest-dev/pytest",
  "created_at": "2021-07-14T08:00:50Z",
  "problem_statement": "Improve handling of skip for module level\nThis is potentially about updating docs, updating error messages or introducing a new API.\r\n\r\nConsider the following scenario:\r\n\r\n`pos_only.py` is using Python 3,8 syntax:\r\n```python\r\ndef foo(a, /, b):\r\n    return a + b\r\n```\r\n\r\nIt should not be tested under Python 3.6 and 3.7.\r\nThis is a proper way to skip the test in Python older than 3.8:\r\n```python\r\nfrom pytest import raises, skip\r\nimport sys\r\nif sys.version_info < (3, 8):\r\n    skip(msg=\"Requires Python >= 3.8\", allow_module_level=True)\r\n\r\n# import must be after the module level skip:\r\nfrom pos_only import *\r\n\r\ndef test_foo():\r\n    assert foo(10, 20) == 30\r\n    assert foo(10, b=20) == 30\r\n    with raises(TypeError):\r\n        assert foo(a=10, b=20)\r\n```\r\n\r\nMy actual test involves parameterize and a 3.8 only class, so skipping the test itself is not sufficient because the 3.8 class was used in the parameterization.\r\n\r\nA naive user will try to initially skip the module like:\r\n\r\n```python\r\nif sys.version_info < (3, 8):\r\n    skip(msg=\"Requires Python >= 3.8\")\r\n```\r\nThis issues this error:\r\n\r\n>Using pytest.skip outside of a test is not allowed. To decorate a test function, use the @pytest.mark.skip or @pytest.mark.skipif decorators instead, and to skip a module use `pytestmark = pytest.mark.{skip,skipif}.\r\n\r\nThe proposed solution `pytestmark = pytest.mark.{skip,skipif}`, does not work  in my case: pytest continues to process the file and fail when it hits the 3.8 syntax (when running with an older version of Python).\r\n\r\nThe correct solution, to use skip as a function is actively discouraged by the error message.\r\n\r\nThis area feels a bit unpolished.\r\nA few ideas to improve:\r\n\r\n1. Explain skip with  `allow_module_level` in the error message. this seems in conflict with the spirit of the message.\r\n2. Create an alternative API to skip a module to make things easier: `skip_module(\"reason\")`, which can call `_skip(msg=msg, allow_module_level=True)`.\r\n\r\n\n",
  "patch": "diff --git a/src/_pytest/python.py b/src/_pytest/python.py\n--- a/src/_pytest/python.py\n+++ b/src/_pytest/python.py\n@@ -608,10 +608,10 @@ def _importtestmodule(self):\n             if e.allow_module_level:\n                 raise\n             raise self.CollectError(\n-                \"Using pytest.skip outside of a test is not allowed. \"\n-                \"To decorate a test function, use the @pytest.mark.skip \"\n-                \"or @pytest.mark.skipif decorators instead, and to skip a \"\n-                \"module use `pytestmark = pytest.mark.{skip,skipif}.\"\n+                \"Using pytest.skip outside of a test will skip the entire module. \"\n+                \"If that's your intention, pass `allow_module_level=True`. \"\n+                \"If you want to skip a specific test or an entire class, \"\n+                \"use the @pytest.mark.skip or @pytest.mark.skipif decorators.\"\n             ) from e\n         self.config.pluginmanager.consider_module(mod)\n         return mod\n"
}