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