{
  "Selected_candidate": {
    "pr_number": 2632,
    "pr_title": "Support PEP-415's Exception.__suppress_context__",
    "pr_body": "PEP-415 states that `exception.__context__` should be suppressed\r\nin traceback outputs, if `exception.__suppress_context__` is\r\n`True`.\r\n\r\nNow if a ``raise exception from None`` is caught by pytest,\r\npytest will no longer chain the context in the test report.\r\n\r\nThe algorithm in `FormattedExcinfo` now better matches the one\r\nin `traceback.TracebackException`.\r\n\r\n`Exception.__suppress_context__` is available in all of the\r\nversions of Python 3 that are supported by pytest.\r\n\r\nFixes #2631.",
    "issue_id": 2631,
    "issue_title": "Support PEP-415's Exception.__suppress_context__",
    "issue_body": "- [x] Include a detailed description of the bug or suggestion\r\n\r\nPEP-415 states that `exception.__context__` should be suppressed in traceback outputs, if `exception.__suppress_context__` is `True`.\r\n\r\nIf a ``raise exception from None`` is caught by pytest, pytest should not chain the context in the test report.\r\n\r\nThe current algorithm in `_pytest._code.code.FormattedExcinfo` is:\r\n\r\n```python\r\nif e.__cause__ is not None:\r\n    # Code to chain the cause.\r\nelif e.__context__ is not None:\r\n    # Code to chain the context.\r\n```\r\n\r\nwhich means that pytest always chains the exception, assuming that `e.__context__` hasn't been set to `None`.\r\n\r\nBy comparison, the algorithm in `traceback.TracebackException` is:\r\n\r\n```python\r\nif e.__cause__ is not None:\r\n    # Code to chain the cause.\r\nelif (e.__context__ is not None and not e.__suppress_context__):\r\n    # Code to chain the context.\r\n```\r\n\r\n`Exception.__suppress_context__` is available in all of the versions of Python 3 that are supported by pytest, so it is trivial to add support for this feature.\r\n\r\n- [x] Minimal example if possible\r\n\r\nHere's a test that has a ``raise exception from None`` in it:\r\n\r\n```python\r\ndef test_raise_from_none():\r\n    try:\r\n        raise ValueError()\r\n    except Exception:\r\n        raise AttributeError() from None\r\n```\r\n\r\nThis is the test output (with the chained exception traceback) that results when running against the pytest feature branch (commit 768edde899fe3629a69d55289f82bb0d95635c06):\r\n\r\n```\r\n_____________________________________________________ test_raise_from_none _____________________________________________________\r\n\r\n    def test_raise_from_none():\r\n        try:\r\n>           raise ValueError()\r\nE           ValueError\r\n\r\ntesting/code/test_excinfo.py:1257: ValueError\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\n    def test_raise_from_none():\r\n        try:\r\n            raise ValueError()\r\n        except Exception:\r\n>           raise AttributeError() from None\r\nE           AttributeError\r\n\r\ntesting/code/test_excinfo.py:1259: AttributeError\r\n```\r\n\r\nBut running the same code in a terminal produces this shorter, non-chained traceback:\r\n\r\n```\r\nAttributeError                            Traceback (most recent call last)\r\n<ipython-input-51-e2e3809c49fb> in <module>()\r\n----> 1 test_raise_from_none()\r\n\r\n<ipython-input-50-d651befdf00e> in test_raise_from_none()\r\n      3         raise ValueError()\r\n      4     except Exception:\r\n----> 5         raise AttributeError() from None\r\n      6\r\n\r\nAttributeError:\r\n```\r\n\r\nAnd this is what the pytest output should look like:\r\n\r\n```\r\n_____________________________________________________ test_raise_from_none _____________________________________________________\r\n\r\n    def test_raise_from_none():\r\n        try:\r\n            raise ValueError()\r\n        except Exception:\r\n>           raise AttributeError() from None\r\nE           AttributeError\r\n\r\ntesting/code/test_excinfo.py:1259: AttributeError\r\n```",
    "issue_closed_at": "2017-07-30T21:46:09Z",
    "base_commit": "768edde899fe3629a69d55289f82bb0d95635c06",
    "changes": [
      {
        "file": "_pytest/_code/code.py",
        "type": "function",
        "name": "repr_excinfo",
        "class_name": "FormattedExcinfo",
        "code": "def repr_excinfo(self, excinfo):\n        if _PY2:\n            reprtraceback = self.repr_traceback(excinfo)\n            reprcrash = excinfo._getreprcrash()\n\n            return ReprExceptionInfo(reprtraceback, reprcrash)\n        else:\n            repr_chain = []\n            e = excinfo.value\n            descr = None\n            while e is not None:\n                if excinfo:\n                    reprtraceback = self.repr_traceback(excinfo)\n                    reprcrash = excinfo._getreprcrash()\n                else:\n                    # fallback to native repr if the exception doesn't have a traceback:\n                    # ExceptionInfo objects require a full traceback to work\n                    reprtraceback = ReprTracebackNative(py.std.traceback.format_exception(type(e), e, None))\n                    reprcrash = None\n\n                repr_chain += [(reprtraceback, reprcrash, descr)]\n                if e.__cause__ is not None:\n                    e = e.__cause__\n                    excinfo = ExceptionInfo((type(e), e, e.__traceback__)) if e.__traceback__ else None\n                    descr = 'The above exception was the direct cause of the following exception:'\n                elif e.__context__ is not None:\n                    e = e.__context__\n                    excinfo = ExceptionInfo((type(e), e, e.__traceback__)) if e.__traceback__ else None\n                    descr = 'During handling of the above exception, another exception occurred:'\n                else:\n                    e = None\n            repr_chain.reverse()\n            return ExceptionChainRepr(repr_chain)"
      }
    ]
  },
  "Justification": "Candidate B is the most helpful because it deals directly with exceptions and their handling within pytest, which is the context for the CURRENT bug. The structural similarity is relevant, as both bugs involve how exceptions are reported and displayed in the pytest framework. The symptoms of these bugs are also related, as they both produce unexpected outputs when exceptions are raised and caught in pytest contexts. Understanding the changes related to suppressing context in exceptions could provide insights into how the reporting behavior in the CURRENT bug can be improved or modified, making it particularly useful for addressing the CURRENT bug.",
  "instance_id": "pytest-dev__pytest-5413",
  "repo": "pytest-dev/pytest",
  "created_at": "2019-06-06T15:21:20Z",
  "problem_statement": "str() on the pytest.raises context variable doesn't behave same as normal exception catch\nPytest 4.6.2, macOS 10.14.5\r\n\r\n```Python\r\ntry:\r\n    raise LookupError(\r\n        f\"A\\n\"\r\n        f\"B\\n\"\r\n        f\"C\"\r\n    )\r\nexcept LookupError as e:\r\n    print(str(e))\r\n```\r\nprints\r\n\r\n> A\r\n> B\r\n> C\r\n\r\nBut\r\n\r\n```Python\r\nwith pytest.raises(LookupError) as e:\r\n    raise LookupError(\r\n        f\"A\\n\"\r\n        f\"B\\n\"\r\n        f\"C\"\r\n    )\r\n\r\nprint(str(e))\r\n```\r\n\r\nprints\r\n\r\n> <console>:3: LookupError: A\r\n\r\nIn order to get the full error message, one must do `str(e.value)`, which is documented, but this is a different interaction. Any chance the behavior could be changed to eliminate this gotcha?\r\n\r\n-----\r\n\r\nPip list gives\r\n\r\n```\r\nPackage            Version  Location\r\n------------------ -------- ------------------------------------------------------\r\napipkg             1.5\r\nasn1crypto         0.24.0\r\natomicwrites       1.3.0\r\nattrs              19.1.0\r\naws-xray-sdk       0.95\r\nboto               2.49.0\r\nboto3              1.9.51\r\nbotocore           1.12.144\r\ncertifi            2019.3.9\r\ncffi               1.12.3\r\nchardet            3.0.4\r\nClick              7.0\r\ncodacy-coverage    1.3.11\r\ncolorama           0.4.1\r\ncoverage           4.5.3\r\ncryptography       2.6.1\r\ndecorator          4.4.0\r\ndocker             3.7.2\r\ndocker-pycreds     0.4.0\r\ndocutils           0.14\r\necdsa              0.13.2\r\nexecnet            1.6.0\r\nfuture             0.17.1\r\nidna               2.8\r\nimportlib-metadata 0.17\r\nipaddress          1.0.22\r\nJinja2             2.10.1\r\njmespath           0.9.4\r\njsondiff           1.1.1\r\njsonpickle         1.1\r\njsonschema         2.6.0\r\nMarkupSafe         1.1.1\r\nmock               3.0.4\r\nmore-itertools     7.0.0\r\nmoto               1.3.7\r\nneobolt            1.7.10\r\nneotime            1.7.4\r\nnetworkx           2.1\r\nnumpy              1.15.0\r\npackaging          19.0\r\npandas             0.24.2\r\npip                19.1.1\r\npluggy             0.12.0\r\nprompt-toolkit     2.0.9\r\npy                 1.8.0\r\npy2neo             4.2.0\r\npyaml              19.4.1\r\npycodestyle        2.5.0\r\npycparser          2.19\r\npycryptodome       3.8.1\r\nPygments           2.3.1\r\npyOpenSSL          19.0.0\r\npyparsing          2.4.0\r\npytest             4.6.2\r\npytest-cache       1.0\r\npytest-codestyle   1.4.0\r\npytest-cov         2.6.1\r\npytest-forked      1.0.2\r\npython-dateutil    2.7.3\r\npython-jose        2.0.2\r\npytz               2018.5\r\nPyYAML             5.1\r\nrequests           2.21.0\r\nrequests-mock      1.5.2\r\nresponses          0.10.6\r\ns3transfer         0.1.13\r\nsetuptools         41.0.1\r\nsix                1.11.0\r\nsqlite3worker      1.1.7\r\ntabulate           0.8.3\r\nurllib3            1.24.3\r\nwcwidth            0.1.7\r\nwebsocket-client   0.56.0\r\nWerkzeug           0.15.2\r\nwheel              0.33.1\r\nwrapt              1.11.1\r\nxlrd               1.1.0\r\nxmltodict          0.12.0\r\nzipp               0.5.1\r\n```\n",
  "patch": "diff --git a/src/_pytest/_code/code.py b/src/_pytest/_code/code.py\n--- a/src/_pytest/_code/code.py\n+++ b/src/_pytest/_code/code.py\n@@ -534,13 +534,6 @@ def getrepr(\n         )\n         return fmt.repr_excinfo(self)\n \n-    def __str__(self):\n-        if self._excinfo is None:\n-            return repr(self)\n-        entry = self.traceback[-1]\n-        loc = ReprFileLocation(entry.path, entry.lineno + 1, self.exconly())\n-        return str(loc)\n-\n     def match(self, regexp):\n         \"\"\"\n         Check whether the regular expression 'regexp' is found in the string\n"
}