{
  "instance_id": "pytest-dev__pytest-5495",
  "repo": "pytest-dev/pytest",
  "created_at": "2019-06-25T23:41:16Z",
  "problem_statement": "Confusing assertion rewriting message with byte strings\nThe comparison with assertion rewriting for byte strings is confusing: \r\n```\r\n    def test_b():\r\n>       assert b\"\" == b\"42\"\r\nE       AssertionError: assert b'' == b'42'\r\nE         Right contains more items, first extra item: 52\r\nE         Full diff:\r\nE         - b''\r\nE         + b'42'\r\nE         ?   ++\r\n```\r\n\r\n52 is the ASCII ordinal of \"4\" here.\r\n\r\nIt became clear to me when using another example:\r\n\r\n```\r\n    def test_b():\r\n>       assert b\"\" == b\"1\"\r\nE       AssertionError: assert b'' == b'1'\r\nE         Right contains more items, first extra item: 49\r\nE         Full diff:\r\nE         - b''\r\nE         + b'1'\r\nE         ?   +\r\n```\r\n\r\nNot sure what should/could be done here.\n",
  "patch": "diff --git a/src/_pytest/assertion/util.py b/src/_pytest/assertion/util.py\n--- a/src/_pytest/assertion/util.py\n+++ b/src/_pytest/assertion/util.py\n@@ -254,17 +254,38 @@ def _compare_eq_iterable(left, right, verbose=0):\n \n \n def _compare_eq_sequence(left, right, verbose=0):\n+    comparing_bytes = isinstance(left, bytes) and isinstance(right, bytes)\n     explanation = []\n     len_left = len(left)\n     len_right = len(right)\n     for i in range(min(len_left, len_right)):\n         if left[i] != right[i]:\n+            if comparing_bytes:\n+                # when comparing bytes, we want to see their ascii representation\n+                # instead of their numeric values (#5260)\n+                # using a slice gives us the ascii representation:\n+                # >>> s = b'foo'\n+                # >>> s[0]\n+                # 102\n+                # >>> s[0:1]\n+                # b'f'\n+                left_value = left[i : i + 1]\n+                right_value = right[i : i + 1]\n+            else:\n+                left_value = left[i]\n+                right_value = right[i]\n+\n             explanation += [\n-                \"At index {} diff: {!r} != {!r}\".format(i, left[i], right[i])\n+                \"At index {} diff: {!r} != {!r}\".format(i, left_value, right_value)\n             ]\n             break\n-    len_diff = len_left - len_right\n \n+    if comparing_bytes:\n+        # when comparing bytes, it doesn't help to show the \"sides contain one or more items\"\n+        # longer explanation, so skip it\n+        return explanation\n+\n+    len_diff = len_left - len_right\n     if len_diff:\n         if len_diff > 0:\n             dir_with_more = \"Left\"\n",
  "similar_bug_items": [
    {
      "pr_number": 2487,
      "pr_title": "Fix internal error when trying to detect the start of a recursive traceback",
      "pr_body": "Fix #2486\r\n",
      "issue_id": 2486,
      "issue_title": "_truncate_recursive_traceback()'s recursionindex can be None in 3.1.2",
      "issue_body": "```\r\ntest runtests: commands[0] | coverage run --parallel --source tested_library -m pytest --doctest-glob=*.md --junit-xml=report.xml\r\n============================= test session starts ==============================\r\nplatform linux2 -- Python 2.7.5, pytest-3.1.2, py-1.4.34, pluggy-0.4.0 -- workspace/.tox/test/bin/python2\r\ncachedir: .cache\r\nrootdir: workspace, inifile: tox.ini\r\nplugins: mock-1.6.0, pylama-7.3.3\r\ncollecting ... INTERNALERROR> Traceback (most recent call last):\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 105, in wrap_session\r\nINTERNALERROR>     session.exitstatus = doit(config, session) or 0\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 140, in _main\r\nINTERNALERROR>     config.hook.pytest_collection(session=session)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 745, in __call__\r\nINTERNALERROR>     return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 339, in _hookexec\r\nINTERNALERROR>     return self._inner_hookexec(hook, methods, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 334, in <lambda>\r\nINTERNALERROR>     _MultiCall(methods, kwargs, hook.spec_opts).execute()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 614, in execute\r\nINTERNALERROR>     res = hook_impl.function(*args)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 150, in pytest_collection\r\nINTERNALERROR>     return session.perform_collect()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 604, in perform_collect\r\nINTERNALERROR>     items = self._perform_collect(args, genitems)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 641, in _perform_collect\r\nINTERNALERROR>     self.items.extend(self.genitems(node))\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 776, in genitems\r\nINTERNALERROR>     rep = collect_one_node(node)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/runner.py\", line 457, in collect_one_node\r\nINTERNALERROR>     rep = ihook.pytest_make_collect_report(collector=collector)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 745, in __call__\r\nINTERNALERROR>     return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 339, in _hookexec\r\nINTERNALERROR>     return self._inner_hookexec(hook, methods, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 334, in <lambda>\r\nINTERNALERROR>     _MultiCall(methods, kwargs, hook.spec_opts).execute()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 613, in execute\r\nINTERNALERROR>     return _wrapped_call(hook_impl.function(*args), self.execute)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 250, in _wrapped_call\r\nINTERNALERROR>     wrap_controller.send(call_outcome)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/capture.py\", line 118, in pytest_make_collect_report\r\nINTERNALERROR>     rep = outcome.get_result()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 280, in get_result\r\nINTERNALERROR>     _reraise(*ex)  # noqa\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 265, in __init__\r\nINTERNALERROR>     self.result = func()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 614, in execute\r\nINTERNALERROR>     res = hook_impl.function(*args)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/runner.py\", line 342, in pytest_make_collect_report\r\nINTERNALERROR>     errorinfo = collector.repr_failure(call.excinfo)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 480, in repr_failure\r\nINTERNALERROR>     return self._repr_failure_py(excinfo, style=\"short\")\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 457, in _repr_failure_py\r\nINTERNALERROR>     style=style, tbfilter=tbfilter)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 429, in getrepr\r\nINTERNALERROR>     return fmt.repr_excinfo(self)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 650, in repr_excinfo\r\nINTERNALERROR>     reprtraceback = self.repr_traceback(excinfo)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 607, in repr_traceback\r\nINTERNALERROR>     traceback, extraline = self._truncate_recursive_traceback(traceback)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 644, in _truncate_recursive_traceback\r\nINTERNALERROR>     traceback = traceback[:recursionindex + 1]\r\nINTERNALERROR> TypeError: unsupported operand type(s) for +: 'NoneType' and 'int'\r\n```\r\n",
      "issue_closed_at": "2017-06-22T11:40:33Z",
      "base_commit": "b2d7c26d80aedb8533430d2e70c3d472614c7c0c",
      "changes": [
        {
          "file": "_pytest/_code/code.py",
          "type": "function",
          "name": "_truncate_recursive_traceback",
          "class_name": "FormattedExcinfo",
          "code": "def _truncate_recursive_traceback(self, traceback):\n        \"\"\"\n        Truncate the given recursive traceback trying to find the starting point\n        of the recursion.\n\n        The detection is done by going through each traceback entry and finding the\n        point in which the locals of the frame are equal to the locals of a previous frame (see ``recursionindex()``.\n\n        Handle the situation where the recursion process might raise an exception (for example\n        comparing numpy arrays using equality raises a TypeError), in which case we do our best to\n        warn the user of the error and show a limited traceback.\n        \"\"\"\n        try:\n            recursionindex = traceback.recursionindex()\n        except Exception as e:\n            max_frames = 10\n            extraline = (\n                '!!! Recursion error detected, but an error occurred locating the origin of recursion.\\n'\n                '  The following exception happened when comparing locals in the stack frame:\\n'\n                '    {exc_type}: {exc_msg}\\n'\n                '  Displaying first and last {max_frames} stack frames out of {total}.'\n            ).format(exc_type=type(e).__name__, exc_msg=safe_str(e), max_frames=max_frames, total=len(traceback))\n            traceback = traceback[:max_frames] + traceback[-max_frames:]\n        else:\n            extraline = \"!!! Recursion detected (same locals & position)\"\n            traceback = traceback[:recursionindex + 1]\n            \n        return traceback, extraline"
        }
      ]
    },
    {
      "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": 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)"
        }
      ]
    },
    {
      "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": 2096,
      "pr_title": "Merge master into features",
      "pr_body": "",
      "issue_id": 2078,
      "issue_title": "Error when --confcutdir is not a directory",
      "issue_body": "Currently `--confcutdir` accepts a filename, which is incorrect because it means no `conftest.py` file will ever match.\r\n\r\nIt should raise an \"invalid option\" error.\r\n\r\n",
      "issue_closed_at": "2016-11-23T12:23:01Z",
      "base_commit": "5ce551e469fa16a292227815e8fb1a8dd779bfa9",
      "changes": [
        {
          "file": "_pytest/assertion/rewrite.py",
          "type": "function",
          "name": "find_module",
          "class_name": "AssertionRewritingHook",
          "code": "def find_module(self, name, path=None):\n        state = self.config._assertstate\n        state.trace(\"find_module called for: %s\" % name)\n        names = name.rsplit(\".\", 1)\n        lastname = names[-1]\n        pth = None\n        if path is not None:\n            # Starting with Python 3.3, path is a _NamespacePath(), which\n            # causes problems if not converted to list.\n            path = list(path)\n            if len(path) == 1:\n                pth = path[0]\n        if pth is None:\n            try:\n                fd, fn, desc = imp.find_module(lastname, path)\n            except ImportError:\n                return None\n            if fd is not None:\n                fd.close()\n            tp = desc[2]\n            if tp == imp.PY_COMPILED:\n                if hasattr(imp, \"source_from_cache\"):\n                    fn = imp.source_from_cache(fn)\n                else:\n                    fn = fn[:-1]\n            elif tp != imp.PY_SOURCE:\n                # Don't know what this is.\n                return None\n        else:\n            fn = os.path.join(pth, name.rpartition(\".\")[2] + \".py\")\n\n        fn_pypath = py.path.local(fn)\n        if not self._should_rewrite(name, fn_pypath, state):\n            return None\n\n        self._rewritten_names.add(name)\n\n        # The requested module looks like a test file, so rewrite it. This is\n        # the most magical part of the process: load the source, rewrite the\n        # asserts, and load the rewritten source. We also cache the rewritten\n        # module code in a special pyc. We must be aware of the possibility of\n        # concurrent pytest processes rewriting and loading pycs. To avoid\n        # tricky race conditions, we maintain the following invariant: The\n        # cached pyc is always a complete, valid pyc. Operations on it must be\n        # atomic. POSIX's atomic rename comes in handy.\n        write = not sys.dont_write_bytecode\n        cache_dir = os.path.join(fn_pypath.dirname, \"__pycache__\")\n        if write:\n            try:\n                os.mkdir(cache_dir)\n            except OSError:\n                e = sys.exc_info()[1].errno\n                if e == errno.EEXIST:\n                    # Either the __pycache__ directory already exists (the\n                    # common case) or it's blocked by a non-dir node. In the\n                    # latter case, we'll ignore it in _write_pyc.\n                    pass\n                elif e in [errno.ENOENT, errno.ENOTDIR]:\n                    # One of the path components was not a directory, likely\n                    # because we're in a zip file.\n                    write = False\n                elif e in [errno.EACCES, errno.EROFS, errno.EPERM]:\n                    state.trace(\"read only directory: %r\" % fn_pypath.dirname)\n                    write = False\n                else:\n                    raise\n        cache_name = fn_pypath.basename[:-3] + PYC_TAIL\n        pyc = os.path.join(cache_dir, cache_name)\n        # Notice that even if we're in a read-only directory, I'm going\n        # to check for a cached pyc. This may not be optimal...\n        co = _read_pyc(fn_pypath, pyc, state.trace)\n        if co is None:\n            state.trace(\"rewriting %r\" % (fn,))\n            source_stat, co = _rewrite_test(self.config, fn_pypath)\n            if co is None:\n                # Probably a SyntaxError in the test.\n                return None\n            if write:\n                _make_rewritten_pyc(state, source_stat, pyc, co)\n        else:\n            state.trace(\"found cached rewritten pyc for %r\" % (fn,))\n        self.modules[name] = co, pyc\n        return self"
        },
        {
          "file": "_pytest/config.py",
          "type": "function",
          "name": "_warn_about_missing_assertion",
          "class_name": "Config",
          "code": "def _warn_about_missing_assertion(self, mode):\n        try:\n            assert False\n        except AssertionError:\n            pass\n        else:\n            if mode == 'plain':\n                sys.stderr.write(\"WARNING: ASSERTIONS ARE NOT EXECUTED\"\n                                 \" and FAILING TESTS WILL PASS.  Are you\"\n                                 \" using python -O?\")\n            else:\n                sys.stderr.write(\"WARNING: assertions not in test modules or\"\n                                 \" plugins will be ignored\"\n                                 \" because assert statements are not executed \"\n                                 \"by the underlying Python interpreter \"\n                                 \"(are you using python -O?)\\n\")"
        },
        {
          "file": "_pytest/config.py",
          "type": "function",
          "name": "_preparse",
          "class_name": "Config",
          "code": "def _preparse(self, args, addopts=True):\n        self._initini(args)\n        if addopts:\n            args[:] = shlex.split(os.environ.get('PYTEST_ADDOPTS', '')) + args\n            args[:] = self.getini(\"addopts\") + args\n        self._checkversion()\n        entrypoint_name = 'pytest11'\n        self._consider_importhook(args, entrypoint_name)\n        self.pluginmanager.consider_preparse(args)\n        self.pluginmanager.load_setuptools_entrypoints(entrypoint_name)\n        self.pluginmanager.consider_env()\n        self.known_args_namespace = ns = self._parser.parse_known_args(args, namespace=self.option.copy())\n        if self.known_args_namespace.confcutdir is None and self.inifile:\n            confcutdir = py.path.local(self.inifile).dirname\n            self.known_args_namespace.confcutdir = confcutdir\n        try:\n            self.hook.pytest_load_initial_conftests(early_config=self,\n                    args=args, parser=self._parser)\n        except ConftestImportFailure:\n            e = sys.exc_info()[1]\n            if ns.help or ns.version:\n                # we don't want to prevent --help/--version to work\n                # so just let is pass and print a warning at the end\n                self._warn(\"could not load initial conftests (%s)\\n\" % e.path)\n            else:\n                raise"
        },
        {
          "file": "_pytest/main.py",
          "type": "function",
          "name": "__getattr__",
          "class_name": "FSHookProxy",
          "code": "def __getattr__(self, name):\n        x = self.pm.subset_hook_caller(name, remove_plugins=self.remove_mods)\n        self.__dict__[name] = x\n        return x"
        },
        {
          "file": "_pytest/main.py",
          "type": "function",
          "name": "_parsearg",
          "class_name": "Session",
          "code": "def _parsearg(self, arg):\n        \"\"\" return (fspath, names) tuple after checking the file exists. \"\"\"\n        parts = str(arg).split(\"::\")\n        if self.config.option.pyargs:\n            parts[0] = self._tryconvertpyarg(parts[0])\n        relpath = parts[0].replace(\"/\", os.sep)\n        path = self.config.invocation_dir.join(relpath, abs=True)\n        if not path.check():\n            if self.config.option.pyargs:\n                msg = \"file or package not found: \"\n            else:\n                msg = \"file not found: \"\n            raise pytest.UsageError(msg + arg)\n        parts[0] = path\n        return parts"
        },
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Function",
          "code": "def __init__(self, name, parent, args=None, config=None,\n                 callspec=None, callobj=NOTSET, keywords=None, session=None,\n                 fixtureinfo=None, originalname=None):\n        super(Function, self).__init__(name, parent, config=config,\n                                       session=session)\n        self._args = args\n        if callobj is not NOTSET:\n            self.obj = callobj\n\n        self.keywords.update(self.obj.__dict__)\n        if callspec:\n            self.callspec = callspec\n            self.keywords.update(callspec.keywords)\n        if keywords:\n            self.keywords.update(keywords)\n\n        if fixtureinfo is None:\n            fixtureinfo = self.session._fixturemanager.getfixtureinfo(\n                self.parent, self.obj, self.cls,\n                funcargs=not self._isyieldedfunction())\n        self._fixtureinfo = fixtureinfo\n        self.fixturenames = fixtureinfo.names_closure\n        self._initrequest()\n\n        #: original function name, without any decorations (for example\n        #: parametrization adds a ``\"[...]\"`` suffix to function names).\n        #:\n        #: .. versionadded:: 3.0\n        self.originalname = originalname"
        }
      ]
    }
  ]
}