Commit 47408ac8 authored by Jason R. Coombs's avatar Jason R. Coombs Committed by GitHub

Merge pull request #2459 from webknjaz/testing/pytest-xdist

Parallelize the test runs via pytest-xdist
parents 347f7497 3b571b08
Tests now run in parallel via pytest-xdist, completing in about half the time. Special thanks to :user:`webknjaz` for hard work implementing test isolation. To run without parallelization, disable the plugin with ``tox -- -p no:xdist``.
...@@ -24,6 +24,9 @@ addopts = "--flake8" ...@@ -24,6 +24,9 @@ addopts = "--flake8"
[pytest.enabler.cov] [pytest.enabler.cov]
addopts = "--cov" addopts = "--cov"
[pytest.enabler.xdist]
addopts = "-n auto"
[tool.towncrier] [tool.towncrier]
package = "setuptools" package = "setuptools"
package_dir = "setuptools" package_dir = "setuptools"
......
...@@ -47,7 +47,7 @@ testing = ...@@ -47,7 +47,7 @@ testing =
pytest-black >= 0.3.7; python_implementation != "PyPy" pytest-black >= 0.3.7; python_implementation != "PyPy"
pytest-cov pytest-cov
pytest-mypy; python_implementation != "PyPy" pytest-mypy; python_implementation != "PyPy"
pytest-enabler pytest-enabler >= 1.0.1
# local # local
mock mock
...@@ -58,6 +58,7 @@ testing = ...@@ -58,6 +58,7 @@ testing =
paver paver
pip>=19.1 # For proper file:// URLs support. pip>=19.1 # For proper file:// URLs support.
jaraco.envs jaraco.envs
pytest-xdist
docs = docs =
# Keep these in sync with docs/requirements.txt # Keep these in sync with docs/requirements.txt
......
import contextlib
import sys
import shutil
import pytest import pytest
from . import contexts from . import contexts
...@@ -21,3 +25,36 @@ def user_override(monkeypatch): ...@@ -21,3 +25,36 @@ def user_override(monkeypatch):
def tmpdir_cwd(tmpdir): def tmpdir_cwd(tmpdir):
with tmpdir.as_cwd() as orig: with tmpdir.as_cwd() as orig:
yield orig yield orig
@pytest.fixture
def tmp_src(request, tmp_path):
"""Make a copy of the source dir under `$tmp/src`.
This fixture is useful whenever it's necessary to run `setup.py`
or `pip install` against the source directory when there's no
control over the number of simultaneous invocations. Such
concurrent runs create and delete directories with the same names
under the target directory and so they influence each other's runs
when they are not being executed sequentially.
"""
tmp_src_path = tmp_path / 'src'
shutil.copytree(request.config.rootdir, tmp_src_path)
return tmp_src_path
@pytest.fixture(autouse=True, scope="session")
def workaround_xdist_376(request):
"""
Workaround pytest-dev/pytest-xdist#376
``pytest-xdist`` tends to inject '' into ``sys.path``,
which may break certain isolation expectations.
Remove the entry so the import
machinery behaves the same irrespective of xdist.
"""
if not request.config.pluginmanager.has_plugin('xdist'):
return
with contextlib.suppress(ValueError):
sys.path.remove('')
...@@ -21,10 +21,10 @@ class VirtualEnv(jaraco.envs.VirtualEnv): ...@@ -21,10 +21,10 @@ class VirtualEnv(jaraco.envs.VirtualEnv):
@pytest.fixture @pytest.fixture
def venv(tmpdir): def venv(tmp_path, tmp_src):
env = VirtualEnv() env = VirtualEnv()
env.root = path.Path(tmpdir) env.root = path.Path(tmp_path / 'venv')
env.req = os.getcwd() env.req = str(tmp_src)
return env.create() return env.create()
......
...@@ -43,11 +43,11 @@ def bare_virtualenv(): ...@@ -43,11 +43,11 @@ def bare_virtualenv():
SOURCE_DIR = os.path.join(os.path.dirname(__file__), '../..') SOURCE_DIR = os.path.join(os.path.dirname(__file__), '../..')
def test_clean_env_install(bare_virtualenv): def test_clean_env_install(bare_virtualenv, tmp_src):
""" """
Check setuptools can be installed in a clean environment. Check setuptools can be installed in a clean environment.
""" """
bare_virtualenv.run(['python', 'setup.py', 'install'], cd=SOURCE_DIR) bare_virtualenv.run(['python', 'setup.py', 'install'], cd=tmp_src)
def _get_pip_versions(): def _get_pip_versions():
...@@ -85,7 +85,7 @@ def _get_pip_versions(): ...@@ -85,7 +85,7 @@ def _get_pip_versions():
@pytest.mark.parametrize('pip_version', _get_pip_versions()) @pytest.mark.parametrize('pip_version', _get_pip_versions())
def test_pip_upgrade_from_source(pip_version, virtualenv): def test_pip_upgrade_from_source(pip_version, tmp_src, virtualenv):
""" """
Check pip can upgrade setuptools from source. Check pip can upgrade setuptools from source.
""" """
...@@ -104,7 +104,7 @@ def test_pip_upgrade_from_source(pip_version, virtualenv): ...@@ -104,7 +104,7 @@ def test_pip_upgrade_from_source(pip_version, virtualenv):
virtualenv.run(' && '.join(( virtualenv.run(' && '.join((
'python setup.py -q sdist -d {dist}', 'python setup.py -q sdist -d {dist}',
'python setup.py -q bdist_wheel -d {dist}', 'python setup.py -q bdist_wheel -d {dist}',
)).format(dist=dist_dir), cd=SOURCE_DIR) )).format(dist=dist_dir), cd=tmp_src)
sdist = glob.glob(os.path.join(dist_dir, '*.zip'))[0] sdist = glob.glob(os.path.join(dist_dir, '*.zip'))[0]
wheel = glob.glob(os.path.join(dist_dir, '*.whl'))[0] wheel = glob.glob(os.path.join(dist_dir, '*.whl'))[0]
# Then update from wheel. # Then update from wheel.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment