Skip to content

Commit 2ad73dd

Browse files
committed
Paralelize the integration tests
This is an attempt to fix fedora-python#45 It does the following: * uses pytest-xdist to paralelize the tests (3 workers) * utilizes custom scheduler that splits the tests acroding to the fixture name * this is needed not to use 1 fixture on multiple workers see pytest-dev/pytest-xdist#18 * use parametrize an all integration tests to enbale our hackish scheduler * mock now creates the roots in pwd (not to pollute the filesystem on /) * note that the roots can take sveral GBs
1 parent 1081859 commit 2ad73dd

File tree

5 files changed

+86
-29
lines changed

5 files changed

+86
-29
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ __pycache__
66
/artifacts-*/
77
/build/
88
/dist/
9+
/mockroots/
910
/.tox/
1011
/.eggs/
1112
.cache

mock.cfg

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,5 @@ config_opts['chroot_setup_cmd'] = 'install ansible dnf'
44
config_opts['use_host_resolv'] = True
55
config_opts['rpmbuild_networking'] = True
66
config_opts['use_nspawn'] = False
7-
config_opts['root'] = 'fedora-27-x86_64-taskotron'
7+
config_opts['root'] = 'taskotron-python-versions-master'
8+
config_opts['plugin_conf']['root_cache_opts']['dir'] = "%(cache_topdir)s/taskotron-python-versions/root_cache/"

test/integration/conftest.py

+18
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,22 @@
1+
from xdist.scheduler import LoadScopeScheduling
2+
3+
14
def pytest_addoption(parser):
25
parser.addoption('--fake', action='store_true', default=False,
36
help='don\'t run the code, reuse the result from '
47
'last tests')
8+
9+
10+
class FixtureScheduling(LoadScopeScheduling):
11+
"""Split by [] value. This is very hackish and might blow up any time!
12+
See https://github.com/pytest-dev/pytest-xdist/issues/18
13+
"""
14+
def _split_scope(self, nodeid):
15+
if '[' in nodeid:
16+
parameters = nodeid.rsplit('[')[-1].replace(']', '')
17+
return parameters.split('-')[0]
18+
return None
19+
20+
21+
def pytest_xdist_make_scheduler(log, config):
22+
return FixtureScheduling(config, log)

test/integration/test_integration.py

+62-26
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
from collections import namedtuple
22
import contextlib
33
import glob
4+
import os
45
import pprint
56
import shutil
67
import subprocess
@@ -16,14 +17,27 @@
1617

1718

1819
class MockEnv:
19-
'''Use this to work with mock. Mutliple instances are not safe.'''
20+
'''Use this to work with mock. Mutliple concurrent instances are safe.'''
2021
mock = ['mock', '-r', './mock.cfg']
2122

22-
def __init__(self):
23+
def __init__(self, worker_id):
24+
self.worker_id = worker_id
2325
self._run(['--init'], check=True)
2426

27+
@property
28+
def root(self):
29+
return 'taskotron-python-versions-{}'.format(self.worker_id)
30+
31+
@property
32+
def rootdir(self):
33+
return os.path.join(os.path.abspath('.'), 'mockroots', self.root)
34+
2535
def _run(self, what, **kwargs):
26-
return subprocess.run(self.mock + what, **kwargs)
36+
command = list(self.mock) # needs a copy not to change in place
37+
command.append('--config-opts=root={}'.format(self.root))
38+
command.append('--rootdir={}'.format(self.rootdir))
39+
command.extend(what)
40+
return subprocess.run(command, **kwargs)
2741

2842
def copy_in(self, files):
2943
self._run(['--copyin'] + files + ['/'], check=True)
@@ -52,12 +66,12 @@ def copy_out(self, directory, target, *, clean_target=False):
5266

5367

5468
@pytest.fixture(scope="session")
55-
def mock(request):
69+
def mock(worker_id, request):
5670
'''Setup a mock we can run Ansible tasks in under root'''
5771
if request.config.getoption('--fake'):
58-
mockenv = FakeMockEnv()
72+
mockenv = FakeMockEnv(worker_id)
5973
else:
60-
mockenv = MockEnv()
74+
mockenv = MockEnv(worker_id)
6175
files = ['taskotron_python_versions'] + glob.glob('*.py') + ['tests.yml']
6276
mockenv.copy_in(files)
6377
yield mockenv
@@ -187,8 +201,10 @@ def test_two_three_passed(results, request):
187201
assert results['dist.python-versions.two_three'].outcome == 'PASSED'
188202

189203

190-
def test_two_three_failed(tracer):
191-
assert tracer['dist.python-versions.two_three'].outcome == 'FAILED'
204+
@pytest.mark.parametrize('results', ('tracer',))
205+
def test_two_three_failed(results, request):
206+
results = request.getfixturevalue(results)
207+
assert results['dist.python-versions.two_three'].outcome == 'FAILED'
192208

193209

194210
@pytest.mark.parametrize('results', ('tracer', 'copr', 'admesh'))
@@ -207,8 +223,10 @@ def test_artifact_is_the_same(results, task, request):
207223
results['dist.python-versions.' + task].artifact)
208224

209225

210-
def test_artifact_contains_two_three_and_looks_as_expected(tracer):
211-
result = tracer['dist.python-versions.two_three']
226+
@pytest.mark.parametrize('results', ('tracer',))
227+
def test_artifact_contains_two_three_and_looks_as_expected(results, request):
228+
results = request.getfixturevalue(results)
229+
result = results['dist.python-versions.two_three']
212230
with open(result.artifact) as f:
213231
artifact = f.read()
214232

@@ -233,8 +251,11 @@ def test_naming_scheme_failed(results, request):
233251
assert results['dist.python-versions.naming_scheme'].outcome == 'FAILED'
234252

235253

