{
  "instance_id": "pytest-dev__pytest-11148",
  "repo": "pytest-dev/pytest",
  "created_at": "2023-06-29T00:04:33Z",
  "problem_statement": "Module imported twice under import-mode=importlib\nIn pmxbot/pmxbot@7f189ad, I'm attempting to switch pmxbot off of pkg_resources style namespace packaging to PEP 420 namespace packages. To do so, I've needed to switch to `importlib` for the `import-mode` and re-organize the tests to avoid import errors on the tests.\r\n\r\nYet even after working around these issues, the tests are failing when the effect of `core.initialize()` doesn't seem to have had any effect.\r\n\r\nInvestigating deeper, I see that initializer is executed and performs its actions (setting a class variable `pmxbot.logging.Logger.store`), but when that happens, there are two different versions of `pmxbot.logging` present, one in `sys.modules` and another found in `tests.unit.test_commands.logging`:\r\n\r\n```\r\n=========================================================================== test session starts ===========================================================================\r\nplatform darwin -- Python 3.11.1, pytest-7.2.0, pluggy-1.0.0\r\ncachedir: .tox/python/.pytest_cache\r\nrootdir: /Users/jaraco/code/pmxbot/pmxbot, configfile: pytest.ini\r\nplugins: black-0.3.12, mypy-0.10.3, jaraco.test-5.3.0, checkdocs-2.9.0, flake8-1.1.1, enabler-2.0.0, jaraco.mongodb-11.2.1, pmxbot-1122.14.3.dev13+g7f189ad\r\ncollected 421 items / 180 deselected / 241 selected                                                                                                                       \r\nrun-last-failure: rerun previous 240 failures (skipped 14 files)\r\n\r\ntests/unit/test_commands.py E\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> traceback >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\n\r\ncls = <class 'tests.unit.test_commands.TestCommands'>\r\n\r\n    @classmethod\r\n    def setup_class(cls):\r\n        path = os.path.dirname(os.path.abspath(__file__))\r\n        configfile = os.path.join(path, 'testconf.yaml')\r\n        config = pmxbot.dictlib.ConfigDict.from_yaml(configfile)\r\n        cls.bot = core.initialize(config)\r\n>       logging.Logger.store.message(\"logged\", \"testrunner\", \"some text\")\r\nE       AttributeError: type object 'Logger' has no attribute 'store'\r\n\r\ntests/unit/test_commands.py:37: AttributeError\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> entering PDB >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\n\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> PDB post_mortem (IO-capturing turned off) >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\n> /Users/jaraco/code/pmxbot/pmxbot/tests/unit/test_commands.py(37)setup_class()\r\n-> logging.Logger.store.message(\"logged\", \"testrunner\", \"some text\")\r\n(Pdb) logging.Logger\r\n<class 'pmxbot.logging.Logger'>\r\n(Pdb) logging\r\n<module 'pmxbot.logging' from '/Users/jaraco/code/pmxbot/pmxbot/pmxbot/logging.py'>\r\n(Pdb) import sys\r\n(Pdb) sys.modules['pmxbot.logging']\r\n<module 'pmxbot.logging' from '/Users/jaraco/code/pmxbot/pmxbot/pmxbot/logging.py'>\r\n(Pdb) sys.modules['pmxbot.logging'] is logging\r\nFalse\r\n```\r\n\r\nI haven't yet made a minimal reproducer, but I wanted to first capture this condition.\r\n\n",
  "patch": "diff --git a/src/_pytest/pathlib.py b/src/_pytest/pathlib.py\n--- a/src/_pytest/pathlib.py\n+++ b/src/_pytest/pathlib.py\n@@ -523,6 +523,8 @@ def import_path(\n \n     if mode is ImportMode.importlib:\n         module_name = module_name_from_path(path, root)\n+        with contextlib.suppress(KeyError):\n+            return sys.modules[module_name]\n \n         for meta_importer in sys.meta_path:\n             spec = meta_importer.find_spec(module_name, [str(path.parent)])\n",
  "similar_bug_items": [
    {
      "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": 10988,
      "pr_title": " Consider testpaths for initial conftests",
      "pr_body": "The `testpaths` option is meant to be identical to execute pytest passing the 'testpaths' directories explicitly.\r\n\r\nThis also fixes #10169, given I had to change the same part of the code in order to get tests to pass on Python 3.7. \r\n\r\nFix #430\r\nFix #10169 \r\nFix #10987",
      "issue_id": 10169,
      "issue_title": "Pytest trying to check if custom argument is a file crashes due to filename being too long",
      "issue_body": "I have a custom flag defined in conftest.py, and when I try to assign it to a value that is too long pytest crashes before ever getting to my code. This reproduces even if the flag isn't defined, and even if the current working directory is `/`.\r\n\r\nFailing example:\r\n```bash\r\n/> pytest --xxxxx_flags=\" --xxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx\"     \r\nTraceback (most recent call last):\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/bin/pytest\", line 8, in <module>\r\n    sys.exit(console_main())\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 188, in console_main\r\n    code = main()\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 146, in main\r\n    config = _prepareconfig(args, plugins)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 325, in _prepareconfig\r\n    config = pluginmanager.hook.pytest_cmdline_parse(\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_hooks.py\", line 265, in __call__\r\n    return self._hookexec(self.name, self.get_hookimpls(), kwargs, firstresult)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_manager.py\", line 80, in _hookexec\r\n    return self._inner_hookexec(hook_name, methods, kwargs, firstresult)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_callers.py\", line 55, in _multicall\r\n    gen.send(outcome)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/helpconfig.py\", line 102, in pytest_cmdline_parse\r\n    config: Config = outcome.get_result()\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_result.py\", line 60, in get_result\r\n    raise ex[1].with_traceback(ex[2])\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_callers.py\", line 39, in _multicall\r\n    res = hook_impl.function(*args)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 1013, in pytest_cmdline_parse\r\n    self.parse(args)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 1301, in parse\r\n    self._preparse(args, addopts=addopts)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 1203, in _preparse\r\n    self.hook.pytest_load_initial_conftests(\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_hooks.py\", line 265, in __call__\r\n    return self._hookexec(self.name, self.get_hookimpls(), kwargs, firstresult)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_manager.py\", line 80, in _hookexec\r\n    return self._inner_hookexec(hook_name, methods, kwargs, firstresult)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_callers.py\", line 60, in _multicall\r\n    return outcome.get_result()\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_result.py\", line 60, in get_result\r\n    raise ex[1].with_traceback(ex[2])\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/pluggy/_callers.py\", line 39, in _multicall\r\n    res = hook_impl.function(*args)\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 1080, in pytest_load_initial_conftests\r\n    self.pluginmanager._set_initial_conftests(\r\n  File \"/home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/config/__init__.py\", line 525, in _set_initial_conftests\r\n    if anchor.exists():  # we found some file object\r\n  File \"/usr/lib/python3.8/pathlib.py\", line 1407, in exists\r\n    self.stat()\r\n  File \"/usr/lib/python3.8/pathlib.py\", line 1198, in stat\r\n    return self._accessor.stat(self)\r\nOSError: [Errno 36] File name too long: '/--xxxxx_flags= --xxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx'\r\n```\r\n\r\nIf I reduce the length of the flag, I get the expected behavior for my project, and this different and expected error from my pytest MVP:\r\n```bash\r\n/> pytest --xxxxx_flags=\" --xxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxx\"\r\n=========================================================================== test session starts ============================================================================\r\nplatform linux -- Python 3.8.10, pytest-7.0.0, pluggy-1.0.0\r\nrootdir: /\r\nplugins: flaky-3.7.0, colcon-core-0.10.0, cov-2.8.1\r\ncollected 0 items                                                                                                                                                          \r\n\r\n============================================================================= warnings summary =============================================================================\r\nhome/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/cacheprovider.py:433\r\n  /home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/cacheprovider.py:433: PytestCacheWarning: could not create cache path /.pytest_cache/v/cache/nodeids\r\n    config.cache.set(\"cache/nodeids\", sorted(self.cached_nodeids))\r\n\r\nhome/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/stepwise.py:52\r\n  /home/ANT.AMAZON.COM/jdckmz/.local/lib/python3.8/site-packages/_pytest/stepwise.py:52: PytestCacheWarning: could not create cache path /.pytest_cache/v/cache/stepwise\r\n    session.config.cache.set(STEPWISE_CACHE_DIR, [])\r\n\r\n-- Docs: https://docs.pytest.org/en/stable/how-to/capture-warnings.html\r\n=========================================================================== 2 warnings in 0.01s ============================================================================\r\nERROR: file or directory not found: --xxxxx_flags= --xxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxx --xxxxxxxxxxxxxxxxxxxxxx\r\n```\r\n\r\nI did a little digging into my version of pytest (7.0.0) to make sure I wasn't doing something wrong, but it looks like there is a blind call to `pathlib.Path.exists()` with a path constructed from the argument in `__init__.py`:\r\n```python\r\n    #\r\n    # Internal API for local conftest plugin handling.\r\n    #\r\n    def _set_initial_conftests(\r\n        self, namespace: argparse.Namespace, rootpath: Path\r\n    ) -> None:\r\n    ...\r\n    testpaths = namespace.file_or_dir\r\n    foundanchor = False\r\n    for testpath in testpaths:\r\n            path = str(testpath)\r\n            i = path.find(\"::\")\r\n            if i != -1:\r\n                path = path[:i]\r\n            anchor = absolutepath(current / path)\r\n            if anchor.exists():  # this throws OSError which is never caught\r\n```\r\nIt seems to me like there should be a try or something here, since in cases like mine the argument may not be a file at all, and that can cause OS level errors.\r\n\r\nOperating System: Ubuntu 20.04 LTS\r\n```\r\n> pytest --version\r\npytest 7.0.0\r\n> python3 --version\r\nPython 3.8.10\r\n```\r\n```\r\n> pip list\r\n/usr/lib/python3/dist-packages/secretstorage/dhcrypto.py:15: CryptographyDeprecationWarning: int_from_bytes is deprecated, use int.from_bytes instead\r\n  from cryptography.utils import int_from_bytes\r\n/usr/lib/python3/dist-packages/secretstorage/util.py:19: CryptographyDeprecationWarning: int_from_bytes is deprecated, use int.from_bytes instead\r\n  from cryptography.utils import int_from_bytes\r\nPackage                       Version\r\n----------------------------- --------------------\r\naiohttp                       3.8.1\r\naiosignal                     1.2.0\r\nalabaster                     0.7.12\r\napturl                        0.5.2\r\nargcomplete                   1.8.1\r\nastroid                       2.9.3\r\nasync-timeout                 4.0.2\r\natomicwrites                  1.4.0\r\nattrs                         21.4.0\r\nautobahn                      17.10.1\r\nAutomat                       0.8.0\r\naws-requests-auth             0.4.3\r\nawscli                        1.22.52\r\nawscrt                        0.13.0\r\nawsiotsdk                     1.9.0\r\nBabel                         2.9.1\r\nbcrypt                        3.2.0\r\nbeautifulsoup4                4.8.2\r\nblack                         22.1.0\r\nblinker                       1.4\r\nboto3                         1.20.52\r\nbotocore                      1.23.52\r\nBrlapi                        0.7.0\r\ncached-property               1.5.1\r\ncatkin-pkg-modules            0.5.2\r\ncbor                          1.0.0\r\ncertifi                       2021.10.8\r\ncffi                          1.15.0\r\nchardet                       4.0.0\r\ncharset-normalizer            2.0.11\r\nclick                         8.0.3\r\ncmakelang                     0.6.13\r\ncmakelint                     1.4.2\r\ncolcon-argcomplete            0.3.3\r\ncolcon-bash                   0.4.2\r\ncolcon-cd                     0.1.1\r\ncolcon-cmake                  0.2.26\r\ncolcon-common-extensions      0.3.0\r\ncolcon-core                   0.10.0\r\ncolcon-defaults               0.2.6\r\ncolcon-devtools               0.2.3\r\ncolcon-library-path           0.2.1\r\ncolcon-metadata               0.2.5\r\ncolcon-notification           0.2.13\r\ncolcon-output                 0.2.12\r\ncolcon-package-information    0.3.3\r\ncolcon-package-selection      0.2.10\r\ncolcon-parallel-executor      0.2.4\r\ncolcon-pkg-config             0.1.0\r\ncolcon-powershell             0.3.7\r\ncolcon-python-setup-py        0.2.7\r\ncolcon-recursive-crawl        0.2.1\r\ncolcon-ros                    0.3.23\r\ncolcon-test-result            0.3.8\r\ncolcon-zsh                    0.4.0\r\ncolorama                      0.4.3\r\ncommand-not-found             0.3\r\nconstantly                    15.1.0\r\ncontrol                       0.9.1\r\ncov-core                      1.15.0\r\ncoverage                      4.5.2\r\ncryptography                  36.0.1\r\ncupshelpers                   1.0\r\ncycler                        0.11.0\r\nCython                        0.29.14\r\ndbus-python                   1.2.16\r\ndefer                         1.0.6\r\ndistlib                       0.3.4\r\ndistro                        1.4.0\r\ndistro-info                   0.23ubuntu1\r\ndocker                        5.0.3\r\ndocker-compose                1.25.0\r\ndockerpty                     0.4.1\r\ndocopt                        0.6.2\r\ndocutils                      0.15.2\r\nduplicity                     0.8.12.0\r\nEasyCluster                   0.22.2\r\nempy                          3.3.2\r\nentrypoints                   0.3\r\nevdev                         1.3.0\r\nfasteners                     0.14.1\r\nfilelock                      3.7.1\r\nfilemagic                     1.6\r\nflake8                        3.7.9\r\nflaky                         3.7.0\r\nfonttools                     4.29.1\r\nfrozenlist                    1.3.0\r\nfuture                        0.18.2\r\ngitdb                         4.0.9\r\ngitdb2                        4.0.2\r\ngithub.py                     0.5.0\r\nGitPython                     3.1.26\r\ngpg                           1.13.1-unknown\r\ngreenlet                      1.1.2\r\nhtml5lib                      1.0.1\r\nhttplib2                      0.14.0\r\nhyperlink                     19.0.0\r\nidna                          3.3\r\nifcfg                         0.18\r\nimagesize                     1.3.0\r\nimportlib-metadata            4.10.1\r\nincremental                   16.10.1\r\ninfluxdb                      5.3.1\r\niniconfig                     1.1.1\r\nisort                         5.10.1\r\nJinja2                        3.0.3\r\njmespath                      0.10.0\r\njsonschema                    3.2.0\r\nkeyring                       18.0.1\r\nkeyrings.alt                  3.4.0\r\nkiwisolver                    1.3.2\r\nlanguage-selector             0.1\r\nlark-parser                   0.8.1\r\nlaunchpadlib                  1.10.13\r\nlazr.restfulclient            0.14.2\r\nlazr.uri                      1.0.3\r\nlazy-object-proxy             1.7.1\r\nlockfile                      0.12.2\r\nlouis                         3.12.0\r\nlxml                          4.5.0\r\nlz4                           3.0.2+dfsg\r\nmacaroonbakery                1.3.1\r\nMako                          1.1.0\r\nMarkupSafe                    2.0.1\r\nmatplotlib                    3.5.1\r\nmccabe                        0.6.1\r\nmock                          3.0.5\r\nmonotonic                     1.5\r\nmore-itertools                8.12.0\r\nmpi4py                        3.0.3\r\nmsgpack                       1.0.3\r\nmulti-key-dict                2.0.3\r\nmultidict                     6.0.2\r\nmypy-extensions               0.4.3\r\nnetifaces                     0.10.4\r\nnose2                         0.9.1\r\nnotify2                       0.3\r\nnumpy                         1.22.2\r\noauthlib                      3.1.0\r\nolefile                       0.46\r\npackaging                     21.3\r\npandas                        1.4.0\r\nparamiko                      2.9.2\r\npathspec                      0.9.0\r\npbr                           5.8.1\r\npexpect                       4.8.0\r\nPillow                        9.0.1\r\npip                           22.1.2\r\npipenv                        2022.6.7\r\nplatformdirs                  2.5.0\r\npluggy                        1.0.0\r\nprotobuf                      3.19.4\r\npsutil                        5.8.0\r\nptyprocess                    0.7.0\r\npy                            1.11.0\r\npy-ubjson                     0.14.0\r\npyasn1                        0.4.8\r\npyasn1-modules                0.2.1\r\npybind11                      2.8.0\r\npycairo                       1.16.2\r\npycodestyle                   2.8.0\r\npycparser                     2.21\r\npycrypto                      2.6.1\r\npycups                        1.9.73\r\npydocstyle                    2.1.1\r\npydot                         1.4.1\r\npyelftools                    0.28\r\npyflakes                      2.1.1\r\nPygments                      2.11.2\r\nPyGObject                     3.36.0\r\nPyHamcrest                    1.9.0\r\nPyJWT                         1.7.1\r\npylint                        2.12.2\r\npymacaroons                   0.13.0\r\nPyNaCl                        1.5.0\r\npyOpenSSL                     19.0.0\r\npyparsing                     3.0.7\r\npypng                         0.0.20\r\nPyQRCode                      1.2.1\r\nPyQt5                         5.14.1\r\npyquaternion                  0.9.9\r\npyRFC3339                     1.1\r\npyrsistent                    0.15.5\r\npyserial                      3.5\r\npytest                        7.0.0\r\npytest-cov                    2.8.1\r\npython-apt                    2.0.0+ubuntu0.20.4.7\r\npython-dateutil               2.8.2\r\npython-debian                 0.1.36ubuntu1\r\npython-dotenv                 0.19.2\r\npython-jenkins                1.7.0\r\npython-magic                  0.4.16\r\npython-snappy                 0.5.3\r\nPyTrie                        0.2\r\npytz                          2021.3\r\npyxdg                         0.26\r\nPyYAML                        5.3.1\r\nreportlab                     3.5.34\r\nrequests                      2.27.1\r\nrequests-unixsocket           0.2.0\r\nroman                         2.0.0\r\nrosdistro-modules             0.9.0\r\nrospkg-modules                1.4.0\r\nrplidar                       0.9.2\r\nrsa                           4.7.2\r\ns3transfer                    0.5.1\r\nscipy                         1.8.0\r\nscreen-resolution-extra       0.0.0\r\nSecretStorage                 2.3.1\r\nservice-identity              18.1.0\r\nsetproctitle                  1.1.10\r\nsetuptools                    45.2.0\r\nsimplejson                    3.16.0\r\nsip                           4.19.21\r\nsix                           1.16.0\r\nsmmap                         5.0.0\r\nsmmap2                        3.0.1\r\nsnowballstemmer               2.2.0\r\nsoupsieve                     1.9.5\r\nSphinx                        4.4.0\r\nsphinx-autoapi                1.8.4\r\nsphinxcontrib-applehelp       1.0.2\r\nsphinxcontrib-devhelp         1.0.2\r\nsphinxcontrib-dotnetdomain    0.4\r\nsphinxcontrib-golangdomain    0.2.0.dev0\r\nsphinxcontrib-htmlhelp        2.0.0\r\nsphinxcontrib-jsmath          1.0.1\r\nsphinxcontrib-qthelp          1.0.3\r\nsphinxcontrib-serializinghtml 1.1.5\r\nsphinxcontrib-websupport      1.2.4\r\nSQLAlchemy                    1.4.35\r\nssh-import-id                 5.10\r\ntensorrt                      8.0.1.6\r\ntexttable                     1.6.2\r\ntoml                          0.10.2\r\ntomli                         2.0.1\r\ntripy                         1.0.0\r\nTwisted                       18.9.0\r\ntxaio                         2.10.0\r\ntyped-ast                     1.5.2\r\ntyping_extensions             4.0.1\r\nu-msgpack-python              2.1\r\nubuntu-advantage-tools        27.9\r\nubuntu-drivers-common         0.0.0\r\nufw                           0.36\r\nunattended-upgrades           0.1\r\nUnidecode                     1.3.2\r\nurllib3                       1.26.8\r\nusb-creator                   0.3.7\r\nvirtualenv                    20.14.1\r\nvirtualenv-clone              0.5.7\r\nwadllib                       1.3.3\r\nwcwidth                       0.1.8\r\nwebencodings                  0.5.1\r\nwebsocket-client              1.2.3\r\nwheel                         0.34.2\r\nwrapt                         1.13.3\r\nwsaccel                       0.6.2\r\nxdot                          1.1\r\nxkit                          0.0.0\r\nxmltodict                     0.12.0\r\nyarl                          1.7.2\r\nzipp                          3.7.0\r\nzope.interface                4.7.1\r\nzstandard                     0.17.0\r\n```\r\n- [x] a detailed description of the bug or problem you are having\r\n- [x] output of `pip list` from the virtual environment you are using\r\n- [x] pytest and operating system versions\r\n- [x] minimal example if possible\r\n",
      "issue_closed_at": "2023-05-12T12:59:01Z",
      "base_commit": "78403237cf5026f23618ea7a867bf8b674116e6f",
      "changes": [
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "pytest_configure",
          "class_name": "PytestPluginManager",
          "code": "def pytest_configure(self, config: \"Config\") -> None:\n        \"\"\":meta private:\"\"\"\n        # XXX now that the pluginmanager exposes hookimpl(tryfirst...)\n        # we should remove tryfirst/trylast as markers.\n        config.addinivalue_line(\n            \"markers\",\n            \"tryfirst: mark a hook implementation function such that the \"\n            \"plugin machinery will try to call it first/as early as possible. \"\n            \"DEPRECATED, use @pytest.hookimpl(tryfirst=True) instead.\",\n        )\n        config.addinivalue_line(\n            \"markers\",\n            \"trylast: mark a hook implementation function such that the \"\n            \"plugin machinery will try to call it last/as late as possible. \"\n            \"DEPRECATED, use @pytest.hookimpl(trylast=True) instead.\",\n        )\n        self._configured = True"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "_set_initial_conftests",
          "class_name": "PytestPluginManager",
          "code": "def _set_initial_conftests(\n        self, namespace: argparse.Namespace, rootpath: Path\n    ) -> None:\n        \"\"\"Load initial conftest files given a preparsed \"namespace\".\n\n        As conftest files may add their own command line options which have\n        arguments ('--my-opt somepath') we might get some false positives.\n        All builtin and 3rd party plugins will have been loaded, however, so\n        common options will not confuse our logic here.\n        \"\"\"\n        current = Path.cwd()\n        self._confcutdir = (\n            absolutepath(current / namespace.confcutdir)\n            if namespace.confcutdir\n            else None\n        )\n        self._noconftest = namespace.noconftest\n        self._using_pyargs = namespace.pyargs\n        testpaths = namespace.file_or_dir\n        foundanchor = False\n        for testpath in testpaths:\n            path = str(testpath)\n            # remove node-id syntax\n            i = path.find(\"::\")\n            if i != -1:\n                path = path[:i]\n            anchor = absolutepath(current / path)\n            if anchor.exists():  # we found some file object\n                self._try_load_conftest(anchor, namespace.importmode, rootpath)\n                foundanchor = True\n        if not foundanchor:\n            self._try_load_conftest(current, namespace.importmode, rootpath)"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "_set_initial_conftests",
          "class_name": "PytestPluginManager",
          "code": "def _set_initial_conftests(\n        self, namespace: argparse.Namespace, rootpath: Path\n    ) -> None:\n        \"\"\"Load initial conftest files given a preparsed \"namespace\".\n\n        As conftest files may add their own command line options which have\n        arguments ('--my-opt somepath') we might get some false positives.\n        All builtin and 3rd party plugins will have been loaded, however, so\n        common options will not confuse our logic here.\n        \"\"\"\n        current = Path.cwd()\n        self._confcutdir = (\n            absolutepath(current / namespace.confcutdir)\n            if namespace.confcutdir\n            else None\n        )\n        self._noconftest = namespace.noconftest\n        self._using_pyargs = namespace.pyargs\n        testpaths = namespace.file_or_dir\n        foundanchor = False\n        for testpath in testpaths:\n            path = str(testpath)\n            # remove node-id syntax\n            i = path.find(\"::\")\n            if i != -1:\n                path = path[:i]\n            anchor = absolutepath(current / path)\n            if anchor.exists():  # we found some file object\n                self._try_load_conftest(anchor, namespace.importmode, rootpath)\n                foundanchor = True\n        if not foundanchor:\n            self._try_load_conftest(current, namespace.importmode, rootpath)"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "_processopt",
          "class_name": null,
          "code": "def _processopt(self, opt: \"Argument\") -> None:\n        for name in opt._short_opts + opt._long_opts:\n            self._opt2dest[name] = opt.dest\n\n        if hasattr(opt, \"default\"):\n            if not hasattr(self.option, opt.dest):\n                setattr(self.option, opt.dest, opt.default)"
        }
      ]
    },
    {
      "pr_number": 4041,
      "pr_title": "Ensure user_properties is a list",
      "pr_body": "Closes #4034.",
      "issue_id": 4034,
      "issue_title": "TestReport.user_properties is documented as a list, actually a tuple",
      "issue_body": "https://github.com/pytest-dev/pytest/blob/253419316ce09b952862811db229a9e9383a332b/src/_pytest/reports.py#L137-L139\r\n\r\nThis makes me nervous when using it in plugins, e.g. to pass Hypothesis statistics around in HypothesisWorks/hypothesis#1577.  My code handles either a list or a tuple here, but it would be nice to - one way or another - make the code and the documentation consistent.",
      "issue_closed_at": "2018-09-27T11:04:30Z",
      "base_commit": "b1fbb2ab9215f1440d46e998ce06818ee6e847ab",
      "changes": [
        {
          "file": "src/_pytest/reports.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CollectErrorRepr",
          "code": "def __init__(self, msg):\n        self.longrepr = msg"
        },
        {
          "file": "src/_pytest/reports.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CollectErrorRepr",
          "code": "def __init__(self, msg):\n        self.longrepr = msg"
        }
      ]
    },
    {
      "pr_number": 4749,
      "pr_title": "Merge master into features",
      "pr_body": "",
      "issue_id": 2895,
      "issue_title": "pytest_report_collectionfinish does not execute with --collect-only",
      "issue_body": "As commented [here](https://github.com/pytest-dev/pytest/pull/2623#issuecomment-341478287).",
      "issue_closed_at": "2019-02-06T22:38:12Z",
      "base_commit": "4cd268dc5d19a5ac362785d3068229559e69ddd7",
      "changes": [
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "_get_plugin_specs_as_list",
          "class_name": null,
          "code": "def _get_plugin_specs_as_list(specs):\n    \"\"\"\n    Parses a list of \"plugin specs\" and returns a list of plugin names.\n\n    Plugin specs can be given as a list of strings separated by \",\" or already as a list/tuple in\n    which case it is returned as a list. Specs can also be `None` in which case an\n    empty list is returned.\n    \"\"\"\n    if specs is not None:\n        if isinstance(specs, str):\n            specs = specs.split(\",\") if specs else []\n        if not isinstance(specs, (list, tuple)):\n            raise UsageError(\n                \"Plugin specs must be a ','-separated string or a \"\n                \"list/tuple of strings for plugin names. Given: %r\" % specs\n            )\n        return list(specs)\n    return []"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "get_actual_log_level",
          "class_name": null,
          "code": "def get_actual_log_level(config, *setting_names):\n    \"\"\"Return the actual logging level.\"\"\"\n\n    for setting_name in setting_names:\n        log_level = config.getoption(setting_name)\n        if log_level is None:\n            log_level = config.getini(setting_name)\n        if log_level:\n            break\n    else:\n        return\n\n    if isinstance(log_level, six.string_types):\n        log_level = log_level.upper()\n    try:\n        return int(getattr(logging, log_level, log_level))\n    except ValueError:\n        # Python logging does not recognise this as a logging level\n        raise pytest.UsageError(\n            \"'{}' is not recognized as a logging level name for \"\n            \"'{}'. Please consider passing the \"\n            \"logging level num instead.\".format(log_level, setting_name)\n        )"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "__init__",
          "class_name": "_LiveLoggingStreamHandler",
          "code": "def __init__(self, terminal_reporter, capture_manager):\n        \"\"\"\n        :param _pytest.terminal.TerminalReporter terminal_reporter:\n        :param _pytest.capture.CaptureManager capture_manager:\n        \"\"\"\n        logging.StreamHandler.__init__(self, stream=terminal_reporter)\n        self.capture_manager = capture_manager\n        self.reset()\n        self.set_when(None)\n        self._test_outcome_written = False"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "__init__",
          "class_name": "_LiveLoggingStreamHandler",
          "code": "def __init__(self, terminal_reporter, capture_manager):\n        \"\"\"\n        :param _pytest.terminal.TerminalReporter terminal_reporter:\n        :param _pytest.capture.CaptureManager capture_manager:\n        \"\"\"\n        logging.StreamHandler.__init__(self, stream=terminal_reporter)\n        self.capture_manager = capture_manager\n        self.reset()\n        self.set_when(None)\n        self._test_outcome_written = False"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "_log_cli_enabled",
          "class_name": "LoggingPlugin",
          "code": "def _log_cli_enabled(self):\n        \"\"\"Return True if log_cli should be considered enabled, either explicitly\n        or because --log-cli-level was given in the command-line.\n        \"\"\"\n        return self._config.getoption(\n            \"--log-cli-level\"\n        ) is not None or self._config.getini(\"log_cli\")"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "pytest_sessionfinish",
          "class_name": "LoggingPlugin",
          "code": "def pytest_sessionfinish(self):\n        with self.live_logs_context():\n            if self.log_cli_handler:\n                self.log_cli_handler.set_when(\"sessionfinish\")\n            if self.log_file_handler is not None:\n                with catching_logs(self.log_file_handler, level=self.log_file_level):\n                    yield\n            else:\n                yield"
        },
        {
          "file": "src/_pytest/logging.py",
          "type": "function",
          "name": "pytest_runtestloop",
          "class_name": "LoggingPlugin",
          "code": "def pytest_runtestloop(self, session):\n        \"\"\"Runs all collected test items.\"\"\"\n        with self.live_logs_context():\n            if self.log_file_handler is not None:\n                with catching_logs(self.log_file_handler, level=self.log_file_level):\n                    yield  # run all the tests\n            else:\n                yield"
        },
        {
          "file": "src/_pytest/main.py",
          "type": "function",
          "name": "filter_",
          "class_name": "Session",
          "code": "def filter_(f):\n                    return f.check(file=1)"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "gethookproxy",
          "class_name": "Package",
          "code": "def gethookproxy(self, fspath):\n        # check if we have the common case of running\n        # hooks with all conftest.py filesall conftest.py\n        pm = self.config.pluginmanager\n        my_conftestmodules = pm._getconftestmodules(fspath)\n        remove_mods = pm._conftest_plugins.difference(my_conftestmodules)\n        if remove_mods:\n            # one or more conftests are not in use at this fspath\n            proxy = FSHookProxy(fspath, pm, remove_mods)\n        else:\n            # all plugis are active for this fspath\n            proxy = self.config.hook\n        return proxy"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "collect",
          "class_name": "Instance",
          "code": "def collect(self):\n        self.session._fixturemanager.parsefactories(self)\n        return super(Instance, self).collect()"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "_find_parametrized_scope",
          "class_name": null,
          "code": "def _find_parametrized_scope(argnames, arg2fixturedefs, indirect):\n    \"\"\"Find the most appropriate scope for a parametrized call based on its arguments.\n\n    When there's at least one direct argument, always use \"function\" scope.\n\n    When a test function is parametrized and all its arguments are indirect\n    (e.g. fixtures), return the most narrow scope based on the fixtures used.\n\n    Related to issue #1832, based on code posted by @Kingdread.\n    \"\"\"\n    from _pytest.fixtures import scopes\n\n    if isinstance(indirect, (list, tuple)):\n        all_arguments_are_fixtures = len(indirect) == len(argnames)\n    else:\n        all_arguments_are_fixtures = bool(indirect)\n\n    if all_arguments_are_fixtures:\n        fixturedefs = arg2fixturedefs or {}\n        used_scopes = [\n            fixturedef[0].scope\n            for name, fixturedef in fixturedefs.items()\n            if name in argnames\n        ]\n        if used_scopes:\n            # Takes the most narrow scope from used fixtures\n            for scope in reversed(scopes):\n                if scope in used_scopes:\n                    return scope\n\n    return \"function\""
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "_idval",
          "class_name": null,
          "code": "def _idval(val, argname, idx, idfn, item, config):\n    if idfn:\n        s = None\n        try:\n            s = idfn(val)\n        except Exception as e:\n            # See issue https://github.com/pytest-dev/pytest/issues/2169\n            msg = \"{}: error raised while trying to determine id of parameter '{}' at position {}\\n\"\n            msg = msg.format(item.nodeid, argname, idx)\n            # we only append the exception type and message because on Python 2 reraise does nothing\n            msg += \"  {}: {}\\n\".format(type(e).__name__, e)\n            six.raise_from(ValueError(msg), e)\n        if s:\n            return ascii_escaped(s)\n\n    if config:\n        hook_id = config.hook.pytest_make_parametrize_id(\n            config=config, val=val, argname=argname\n        )\n        if hook_id:\n            return hook_id\n\n    if isinstance(val, STRING_TYPES):\n        return ascii_escaped(val)\n    elif isinstance(val, (float, int, bool, NoneType)):\n        return str(val)\n    elif isinstance(val, REGEX_TYPE):\n        return ascii_escaped(val.pattern)\n    elif enum is not None and isinstance(val, enum.Enum):\n        return str(val)\n    elif (isclass(val) or isfunction(val)) and hasattr(val, \"__name__\"):\n        return val.__name__\n    return str(argname) + str(idx)"
        },
        {
          "file": "src/_pytest/python_api.py",
          "type": "function",
          "name": "raises",
          "class_name": null,
          "code": "def raises(expected_exception, *args, **kwargs):\n    r\"\"\"\n    Assert that a code block/function call raises ``expected_exception``\n    or raise a failure exception otherwise.\n\n    :kwparam match: if specified, asserts that the exception matches a text or regex\n\n    :kwparam message: **(deprecated since 4.1)** if specified, provides a custom failure message\n        if the exception is not raised\n\n    .. currentmodule:: _pytest._code\n\n    Use ``pytest.raises`` as a context manager, which will capture the exception of the given\n    type::\n\n        >>> with raises(ZeroDivisionError):\n        ...    1/0\n\n    If the code block does not raise the expected exception (``ZeroDivisionError`` in the example\n    above), or no exception at all, the check will fail instead.\n\n    You can also use the keyword argument ``match`` to assert that the\n    exception matches a text or regex::\n\n        >>> with raises(ValueError, match='must be 0 or None'):\n        ...     raise ValueError(\"value must be 0 or None\")\n\n        >>> with raises(ValueError, match=r'must be \\d+$'):\n        ...     raise ValueError(\"value must be 42\")\n\n    The context manager produces an :class:`ExceptionInfo` object which can be used to inspect the\n    details of the captured exception::\n\n        >>> with raises(ValueError) as exc_info:\n        ...     raise ValueError(\"value must be 42\")\n        >>> assert exc_info.type is ValueError\n        >>> assert exc_info.value.args[0] == \"value must be 42\"\n\n    .. deprecated:: 4.1\n\n        In the context manager form you may use the keyword argument\n        ``message`` to specify a custom failure message that will be displayed\n        in case the ``pytest.raises`` check fails. This has been deprecated as it\n        is considered error prone as users often mean to use ``match`` instead.\n\n    .. note::\n\n       When using ``pytest.raises`` as a context manager, it's worthwhile to\n       note that normal context manager rules apply and that the exception\n       raised *must* be the final line in the scope of the context manager.\n       Lines of code after that, within the scope of the context manager will\n       not be executed. For example::\n\n           >>> value = 15\n           >>> with raises(ValueError) as exc_info:\n           ...     if value > 10:\n           ...         raise ValueError(\"value must be <= 10\")\n           ...     assert exc_info.type is ValueError  # this will not execute\n\n       Instead, the following approach must be taken (note the difference in\n       scope)::\n\n           >>> with raises(ValueError) as exc_info:\n           ...     if value > 10:\n           ...         raise ValueError(\"value must be <= 10\")\n           ...\n           >>> assert exc_info.type is ValueError\n\n    **Legacy form**\n\n    It is possible to specify a callable by passing a to-be-called lambda::\n\n        >>> raises(ZeroDivisionError, lambda: 1/0)\n        <ExceptionInfo ...>\n\n    or you can specify an arbitrary callable with arguments::\n\n        >>> def f(x): return 1/x\n        ...\n        >>> raises(ZeroDivisionError, f, 0)\n        <ExceptionInfo ...>\n        >>> raises(ZeroDivisionError, f, x=0)\n        <ExceptionInfo ...>\n\n    The form above is fully supported but discouraged for new code because the\n    context manager form is regarded as more readable and less error-prone.\n\n    .. note::\n        Similar to caught exception objects in Python, explicitly clearing\n        local references to returned ``ExceptionInfo`` objects can\n        help the Python interpreter speed up its garbage collection.\n\n        Clearing those references breaks a reference cycle\n        (``ExceptionInfo`` --> caught exception --> frame stack raising\n        the exception --> current frame stack --> local variables -->\n        ``ExceptionInfo``) which makes Python keep all objects referenced\n        from that cycle (including all local variables in the current\n        frame) alive until the next cyclic garbage collection run. See the\n        official Python ``try`` statement documentation for more detailed\n        information.\n\n    \"\"\"\n    __tracebackhide__ = True\n    for exc in filterfalse(isclass, always_iterable(expected_exception, BASE_TYPE)):\n        msg = (\n            \"exceptions must be old-style classes or\"\n            \" derived from BaseException, not %s\"\n        )\n        raise TypeError(msg % type(exc))\n\n    message = \"DID NOT RAISE {}\".format(expected_exception)\n    match_expr = None\n\n    if not args:\n        if \"message\" in kwargs:\n            message = kwargs.pop(\"message\")\n            warnings.warn(deprecated.RAISES_MESSAGE_PARAMETER, stacklevel=2)\n        if \"match\" in kwargs:\n            match_expr = kwargs.pop(\"match\")\n        if kwargs:\n            msg = \"Unexpected keyword arguments passed to pytest.raises: \"\n            msg += \", \".join(kwargs.keys())\n            raise TypeError(msg)\n        return RaisesContext(expected_exception, message, match_expr)\n    elif isinstance(args[0], str):\n        warnings.warn(deprecated.RAISES_EXEC, stacklevel=2)\n        code, = args\n        assert isinstance(code, str)\n        frame = sys._getframe(1)\n        loc = frame.f_locals.copy()\n        loc.update(kwargs)\n        # print \"raises frame scope: %r\" % frame.f_locals\n        try:\n            code = _pytest._code.Source(code).compile(_genframe=frame)\n            six.exec_(code, frame.f_globals, loc)\n            # XXX didn't mean f_globals == f_locals something special?\n            #     this is destroyed here ...\n        except expected_exception:\n            return _pytest._code.ExceptionInfo.from_current()\n    else:\n        func = args[0]\n        try:\n            func(*args[1:], **kwargs)\n        except expected_exception:\n            return _pytest._code.ExceptionInfo.from_current()\n    fail(message)"
        },
        {
          "file": "src/_pytest/terminal.py",
          "type": "function",
          "name": "pytest_report_header",
          "class_name": "TerminalReporter",
          "code": "def pytest_report_header(self, config):\n        inifile = \"\"\n        if config.inifile:\n            inifile = \" \" + config.rootdir.bestrelpath(config.inifile)\n        lines = [\"rootdir: %s, inifile:%s\" % (config.rootdir, inifile)]\n\n        plugininfo = config.pluginmanager.list_plugin_distinfo()\n        if plugininfo:\n\n            lines.append(\"plugins: %s\" % \", \".join(_plugin_nameversions(plugininfo)))\n        return lines"
        },
        {
          "file": "src/_pytest/unittest.py",
          "type": "function",
          "name": "_make_xunit_fixture",
          "class_name": null,
          "code": "def _make_xunit_fixture(obj, setup_name, teardown_name, scope, pass_self):\n    setup = getattr(obj, setup_name, None)\n    teardown = getattr(obj, teardown_name, None)\n    if setup is None and teardown is None:\n        return None\n\n    @pytest.fixture(scope=scope, autouse=True)\n    def fixture(self, request):\n        if setup is not None:\n            if pass_self:\n                setup(self, request.function)\n            else:\n                setup()\n        yield\n        if teardown is not None:\n            if pass_self:\n                teardown(self, request.function)\n            else:\n                teardown()\n\n    return fixture"
        }
      ]
    },
    {
      "pr_number": 1734,
      "pr_title": "setup_* and teardown_* functions argument now optional",
      "pr_body": "setup_module, setup_function and setup_method extra argument are now optional and may be omitted.\n\nFix #1728\n",
      "issue_id": 1728,
      "issue_title": "teardown_function() with no argument list crashes",
      "issue_body": "- [x] Include a detailed description of the bug or suggestion\n\nThese functions appear to work even if someone forgets to list the module/function argument:\n- setup_module()\n- teardown_module()\n- setup_function()\n\nHowever, this crashes:\n- teardown_function()\n\nExample file test_foo.py, listing at end.\n\n```\n> py.test test_foo.py\n<some output deleted>\n>   self.addfinalizer(lambda: fin(self.obj))\nE   TypeError: teardown_function() takes no arguments (1 given)\n<some output deleted>\n```\n\nSuggestion. look at how setup/teardown module and setup_function are handled, because they don't crash.\n- [ ] `pip list` of the virtual environment you are using\n\nCurrently looking at an anaconda install, and the list is huge.\n- [x] py.test and operating system versions\n\n> python --version\n> Python 2.7.10 :: Anaconda 2.4.0 (64-bit)\n> py.test --version\n> This is pytest version 2.8.1, imported from C:\\Python\\lib\\site-packages\\pytest.pyc\n\nHowever, I've tried it with Python 3.5.2 and most recent pytest on a mac without anaconda, and that's broken too.\n- [x] Minimal example if possible\n\ntest_foo.py:\n\n``` python\nfrom __future__ import print_function\n\ndef test_something():\n    print('\\nIn test_something()')\n\ndef teardown_function():\n    print('\\nIn teardown_function()')\n```\n",
      "issue_closed_at": "2016-07-15T13:00:42Z",
      "base_commit": "ee374e3b8059351719a85fa3ba7abd5ec6b632ea",
      "changes": [
        {
          "file": "_pytest/python.py",
          "type": "line",
          "name": "line 25",
          "code": "cutdir1 = py.path.local(pluggy.__file__.rstrip(\"oc\"))\n\n\ndef _has_positional_arg(func):\n    return func.__code__.co_argcount\n\n\ndef filter_traceback(entry):\n    # entry.path might sometimes return a str object when the entry\n    # points to dynamically generated code"
        },
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "_importtestmodule",
          "class_name": "Module",
          "code": "def _importtestmodule(self):\n        # we assume we are only called once per module\n        importmode = self.config.getoption(\"--import-mode\")\n        try:\n            mod = self.fspath.pyimport(ensuresyspath=importmode)\n        except SyntaxError:\n            raise self.CollectError(\n                _pytest._code.ExceptionInfo().getrepr(style=\"short\"))\n        except self.fspath.ImportMismatchError:\n            e = sys.exc_info()[1]\n            raise self.CollectError(\n                \"import file mismatch:\\n\"\n                \"imported module %r has this __file__ attribute:\\n\"\n                \"  %s\\n\"\n                \"which is not the same as the test file we want to collect:\\n\"\n                \"  %s\\n\"\n                \"HINT: remove __pycache__ / .pyc files and/or use a \"\n                \"unique basename for your test file modules\"\n                 % e.args\n            )\n        except ImportError:\n            exc_class, exc, _ = sys.exc_info()\n            raise self.CollectError(\n                \"ImportError while importing test module '%s'.\\n\"\n                \"Original error message:\\n'%s'\\n\"\n                \"Make sure your test modules/packages have valid Python names.\"\n                % (self.fspath, exc or exc_class)\n            )\n        except _pytest.runner.Skipped:\n            raise self.CollectError(\n                \"Using @pytest.skip outside a test (e.g. as a test function \"\n                \"decorator) is not allowed. Use @pytest.mark.skip or \"\n                \"@pytest.mark.skipif instead.\"\n            )\n        #print \"imported test module\", mod\n        self.config.pluginmanager.consider_module(mod)\n        return mod"
        },
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "collect",
          "class_name": "Generator",
          "code": "def collect(self):\n        # test generators are seen as collectors but they also\n        # invoke setup/teardown on popular request\n        # (induced by the common \"test_*\" naming shared with normal tests)\n        self.session._setupstate.prepare(self)\n        # see FunctionMixin.setup and test_setupstate_is_preserved_134\n        self._preservedparent = self.parent.obj\n        l = []\n        seen = {}\n        for i, x in enumerate(self.obj()):\n            name, call, args = self.getcallargs(x)\n            if not callable(call):\n                raise TypeError(\"%r yielded non callable test %r\" %(self.obj, call,))\n            if name is None:\n                name = \"[%d]\" % i\n            else:\n                name = \"['%s']\" % name\n            if name in seen:\n                raise ValueError(\"%r generated tests with non-unique name %r\" %(self, name))\n            seen[name] = True\n            l.append(self.Function(name, self, args=args, callobj=call))\n            msg = 'yield tests are deprecated, and scheduled to be removed in pytest 4.0'\n            self.config.warn('C1', msg, fslocation=self.fspath)\n        return l"
        },
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "setup",
          "class_name": "Function",
          "code": "def setup(self):\n        super(Function, self).setup()\n        fixtures.fillfixtures(self)"
        },
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "setup",
          "class_name": "Function",
          "code": "def setup(self):\n        super(Function, self).setup()\n        fixtures.fillfixtures(self)"
        }
      ]
    }
  ]
}