{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate D relates to assertion rewriting, which is directly relevant to the CURRENT bug report's problem of confusing assertion output. The structural similarity lies in the fact that both reports involve modifications to assertion handling in pytest, and thus may share common underlying issues or behaviors regarding how assertions are reported. By reviewing Candidate D, the developer can gain insights into enhancing the assertion reporting, helping to resolve the current confusion surrounding byte string comparisons.",
  "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"
}