1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
|
import json
import os
import sys
import types
import logging
import py
import pytest
import _pytest.doctest
import six
from library.python.pytest import module_utils
import library.python.testing.filter.filter as test_filter
import yatest.common
import __res
if six.PY3:
import pathlib
def _make_module_with_single_skipped_test(module_name, file_name, exc):
if exc.msg is not None:
mark = pytest.mark.skip(reason=exc.msg)
else:
mark = pytest.mark.skip()
module = types.ModuleType(module_name)
module.__file__ = file_name
@mark
def module_level_skip():
pass
module_level_skip.__test__ = True
setattr(module, 'pytest.skip', module_level_skip)
return module
class LoadedModule(pytest.Module):
def __init__(self, module_name, **kwargs):
self.ya_module_name = module_name
if module_name.startswith('__tests__.'):
self.ya_display_module_name = module_name[len('__tests__.') :]
else:
self.ya_display_module_name = module_name
# Always passed to `super().__init__` explicitly.
kwargs.pop('nodeid', None)
nodeid = module_utils.get_proper_module_path(self.ya_module_name)
self.ya_is_fake_module = False
if nodeid is None:
self.ya_is_fake_module = True
return
# Avoid specializing on `name`. It may change in the future if we add virtual collection nodes for directories.
name = nodeid
# `path` and `fspath` are required to be absolute paths.
# The original source or build directory may be unavailable (at least in CI) and is irrelevant at this point.
# Meanwhile, `source_path` is useful, because it allows to find nearby test data, like with vanilla pytest.
path = yatest.common.source_path(nodeid)
if six.PY3:
kwargs['path'] = kwargs.get('path') or pathlib.Path(path)
super().__init__(name=name, nodeid=nodeid, **kwargs)
else:
kwargs['fspath'] = kwargs.get('fspath') or py.path.local(path)
super(LoadedModule, self).__init__(nodeid=nodeid, **kwargs)
self.name = name
@classmethod
def from_parent(cls, **kwargs):
return getattr(super(LoadedModule, cls), 'from_parent', cls)(**kwargs)
def _getobj(self):
# A simplified version of pytest `importtestmodule` that works with resfs.
try:
__import__(self.ya_module_name)
except pytest.skip.Exception as e:
if not e.allow_module_level:
raise RuntimeError(
"Using pytest.skip outside of a test will skip the entire module. If that's your intention, pass `allow_module_level=True`."
)
# DEVTOOLSSUPPORT-79470: reraising pytest.skip.Exception from here seems to break reporting in ya plugin.
# Instead, pretend to be a module with a single skipped test.
return _make_module_with_single_skipped_test(self.ya_module_name, self.nodeid, e)
except Exception as e:
msg = 'Failed to load module "{}" and obtain list of tests due to an error'.format(
self.ya_display_module_name
)
logging.exception('%s: %s', msg, e)
etype, exc, tb = sys.exc_info()
six.reraise(etype, type(exc)('{}\n{}'.format(exc, msg)), tb)
return sys.modules[self.ya_module_name]
class DoctestModule(LoadedModule):
def collect(self):
import doctest
module = self._getobj()
# uses internal doctest module parsing mechanism
finder = doctest.DocTestFinder()
if six.PY3:
optionflags = _pytest.doctest.get_optionflags(self.config)
else:
optionflags = _pytest.doctest.get_optionflags(self)
runner = doctest.DebugRunner(verbose=0, optionflags=optionflags)
try:
for test in finder.find(module, self.ya_display_module_name):
if test.examples: # skip empty doctests
yield getattr(_pytest.doctest.DoctestItem, 'from_parent', _pytest.doctest.DoctestItem)(
name=test.name, parent=self, runner=runner, dtest=test
)
except Exception:
logging.exception('DoctestModule failed, probably you can add NO_DOCTESTS() macro to ya.make')
etype, exc, tb = sys.exc_info()
msg = 'DoctestModule failed, probably you can add NO_DOCTESTS() macro to ya.make'
six.reraise(etype, type(exc)('{}\n{}'.format(exc, msg)), tb)
# NOTE: Since we are overriding collect method of pytest session, pytest hooks are not invoked during collection.
# This is solely a helper function, not an implementation of a pytest hook.
def _pytest_ignore_collect(module, config, filenames_from_full_filters, accept_filename_predicate):
if module.ya_is_fake_module:
return True
# TODO refactor to use meaningful attributes like `path` instead?
# test_file_filter would need to be relative to repository root.
legacy_name = module.ya_display_module_name + '.py'
if config.option.mode == 'list':
return not accept_filename_predicate(legacy_name)
if filenames_from_full_filters is not None and legacy_name not in filenames_from_full_filters:
return True
test_file_filter = getattr(config.option, 'test_file_filter', None)
if test_file_filter and legacy_name != test_file_filter.replace('/', '.'):
return True
return False
def _patch_set_initial_conftests(pluginmanager):
"""
Avoids attempts to access filesystem directly in built-in `pytest_load_initial_conftests` hook impl.
"""
if six.PY3:
def _set_initial_conftests_py3(pyargs, noconftest, *args, **kwargs):
pluginmanager._noconftest = noconftest
pluginmanager._using_pyargs = pyargs
pluginmanager._set_initial_conftests = _set_initial_conftests_py3
else:
def _set_initial_conftests_py2(namespace):
pluginmanager._noconftest = namespace.noconftest
pluginmanager._using_pyargs = namespace.pyargs
pluginmanager._set_initial_conftests = _set_initial_conftests_py2
def _find_resfs_test_and_doctest_modules():
prefix = '__tests__.'
test_modules = [
# fmt: off
name
for name in sys.extra_modules
if name.startswith(prefix) and not name.endswith('.conftest')
# fmt: on
]
doctest_packages = __res.find("PY_DOCTEST_PACKAGES") or ""
if isinstance(doctest_packages, bytes):
doctest_packages = doctest_packages.decode('utf-8')
doctest_packages = doctest_packages.split()
def is_doctest_module(name):
for package in doctest_packages:
if name == package or name.startswith(str(package) + "."):
return True
return False
doctest_modules = [
# fmt: off
name
for name in sys.extra_modules
if is_doctest_module(name)
# fmt: on
]
return test_modules, doctest_modules
@pytest.hookimpl(tryfirst=True)
def pytest_load_initial_conftests(early_config):
_patch_set_initial_conftests(early_config.pluginmanager)
def pytest_sessionstart(session):
resfs_test_modules, resfs_doctest_modules = _find_resfs_test_and_doctest_modules()
def collect(*args, **kwargs):
config = session.config
# A custom function that is set in conftests.py.
config._ya_register_non_initial_conftests()
accept_filename_predicate = test_filter.make_py_file_filter(config.option.test_filter)
full_test_names_file_path = config.option.test_list_path
filenames_filter = None
if full_test_names_file_path and os.path.exists(full_test_names_file_path):
with open(full_test_names_file_path, 'r') as afile:
# in afile stored 2 dimensional array such that array[modulo_index] contains tests which should be run in this test suite
full_names_filter = set(json.load(afile)[int(config.option.modulo_index)])
filenames_filter = set(map(lambda x: x.split('::')[0], full_names_filter))
for test_module in resfs_test_modules:
module = LoadedModule.from_parent(module_name=test_module, parent=session)
if not _pytest_ignore_collect(module, config, filenames_filter, accept_filename_predicate):
yield module
if os.environ.get('YA_PYTEST_DISABLE_DOCTEST', 'no') == 'no':
module = DoctestModule.from_parent(module_name=test_module, parent=session)
if not _pytest_ignore_collect(module, config, filenames_filter, accept_filename_predicate):
yield module
if os.environ.get('YA_PYTEST_DISABLE_DOCTEST', 'no') == 'no':
for doctest_module in resfs_doctest_modules:
module = DoctestModule.from_parent(module_name=doctest_module, parent=session)
if not module.ya_is_fake_module:
yield module
session.collect = collect
|