forked from OpenCyphal/nunavut
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconftest.py
430 lines (354 loc) · 15.7 KB
/
conftest.py
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
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
#
# Copyright (C) OpenCyphal Development Team <opencyphal.org>
# Copyright Amazon.com Inc. or its affiliates.
# SPDX-License-Identifier: MIT
#
"""
Configuration for pytest tests including fixtures and hooks.
"""
import logging
import os
import pathlib
import re
import subprocess
import tempfile
import textwrap
import typing
import urllib
from doctest import ELLIPSIS
import pydsdl
import pytest
from sybil import Sybil
from sybil.parsers.rest import DocTestParser, PythonCodeBlockParser
from nunavut import Namespace
# +-------------------------------------------------------------------------------------------------------------------+
# | TEST FIXTURES
# +-------------------------------------------------------------------------------------------------------------------+
@pytest.fixture
def run_nnvg(request: pytest.FixtureRequest) -> typing.Callable: # pylint: disable=unused-argument
"""
Test helper for invoking the nnvg command-line script as part of a unit test.
"""
def _run_nnvg(
_: typing.Any,
args: typing.List[str],
check_result: bool = True,
env: typing.Optional[typing.Dict[str, str]] = None,
raise_called_process_error: bool = False,
) -> subprocess.CompletedProcess:
"""
Helper to invoke nnvg for unit testing within the proper python coverage wrapper.
"""
coverage_args = ["coverage", "run", "--parallel-mode", "-m", "nunavut"]
this_env = os.environ.copy()
if env is not None:
this_env.update(env)
try:
return subprocess.run(
coverage_args + args, check=check_result, stdout=subprocess.PIPE, stderr=subprocess.PIPE, env=this_env
)
except subprocess.CalledProcessError as e:
if raise_called_process_error:
raise e
raise AssertionError(e.stderr.decode("utf-8")) from e
return _run_nnvg
class GenTestPaths:
"""Helper to generate common paths used in our unit tests."""
def __init__(self, test_file: str, keep_temporaries: bool, node_name: str):
test_file_path = pathlib.Path(test_file)
self.test_name = f"{test_file_path.parent.stem}_{node_name}"
self.test_dir = test_file_path.parent
search_dir = self.test_dir.resolve()
while search_dir.is_dir() and not (search_dir / pathlib.Path("src")).is_dir():
search_dir = search_dir.parent
self.root_dir = search_dir
self.templates_dir = self.test_dir / pathlib.Path("templates")
self.support_templates_dir = self.test_dir / pathlib.Path("support")
self.dsdl_dir = self.test_dir / pathlib.Path("dsdl")
self.lang_src_dir = self.root_dir / pathlib.Path("src") / pathlib.Path("nunavut") / pathlib.Path("lang")
self._keep_temp = keep_temporaries
self._out_dir: typing.Optional[pathlib.Path] = None
self._build_dir: typing.Optional[pathlib.Path] = None
self._dsdl_dir: typing.Optional[pathlib.Path] = None
self._temp_dirs: typing.List[tempfile.TemporaryDirectory] = []
print(f'Paths for test "{self.test_name}" under dir {self.test_dir}')
print(f"(root directory: {self.root_dir})")
def test_path_finalizer(self) -> None:
"""
Finalizer to clean up any temporary directories created during the test.
"""
for temporary_dir in self._temp_dirs:
temporary_dir.cleanup()
self._temp_dirs.clear()
def create_new_temp_dir(self, dir_key: str) -> pathlib.Path:
"""
Create a new temporary directory for the test case.
"""
if self._keep_temp:
result = self._ensure_dir(self.build_dir / pathlib.Path(dir_key))
else:
temporary_dir = tempfile.TemporaryDirectory(dir=str(self.build_dir))
result = pathlib.Path(temporary_dir.name)
self._temp_dirs.append(temporary_dir)
return result
@property
def out_dir(self) -> pathlib.Path:
"""
The directory to place test output under for this test case.
"""
if self._out_dir is None:
self._out_dir = self.create_new_temp_dir(urllib.parse.quote_plus(self.test_name))
return self._out_dir
@property
def build_dir(self) -> pathlib.Path:
"""
The directory to place build artifacts under for this test case.
"""
if self._build_dir is None:
self._build_dir = self._ensure_dir(self.root_dir / pathlib.Path("build"))
return self._build_dir
@staticmethod
def find_outfile_in_namespace(
typename: str, namespace: Namespace, type_version: pydsdl.Version = None
) -> typing.Optional[str]:
"""
Find the output file for a given type in a namespace.
"""
found_outfile: typing.Optional[str] = None
for dsdl_type, outfile in namespace.get_all_types():
if dsdl_type.full_name == typename:
if type_version is not None:
if isinstance(dsdl_type, pydsdl.CompositeType) and type_version == dsdl_type.version:
found_outfile = str(outfile)
break
# else ignore this since it's either a namespace or it's not the version
# of the type we're looking for.
elif found_outfile is not None:
raise RuntimeError(
f"Type {typename} had more than one version for this test but no type version argument"
" was provided."
)
else:
found_outfile = str(outfile)
return found_outfile
@staticmethod
def _ensure_dir(path_dir: pathlib.Path) -> pathlib.Path:
try:
path_dir.mkdir()
except FileExistsError:
pass
if not path_dir.exists() or not path_dir.is_dir():
raise RuntimeWarning(f'Test directory "{path_dir}" was not setup properly. Tests may fail.')
return path_dir
@pytest.fixture(scope="function")
def gen_paths(request: pytest.FixtureRequest) -> GenTestPaths:
"""
Used by the "gentest" unit tests in Nunavut to standardize output paths for generated code created as part of
the tests. Use the --keep-generated argument to disable the auto-clean behaviour this fixture provides by default.
"""
g = GenTestPaths(str(request.fspath), request.config.option.keep_generated, request.node.name)
request.addfinalizer(g.test_path_finalizer)
return g
@pytest.fixture(scope="module")
def gen_paths_for_module(request: pytest.FixtureRequest) -> GenTestPaths: # pylint: disable=unused-argument
"""
Used by our Sybil doctests in Nunavut to standardize output paths for generated code created as part of
the tests. Use the --keep-generated argument to disable the auto-clean behaviour this fixture provides by default.
Note: this fixture is different than gen_paths because it is scoped to the module level. This is useful for
Sybil tests that share temporary files across different test blocks within the same document.
"""
g = GenTestPaths(str(request.fspath), request.config.option.keep_generated, request.node.name)
request.addfinalizer(g.test_path_finalizer)
return g
class _UniqueNameEvaluator:
def __init__(self) -> None:
self._found_names: typing.Set[str] = set()
def __call__(self, expected_pattern: str, actual_value: str) -> None:
assert re.match(expected_pattern, actual_value) is not None
assert actual_value not in self._found_names
self._found_names.add(actual_value)
@pytest.fixture(scope="function")
def unique_name_evaluator(request: pytest.FixtureRequest) -> _UniqueNameEvaluator: # pylint: disable=unused-argument
"""
Class that defined ``assert_is_expected_and_unique`` allowing assertion that a set of values
in a single test adhere to a provided pattern and are unique values (compared to other values
provided to this method).
.. code-block:: python
def test_is_unique(unique_name_evaluator) -> None:
value0 = '_foo0_'
value1 = '_foo1_'
unique_name_evaluator(r'_foo\\d_', value0)
unique_name_evaluator(r'_foo\\d_', value1)
# This next line should fail because value 0 was already evaluated so it
# is not unique
unique_name_evaluator(r'_foo\\d_', value0)
"""
return _UniqueNameEvaluator()
@pytest.fixture
def assert_language_config_value(request: pytest.FixtureRequest) -> typing.Callable: # pylint: disable=unused-argument
"""
Assert that a given configuration value is set for the target language.
"""
from nunavut.lang import LanguageContext, LanguageContextBuilder # pylint: disable=import-outside-toplevel
def _assert_language_config_value(
target_language: typing.Union[str, LanguageContext],
key: str,
expected_value: typing.Any,
message: typing.Optional[str],
) -> None:
if isinstance(target_language, LanguageContext):
lctx = target_language
else:
lctx = (
LanguageContextBuilder(include_experimental_languages=True)
.set_target_language(target_language)
.create()
)
language = lctx.get_target_language()
if language is None:
raise AssertionError("Unable to determine target language from provided arguments.")
if expected_value != language.get_config_value(key):
raise AssertionError(message)
return _assert_language_config_value
@pytest.fixture
def jinja_filter_tester(request: pytest.FixtureRequest): # pylint: disable=unused-argument
"""
Use to create fluent but testable documentation for Jinja filters and tests
Example:
.. code-block: python
from nunavut._templates import template_environment_filter
@template_environment_filter
def filter_dummy(env, input):
return input
# Given
I = 'foo'
# and
template = '{{ I | dummy }}'
# then
rendered = I
jinja_filter_tester(filter_dummy, template, rendered, 'c', I=I)
You can also control the language context:
.. code-block: python
from nunavut.lang import LanguageContextBuilder, Language
lctx = (
LanguageContextBuilder()
.set_target_language("c")
.set_target_language_configuration_override(Language.WKCV_ENABLE_STROPPING, False)
.create()
)
jinja_filter_tester(filter_dummy, template, rendered, lctx, I=I)
"""
from nunavut.jinja.jinja2 import DictLoader # pylint: disable=import-outside-toplevel
from nunavut.lang import LanguageContext, LanguageContextBuilder # pylint: disable=import-outside-toplevel
def _make_filter_test_template(
filter_or_list_of_filters: typing.Union[None, typing.Callable, typing.List[typing.Callable]],
body: str,
expected: str,
target_language_or_language_context: typing.Union[str, LanguageContext],
**additional_globals: typing.Optional[typing.Dict[str, typing.Any]],
) -> str:
from nunavut.jinja import CodeGenEnvironmentBuilder # pylint: disable=import-outside-toplevel
if isinstance(target_language_or_language_context, LanguageContext):
lctx = target_language_or_language_context
else:
lctx = (
LanguageContextBuilder(include_experimental_languages=True)
.set_target_language(target_language_or_language_context)
.create()
)
if filter_or_list_of_filters is None:
additional_filters: typing.Optional[typing.Dict[str, typing.Callable]] = {}
elif isinstance(filter_or_list_of_filters, list):
additional_filters = {}
for filter_method in filter_or_list_of_filters:
additional_filters[filter_method.__name__] = filter_method
else:
additional_filters = {filter_or_list_of_filters.__name__: filter_or_list_of_filters}
e = (
CodeGenEnvironmentBuilder(DictLoader({"test": body}), lctx)
.set_allow_filter_test_or_use_query_overwrite(True)
.add_filters(**additional_filters)
.add_globals(**additional_globals)
.create()
)
e.update_nunavut_globals(
*lctx.get_target_language().get_support_module(), omit_serialization_support=True, embed_auditing_info=True
)
rendered = str(e.get_template("test").render())
if expected != rendered:
msg = "Unexpected template output\n\texpected : {}\n\twas : {}".format(
expected.replace("\n", "\\n"), rendered.replace("\n", "\\n")
)
raise AssertionError(msg)
return rendered
return _make_filter_test_template
@pytest.fixture
def mock_environment(request: pytest.FixtureRequest) -> typing.Any: # pylint: disable=unused-argument
"""
A MagicMock that can be used where a jinja environment is needed.
"""
from unittest.mock import MagicMock # pylint: disable=import-outside-toplevel
magic_mock_environment = MagicMock()
support_mock = MagicMock()
magic_mock_environment.globals = {"nunavut": support_mock}
support_mock.support = {"omit": True}
return magic_mock_environment
# +-------------------------------------------------------------------------------------------------------------------+
# | PYTEST HOOKS
# +-------------------------------------------------------------------------------------------------------------------+
def pytest_configure(config: typing.Any) -> None: # pylint: disable=unused-argument
"""
See https://docs.pytest.org/en/6.2.x/reference.html#initialization-hooks
"""
# pydsdl._dsdl_definition is reeeeeeealy verbose at the INFO level and below. Turn this down to reduce
# scroll-blindness.
logging.getLogger("pydsdl._dsdl_definition").setLevel(logging.WARNING)
# A lot of DEBUG noise in the other loggers so we'll tune this down to INFO and higher.
logging.getLogger("pydsdl._namespace").setLevel(logging.INFO)
logging.getLogger("pydsdl._data_type_builder").setLevel(logging.INFO)
def pytest_addoption(parser: pytest.Parser) -> None:
"""
See https://docs.pytest.org/en/6.2.x/reference.html#initialization-hooks
"""
parser.addoption(
"--keep-generated",
action="store_true",
help=textwrap.dedent(
"""
If set then the temporary directory used to generate files for each test will be left after
the test has completed. Normally this directory is temporary and therefore cleaned up automatically.
:: WARNING ::
This will leave orphaned files on disk. They won't be big but there will be a lot of them.
:: WARNING ::
Do not run tests in parallel when using this option.
"""
),
)
# +-------------------------------------------------------------------------------------------------------------------+
# | SYBIL
# +-------------------------------------------------------------------------------------------------------------------+
pytest_collect_file = Sybil(
parsers=[
DocTestParser(optionflags=ELLIPSIS),
PythonCodeBlockParser(),
],
pattern="**/*",
excludes=[
"**/markupsafe/*",
"**/jinja2/*",
"**/static/*",
"**/.*/*",
"**/.*",
"**/CONTRIBUTING.rst",
"**/verification/*",
"**/prof/*",
"*.j2",
"*.png",
],
fixtures=[
"jinja_filter_tester",
"gen_paths",
"gen_paths_for_module",
"assert_language_config_value",
],
).pytest()