236-
def test_artifact_contains_naming_scheme_and_looks_as_expected(copr):
237-
result = copr['dist.python-versions.naming_scheme']
254+
@pytest.mark.parametrize('results', ('copr',))
255+
def test_artifact_contains_naming_scheme_and_looks_as_expected(results,
256+
request):
257+
results = request.getfixturevalue(results)
258+
result = results['dist.python-versions.naming_scheme']
238259
with open(result.artifact) as f:
239260
artifact = f.read()
240261

@@ -258,9 +279,11 @@ def test_requires_naming_scheme_failed(results, request):
258279
assert task_result.outcome == 'FAILED'
259280

260281

282+
@pytest.mark.parametrize('results', ('tracer',))
261283
def test_artifact_contains_requires_naming_scheme_and_looks_as_expected(
262-
tracer):
263-
result = tracer['dist.python-versions.requires_naming_scheme']
284+
results, request):
285+
results = request.getfixturevalue(results)
286+
result = results['dist.python-versions.requires_naming_scheme']
264287
with open(result.artifact) as f:
265288
artifact = f.read()
266289

@@ -281,8 +304,10 @@ def test_artifact_contains_requires_naming_scheme_and_looks_as_expected(
281304
""").strip() in artifact.strip()
282305

283306

284-
def test_requires_naming_scheme_contains_python(yum):
285-
result = yum['dist.python-versions.requires_naming_scheme']
307+
@pytest.mark.parametrize('results', ('yum',))
308+
def test_requires_naming_scheme_contains_python(results, request):
309+
results = request.getfixturevalue(results)
310+
result = results['dist.python-versions.requires_naming_scheme']
286311
with open(result.artifact) as f:
287312
artifact = f.read()
288313

@@ -306,9 +331,11 @@ def test_executables_failed(results, request):
306331
assert task_result.outcome == 'FAILED'
307332

308333

334+
@pytest.mark.parametrize('results', ('docutils',))
309335
def test_artifact_contains_executables_and_looks_as_expected(
310-
docutils):
311-
result = docutils['dist.python-versions.executables']
336+
results, request):
337+
results = request.getfixturevalue(results)
338+
result = results['dist.python-versions.executables']
312339
with open(result.artifact) as f:
313340
artifact = f.read()
314341

@@ -352,9 +379,11 @@ def test_unvesioned_shebangs_failed(results, request):
352379
assert result.outcome == 'FAILED'
353380

354381

382+
@pytest.mark.parametrize('results', ('tracer',))
355383
def test_artifact_contains_unversioned_shebangs_and_looks_as_expected(
356-
tracer):
357-
result = tracer['dist.python-versions.unversioned_shebangs']
384+
results, request):
385+
results = request.getfixturevalue(results)
386+
result = results['dist.python-versions.unversioned_shebangs']
358387
with open(result.artifact) as f:
359388
artifact = f.read()
360389

@@ -378,9 +407,11 @@ def test_unvesioned_shebangs_mangled_failed(results, request):
378407
assert result.outcome == 'FAILED'
379408

380409

410+
@pytest.mark.parametrize('results', ('bucky',))
381411
def test_artifact_contains_mangled_unversioned_shebangs_and_looks_as_expected(
382-
bucky):
383-
result = bucky['dist.python-versions.unversioned_shebangs']
412+
results, request):
413+
results = request.getfixturevalue(results)
414+
result = results['dist.python-versions.unversioned_shebangs']
384415
with open(result.artifact) as f:
385416
artifact = f.read()
386417

@@ -420,15 +451,17 @@ def test_py3_support_failed(results, request):
420451
assert task_result.outcome == 'FAILED'
421452

422453

454+
@pytest.mark.parametrize('results', ('bucky',))
423455
def test_artifact_contains_py3_support_and_looks_as_expected(
424-
bucky):
456+
results, request):
425457
"""Test that py3_support check fails if the package is mispackaged.
426458
427459
NOTE: The test will start to fail as soon as python-bucky
428460
gets ported to Python 3 and its Bugzilla gets closed.
429461
See https://bugzilla.redhat.com/show_bug.cgi?id=1367012
430462
"""
431-
result = bucky['dist.python-versions.py3_support']
463+
results = request.getfixturevalue(results)
464+
result = results['dist.python-versions.py3_support']
432465
with open(result.artifact) as f:
433466
artifact = f.read()
434467

@@ -459,8 +492,11 @@ def test_python_usage_failed(results, request):
459492
assert task_result.outcome == 'FAILED'
460493

461494

462-
def test_artifact_contains_python_usage_and_looks_as_expected(jsonrpc):
463-
result = jsonrpc['dist.python-versions.python_usage']
495+
@pytest.mark.parametrize('results', ('jsonrpc',))
496+
def test_artifact_contains_python_usage_and_looks_as_expected(results,
497+
request):
498+
results = request.getfixturevalue(results)
499+
result = results['dist.python-versions.python_usage']
464500
with open(result.artifact) as f:
465501
artifact = f.read()
466502

tox.ini

+3-2
Original file line numberDiff line numberDiff line change
@@ -14,13 +14,14 @@ sitepackages = True
1414
[testenv:integration]
1515
deps =
1616
pytest
17+
pytest-xdist
1718
pyyaml
1819
basepython = python3
19-
commands = python -m pytest -v {posargs} test/integration
20+
commands = python -m pytest -v -n3 {posargs} test/integration
2021
sitepackages = False
2122

2223
[testenv:style]
2324
deps = flake8
2425
basepython = python3
25-
commands = python -m flake8 . --ignore=E402
26+
commands = python -m flake8 . --ignore=E402 --exclude=.git,__pycache__,.tox,.eggs,dist,build,mockroots
2627
sitepackages = False

0 commit comments

Comments
 (0)